Стивен Барретт - Встраиваемые системы. Проектирование приложений на микроконтроллерах семейства 68HC12/HCS12 с применением языка С
Рис. 8.20. Имитатор усилителя
8.9. Другая прикладная программа: цикл опроса с прерываниями
В этой программе мы разработаем приоритетную часть системы опроса с передним планом и фоном, позволяющей предотвратить перегрев транзисторов в стереоусилителе. Система приведена на рис. 8.21. Температура транзистора постоянно контролируется датчиком температуры LM34 (в пластмассовом корпусе), приклеенным к металлическом корпусу ТО K-220 мощного транзистора. Напряжение на выходе датчика линейно связано с его температурой (коэффициент 10 мВ/С).
Рис. 8.21. Транзисторные системы обнаружения перегрева
Выход LM34 подан к один из входов аналогового компаратора, построенного на ОУ. На другой вход подается опорное напряжение, задающее порог температуры. Когда температура мощного транзистора достигает этого порога, на входе системы прерывания микроконтроллера появляется активный сигнал низкого уровня формирующий запрос на прерывание IRQ.
При обработке запроса на прерывание, PORTX[0] формируется импульс с активным высоким уровнем, подключающий дополнительный вентилятор через мощный МОП-транзистор IRF530. Можно предусмотреть определенную задержку в программе обработки прерывания и затем и затем выключить вентилятор. Когда температура транзистора упадет ниже установленного предела, вывод запроса на прерывание перейдет на высокий логический уровень, возвращая процессор к выполнению основной программы.
Если температура транзистора упадет до безопасного уровня процессор продолжит нормальную работу, (при этом на входе запроса на прерывание установится сигнал высокого уровня), а при превышении температуры снова инициализирует прерывание (установится непрерывный низкий уровень на входе запроса на прерывание). Мы предлагаем вам самостоятельно разработать программу для системы, использующей прерывания, в качестве задания для самостоятельной работы (задание 2).
8.10. Сложное прикладное устройство: имитатор ОСРВ
8.10.1. Краткий обзор проекта
Некоторые из продвинутых алгоритмов планирования в режиме реального времени достаточно сложно выполнить. Сложность частично заключается в том, что внутренняя операция этих программ не визуализирована. Имитатор ОСРВ, показанный на рис. 8.22 может использоваться как помощь в разработка ОСРВ.
Рис. 8.22. Имитатор ОСРВ
Имитатор состоит из набора блоков для 16 задач: пронумерованных в шестнадцатеричной системе от задачи 0 до задачи F. Каждый из таких блоков выполнен на микросхеме регистра с защелкой 74HC573. Состояние пересылается в регистр, где оно сохраняется вплоть до изменения. Состояние каждой задачи отображается набором светодиодов. Только один светодиод высвечивается в любой момент для каждой задачи. Мы использовали схему драйвера для светодиодов, рассмотренную в главе 5 для каждого из регистров 74HC573.
Схема симулятора приведена на рис. 8.23. Демультиплексор 4 в 16 (74HC154) используется, чтобы выбрать отдельную задачу. Двоичный «адрес» задачи задается микропроцессором на входных линиях демультиплексора. В любой момент времени только один его выход находится в активном низком состоянии и, следовательно, выбирается только одна задача.
Рис. 8.23. Аппаратное обеспечение симулятора ОСРВ
Обычно после инициировании все задачи находятся в состоянии бездействия. Чтобы выбрать различные состояния системы используется вспомогательная клавиатура, моделирующая работу системы ОСРВ. С вспомогательной клавиатуры набирается сначала конкретное состояние, отображающее набор типов активности различных задач. Затем с той же клавиатуры набирается информация о том, активность какой из задач будет изменена.
Чтобы облегчить понимании концепций ОСРВ, обратимся снова к относительно гибким, быстро изменяющимся и просто визуализируемым сценариям с дилерским обслуживанием автомобиля, обслуживанием многих заказчиков официантом или даже к системе защиты в большой гостинице. Любые из этих примеров могут использоваться, чтобы иллюстрировать различные алгоритмы планирования.
Например, используемый автомобиль (задача) может иметь свойственный контекст, состоящий из года изготовления, модели, номера и показаний одометра. Автомобиль (задача) может находится в различных состояниях, идентичных состояниям задач (готовности, бездействия и т.д). После изготовления автомобиля, он может быть готов для продажи (готовность), проходить испытания (активность), недоступен из-за обслуживания (ожидание), продан (бездействие), и т.д. Состояние нескольких автомобилей (задач) может быть визуально отображено на имитаторе ОСРВ (до 16 задач). Новое состояние сценария может вводиться пользователем с шестнадцатеричной вспомогательной клавиатуры. Новое состояние можно показывать на ЖКД, а изменение списков с указателями получаемых при изменении состояния может быть отображено на экране ПК [Barrett 20041].
8.10.2. Типовой код
Детальное описание имитатора ОСРВ выходит за рамки данной книги. В этом разделе мы просто приведем код, позволяющий отображать и изменять состояние каждой из задач.
Функция update_task_status требует в качестве параметров идентификационного номера задачи и кода желательного изменения состояния задачи. Специфическая задача используется, чтобы конфигурировать PORTA[3:0] правильным четырехразрядным двоичным кодом, который подается на микросхему 74HC154 декодера 4–16. Когда на демультиплексор подается код через выходной порта PORTA[7], активируется триггер-защелка, соответствующий выбранной задаче. Это позволяет передать модифицированное состояние задачи, представленное на выходах порта T контроллера 68HC12 на светодиоды и, таким образом, показать состояние задачи на дисплее.
Алгоритм программы на UML для реализации этой функции представлен на рис. 8.24.
Рис. 8.24. Алгоритм UML для функции update_task_status
//******************************************************************
//имя файла: realtime.с
//авторы: Steve Barrett and Daniel Pack
//разработан: 1 июля, 2004
//последняя редакция: 1 июля, 2004
//******************************************************************
//включенные файлы ****************************************
#include <912b32.h>
//функции-прототипы****************************************
void initialize_ports(void); //инициализация портов
void initialize_LCD(void); //инициализация ЖКД
void putchars(unsigned char); //функция поддержки ЖКД - ввести символ
void putcommands(unsigned char); // функция поддержки ЖКД - ввести команду
void delay_5ms(void); //задержка на 5 мс
void delay_100us(void); //задержка на 100 мкс
void update_task_status(unsigned char task, char task_status);
//изменить состояние задачи
//глобальные переменные************************************************
//главная программа****************************************************
void main(void) {
asm(" .area vectors(abs)n" //inline assembly statement
" .org 0xFFF8n" //initialize 68HC12 B32 reset vector
" .word 0x8000, 0x8000, 0x8000, 0x8000n"
" .text");
initialize_ports(); //инициализировать порты
initialize_LCD(); // инициализировать ЖКД
update_task_status(0x00, 'R') ; //изменить состояние задачи
}
//******************************************************************
//initialize_ports: provides initial configuration for I/O ports
//******************************************************************
void initialize_ports(void) {
DDRA = 0xFF; //установить PORTA как выходной - управляющие линии
// демультиплексора
DDRT = 0xFF; // установить PORTT как выходной - состояние задачи
DDRB = 0xFF; // установить PORTB как выходной - порт данных для ЖКД
CTDRDLC = 0xFF; // установить PORT DLC как выходной - сигналы
//управления для ЖКД
DDRP = 0x0F; // установить PORTP[3:0] как выходной, PORT[7:4] как
//входной - для клавиатуры
PORTA = 0xFF; //установить для PORTA все линии декодера высокоомными (Hi-Z)
}
/****************************************************************/
/****************************************************************/
/*update_task_status: изменить состояние конкретной задачи */
/*в соответствии с допустимым переходом (рис. 8.14) */
/****************************************************************/
void update_task_status(unsigned char task, char task_status) {
//установить состояние задачи на выходе порта T
switch(task_status) {
case 'D': //бездействие (D)
PORTT = 0x01;
break;
case 'R': //готовность (R)
PORTT = 0x02;