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

 
 
> Оператор "printf" и функция "putchar" для работы с символьным LCD
Буратино
сообщение Mar 19 2010, 11:47
Сообщение #1


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

Группа: Свой
Сообщений: 1 433
Регистрация: 27-10-08
Из: Украина, Киев
Пользователь №: 41 215



не понимаю, что за беда с выводом.
Прописал "putchar" в виде:
Код
void putchar( char c ) {
    if ( !c )
    return( 0 );
    Transmit_Byte( c );  /*функция отправки сообщения в последовательный порт*/
}


LCD оснащен контроллером, который принимает данные по последовательному порту и выводит на экран дисплея 16*02
Не могу разобраться почему:

printf("%a", 30.0) выводит "???"
printf("%.2A", 30.0) выводит "???"
printf("%c", 'a') ничего выводит
printf("<%3c|%-3c>", 'a', 'b') выводит "< крокозябра >"
и т.д. совсем другими словами не то что я ожидаю.

printf("Hello World!") выводит "Hello World!", но если использовать длинную строку или несколько операторов "printf" со строками, то выводятся куски строк в хаотичном порядке.
размер Data Stack в компиляторе IAR установил в 0x300.

Мне нужно выводить на экран простые конструкции состоящие из текста и числовых данных, что делать? smile.gif

Сообщение отредактировал Буратино - Mar 19 2010, 12:46


--------------------
Брак - это такой вид отношений, в которых один всегда прав, - а другой - муж.
Go to the top of the page
 
+Quote Post
 
Start new topic
Ответов
Буратино
сообщение Mar 19 2010, 18:20
Сообщение #2


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

Группа: Свой
Сообщений: 1 433
Регистрация: 27-10-08
Из: Украина, Киев
Пользователь №: 41 215



Все, разобрался (хотя и остались некоторые непонятки):
Не включил в проект #include <stdio.h> rolleyes.gif
---
zltigo, прошу прощения, что наорал на Вас, злой очень был.
Все никак не решусь на новый виток в образовании, вот если бы ногу или руку сломать (на крайний случай ветрянка сойдет), чтоб дома в кровати валяться и книги умные читать..

Сообщение отредактировал Буратино - Mar 19 2010, 18:28


--------------------
Брак - это такой вид отношений, в которых один всегда прав, - а другой - муж.
Go to the top of the page
 
+Quote Post
zltigo
сообщение Mar 19 2010, 18:37
Сообщение #3


Гуру
******

Группа: Свой
Сообщений: 13 372
Регистрация: 27-11-04
Из: Riga, Latvia
Пользователь №: 1 244



Цитата(Буратино @ Mar 19 2010, 21:20) *
Все, разобрался (хотя и остались некоторые непонятки):

Некошерно смотрится дергание бита разрешения прерывания sad.gif. Ну и volatile для UART_TxHead0 лишнее - только warning вызывает. tmptail0 лишняя сущность... Про имена помолчу.... Прототипы своих функций тоже игнорируем sad.gif


--------------------
Feci, quod potui, faciant meliora potentes
Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение Mar 19 2010, 21:20
Сообщение #4


Гуру
******

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



Цитата(zltigo @ Mar 19 2010, 20:37) *
Некошерно смотрится дергание бита разрешения прерывания sad.gif
Нормально смотрится. Нужны прервания - разрешаем, не нужны - запрещаем. Флаг-то аппаратно торчит всегда, когда свободен буфер передачи. Это вам не '550.
Цитата(zltigo @ Mar 19 2010, 20:37) *
Ну и volatile для UART_TxHead0 лишнее - только warning вызывает.
Формально - не лишняя. Переменная меняется в другой относительно прерывания нити.
Цитата(zltigo @ Mar 19 2010, 20:37) *
tmptail0 лишняя сущность...
Вовсе нет. Без нее получаем предупреждение компилятора на неопределенный порядок чтения двух volatile в сравнении.


--------------------
На любой вопрос даю любой ответ
"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
zltigo
сообщение Mar 19 2010, 21:53
Сообщение #5


