Архитектура Аудит Военная наука Иностранные языки Медицина Металлургия Метрология
Образование Политология Производство Психология Стандартизация Технологии


Описание алгоритма программы. Функция readvect() читает вектор заданного прерывания. Текст программы



Программу можно назвать моделью процесса АЦП. Главная программа постоянно вычисляет значения заданной функции F(x) при переменном аргументе, что имитирует непрерывный сигнал, а обработчик прерывания 8 имитирует преобразователь с постоянным шагом дискретизации по времени. Перед началом работы канал 0 таймера программируется на частоту в 2 раза большую обычной (записью в порт 43h управляющего байта 00110110b=36h, а потом посылкой в порт 40h новогозначения коэффициента деления), таким образом, «частота дискретизации') составляет около 36.4 Гц. При поступлении следующего прерывания запоминается текущее значение функции F(x), старый обработчик прерывания oldtime вызывается не при кожному прерывании, а лишь один раз из двух (переменная kf - счетчик по модулю 2), когда oldtime не вызывается, наш обработчик сам сбрасывает контроллер прерываний посылкой значения 20h в порт 20h. После набора 100 «показаний АЦП» восстанавливается старый вектор обработчика таймера, а результат аналого-цифрового преобразование выводится на терминал в графическом режиме в виде решетчатой функции.

Функция readvect() читает вектор заданного прерывания. Для чтения вектора при меняется функция 35h DOS (прерывания 21h):

Вход: АН = 35h;

AL = номер вектора прерывания,

Выход: ES:BX = адрес программы обработки прерывания.

Функция writevect() устанавливает новый вектор прерывания по заданному адресу. Для записи вектора применяется функция 25h DOS:

Вход: АН = 25h;

AL = номер вектора прерывания;

DS:BX = 4-байтный адрес нового обработчика прерывания.

 

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


#inсludе <dos.h>

#inсludе <math.h>

#inсludе <stdlib.h>

#inсludе <graphics.h>

#inсludе <time.h>

#include <conio.h>

 

#define TIMEINT 8 /* Прерывание таймера */

#define NN 100 /* максимальное количество показаний */

 

void interrupt (*oldtime)(); /* Новый обработчик прерываний таймера */

void interrupt newtime(); /* Старый обработчик прерываний таймера */

static int y[NN]; /* Накопитель показаний */'

static int nу; /* Индекс в массиве у */

static int ус; /* Текущее значение        */,

static int kf; /* Счетчик вызовов oldtime */

union REGS rr; /* Запись нового вектора */

struct SREGS sr;

void *readvect(int in); /*Получение старого вектора */

void writevect(int in,void*h); /* Запись нового вектора */

/*------------------------------------------------------------------------------*/

void main()

{

unsigned oldtic=65535u; /* Старый коэффициент деления*/

unsigned newtic=32768u; /* Новый козфф. деления */

unt dd,      /* Графический драйвер */

 

m,            /* Графический режим */

errorcode;      /* Код ошибки       */

double х; /* Аргумент функций sin и cos */

 

textbackground(0);

clrscr();

textattr(0x0a);

cprintf(" Лабораторная работа “)

срrintf ("\n Управление таймером” )

textattr(0x8e); ,

gotoxy(35, 12);

cprintf("Please wait");

/* Программирование канала 0 */

оutроrtb(0х43,0хЗ6); /* Управляющий байт */ outportb(0x40,newtic&0x00ff); /* Младший байт счетчика */ outportb(0x40,newtic>>8); /* Старший байт счетчика */

nу= -1; /* Признак того, что АЦП еще не началось */

kf=15;

/* Подключение к вектору */

oldtime=readvect(TIMEINT);

writevect(TIMEINT,newtime);

/* Запуск "непрерывного процесса" */

randomize ( );

for (х=nу=О; ny<NN; х+=1)

yc=(int)(50*(sin(x/10)+cos(x/8))+random(11)+150);

/* Восстановление вектора */

writevect(TIMEINT,oldtime);

/* Восстановление канала 0 */

outportb(0x43,Ox36); /* Управляющий байт */ outportb(0x40,01dtic&0x00ff); /* Младший байт счетчика */ outportb(0x40,01dtic>>8); /* Старший байт счетчика */

 

/* Вывод запомненных результатов */

dd=З; /* EGA, 16 цветов */

m=1; /* Режим 640*З50 */

initgraph(&dd,&m, "");

/* проверка результата инициализации */

errorcode = graphresult();

if (errorcode != gr0k) /* ошибка графического режима */

{

printf ("Graphics error: %s\n", grapherrormsg( errorcode));

printf( "Press аny key to halt:");

getch() ;

exit (1) ; /* аварийное завершение */

}

setcolor( 10);

settextstyle(0, 0, 2);

outtextxy(15, 10, "Результати аналого-цифрового преобразования:");

 

setcolor(9) ;

rесtаnglе(15,40, 624,330);

setcolor(11);

for(ny=0; ny<NN; nу++)

{

сirсlе(22+nу*6,330-у[nу]*1,2);

linе(22+nу*6,330,22+nу*6,330-у[nу]*1);

}

setcolor( 12);

settextstyle(0,0,1);

оuttехtху(260,З40, "Нажмите любую клавишу ….”);

getch() ;

closegraph() ;

}

/* Новый обработчик прерываний таймера */

void intеrruрt nеwtimе()

{

if (--kf<0) {

/* Виклик oldtime - на 2-й раз */

(*oldtime)();

kf=1;

}

else /* иначе - сброс контроллера */

outportb(0x20,0x20);

if ((nу>=0) /* Если АЦП началось, */

&&(nу<NN)) /* и NN показаний еще не набрано, */

у[nу++]=ус; /* запоминание очередного показания */

}

 

/* Получение старого вектора */

void *rеаdvесt(int in)

{

rr.h.аh=0хЗ5; rr.h.al=in;

intdоsх(&rr,&rr,&sr);

rеturn(МК_FР(sr.еs, rr.x.bх));

}

/* Запись нового вектора */

void writеvесt(int in, void *h)

{

rr.h.ah=0x25;

rr.h.аl=in;

sr.ds=FP_SEG(h);

rr.x.dx=FP_OFF(h);

intdosx(&rr,&rr,&sr);

}


Поделиться:



Последнее изменение этой страницы: 2019-05-08; Просмотров: 202; Нарушение авторского права страницы


lektsia.com 2007 - 2024 год. Все материалы представленные на сайте исключительно с целью ознакомления читателями и не преследуют коммерческих целей или нарушение авторских прав! (0.013 с.)
Главная | Случайная страница | Обратная связь