Язык Си - руководство для начинающих
Шрифт:
Продолжительность звучания
Мы можем регулировать продолжительность так, как было указано в гл. 6. Вспомним, что громкоговоритель управляется устройством, называемым "Программируемый параллельный интерфейсный контроллер 8255". Специальные каналы ввода-вывода, называемые портами, связывают этот и другие контроллеры с "мозгом" системы, микропроцессором 8088. Мы используем порт 97 для включения громкоговорителя, цикл, чтобы отмечать время, и затем порт 97 для отключения громкоговорителя. Вот фрагмент программы, которая будет выполнять эти действия:
#define TIMESCALE
1270 /*
#define BEEPPORT 97 /* порт управляет громкоговорителем */
#define ON 79 /* сигнал включения громкоговорителя */
count = TIMESCALE *time; /* преобразование времени
в единицы таймера */
port = inp(BEEPPORT); /* запоминание состояния порта */
outp(BEEPPORT, ON); /* включение громкоговорителя */
for(i = 0; i < count; i++)
; /* отметка времени */
outp(ВEEPPORT, port); /* выключение громкоговорителя, восстановление состояния */
Значение count (число отсчетов) дает время, в течение которого громкоговоритель включен. Коэффициент TIMESCALE преобразует десятые доли секунды в эквивалентное количество отсчетов времени. Конечно, мы должны установить требуемую частоту звука до того, как зазвучит громкоговоритель, поэтому рассмотрим этот параметр.
Частота звука
Частоту звука можно установить при помощи другого устройства, называемого "Программируемым интервальным таймером 8253". Этот контроллер в числе прочего определяет, сколько импульсов в секунду следует послать на громкоговоритель. Устройство 8253 вырабатывает базовую частоту 1,190,000 Гц, которая значительно выше граничной частоты восприятия звука человеком. Однако мы можем послать на устройство 8253 число для деления этой базовой частоты. Например, если мы направляем туда 5000, то получаем частоту, следования импульсов
1,190,000/5000 = 238 Гц,
которая немного ниже среднего звука си (нота, а не версии более низкого класса рассматриваемого языка). Если мы знаем, какая частота freq нам нужна, можно вычислить требуемый делитель, скажем, так:
divisor = 1,190,000/freq;
Наша функция позволяет сделать это, в связи с чем нам нужно только знать, как подать значение переменной divisor на устройство 8253. Теперь требуется использовать еще два порта.
Первый шаг заключается в установке таймера 8253 в правильный рабочий режим для приема делителя. Это достигается посылкой значения 182 (0´В6 в шестнадцатеричном коде) через порт 67. Как только такая посылка будет выполнена, можно использовать порт 66 для передачи делителя.
Посылка делителя представляет собой несложную задачу. Сам делитель
lobyt = divisor % 256;
hibyt = divisor % 256;
Можно также использовать поразрядные операции:
lobyt = divisor & 255;
hibyt = divisor >> 8;
Первый оператор в каждой паре строк примеров устанавливает первые восемь разрядов в 0, оставляя в последних восьми разрядах первого байта число. Проверьте результаты операцией получения модуля и поразрядной операцией И, чтобы увидеть, как это делается. Второй оператор каждой пары берет исходное значение divisor и сдвигает его на 8 позиций вправо (что эквивалентно делению на 28, или на 256). Восемь левых разрядов устанавливаются в 0, сохраняя 8-разрядное число, содержащее исходные значения восьми левых разрядов.
Ниже показана такая функция целиком:
/* tone(freq, time) -- устанавливает звук заданной частоты и продолжительности */
#define TIMERMODE 182 /* код установки таймера в нужный режим */
#define FREQSCALE 119000L /* базовая частота в герцах */
#define TIMESCALE 1230L /* число отсчетов времени в 0,1 с */
#define T_MODEPORT 67 /* порт управляет режимом работы таймера */
#define FREQPORT 66 /* порт регулирует частоту звука*/
#define BEEPPORT 97 /* порт управляет громкоговорителем */
#define ON 97 /* сигнал включения громкоговорителя */
tone(freq, time)
int freq, time;
{
int hibyt, lobyt, port;
long i, count, divisor;
divisor = FREQSCALE/freq; /* масштабирование частоты в единицах таймера */
lobyt = divisor % 256; /* разбивает целое */
hibyt = divisor / 256; /* на два байта */
count = TIMESCALE * time; /* преобразует время в единицы таймера */
outp(T_MODEPORT, TIMERMODE); /* подготавливает таймер к вводу */
outp(FREQPORT, lobyt); /* устанавливает младший байт регистра таймера */
outp(FREQPORT, hibyt); /* устанавливает старший байт регистра таймера */