Гуру
******

Группа: Свой
Сообщений: 13 372
Регистрация: 27-11-04
Из: Riga, Latvia
Пользователь №: 1 244



Цитата(Сергей Борщ @ Mar 20 2010, 00:20) *
Нормально смотрится. Нужны прервания - разрешаем, не нужны - запрещаем. Флаг-то аппаратно торчит всегда, когда свободен буфер передачи.

Свободен сдвиговый регистр - записываем прямо в UART, не свободен в буфер и по прерыванию уже вычитываем, либо по анализу указателей, либо флаг можно завести.
Цитата
Формально - не лишняя. Переменная меняется в другой относительно прерывания нити.

Абсолютно лишняя, ибо пока выполняется обработчик она измениться за его пределами не может. для Tail volatile нужен, Head - нет.
Цитата
Вовсе нет. Без нее получаем предупреждение компилятора на неопределенный порядок чтения двух volatile в сравнении.

Два volatile, повторяю, ну совсем не нужны, а warning, гарантировано при if ( UART_TxHead0 != UART_TxTail0 ) вылезет хоть есть tmptail0, хоть нет.


--------------------
Feci, quod potui, faciant meliora potentes
Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение Apr 20 2010, 21:59
Сообщение #6


Гуру
******

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



Цитата(zltigo @ Mar 20 2010, 01:08) *
Абсолютно лишняя, ибо пока выполняется обработчик она измениться за его пределами не может. для Tail volatile нужен, Head - нет.
Вспомнил, зачем Head должен быть volatile. Долго же в мозгу зрело, не давало покоя - ну не совсем же сдуру я его делал volatile! Если компилятор достаточно умный и решит встроить uart_t::send (например, если он вызывается только из одной точки программы), то без volatile он имеет полное право закешировать в основном коде Head в регистре и обработчик вместо одного байта будет каждый раз выстреливать весь буфер от начала до конца.

Кстати, и буфер тоже должен быть volatile. В противном случае компилятор имеет полное право перенести запись в буфер после изменения Head и разрешения прерывания.


--------------------
На любой вопрос даю любой ответ
"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
zltigo
сообщение Apr 20 2010, 22:13
Сообщение #7


Гуру
******

Группа: Свой
Сообщений: 13 372
Регистрация: 27-11-04
Из: Riga, Latvia
Пользователь №: 1 244



Цитата(Сергей Борщ @ Apr 21 2010, 00:00) *
то без volatile он имеет полное право закешировать в основном коде Head в регистре...

Это по любому глобальная переменная. Даже при встраивании функции область работы с этой переменной и область cуществования локальной переменной Tmp четко определена телом бывшей функции и при выходе за ее пределы должна быть синхронизирована с памятью. Если этого не будет сделано, то компилятор не умный, а вовсе даже с ошибками.
Даже допустим, что компилятор плюет на все и всех, но даже в этом случае uart_t::send должна встроися в некий вечный цикл, причем ничего особо кроме вызова send() не делающий, дабы была куча неиспользуемых регистров и компилятор мог ими разбрасываться. Что-то за уши притянутое.
Цитата
Кстати, и буфер тоже должен быть volatile. В противном случае компилятор имеет полное право перенести запись в буфер после изменения Head и разрешения прерывания.

Ну это вообще понесло sad.gif
Вместо:
Код
  TxBuffer[ TxHead ] = byte;
    TxHead = Tmp;
    UCSRB |= (1<<UDRIE);

Сделать
Код
    TxHead = Tmp;
    UCSRB |= (1<<UDRIE);
  TxBuffer[ TxHead ] = byte;

Может он тогда имеет право и вообще генерить любые побочные эффекты и точки следования ему не указ?


--------------------
Feci, quod potui, faciant meliora potentes
Go to the top of the page
 
+Quote Post
ReAl
сообщение Apr 24 2010, 21:28
Сообщение #8


Нечётный пользователь.
******

