реклама на сайте
подробности

 
 
 
Reply to this topicStart new topic
> Задержка в несколько системных тиков на ассемблере
Haamu
сообщение Sep 26 2014, 12:04
Сообщение #1


Частый гость
**

Группа: Участник
Сообщений: 90
Регистрация: 12-12-13
Пользователь №: 79 587



Есть простой код на ассемблере:
Код
#define delay(tick) __asm("mov r0, %0 \n" "cycle: subs r0, #1 \n" " bhi cycle"::"r"(tick / 2))

Главная его проблема в том, что переход происходит на метку (cycle:), из за чего получается использовать его только один раз. При повторном использовании компилятор ругается, что такая метка уже существует.
Подскажите, как можно уйти от метки, не увеличивая сильно число команд?
Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение Sep 26 2014, 12:07
Сообщение #2


Гуру
******

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



Не знаю какой у вас компилятор, но в gas (из комплекта gcc) существует понятие "локальная метка". Попробуйте так:
Код
#define delay(tick) __asm("mov r0, %0 \n" "1: subs r0, #1 \n" " bhi 1b"::"r"(tick / 2))


--------------------
На любой вопрос даю любой ответ
"Write code that is guaranteed to work, not code that doesn’t seem to break" (C++ FAQ)
Go to the top of the page
 
+Quote Post
Haamu
сообщение Sep 26 2014, 12:21
Сообщение #3


Частый гость
**

Группа: Участник
Сообщений: 90
Регистрация: 12-12-13
Пользователь №: 79 587



Отлично, то что надо! Спасибо!
Вот, с небольшими поправками для точности:
Код
#define delay(tick) __asm("mov r0, %0 \n" "1: subs r0, #1 \n" " bhi 1b"::"r"((tick - 2) / 2))


А теперь, если не сложно, поясните пожалуйста, что значит 1b и чем такой вариант отличается от использования метки cycle?
Go to the top of the page
 
+Quote Post
WitFed
сообщение Sep 26 2014, 12:26
Сообщение #4


Местный
***

Группа: Свой
Сообщений: 271
Регистрация: 6-12-11
Из: Taganrog
Пользователь №: 68 701



Это я не знаю, могу только из вариантов предложить использовать текущий счётчик команд -- он обычно виден в ассемблере типа $PC, ну или к имени метки приделывать макрос __LINE__ через # или ## -- не помню точно.
Go to the top of the page
 
+Quote Post
demiurg_spb
сообщение Sep 26 2014, 13:40
Сообщение #5


неотягощённый злом
******

Группа: Свой
Сообщений: 2 746
Регистрация: 31-01-08
Из: Санкт-Петербург
Пользователь №: 34 643



2TC: Так ведь намного читабельней. Не находите?
Код
    static __inline void __attribute__((always_inline)) delay_4cycles(uint32_t cy)  // +1 cycle
    {    
        __asm__ __volatile__
        (
            "loop%=: subs %[cnt],#1"  "\n\t"
            "        bne  loop%="     "\n\t"
            : [cnt]"+r"(cy)           // output:  +r means input+output   http://www.nongnu.org/avr-libc/user-manual/inline_asm.html
            :                         // input:
            : "cc"                    // clobbers:
        );
    }


--------------------
“Будьте внимательны к своим мыслям - они начало поступков” (Лао-Цзы)
Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение Sep 26 2014, 13:54
Сообщение #6


Гуру
******

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



Цитата(Haamu @ Sep 26 2014, 15:21) *
А теперь, если не сложно, поясните пожалуйста, что значит 1b и чем такой вариант отличается от использования метки cycle?
Отличается тем, что вы создавали глобальную метку Cycle, а метка 1 является локальной. Гугление по "gas local label" дает ссылку на документацию:
Цитата
Local Symbol Names

