Глава 3
ОСНОВЫ ПРОГРАММИРОВАНИЯ МИКРОКОНТРОЛЛЕРОВ НА СИ
ПОСЛЕ ИЗУЧЕНИЯ ГЛАВЫ ВЫ СМОЖЕТЕ:
• Описать основные конструкции языка Си;
• Написать на Си простые программы для встроенных систем на основе микроконтроллеров;
• Объяснять последовательность действий, необходимую для получения исполняемого фрагмента кодов для МК 68HC12 и HCS12;
• Объяснить назначение отдельных программ в составе интегрированной среды разработки, позволяющие редактировать, компилировать, ассемблировать, объединять и отлаживать программные фрагменты управляющего кода будущих встраиваемых систем.
В главе 3 рассматривается технология написания и отладки программ на языке Си для микроконтроллеров 68НС12 и HCS12. На протяжении всей главы мы не будем делать различия между этими двумя типами МК и, как следствие, будем использовать для ссылок общую аббревиатуру 68HC12. Основное наше внимание будет уделено технике программирования на языке Си. Однако это не означает, что программирование на языке ассемблера может быть полностью забыто. Включение в исходный текст Си-программы фрагментов на языке ассемблера позволяет создать эффективный исполняемый код в критичных по времени исполнения задачах. Вопросы объединения программных фрагментов, записанных на Си и на ассемблере, также рассматриваются в данной главе.
Мы покажем, как из исходного текста программы, записанного на Си, создать файл исполняемого кода для микроконтроллера, используя для этого программы компилятора, ассемблера, линковщика и загрузчика в составе программного пакета ICC12 от компании Imagecraft (http:/www.imagekraft.com). Программный продукт ICC12 - достаточно простой в управлении, недорогой, но обладающий всеми необходимыми типовыми функциями программный пакет класса "интегрированная среда разработки и отладки программ управления для встраиваемых систем". Технология создания программного обеспечения для встраиваемых микроконтроллерных систем с использованием перечисленных выше программ в составе интегрированной среды разработки IDE (Integrated Development Environment) обладает достаточной степенью универсальности. Поэтому навыки создания и отладки программ для МК 68HC12, полученные с использованием пакета ICC12, могут быть использованы читателем при программировании 68HC12 или иных типов микроконтроллеров с использованием других более развитых программных пакетов IDE, например Code Warrior от компании Metrowerks (http:/www.metrowerks.com).
В этой главе мы обсудим технологию отладки программы, написанной на языке Си, в процессе ее исполнения реальным микроконтроллером. Все МК семейства 68HC12 обладают специальным режимом отладки в реальном времени BDM (Background Debug Mode). Мы рассмотрим основные свойства режима BDM в этой главе. В заключение мы приведем подробный пример преобразования исходного текста программы на Си в файл исполняемого кода для выбранного типа МК средствами программ, входящих в пакет интегрированной среды разработки IDE ICC12.
3.1. Введение в программирование на Си
В главе 2 было отмечено, что язык Си стал одним из наиболее часто используемых в техническом сообществе языков программирования высокого уровня. Связано это с тем, что при сохранении абстрактного характера представления алгоритмов и данных, которое свойственно языкам программирования высокого уровня, язык Си позволяет программистам непосредственно управлять компонентами аппаратных средств компьютеров и микроконтроллеров. Например, в Си программист имеет возможность непосредственного обращения к содержимому ячеек памяти и регистров периферийных модулей с конкретными физическими адресами, что свойственно языку ассемблер.
Для того чтобы начать программировать микроконтроллеры 68НС12 на Си, мы должны сначала изучить основные конструкции этого языка, которые и рассматриваются в следующих параграфах главы 3. Следует заметить, что в этих параграфах приведен лишь минимально необходимый набор сведений по Си. Для более полного знакомства с этим языком программирования читателю следует обратиться к специальной литературе. Список рекомендуемых авторами изданий приведен в конце главы.
Си - это структурированный язык программирования. Для записи программ на языке Си используются идентификаторы, ключевые слова, числа и операторы, которые располагаются в тексте программы в соответствие с предписанными правилами. Идентификаторы - это определяемые пользователем имена переменных, функций, меток и объектов. В языке Си для того, чтобы использовать переменную в программе, её необходимо предварительно создать. Создать переменную - это значит выделить для её хранения одну или несколько ячеек памяти и присвоить ей имя, с помощью которого можно будет ссылаться на эту. В отличие от ассемблера, в языке Си необходимо указать также тип переменной. Тип служит для того, чтобы сообщить компилятору о том, как интерпретировать значение, хранящееся в ячейках памяти, отводимых под переменную. Например, имеет ли число знак (старший бит отводится для хранения знака) или старший бит является значащим разрядом числа и т. д.
3.1.1. Глобальные и локальные переменные
Каждая объявленная в начале программы переменная должна храниться в одной или нескольких ячейках памяти в течение времени исполнения программы. В микроконтроллерах переменные могут размещаться в ОЗУ, ПЗУ и регистрах центрального процессора.
В языке Си различают два типа переменных: глобальные переменные, к которым можно обратиться из любой функции программы, и локальные переменные, которые доступны только при исполнении той функции, в которой они были объявлены. В системах на микроконтроллерах, локальные переменные обычно размещаются в области стека.
Объявление различных идентификаторов в начале программы - особенность языка Си по сравнению с языком ассемблер. Служебные слова, которые будут сопровождать объявление идентификатора, определяют правила доступа к переменной, функции, макросу или структуре, которая именована этим идентификатором. Не все идентификаторы будут доступны из любого места программы на Си. Подробное описание служебных слов для объявления классов хранения переменных выходит за рамки этой главы. Вы можете найти его в [3].
В микроконтроллерах семейства 68НС12, переменные размещаются в ячейках памяти ПЗУ или ОЗУ. Переменная, которая не изменяет своего значения в течение выполнения программы, именуется константой и хранится в ПЗУ. Переменная, значение которой изменяется в процессе исполнения программы, должна храниться в ОЗУ. Например, предположим, что некоторый контроллер связан с датчиком температуры посредством аналого цифрового преобразователя. Прикладная программа должна постоянно обновлять значение температуры в памяти контроллера. Для этого объявим переменную с именем temp (от слова temperature - температура) и разместим ее в ОЗУ МК. Приведенный ниже фрагмент программы на Си демонстрирует, как обратиться к размещенной в ОЗУ переменной temp и отобразить ее значение на дисплее. Заметим, что номера строк в приводимых фрагментах программ не являются частью записи операторов языка Си и не должны присутствовать в исходном тексте программы, подлежащем компиляции. Они введены искусственно для ссылки на отдельные операторы при обсуждении конструкций языка Си.
1 while (1)
2 {
3 temp = *(unsigned char volatile*)(0х1000);
4 printf(The current temperature is %d\n, temp);
5 }
Данный фрагмент кода предполагает, что переменная с именем temp создана (объявлена) ранее, и также ранее реализован фрагмент кода на Си для опроса аналого цифрового преобразователя с целью обновления значения температуры. Для определенности в строке 3 мы сами присваиваем этой переменной значение $1000 в шестнадцатеричном коде (или 4096 в десятичной системе счисления). Префикс 0x служит в Си для обозначения шестнадцатеричной системы счисления. Не следует тревожиться, если не все записи в данном примере ясны для Вас. Пройдет немного времени, Вы закончите изучение материалов данной главы, и все рассмотренные примеры станут для Вас простыми и понятными.
В приведенном фрагменте переменная temp может быть объявлена как глобальная или как локальная переменная. Назначение статуса переменной (глобальная или локальная) определяет программист посредством записи оператора для объявления переменной в определенном месте текста программы. Глобальная переменная должна быть объявлена вне всех функций данной программы. В то время как локальная переменная объявляется внутри той функции, в которой используется. Мы рассмотрим способы объявления переменных после обсуждения в следующем параграфе типов данных, используемых в программах на Си.
3.2. Типы данных в Си
Язык Си оперирует с восемью основными типами данных, которые представлены в табл. 3.1. Тип переменной определяет число байтов в памяти микроконтроллера, которые выделяются компилятором для ее хранения.