|
|
  |
Вопрос по HI-Tech C, Помогите разобраться... |
|
|
|
Apr 12 2009, 11:10
|

Участник

Группа: Свой
Сообщений: 68
Регистрация: 11-12-08
Из: Республика Беларусь, г.Минск
Пользователь №: 42 380

|
Вобщем есть такой вопросик: Есть подпрограмма(функция или процедура-не важно). В ней таблица вида Код void Table1(void) { #asm dw 0x3FF,0x3FF// и так далее... #endasm } почему asm,dw 0x... спросите вы: для того, что бы я мог использовать все 14 бит flash памяти. можно задать как массив во флэш,но при реализации компилятор сделает таблицу вида: Код retlw 0xFF retlw 0x1F и т.д. собственно, сам вопрос: как можно считать эти данные внутри функции Table1 и если не вызывается данная функция, как заставить компилятор её не исключать при сборке. Считать флэш у меня получается, а присвоить присвоить переменной адрес функции-не получается. Да, проц = PIC16F876A и(или) PIC16F886
Сообщение отредактировал DenisIV - Apr 12 2009, 11:11
|
|
|
|
|
Apr 12 2009, 19:13
|
Группа: Новичок
Сообщений: 6
Регистрация: 13-07-08
Пользователь №: 38 904

|
если это просто данные, зачем их описывать как функцию ? нужно делать массив констант
|
|
|
|
|
Apr 12 2009, 19:48
|

Участник

Группа: Свой
Сообщений: 68
Регистрация: 11-12-08
Из: Республика Беларусь, г.Минск
Пользователь №: 42 380

|
Цитата(Zman @ Apr 12 2009, 22:13)  если это просто данные, зачем их описывать как функцию ? нужно делать массив констант Объясняю: данные размером 8 килобайт. в памяти FLASH контроллера 14 килобайт (8 килослов размером 14 бит) если я задам массив из 8 килобайт, компилятор сгенерит код типа retlw 0xFF и т.д. т.е. будет использовано 8 бит из 14-и. этот код будет весить 8 килослов(14-битных, а не 8!!!), т.е. 100% FLASH памяти контроллера. мне же мужно считать 14-битные данные из FLASH, выровнять на границу в 8 бит и т.д (дальше не вопрос) я думаю, в 6 килобайт оставшихся точно помещу распаковку(выравнивание) =)) ваш совет полезен только в случае, если массив менее 8кб (8192-длина программы) а здесь другое...
|
|
|
|
|
Apr 12 2009, 20:38
|

Участник

Группа: Участник
Сообщений: 54
Регистрация: 7-08-08
Из: SPb
Пользователь №: 39 471

