Пример проектирования микропроцессорной системы

ИЗМЕРИТЕЛЬ ЧАСТОТЫ СЕТИ

        Техническое задание. Спроектировать цифровой измеритель частоты сети. На цифровые индикаторы выводятся три цифры (например, 51.7), т.е. значение частоты с точностью до десятых долей герца. Диапазон измеряемых частот от 20 до 99 Гц. Время измерения – не более 1 с.

Обоснование алгоритма решения задачи

        Для получения требуемой точности можно подсчитать количество импульсов сетевого напряжения за 10 с. Такой алгоритм не проходит по быстродействию.
        Другой способ основан на измерении периода сетевого напряжения и оценке частоты по формуле f = 1/T. Если период измерять в секундах, то получаем значение частоты в герцах. Точность определения частоты зависит от точности измерения периода и точности выполнения операции деления. При использовании микроконтроллеров следует ориентироваться на работу с целыми числами. Типичные погрешности, учитываемые при этом – погрешность дискретизации и погрешность округления.
        Будем измерять период сети в микросекундах. Для типового значения частоты 50 Гц период T составляет 20000 мкс. С помощью 16-разрядного таймера, работающего на частоте 1 МГц, можно измерить период до 65 мс, что соответствует частоте порядка 17 Гц. При частоте 99 Гц период равен 10101 мкс, т.е. таймер позволит измерить его с высокой точностью (погрешность дискретизации порядка 0,01%).

Основное рабочее соотношение

        Для того чтобы работать с целыми числами и гарантировать точность цифры десятых долей герца после выполнения целочисленного деления и округления будем рассчитывать значение частоты по формуле

и после вычисления f в виде трехразрядного десятичного числа добавлять точку перед последней цифрой при выводе информации на табло. Например, при Т=20500 мкс получаем и . На цифровых индикаторах нужно зажечь значение частоты 48.8 Гц.
        Выбираем микроконтроллер ATmega16 с кварцем на 8 МГц (время выполнения простейших команд 1/8 мкс). Функциональная схема устройства с распределением функций портов показана на рис.9. Порт РА используется для вывода информации на индикаторную панель. Порт РВ используется для программирования микроконтроллера (например, с помощью интерфейса «пять проводков» он подключается к СОМ-порту персонального компьютера).
        Прямоугольные импульсы с частотой сети, формируемые на выходе триггера Шмитта, будем подавать на вход INT0 (вторая линия порта PD). Запустив в основной программе таймер/счетчик Т1 в требуемый режим работы, поручим основные функции по решению задачи подпрограмме внешнего аппаратного прерывания по нарастающему фронту INT0:

        Полагаем, что на реализацию указанных действий будет затрачено время не более 10 мс и к началу нового периода микроконтроллер будет готов повторить описанный алгоритм в новой подпрограмме прерываний.



Рисунок 1 – Функциональная схема измерителя частоты сети

Разработка прикладной программы

        Последовательность шагов по разработке и отладке программы может быть следующей (проект формируем в папке z10):

  1. Создаем новый проект выбирая:
    File>New>Select Project
  2. Выбираем использование CodeWizardAVR:
    Use the CodeWizard?>Yes
  3. В окнах CodeWizardAVR фиксируем тип МК и рабочую частоту:
    Chip>Chip: ATmega16>Clock: 8MHz
  4. Конфигурируем LCD-дисплей:
    LCD>PORTA>16
  5. Конфигурируем Timer1:
    Timers>Timer1>Clock Value: 1000kHz>Interrupt off: Timer1 Overflow>Val: 0xFFFF
  6. Настраиваем прерывания:
    External IRQ>INT0 Enabled>Rising Edge
  7. Генерируем файлы C source, C project и CodeWizardAVR project выбирая:
    File|Generate, Save and Exit>
    Create new directory: C:\cvavr\z10>
    Save: z10.c>Save: z10.prj >Save: z10.cwp
  8. Редактируем код C source (основная творческая часть по реализации алгоритма)
  9. Смотрим или модифицируем конфигурацию проекта выбирая:
    Project>Configure>After Make>Program the Chip
  10. Компилируем программу выбирая:
    Project>Make
  11. Автоматически программируем ATmega16 на STK500:
    Apply power>Information>Program.

        Ниже приведен текст программы на языке Си. Для обеспечения возможности выводить на LCD-дисплей русские буквы в программу добавлена таблица кодов и функция void putchar(char c). Большую часть времени МК находится в цикле основной программы.

/*****************************************************
This program was produced by the
CodeWizardAVR V1.25.7 beta 5 Standard
Automatic Program Generator
© Copyright 1998-2007 Pavel Haiduc, HP InfoTech s.r.l.
http://www.hpinfotech.com

Project : Измеритель частоты сети
Version : от 17 до 99 Гц
Date    : 10.11.2008
Author  : Шарапов А.В.                          
Company : ПрЭ                           
Comments: курсовой проект

Chip type           : ATmega16
Program type        : Application
Clock frequency     : 1,000000 MHz
Memory model        : Small
External SRAM size  : 0
Data Stack size     : 256
*****************************************************/

#include <mega16.h>

// Alphanumeric LCD Module functions
#asm
.equ __lcd_port=0x1B ;PORTA
#endasm
#include <lcd.h>
#include <stdio.h>

flash char Decode2Rus[255-192+1]= {

0x41,0xA0,0x42,0xA1,0xE0,0x45,0xA3,0xA4,
0xA5,0xA6,0x4B,0xA7,0x4D,0x48,0x4F,0xA8,
0x50,0x43,0x54,0xA9,0xAA,0x58,0xE1,0xAB,
0xAC,0xE2,0xAD,0xAE,0xAD,0xAF,0xB0,0xB1,
0x61,0xB2,0xB3,0xB4,0xE3,0x65,0xB6,0xB7,
0xB8,0xB9,0xBA,0xBB,0xBC,0xBD,0x6F,0xBE,
0x70,0x63,0xBF,0x79,0xE4,0x78,0xE5,0xC0,
0xC1,0xE6,0xC2,0xC3,0xC4,0xC5,0xC6,0xC7

};

