КаталогИндекс раздела
НазадОглавлениеВперед


Лабораторная работа N10
УПРАВЛЕНИЕ ПРОГРАММАМИ

1. Цель работы

Изучение принципов управления программами в MS DOS и приобретение практических навыков работы с префиксом программного сегмента и его полями.

2. Темы для предварительной проработки

3. Постановка задачи

Разработать программу, производящую форматный вывод на печать своего Префикса Программного Сегмента.

4. Порядок выполнения работы

Порядок выполнения работы и содержание отчета определены в общих указаниях.

5. Пример решения задачи

5.1. Разработка алгоритма решения

5.1.1. Структура программы

Программа состоит из основной программы и двух функций:

5.1.2. Описание переменных

Переменные, глобальные для всей программы:

5.1.3. Описание алгоритма программы

Данная программа производит распечатку основных полей своего PSP. Для этого префикс программного сегмента представим в виде следующей структуры:

       struct psp
       {        /* ФОРМАТ PSP */
         byte ret_op[2];       /* команда INT 20h                */
         word end_of_mem;      /* вершина доступной памяти       */
         byte reserved1;
         byte old_call_dos[5]; /* старый вызов DOS               */
         void *term_ptr;       /* адрес завершения               */
         void *ctrlbrk_ptr;    /* адрес обработчика Ctrl+Break   */
         void *criterr_ptr;    /* адрес обработчика крит.ошибок  */
         word father_psp;      /* PID родителя                   */
         byte JFT[20];         /* таблица файлов программы       */
         word env_seg;         /* адрес окружения                */
         void *stack_ptr;      /* адрес стека                    */
         word JFT_size;        /* размер таблицы файлов          */
         byte *JFT_ptr;        /* адрес таблицы файлов           */
         byte reserved2[24];
         byte new_call_dos[3]; /* новый вызов DOS                */
         } *p_psp;

Поле ret_op используется для возможного завершения программы по команде RET 0, поле old_call_dos, содержит команду вызова диспетчера функций DOS. Обращение к этому полю в программе может использоваться вместо команды INT 21h, но в современных версиях DOS для этих целей лучше обращаться к полю new_call_dos.

Поле end_of_mem содержит сегментный адрес конца доступной памяти в системе. В три поля: term_ptr, ctrlbrk_ptr, criterr_ptr DOS при загрузке программы копирует содержимое векторов прерываний: 22h, 23h, 24, представляющее собой адреса обработчиков: завершения программы, комбинации клавиш Ctrl+Break, критической ошибки - соответственно. Предполагается, что программа может свободно перенаправить эти векторы на собственные обработчики соответствующих ситуаций, но от забот по восстановлению векторов программа избавляется, так как при ее завершении DOS сама восстанавливает векторы из соответствующих полей PSP завершаемой программы. Для аналогичных целей предназначено и поле stack_ptr - в нем сохраняется (а при завершении - из него восстанавливается) адрес стека, использовавшегося до вызова программы. Поле, именуемое father_psp, содержит сегментный адрес PSP родителя - программы, запустившей данную программу, обычно родителем является COMMAND.COM.

При загрузке программы DOS, кроме программного сегмента, создает для нее еще и сегмент окружения. Сегмент окружения содержит ASCIIZ-строки, задающие значения некоторых глобальных переменных, эти значения могут устанавливаться командой DOS SET, они доступны командным файлам и - через PSP - программам. Набор строк окружения заканчивается пустой ASCIIZ-строкой (нулем). В DOS 3.0 и выше за ним следует еще 2-байтное число строк вызова (обычно 1) и далее - строка (или строки) вызова программы. Обычно в первую (до строк вызова) часть порождаемой программы копируется содержимое окружения программы-родителя. Программа имеет доступ к своему сегменту окружения через поле env_seg PSP, содержащее сегментный адрес окружения.

Поле JFT (Job File Table - Таблица Файлов Задачи) представляет собой массив из 20 однобайтных элементов. При открытии программой файла DOS формирует для него блок-описатль в системной таблице файлов и помещает ссылку на него (его номер) в свободный элемент JFT. Дескриптор файла, возвращаемый программе DOS при открытии файла, является номером элемента в JFT. При запуске программы первые пять элементов создаваемой для нее JFT содержат ссылки на системные файлы, остальные свободны. При обработке JFT DOS использует не прямое обращение к полю JFT PSP, а косвенное - через поле JFT_ptr, а в качастве ограничителя размера JFT - не константу 20, а значение поля JFT_size PSP.