|
Объявление: Код void Table1(void) { #asm global MyTable // Используем отдельную метку на тот случай, когда компилятор вдруг // решит добавить какой-нибудь код в начало функции (вроде clrf STATUS) MyTable: dw 0x3FFF,0x3FFF// и так далее... #endasm } Считывание адреса: Код void main (void) { asm(" fncall _main, _Table"); // Говорим компилятору, что функция используется без прямого вызова, // чтобы он ее включил в код ... #asm global MyTable bsf _STATUS, 6 movlw low(MyTable) movwf EEADR movlw high(MyTable) movwf EEADRH bcf _STATUS, 6 #endasm ... } P.S. А вариант со сжатием данных по подойдет? Т.е. сделать обычный массив через const char Table[]={...}, и добавить функцию-распаковщик.
Сообщение отредактировал testerplus - Apr 12 2009, 20:39
|
|
|
|
|
Apr 13 2009, 00:10
|

Участник

Группа: Свой
Сообщений: 68
Регистрация: 11-12-08
Из: Республика Беларусь, г.Минск
Пользователь №: 42 380

|
Огромное вам сапасибо, testerplus , я более ёмкого и конкретного ответа даже не ожидал. Буду тестировать на проекте как проснусь, сразу отпишусь. Алгоритм со сжатием не подойдёт, так как данные уже упаковны не хуже GSM упаковки... Но всё равно биг спс !!! Я думаю, считывание типа Temp=FLASH_READ(MyTable+OffSet) должно пройти... Есть ещё пара мелких локальных вопросов, которые я хотел бы задать, но это потом... Адреналин, оказывается, вырабатывается не только в случае опасности, а так же в случае удачной компиляци...
|
|
|
|
|
Apr 13 2009, 05:58
|

Участник

Группа: Участник
Сообщений: 54
Регистрация: 7-08-08
Из: SPb
Пользователь №: 39 471

|
Цитата(DenisIV @ Apr 13 2009, 03:10)  Я думаю, считывание типа Temp=FLASH_READ(MyTable+OffSet) должно пройти... Такой фокус не пройдет (пройдет только на STD-версии компилятора и для ПИКа с ROM < 2k, да и то, если переобъявить метки "MyTable" -> "_MyTable" и в начале файла добавить объявление прототипа: void MyTable(void)). Метка, объявленная в asm из Си не будет видна. Так что только через асм. Удобно будет организовать в виде функции, но это уже Вам решать: Код int GetMyTable (int offset) { #asm global MyTable movlw low(MyTable) // Прибавляем адрес MyTable к offset addwf _Table$offset + 0, f movlw high(MyTable) btfsc _STATUS, 0 addlw 1 addwf _Table$offset + 1, f #endasm return FLASH_READ(offset); }
...
// обращаемся так:
i = GetMyTable(100); ... j = GetMyTable(i + 18); ... и т.д.
|
|
|
|
|
Apr 13 2009, 14:07
|

Участник

Группа: Свой
Сообщений: 68
Регистрация: 11-12-08
Из: Республика Беларусь, г.Минск
Пользователь №: 42 380

|
Цитата(testerplus @ Apr 12 2009, 23:38)  У меня прошло так: Код void MyArray(void) { #asm dw 0x3FF0,0x1A55 #endasm }
void main(void) {unsigned int Temp,Tmp; asm(" fncall _main, _MyArray"); Temp=&MyArray; Tmp=FLASH_READ(Temp); } Только вот указатель на функцию компилируется неправильно. Либо я его неправильно описал. Как я понимаю: Temp - прямая адресация к ячейке. *Teмp - адресация к данным через указатель (как к регистру INDF) &Temp - адресация к самому указателю (как к регистру FSR) в итоге у меня почему-то значение в Temp не совпадает с адресом самой функции. А так строчка Код asm(" fncall _main, _MyArray"); мне очень помогла. Не хотелось бы прибегать к асму, т.к. программа планируется быть носимой на другие платформы. предыдущий рабочий вариант был таков: Код unsigned int MyArray(void) { #asm movlw low ($+5) movwf ?_MyArray movlw high($+3) movwf ?_MyArray+1 return dw 0x3FF0,0x1A55 #endasm return 0;// для того, чтобы компилятор увидел явный возврат из функции на С (команды asm он в расчёт не берёт) }
void main(void) { unsigned int i,Tmp,Temp; Temp=MyArray(); Tmp=0; for(i=0;i<1024;i++) {Tmp=Tmp+FLASH_READ(Temp+i);} } но в начале функции команды asm не дадут перейти на другую платформу... =((
|
|
|
|
|
Apr 13 2009, 14:53
|

Участник

Группа: Участник
Сообщений: 54
Регистрация: 7-08-08
Из: SPb
Пользователь №: 39 471

|
Цитата(DenisIV @ Apr 13 2009, 17:07)  Только вот указатель на функцию компилируется неправильно. Либо я его неправильно описал. Как я понимаю: Temp - прямая адресация к ячейке. *Teмp - адресация к данным через указатель (как к регистру INDF) &Temp - адресация к самому указателю (как к регистру FSR) в итоге у меня почему-то значение в Temp не совпадает с адресом самой функции. Я выше описывал, что так не получится. HTPICC (и STD, и PRO) при объеме ROM > 2K будут строить таблицы переходов для всех функций, к адресам которых есть обращения в программе. Т.е. Ваша операция &MyArray вернет не адрес самой функции, а адрес вызова через таблицу переходов для STD-версии или номер элемента в таблице переходов для PRO-версии (кстати, у Вас каккая?) Кроме того, HTPICC легко первыми инструкциями в функции вставит: Код bcf status, rp0 bcf status, pr1 Если они будут добавлены, то вся таблица сместится, поэтому обращение может быть только через внутреннюю метку. на какие платформы планируется перенос? dw не везде задается одинаково.
|
|
|
|
|
Apr 13 2009, 15:43
|

Участник

Группа: Свой
Сообщений: 68
Регистрация: 11-12-08
Из: Республика Беларусь, г.Минск
Пользователь №: 42 380

|
Цитата(testerplus @ Apr 13 2009, 17:53)  Я выше описывал, что так не получится. HTPICC (и STD, и PRO) при объеме ROM > 2K будут строить таблицы переходов для всех функций, к адресам которых есть обращения в программе. Т.е. Ваша операция &MyArray вернет не адрес самой функции, а адрес вызова через таблицу переходов для STD-версии или номер элемента в таблице переходов для PRO-версии (кстати, у Вас каккая?) Можно поподробнее? Есть ли доступ к таблице переходов? PRO 9.65 Цитата Кроме того, HTPICC легко первыми инструкциями в функции вставит: Код bcf status, rp0 bcf status, pr1 Если они будут добавлены, то вся таблица сместится, поэтому обращение может быть только через внутреннюю метку. даже если сместится, можно хоть как-нибудь не прибегая к асму вытащить адрес функции и засунуть в какую-нибудь переменную? Цитата на какие платформы планируется перенос? dw не везде задается одинаково. на AVR и x51. ну в AVR с dw нормально, только точка перед ней нужна, с 51 проблем никаких вроде. dw не самая большая проблема.
|
|
|
|
|
Apr 13 2009, 16:12
|

Участник

Группа: Участник
Сообщений: 54
Регистрация: 7-08-08
Из: SPb
Пользователь №: 39 471

|
Цитата(DenisIV @ Apr 13 2009, 18:43)  Можно поподробнее? Есть ли доступ к таблице переходов? PRO 9.65 Тут все не очень просто. В ПРО-версиях (кажется с версии 9.60pl4) есть внутренняя константа fpbase (не доступная через Си, по крайней мере, я не добрался до нее). К ней можно обратиться только через asm: Код int i; ... asm(" movlw low(fpbase)"); asm(" movwf _i + 0"); asm(" movlw high(fpbase)"); asm(" movwf _i + 1"); Переменная существует и доступна только тогда, когда в программе есть обращение к адресу функции. Сами понимаете, что это не выход, т.к., во-первых, по-любому получается использование асма, а во-вторых, это всего лишь адрес таблицы переходов. Но также доступой становится константа с префиксом fp__ (опять же через ассемблер). Если функция называется MyArray, то адрес перехода на эту функцию в таблице переходов будет доступен через константу fp__MyArray. Цитата даже если сместится, можно хоть как-нибудь не прибегая к асму вытащить адрес функции и засунуть в какую-нибудь переменную? Я пока не находил таких механизмов. Цитата на AVR и x51. ну в AVR с dw нормально, только точка перед ней нужна, с 51 проблем никаких вроде. dw не самая большая проблема. Я про разные платформы спросил вот к чему: учитывая, что в трех перечисленных платформах dw описывается по-разному, то все равно придется делать какие-то платформозависимые куски кода и выбирать их директивами препроцессора #ifdef. А раз так, то совершенно спокойно можно использовать встроенный asm по той схеме, которую я привел в функции GetMyTable() (тремя-четырьмя постами выше). Вот эту функцию и нужно будет сделать платформозависимой. Весь остальной код будет переносимым, т.к. будет содержать только вызов этой функции.
|
|
|
|
|
Apr 14 2009, 01:27
|

Участник

Группа: Свой
Сообщений: 68
Регистрация: 11-12-08
Из: Республика Беларусь, г.Минск
Пользователь №: 42 380

|
Ясно. ну в этом случае мне выгоднее использовать конструкцию, которую я ранее приводил, когда я вызываю функцию, а в теле функции асмом отдаётся собственный код адреса и далее dw 0x.. и т.д. Да, позвольте дать совет не использовать конструкции вида Код asm ('movlw 0x00'); asm ('movwf MyVar'); дело в том, что компилятор когда начинает свою оптимизацию, может вообще поменять местами ваши команды по своему усмотрению. лечше используйте блоками: Код #asm movlw 0x00 movwf MyVar #endasm Это хоть какая-то гарантия будет от перемешивания команд... (IAR для AVR точно очень любит такие фокусы, я насмотрелся...) Ну тема с доступом к массиву полагаю,закрыта... Хотелось бы ещё спросить про аппаратный I2C мастер. У меня ну никак не получилось заставить его работать. Может какие тонкости есть в этом деле? Я попытался повторить эту демку на PIC16F886 и сколько не бился, так и не смог запустить аппаратно I2C. Пришлось программно симулировать. В даташите вроде не отличаются I2C у 876A и 886.
|
|
|
|
|
Apr 14 2009, 05:00
|

Участник

Группа: Участник
Сообщений: 54
Регистрация: 7-08-08
Из: SPb
Пользователь №: 39 471

|
Цитата(DenisIV @ Apr 14 2009, 04:27)  Да, позвольте дать совет не использовать конструкции вида Код asm ('movlw 0x00'); asm ('movwf MyVar'); дело в том, что компилятор когда начинает свою оптимизацию, может вообще поменять местами ваши команды по своему усмотрению. лечше используйте блоками: Код #asm movlw 0x00 movwf MyVar #endasm Это хоть какая-то гарантия будет от перемешивания команд... (IAR для AVR точно очень любит такие фокусы, я насмотрелся...) Спасибо, хотя я с этим никогда не сталкивался. Цитата Хотелось бы ещё спросить про аппаратный I2C мастер. Если честно, никогда не работал с аппаратным i2c в ПИКах
|
|
|
|
|
  |
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0
|
|
|