Группа: Свой
Сообщений: 2 033
Регистрация: 26-05-05
Из: Бровари, Україна
Пользователь №: 5 417



Цитата(zltigo @ Apr 21 2010, 01:28) *
Это по любому глобальная переменная. Даже при встраивании функции область работы с этой переменной и область cуществования локальной переменной Tmp четко определена телом бывшей функции и при выходе за ее пределы должна быть синхронизирована с памятью. Если этого не будет сделано, то компилятор не умный, а вовсе даже с ошибками.

Код
#include <avr/io.h>
uint8_t i;
inline void ppi() {
        uint8_t temp = i + 1;
        i = temp;
}

void foo(uint8_t n) {
        while(n--) {
                ppi();
                PORTB = 0;
        }
}

Код
foo:
    tst r24
    breq .L6
    lds r18,i
    mov r25,r24
.L5:
    out 56-32,__zero_reg__
    subi r25,lo8(-(-1))
    brne .L5
    add r18,r24
    sts i,r18
.L6:
    ret

Всегда считал, что тут всё правильно. Хотя тут нет ни намёка на работу с i "в пределах тела бывшей функции".
Аналогично при встраивании uart::send при отправке строки вполне может на первом байте разрешить прерывания (записать в UDR - не важно), но обновить head в глобальной переменной только в конце строки.

Цитата(zltigo @ Apr 21 2010, 01:28) *
Вместо:
Код
  TxBuffer[ TxHead ] = byte;
    TxHead = Tmp;
    UCSRB |= (1<<UDRIE);

Сделать
Код
    TxHead = Tmp;
    UCSRB |= (1<<UDRIE);
  TxBuffer[ TxHead ] = byte;

Может он тогда имеет право и вообще генерить любые побочные эффекты и точки следования ему не указ?
Ну аж так нельзя, так как у массива индекс выходит другой. А со сгенерированным компилятором временным указателем или индексом, сохраняющем предыдущее значение - вполне.
Точки следования не к этому. Вон в примере выше от точек следования в ppi() вообще ничего не осталось. И инвариантный код из цикла оптимизатором выносится с полным правом это делать, хоть при этом точки следования и меняются местами - от _независимых_ операций.
Там в стандарте есть такое слово "observable behavior", такая перестановка
Код
  uint8_t * ptr = &TxBuffer[ TxHead ];
    TxHead = Tmp;
    UCSRB |= (1<<UDRIE);
    *ptr = byte;
для не-volatile TxBuffer "видимое поведение" не меняет. А если ещё и TxHead не-volatile, то можно и так:
Код
    UCSRB |= (1<<UDRIE);
    TxBuffer[ TxHead ] = byte;
    TxHead = Tmp;
и тут даже внтренняя переменная для индекса/адреса не нужна. И в этом смысле TxHead должен быть volatile как у Сергея, так и в 11-том сообщении автора темы.

Как-то очередной версией копилятора в scmRTOS
Код
class TCritSect
{
public:
    TCritSect () : StatusReg( SREG ) { cli(); }
    ~TCritSect() { SREG = StatusReg; }
private:
    TStatusReg StatusReg;
};

    INLINE inline dword GetTickCount() { TCritSect cs; return Kernel.SysTickCount; }
при не-volatile на тот момент SysTickCount начало компилироваться не как
Код
   dword tmp;
   cs.StatusReg = SREG;
   cli();
   tmp = Kernel.SysTickCount;
   SREG = cs.StatusReg;
   return tmp;
а как
Код
   cs.StatusReg = SREG;
   cli();
   SREG = cs.StatusReg;
   return Kernel.SysTickCount;

Тут http://forum.sources.ru/index.php?showtopic=201904 эти все перестановки кода обсуждались. Тему неплохо почистили от ругни, но суть оталась :-)
Уже нужно было где-то процитировать - тогда нашлась в последнем сообщении опечатка, исправлялсь, поэтому дата у него оторванная от остальных.


--------------------
Ну, я пошёл… Если что – звоните…
Go to the top of the page
 
+Quote Post

