|
AVR Studio ругается на порты F и G, mega128 |
|
|
|
May 26 2015, 12:58
|
Профессионал
    
Группа: Свой
Сообщений: 1 879
Регистрация: 20-06-11
Из: Карелия, Петрозаводск
Пользователь №: 65 799

|
Цитата Если на ассемблере, то, надеюсь Вы в курсе, что команды IN/OUT для доступа к этим регистрам не применимы Это только к ассемблеру относится?
--------------------
путь наименьшего сопротивления проходит по пути наитолстого провода (с)
|
|
|
|
|
May 26 2015, 13:10
|
Профессионал
    
Группа: Свой
Сообщений: 1 879
Регистрация: 20-06-11
Из: Карелия, Петрозаводск
Пользователь №: 65 799

|
Upd: проект на Mega128 с компилятором AVR GCC тоже не позволяет работать с портами F и G командами I/O с портами А...Е такой фигни нет - сразу определены адреса и доступ есть Xenia, я до фузов не дошёл ещё - у меня пустой лист, даже без программы. А доступа уже нет. При попытке настроить порты, А...Е настраиваются, а про F и G компилятор ругается \\_PROGRAM\AVRStudio\MAIN.asm(71): error: Operand 1 out of range: 0x61
Прикрепленные изображения
--------------------
путь наименьшего сопротивления проходит по пути наитолстого провода (с)
|
|
|
|
|
May 26 2015, 14:12
|

Гуру
     
Группа: Модератор FTP
Сообщений: 4 479
Регистрация: 20-02-08
Из: Москва
Пользователь №: 35 237

|
Цитата(Палыч @ May 26 2015, 17:19)  Инженеры Atmel'а при проектировании микроконтроллеров AVR когда-то совершили стратегическую ошибку - отвели под область ввода-вывода слишком маленькое пространство адресов. Очень скоро количество регистров ввода-вывода в отдельных AVR стало превышать адресное пространство области ввода-вывода. Некоторые регистры ввода-вывода стали размещаться за пределами области ввода-вывода - команды IN/OUT для таких регистров применять нельзя. В командах IN и OUT для адреса/номера регистра ввода-вывода есть всего лишь 6 бит. Именно потому эти команды не могут охватить более 64 регистров. К остальным же приходится обращаться (читать или писать), как к памяти, а не через инструкции IN и OUT. Ругать инженеров Atmel'а за непредусмотрительность глупо, т.к. в одном слове больше места не выкроишь. Ведь там же еще и номер регистра общего назначения находится, в который IN читает, а OUT пишет. Т.е. проблема тут отнюдь не в том, что памяти отвели мало, а в том, что в 16-разрядную инструкцию (тогда еще 32-разрядных не было) длинное число не засунешь. Тогда как для инструкций IN и OUT таких чисел надо сразу два. Делать инструкции IN и OUT двубайтными "инженеры" остереглись, т.к. тогда бы и выполнялась такая инструкция за два такта, а не за один. А это могло стать очень существенной потерей для архитектуры AVR, т.к. свое место под солнцем она отвоевала именно благодаря выполнению большинства инструкций за 1 такт, чем и замогилила  51-ю архитектуру. Кстати, раз уж разговор об этом зашел, замечу, что у Х-Мег (у которых разных регистров видимо-невидимо) есть, так называемый, "виртуальный порт". Это что-то вроде окошка, накладываемого на любой из портов, внутри которого действуют команды IN и OUT. Т.е. идея тут такова, что если требуется настолько быстрый ввод-вывод, что становится существенным, за сколько тактов он выполяется, то используемый порт совмещают с виртуальным, а дальше работают на вируальном порте однотактовыми командами IN и OUT.
|
|
|
|
|
May 28 2015, 09:59
|
Частый гость
 
Группа: Участник
Сообщений: 168
Регистрация: 14-02-10
Пользователь №: 55 490