#define _ALTERNATE_PUTCHAR_
void putchar(char c)
{
if(c>=192) lcd_putchar(Decode2Rus[c-192]); else lcd_putchar(c);
}

// Declare your global variables here
unsigned int period=20000;
unsigned int f=500;
char lcd_buffer[33];// Буфер в ОЗУ для LCD-дисплея

// External Interrupt 0 service routine
interrupt [EXT_INT0] void ext_int0_isr(void)
{
// Place your code here

TCCR1B=0; //останов таймера
period=TCNT1L+256*TCNT1H;// Вычисление периода сети
TCNT1H=0;//обнуление таймера
TCNT1L=0;
TCCR1B=0x02;//запуск измерения нового периода

f=(100000000/period+5)/10;// Вычисление частоты сети
sprintf(lcd_buffer," power circuit frequency %u.%uHz",f/10,f%10);// Подготовка строки для LCD-дисплея
lcd_gotoxy(1,0);// Вывод строки со второй позиции в 1 строке
lcd_puts(lcd_buffer);
}

void main(void)
{
// Declare your local variables here

// Timer/Counter 1 initialization
// Clock source: System Clock
// Clock value: 1000,000 kHz
// Mode: Normal top=FFFFh
// OC1A output: Discon.
// OC1B output: Discon.
// Noise Canceler: Off
// Input Capture on Falling Edge
// Timer 1 Overflow Interrupt: Off
// Input Capture Interrupt: Off
// Compare A Match Interrupt: Off
// Compare B Match Interrupt: Off

TCCR1A=0x00;
TCCR1B=0x02;
TCNT1H=0x00;
TCNT1L=0x00;
ICR1H=0x00;
ICR1L=0x00;
OCR1AH=0x00;
OCR1AL=0x00;
OCR1BH=0x00;
OCR1BL=0x00;

// External Interrupt(s) initialization
// INT0: On
// INT0 Mode: Falling Edge
// INT1: Off
// INT2: Off

GICR|=0x40;
MCUCR=0x02;
MCUCSR=0x00;
GIFR=0x40;

// LCD module initialization
lcd_init(16);

// Global enable interrupts
#asm("sei")

while (1)
{
// Place your code here

};}

Моделирование работы устройства с помощью VMLAB

        Ниже приведен файл проекта для симулятора. Для моделирования работы измерителя частоты сети к входу INT0 подключается генератор прямоугольных импульсов, период которого можно задавать программно. К порту РА подключен LCD-дисплей (две строки по 16 символов).

; файл z10_vm.prj
.MICRO "ATmega16"
.TOOLCHAIN "GENERIC"
.TARGET "z.hex"
.COFF "z.cof"
.SOURCE "z__.c"
.CLOCK 8meg
.POWER VDD = 5  VSS = 0

;V[inst_name]  node VSS PULSE(v_initial v_final t_delay t_rise ;t_fall t_width ;t_period)
V1 PD2 VSS PULSE(0 5 0 0 0 5m 21m)

;X[inst_name]  LCD(chars lines oscil_freq) RS RW E D7 D6 D5 D4 D3 D2 D1D0
Xdisp LCD(16 2 250K) PA0 PA1 PA2 PA7 PA6 PA5 PA4 nc3 nc2 nc1 nc0

        Цель моделирования – проверка работоспособности выбранного алгоритма и оценка времени, которое микроконтроллер затрачивает на обработку информации. При проведении эксперимента раскрываем рабочие окна: Peripherals (окно периферийных устройств, чтобы пронаблюдать значение таймера Т1), I/O Ports (окно портов ввода/вывода, чтобы увидеть подачу импульсов на линию PD2), Control Panel (экран жидкокристаллического индикатора).

        При моделировании подключался файл Z10.hex, сформированный при трансляции программы, подготовленной на языке Си, с выводом информации на русском языке  (частота сети 47.6 Гц).


Рисунок 11.2 – Рабочие окна симулятора VMLAB

        В окне Code (текст программы на языке Си) ставим точки останова перед открывающей и закрывающей скобкой подпрограммы прерывания и, проведя цикл измерения частоты (для этого дважды нажимаем зеленую кнопку светофора), определяем по данным окна Messages время выполнения подпрограммы прерывания. Оно составило 3,81 мс, что говорит о работоспособности выбранного алгоритма определения частоты. Практически каждый период сетевого напряжения обновляется значение рассчитываемой частоты.
        В окне Program Memory можно увидеть машинные коды выполняемых команд и соответствующие им команды на языке ассемблера микроконтроллера. Можно реализовать пошаговое выполнение команд, нажимая кнопочку правее светофора.

Моделирование работы устройства с помощью симулятора PROTEUS VSM

        При моделировании на вход прерывания INT0 подавался сигнал с генератора прямоугольных импульсов, частоту колебаний которого можно задавать с помощью соответствующих регулировок (сигнал униполярный амплитудой 5 В). Убеждаемся, что сразу после изменения частоты в диапазоне 17-120 Гц аналогичные изменения происходят на табло LCD-дисплея. При моделировании не требуется подключение кварцевого резонатора и источников питания микроконтроллера и LCD-дисплея.


Рисунок 11.3 –  Рабочие окна симулятора Proteus VSM

        При моделировании подключался файл Z10.hex, сформированный при трансляции программы, подготовленной на языке Си, с выводом информации на английском языке  (power circuit frequency 49.0 Hz).