Local symbols help compilers and programmers use names temporarily. They create symbols which are guaranteed to be unique over the entire scope of the input source code and which can be referred to by a simple notation. To define a local symbol, write a label of the form N: (where N represents any positive integer). To refer to the most recent previous definition of that symbol write Nb, using the same number as when you defined the label. To refer to the next definition of a local label, write Nf - The b stands for "backwards" and the f stands for "forwards".

There is no restriction on how you can use these labels, and you can reuse them as well. So it is possible to repeatedly define the same local label (using the same number N), although you can only refer to the most recently defined local label of that number (for a backwards reference) or the next definition of a specific local label for a forward reference. It is also worth noting that the first 10 local labels (0:...9:) are implemented in a slightly more efficient manner than the others.

Here is an example:
1: jra 1f
2: jra 1b
1: jra 2f
2: jra 1b



--------------------
На любой вопрос даю любой ответ
"Write code that is guaranteed to work, not code that doesn’t seem to break" (C++ FAQ)
Go to the top of the page
 
+Quote Post
scifi
сообщение Sep 26 2014, 16:31
Сообщение #7


Гуру
******

Группа: Свой
Сообщений: 3 020
Регистрация: 7-02-07
Пользователь №: 25 136



Цитата(Haamu @ Sep 26 2014, 16:04) *
Подскажите, как можно уйти от метки, не увеличивая сильно число команд?

Наверное, можно вычислить абсолютное смещение для относительного перехода и указать его в команде.
Go to the top of the page
 
+Quote Post
Genadi Zawidowsk...
сообщение Sep 26 2014, 17:25
Сообщение #8


Профессионал
*****

Группа: Участник
Сообщений: 1 620
Регистрация: 22-06-07
Из: Санкт-Петербург, Россия
Пользователь №: 28 634



Цитата(scifi @ Sep 26 2014, 20:31) *
Наверное, можно вычислить абсолютное смещение для относительного перехода и указать его в команде.

Ох уж это наследие трансляции программ в клетчатой тетради... Да вычисляется смещение само!
Цитата
не увеличивая сильно число команд
Метка в машинную команду не транслируется.

Сообщение отредактировал Genadi Zawidowski - Sep 27 2014, 08:15
Go to the top of the page
 
+Quote Post
Haamu
сообщение Oct 1 2014, 05:07
Сообщение #9


Частый гость
**

Группа: Участник
Сообщений: 90
Регистрация: 12-12-13
Пользователь №: 79 587



Цитата(Сергей Борщ @ Sep 26 2014, 17:54) *
Отличается тем, что вы создавали глобальную метку Cycle, а метка 1 является локальной. Гугление по "gas local label" дает ссылку на документацию:

Спасибо. Всё предельно ясно. Вот собственно та самая ссылка. Там еще несколько строчек примера, из который всё становится понятно.

Цитата(scifi @ Sep 26 2014, 20:31) *
Наверное, можно вычислить абсолютное смещение для относительного перехода и указать его в команде.

Наверно можно. Но как?
Пробовал вместо "bhi cycle" писать "bxhi rX", где X - номер регистра с адресом перехода, но как-то так и не добился правильного результата, то непонятно куда улетает, то вообще в HardFault сваливается.

Сообщение отредактировал Haamu - Oct 1 2014, 05:14
Go to the top of the page
 
+Quote Post
scifi
сообщение Oct 1 2014, 11:49
Сообщение #10


Гуру
******

Группа: Свой
Сообщений: 3 020
Регистрация: 7-02-07
Пользователь №: 25 136



Цитата(Haamu @ Oct 1 2014, 09:07) *
Наверно можно. Но как?

Вот так:
Код
#define delay(tick) __asm("mov r0, %0 \n" "cycle: subs r0, #1 \n" " bhi . - 2"::"r"(tick / 2))
Go to the top of the page
 
+Quote Post

Reply to this topicStart new topic
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0

 


RSS Текстовая версия Сейчас: 18th July 2025 - 15:24
Рейтинг@Mail.ru


Страница сгенерированна за 0.01481 секунд с 7
ELECTRONIX ©2004-2016