После всего, сказанного выше, не составляет труда написать программу, осуществляющую форматный вывод своего префикса программного сегмента. Для в начале необходимо определить версию DOS (с помощью функции get_DOS_version_h() и получить адрес PSP (с помощью функции addr_PSP()).

Функция get_DOS_version_h() определяет старшее число номера версии DOS, используя для этого функцию DOS 30h (прерывание 21h), которая возвращает в регистре AL старшее число номера версии, а в регистре AH - младшее число. Нас интересует только значение регистра AL.

Функция addr_PSP() возвращает сегментный адрес PSP путем использования функции DOS 62h:

          Вход:  AH = 62h
          Выход:  BX = сегментный адрес PSP текущего процесса.

5.2. Текст программы

/*----------------Лабораторная работа N10-----------------*/
/*----------------"Управление программами"----------------*/

/* Подключение стандартных заголовков */
#include <dos.h>
#include <conio.h>

/* Типы данных */
#define byte unsigned char
#define word unsigned int

/* Описание функций */
void get_DOS_version_h(void);  /* Определение версии DOS  */
void addr_PSP (void);          /* Получение адреса PSP    */

struct psp
{        /* ФОРМАТ PSP */
  byte ret_op[2];       /* команда INT 20h                */
  word end_of_mem;      /* вершина доступной памяти       */
  byte reserved1;
  byte old_call_dos[5]; /* старый вызов DOS               */
  void *term_ptr;       /* адрес завершения               */
  void *ctrlbrk_ptr;    /* адрес обработчика Ctrl+Break   */
  void *criterr_ptr;    /* адрес обработчика крит.ошибок  */
  word father_psp;      /* PID родителя                   */
  byte JFT[20];         /* таблица файлов программы       */
  word env_seg;         /* адрес окружения                */
  void *stack_ptr;      /* адрес стека                    */
  word JFT_size;        /* размер таблицы файлов          */
  byte *JFT_ptr;        /* адрес таблицы файлов           */
  byte reserved2[24];
  byte new_call_dos[3]; /* новый вызов DOS                */
  } *p_psp;

word pid;               /* сегм.адрес PSP                 */
int dos_ver,            /* версия DOS                     */
    i, l, j;
char *s;
union REGS rr;

main()
{
  textbackground(0);
  clrscr();
  textattr(0x0a);
  cprintf("---------------");
  cprintf("             Лабораторная  работа N10             ");
  cprintf("---------------");
  cprintf("---------------");
  cprintf("              Управление программами              ");
  cprintf("---------------");
  textcolor(11);
  get_DOS_version_h();
  addr_PSP();
  /* распечатка PSP */
  cprintf("\n\n            Адрес PID = %04X\n\n\r",pid);
  p_psp=(struct psp *)MK_FP(pid,0);
  textcolor(10);
  cprintf("Команды:\n\r");
  cprintf("--------\n\r");
  textcolor(14);
  cprintf("     Завершение - int 20h:");
  textcolor(12);
  cprintf("  %02X %02X\n\r",p_psp->ret_op[0],p_psp->ret_op[1]);
  textcolor(14);
  cprintf("     Старый вызов DOS:      ");
  textcolor(12);
  for (i=0;i<5;cprintf("%02X ",p_psp->old_call_dos[i++]));
  textcolor(14);
  cprintf("\n\r     Новый вызов DOS:       ");
  textcolor(12);
  for(i=0;i<3;cprintf("%02X ",p_psp->new_call_dos[i++]));
  textcolor(10);
  cprintf("\n\n\rАдреса:\n\r");
  cprintf("-------\n\r");
  textcolor(14);
  cprintf("    Конец памяти:             ");
  textcolor(12);
  cprintf("%04X:0000\n\r",p_psp->end_of_mem);
  textcolor(14);
  cprintf("    Обработчик завершения:    ");
  textcolor(12);
  cprintf("%Fp\n\r",p_psp->term_ptr);
  textcolor(14);
  cprintf("    Обработчик Ctrl+Break:    ");
  textcolor(12);
  cprintf("%Fp\n\r",p_psp->ctrlbrk_ptr);
  textcolor(14);
  cprintf("    Обработчик критич.ошибки: ");
  textcolor(12);
  cprintf("%Fp\n\r",p_psp->criterr_ptr);
  textcolor(14);
  cprintf("    Стек:                     ");
  textcolor(12);
  cprintf("%Fp\n\n\r",p_psp->stack_ptr);
  textcolor(14);
  cprintf("\n\rРодитель: ");
  textcolor(12);
  cprintf("%04X  ",p_psp->father_psp);
  textcolor(0x8b);
  cprintf("\n\n\rНажмите любую клавишу ...\n\r\7");
  getch();
  clrscr();
  textattr(0x0a);
  cprintf("---------------");
  cprintf("             Лабораторная  работа N10             ");
  cprintf("---------------");
  cprintf("---------------");
  cprintf("              Управление программами              ");
  cprintf("---------------");
  /* Распечатка таблицы файлов */
  s=p_psp->JFT_ptr;
  textcolor(10);
  cprintf("\n\n\rТаблица файлов: ");
  textcolor(12);
  cprintf("%Fp (%d) ",s,p_psp->JFT_size);
  textcolor(11);
  if (s==(byte *)p_psp+0x18)
    cprintf(" -  в этом же PSP");
  cprintf("\n\r");
  for (i=0; ++i<=p_psp->JFT_size; cprintf("%d ",*(s++)));
  textcolor(10);
  cprintf("\n\n\rОкружение DOS: ");
  textcolor(12);
  cprintf("%04X\n\r",p_psp->env_seg);
  s=(char *)MK_FP(p_psp->env_seg,0);
  textcolor(11);
  while(l=strlen(s))
  {
    cprintf("      %s\n\r",s);
    s+=l+1;

  }
  if (dos_ver>2)
  {
    /* для DOS 3.0 и дальше можно получить строку вызова */
    s++;
    l=*((int *)s);
    textcolor(10);
    cprintf("\n\rЧисло строк вызова: ");
    textcolor(12);
    cprintf("%d\n\r",l);
    s+=2;
    textcolor(11);
    for(i=0; i<l; i++)
    {
      cprintf("%s\n\r",s);
      s+=strlen(s)+1;
    }
  }
  textattr(0x8b);
  cprintf("\n\n\n\n\rНажмите любую клавишу ...\7");
  textattr(0x07);
  cprintf("\n\r");
  getch();
  clrscr();
}

/* Определение версии DOS */
void get_DOS_version_h(void)
{
  rr.h.ah=0x30;
  intdos(&rr,&rr);
  dos_ver=rr.h.al;
}

/* Получение адреса PSP */
void addr_PSP (void)
{
  rr.h.ah=0x62;
  intdos(&rr,&rr);
  pid=rr.x.bx;
}

5.6. Результаты работы программы

В процессе работы программы на экран была выведена следующая информация:

--------------------------------------------------------------------------------
---------------             Лабораторная  работа N10             ---------------
---------------              Управление программами              ---------------


            Адрес PID = 0BA0

Команды:
--------
     Завершение - int 20h:  CD 20
     Старый вызов DOS:      9A F0 FE 1D F0
     Новый вызов DOS:       CD 21 CB

Адреса:
-------
    Конец памяти:             9FC0:0000
    Обработчик завершения:    0AFA:02B1
    Обработчик Ctrl+Break:    0AFA:014A
    Обработчик критич.ошибки: 0AFA:0155
    Стек:                     0E04:0F94


Родитель: 0AFA

Таблица файлов: 0BA0:0018 (20)  -  в этом же PSP
1 1 1 0 2 -1 -1 -1 -1 -1 -1 -1 -1 -1 -1 -1 -1 -1 -1 -1

Окружение DOS: 0A1E
      CONFIG=STD
      COMSPEC=C:\DOS\COMMAND.COM
      PROMPT=$p$g
      PATH=D:\WIN;C:\;C:\DOS;C:\ARH;C:\NC;C:\BAT;D:\TP;D:\TP7;D:\BC\BIN
      TEMP=d:\~TMP

Число строк вызова: 1
D:\TC\TC_LAB10.EXE
--------------------------------------------------------------------------------


КаталогИндекс раздела
НазадОглавлениеВперед