|
Цитата(RA9YSS @ May 28 2015, 13:39)  А никто не подскажет мануальчик, как работать с портом G на Си, проблема таже что и у Ydaloj, студия 4.19  Компилятор знает об отличиях портов, смотрите: Код #include <avr/io.h>
int main ( void ) { PORTA |= 0x01; PORTG |= 0x01; } превращается в Код int main ( void ) { PORTA |= 0x01; be: d8 9a sbi 0x1b, 0; 27 PORTG |= 0x01; c0: e5 e6 ldi r30, 0x65; 101 c2: f0 e0 ldi r31, 0x00; 0 c4: 80 81 ld r24, Z c6: 81 60 ori r24, 0x01; 1 c8: 80 83 st Z, r24 }
--------------------
#define TRUE (4==(2*2))
|
|
|
|
|
May 28 2015, 11:42
|
Участник

Группа: Участник
Сообщений: 31
Регистрация: 12-05-12
Из: Барнаул
Пользователь №: 71 809

|
И в правду всё норм. Извините, перепаниковал((( Просто я весьма и весьма начинающий. На половину готовый проект, который пробовал на отладке с 32ой мегой собрался переносить на плату и решил взять мегу128 ( и в наличии есть, и памяти впрок) и вот случайно наткнулся на эту тему. Плату то уже развел. Еще раз прошу прощения
|
|
|
|
|
Jun 2 2015, 21:21
|

Профессионал
    
Группа: Свой
Сообщений: 1 940
Регистрация: 16-12-07
Из: Москва
Пользователь №: 33 339

|
Цитата(Владивольт @ May 28 2015, 12:59)  Компилятор знает об отличиях портов, смотрите: Код #include <avr/io.h>
int main ( void ) { PORTA |= 0x01; PORTG |= 0x01; } превращается в Код int main ( void ) { PORTA |= 0x01; be: d8 9a sbi 0x1b, 0; 27 PORTG |= 0x01; c0: e5 e6 ldi r30, 0x65; 101 c2: f0 e0 ldi r31, 0x00; 0 c4: 80 81 ld r24, Z c6: 81 60 ori r24, 0x01; 1 c8: 80 83 st Z, r24 } Охренеть можно , как Си "пахабит" благороднейший asm Код lds R16,PORTG ori R16,0x01 sts PORTG,R16
--------------------
Закон Мерфи:
Чем тщательнее составлен проект, тем больше неразбериха, если что-то пошло не так
|
|
|
|
|
Jun 3 2015, 05:59
|
Участник
  
Группа: Свой
Сообщений: 462
Регистрация: 2-04-07
Из: Иркутск
Пользователь №: 26 695

|
Цитата(ILYAUL @ Jun 3 2015, 06:21)  Охренеть можно , как Си "пахабит" благороднейший asm Код lds R16,PORTG ori R16,0x01 sts PORTG,R16 Код ldi r30, 0x65; 101 ldi r31, 0x00; 0 ld r24, Z ori r24, 0x01; 1 st Z, r24 Лишь 3 строчки вместо 5, вроде бы компактнее, быстрее… но в памяти занимает те же 5 слов, выполнение — 5 тактов и там, и там. Минус — в дополнительном использовании Z регистра, но если работать не только с одним портом, но и +63 соседними адресами тоже, то минус становится плюсом: в Z единожды загружаем базовый адрес, а при обращении к соседним адресам используется смещение Z+n Код .equ base=0x60 .def temp=R16
ldi ZL, low(base) ldi ZH, high(base)
ldd temp, Z+(PORTF-base) ori temp, 0x01; PORTF |= 1 std Z+(PORTF-base), temp
ldd temp, Z+(PORTG-base) ori temp, 0x03; PORTG |= 3 std Z+(PORTG-base), temp Такое решение уже для двух портов будет и короче, и быстрее. Если я не ошибаюсь, компиляторы умеют такое делать самостоятельно, если адреса/переменные расположены рядом.
|
|
|
|
|
Jun 17 2015, 14:34
|
Профессионал
    
Группа: Свой
Сообщений: 1 879
Регистрация: 20-06-11
Из: Карелия, Петрозаводск
Пользователь №: 65 799

|
вот теперь у меня проблема с таймером0 мне надо, чтобы 10 раз в секунду у него вызывалось прерывание Код ldi tmp,0
out ASSR,tmp out TCNT0,tmp
ldi tmp,0x60 out OCR0,tmp
ldi tmp,0x0F out TCCR0,tmp
ldi tmp,0x02 out TIMSK,tmp
ldi tmp,high(RamEnd) out SPH,tmp
ldi tmp,low(RamEnd) out SPL,tmp sei соответственно, в его обработчике лежит Код TIMER0_COMP:
in sreg_tmp,SREG push tmp
бла-бла-бла некоторые действия
pop tmp out SREG,sreg_tmp reti И вот эти действия в обработчике выполняются через ж. Вернее, не выполняются. В портах - ересь. Эти же действия, выполненные в теле программы, работают без ошибок. Значит, у меня подозрение на настройку таймера или его компаратора. Что можно (или нужно) сделать?
Сообщение отредактировал Ydaloj - Jun 17 2015, 14:35
--------------------
путь наименьшего сопротивления проходит по пути наитолстого провода (с)
|
|
|
|
|
Jun 17 2015, 14:40
|

Гуру
     
Группа: Модераторы
Сообщений: 8 455
Регистрация: 15-05-06
Из: Рига, Латвия
Пользователь №: 17 095

|
Цитата(Ydaloj @ Jun 17 2015, 17:34)  Код pop tmp out SREG,sreg_tmp reti А в основной программе регистр sreg_tmp не используется?
--------------------
На любой вопрос даю любой ответ"Write code that is guaranteed to work, not code that doesn’t seem to break" ( C++ FAQ)
|
|
|
|
|
Jun 19 2015, 13:47
|
Профессионал
    
Группа: Свой
Сообщений: 1 879
Регистрация: 20-06-11
Из: Карелия, Петрозаводск
Пользователь №: 65 799

|
Сергей Борщ, sreg нигде больше не используется. sreg_tmp тоже.
--- мне сложно объяснять причину таких вещей. я нутром-то чую, что это наверное я сам что-то делаю не так, но...
проблема решилась наистраннейшим образом в разделе .dseg был объявлен десяток переменных в ОЗУ с выделением байта под каждую. я одну из переменных в этом списке перенёс в конец списка. и всё. заработало. тут наверное вспышки на солнце виноваты.
Ну да ладно.
Следующая проблема. У меня их много - я дуб в микроконтроллерах, а работать надо. Итак. Есть основной цикл, в нём эти переменные крутятся-вертятся, обрабатываются. 10 раз в секунду вызывается разносчик пиццы обработчик прерывания таймера, который собирает все эти переменные и толкает их по портам. Это всё работало, до вчера. Программа пишется под мегу128. Пришлось (по некоторым причинам) заменить её на мегу128А. И теперь, каждый раз после обработки прерывания, ВСЕ переменные обнуляются.
Как лечить?
УРА, решил - при прошивке отключил Watchdog
Сообщение отредактировал Ydaloj - Jun 19 2015, 14:01
--------------------
путь наименьшего сопротивления проходит по пути наитолстого провода (с)
|
|
|
|
|
Jun 24 2015, 08:28
|
Местный
  
Группа: Участник
Сообщений: 465
Регистрация: 13-05-15
Из: Запорожье
Пользователь №: 86 663

|
Цитата(mcheb @ Jun 19 2015, 17:00) Мегу 128 на ассемблере программировать гиблое дело. Цитата(zombi @ Jun 24 2015, 01:29)  Глупость Вы написали! Согласен. Я для своих задач использую ассемблер: просто и понятно. А может у меня задачи такие?  . В AVRStudio можно прогнать и посмотреть где программа сворачивает не туда. А если проганять прогр на С или др языке, то как узнать какими командами расписана та или иная функция? И туда ли свернула программа? А чужие библиотеки! Кто и как их писал и как они работают?
|
|
|
|
2 чел. читают эту тему (гостей: 2, скрытых пользователей: 0)
Пользователей: 0
|
|
|