Сообщений в этой теме
- Буратино   Оператор "printf" и функция "putchar" для работы с символьным LCD   Mar 19 2010, 11:47
- - barabek   Цитата(Буратино @ Mar 19 2010, 21:47) pri...   Mar 19 2010, 12:09
- - Буратино   printf("%f", 30.0); выводит "0.0000...   Mar 19 2010, 12:12
- - zltigo   Цитата(Буратино @ Mar 19 2010, 14:47) Про...   Mar 19 2010, 12:34
- - Буратино   А почему эта функция не putchar()? --- Буратино ...   Mar 19 2010, 12:43
|- - zltigo   Цитата(Буратино @ Mar 19 2010, 15:43) А п...   Mar 19 2010, 12:46
|- - Буратино   Цитата(zltigo @ Mar 19 2010, 15:46) А вот...   Mar 19 2010, 12:51
- - Буратино   Пересмотрел форум, и понял ,что функция printf() ...   Mar 19 2010, 15:36
|- - zltigo   Цитата(Буратино @ Mar 19 2010, 18:36) Я у...   Mar 19 2010, 15:49
- - one_eight_seven   А в контроллер уже зашиты ASCII коды? Может, я не...   Mar 19 2010, 15:49
- - Буратино   zltigo ,почему Вы такой злой? Ну нет желания помоч...   Mar 19 2010, 15:59
|- - zltigo   Цитата(Буратино @ Mar 19 2010, 18:59) zlt...   Mar 19 2010, 16:34
|- - Буратино   Цитата(zltigo @ Mar 19 2010, 21:37) Некош...   Mar 19 2010, 18:45
||- - zltigo   Цитата(Буратино @ Mar 19 2010, 21:45) Что...   Mar 19 2010, 18:52
|- - Сергей Борщ   Цитата(zltigo @ Mar 19 2010, 23:53) Свобо...   Mar 20 2010, 09:54
||- - zltigo   Цитата(Сергей Борщ @ Mar 20 2010, 12:54) ...   Mar 20 2010, 10:34
||- - Сергей Борщ   Цитата(zltigo @ Mar 20 2010, 12:34) Очень...   Mar 20 2010, 11:11
||- - zltigo   Цитата(Сергей Борщ @ Mar 20 2010, 14:08) ...   Mar 20 2010, 11:14
|||- - Сергей Борщ   Цитата(zltigo @ Mar 20 2010, 13:14) При н...   Mar 20 2010, 15:08
|||- - zltigo   Цитата(Сергей Борщ @ Mar 20 2010, 18:08) ...   Mar 20 2010, 15:35
||- - ReAl   Цитата(Сергей Борщ @ Mar 20 2010, 13:11) ...   Mar 20 2010, 18:38
||- - zltigo   Цитата(Сергей Борщ @ Mar 20 2010, 14:11) ...   Mar 20 2010, 18:51
||- - Сергей Борщ   Цитата(zltigo @ Apr 21 2010, 01:28) Это п...   Apr 21 2010, 07:31
|- - zltigo   Цитата(Сергей Борщ @ Apr 21 2010, 00:14) ...   Apr 24 2010, 22:39
- - sonycman   Кстати, можно вопрос по поводу UARTов микроконтрол...   Mar 20 2010, 12:57
|- - zltigo   Цитата(sonycman @ Mar 20 2010, 15:57) В о...   Mar 20 2010, 13:10
|- - sonycman   Цитата(zltigo @ Mar 20 2010, 17:10) Пальц...   Mar 20 2010, 14:00
|- - zltigo   Цитата(sonycman @ Mar 20 2010, 17:00) У В...   Mar 20 2010, 14:41
- - Буратино   RE: Оператор "printf" и функция "putchar" для работы с символьным LCD   Mar 20 2010, 16:49
- - zltigo   Цитата(Буратино @ Mar 20 2010, 19:49) А ...   Mar 20 2010, 17:01


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

 


RSS Текстовая версия Сейчас: 5th August 2025 - 08:47
Рейтинг@Mail.ru


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