|
Проблема с volatile переменной - помогите плиз. |
|
|
|
Aug 19 2008, 14:21
|
Участник

Группа: Участник
Сообщений: 39
Регистрация: 18-10-07
Пользователь №: 31 472

|
Добрый день господа форумчане. У меня возникла проблема - очень надеюсь на вашу помощь. Пишу программу под PIC16F876 на PICC от HT-PICC. В программе захотел сделать функцию антидребезг. unsigned char do_antiripple_pin(volatile struct data_unit * data , char time_conversion);В эту функцию передаю время проверки на дребезг и указатель на структуру в которой содержиться иформация о ножке процессора. volatile struct data_unit { volatile unsigned char pin_addres; // Адрес пина на цоколевке процессора. unsigned char time_to_end; // Показывает сколько времени до конца преобразования unsigned char temp_pin_value; // Сохраняет временное значение на меняющимся пине проца. unsigned char changing_data; // Переменная с которой прграмма работает как с ножкой проца unsigned char diferences_counter; // Счетчик несовпадений пина и значения переменой за интервал };Проблема с полем pin_addres - это у меня просто имя пина ( допустим RA0), я объявил глобальный экземпляр структуры volatile struct data_unit button_data = {0,0,1,1,0};потом в main(); присвоил полю pin_addres значение RA0 button_data.pin_addres = RA0;ниже вызываю функцию do_antiripple_pin(&button_data,100);и надеюсь что у меня в функцию передасться значение с ножки порта, причем не просто значение а мгновенное, так как я везде где можно поставил volatile. Но этого не происходит. Программа как один раз зафиксировала в памяти значение button_data.pin_addres = RA0;так больше его не меняет, хотя значение на ножке меняется (это я вижу через IDC2). Выкладываю программу , там в листинге более понятно, чем я тут объясняю. Программма как следует не работает, но если закоментировать стр 31, и раскоментировать стр 92 тогда все работает нормально. Нормально так же все работает если передавать в функцию просто назвние ножки порта. Но вот почему не работает как мне надо??? (Т.е в том виде в котором выложил!) Помогите пожалуйста разобраться.
|
|
|
|
|
 |
Ответов
|
Aug 20 2008, 06:46
|
Участник

Группа: Участник
Сообщений: 39
Регистрация: 18-10-07
Пользователь №: 31 472

|
Уважаемый rezident. Я попробовал сделать как вы советовали. Результат тот же - ничего не получается. Поле pin_addres не обновляеися согласно со значением ножки RA0. А вот насчет того что истользовать volatile где попало не следует, я с вами согласен. Просто в программе пытался найти верный вариант методом проб и ошибок, по этому и поставил volatile где только возможно. И еще я передаю в функцию только указатель на структуру, что бы избежать лишнего копирования при передачи аргументов в функцию. Но главный вопрос так и остался открытым : "Почему переменная обьявлення как volatile и которой присвоено значение ножки процессора, на меняется согласно с изменением сигнала на этой ноге процессора ??? " Прикладываю новую версию программв с изменениями которые советовал внести г-н rezident.
|
|
|
|
|
Aug 20 2008, 18:36
|
Профессионал
    
Группа: Свой
Сообщений: 1 481
Регистрация: 10-04-05
Пользователь №: 4 007

|
Цитата(Илья_ @ Aug 20 2008, 10:46)  Уважаемый rezident.
Я попробовал сделать как вы советовали. Результат тот же - ничего не получается. Поле pin_addres не обновляеися согласно со значением ножки RA0.
А вот насчет того что истользовать volatile где попало не следует, я с вами согласен.
Просто в программе пытался найти верный вариант методом проб и ошибок, по этому и поставил volatile где только возможно.
И еще я передаю в функцию только указатель на структуру, что бы избежать лишнего копирования при передачи аргументов в функцию.
Но главный вопрос так и остался открытым : "Почему переменная обьявлення как volatile и которой присвоено значение ножки процессора, на меняется согласно с изменением сигнала на этой ноге процессора ??? "
Прикладываю новую версию программв с изменениями которые советовал внести г-н rezident. Уважаемый Илья! Поймите раз и навсегда. Переменная (или поле в структуре) - это значение, лежащее в некоторой ячейке памяти. Указатель на эту переменную - адрес этой ячейки памяти. Улавливаете, что никакими ножками порта тут не пахнет? Значит надо обеспечить программно чтение значения этой ноги и ее сохранение в вашей переменной. Volatile к этому процессу ну никаким образом. Volatile лишь запрещает компилятору оптимизировать эту переменную. Опрашивать состояние ножки можно либо в каком-либо программном цикле, либо по прерыванию, если можно организовать такое, по изменению значения на ноге (зависит от конкретного процессора)
|
|
|
|
|
Aug 20 2008, 20:15
|
Участник

Группа: Участник
Сообщений: 39
Регистрация: 18-10-07
Пользователь №: 31 472

|
Уважаемые господа, sergeeff и rezident. То что указатель содержит адрес ячейки памяти, в которой храниться значение переменной на которую этот указатель указывае - это азы Си. И они мне уверяю вас известны. А то что модификатор voltile в обьявлениии указателя может указывать что указатель, так сказать волатильный( т.е. может измениться не только из программы, как регистры переферии ), и может модификатор voltile указывать что указатель указывае на волатильную перменную. Вот цитата из прервода документации по компилятору HT-PICC: "Чтобы четче определить поведение и особенности указателя и объекта на который он указывает можно комбинировать различные квалификаторы. К наиболее часто употребимым квалификаторам относятся const, volatile, persistent. Применяя комбинации этих квалификаторов необходимо следить за соответствием (отсутствием противоречий) квалификаторов, оказывающих воздействие на свойства собственно указателя и объекта на котрый он указывает. Правила, которые помогут сделать все правильно, просты: если квалификатор находится слева от “*” в описании указателя, то он воздействует на объект, адресуемый указателем. Если квалификатор находится справа, то он воздействует на собственно указатель. Проиллюстрируем это примерами: volatile char * nptr; объявляет указатель на volatile символ. Другими словами в этом примере квалификатор воздействует на объект, адресуемый указателем nptr. char * volatile ptr; так как квалификатор располагается справа от “*”, то он будет воздействовать на собственно указатель ptr, а не на объект. И заключительный пример по этому поводу: volatile char * volatile nnptr; теперь будет описан volatile указатель на volatile переменную. Рассмотрим некоторые аспекты применения константных указателей. Они применяются для косвенного обращения к переменным описанным как const. В общем их поведение (константных указателей) не отличается от поведения обычных указателей, но компилятор препятствует выполнению операций записи с использованием этих указателей. Вот несколько примеров: const char * cptr; при этом выражение: char ch = *cptr; абсолютно легально, в то время как выражение: *cptr = ch; недопустимо и вызовет ошибку"Судя по тому что написано в описании, я совершенно правильно использовал квалификатор volatile, а то что сам указатель не меняется так это тоже верно. А я написал функцию которая выполняет антидребезг. И передаю в нее время , в течении которого проводить проверку на дребезг, и так же передаю в функцию структуру которая содержит информацию о пине. Так как антидребезг мы можем производить для любой из ножек контроллера к которой подклюен датчик или кнопка. Так вот я мог бы пердавать в функцию явно номер пина RA0 или RC5 и это бы работало. А если передавать структуру, в которой одним из полей будет номер пина (RA0 или RC5) то функция работает некоретно. И вот вопрос ПОЧЕМУ??? Кстати я пробовал работать и без указателей, тоже ничего не выходит, если номер пина я вляется полем структуры - так что видите что вазговоры о том что указатель волатильный, а сама переменная нет, по моему беспочвенны. Хотя быть может вы меня переубедите, если предложите другие аргументы.
|
|
|
|
|
Aug 20 2008, 21:26
|

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

|
Цитата(rezident @ Aug 20 2008, 23:51)  Кстати, вы так и не привели выдержку о том, что представляет из себя символьное обозначение RA0. Дедуктивно предположу, что это разыменованный указатель на volatile переменную типа bit. Этот тип является расширением языка HI-TECH компилятора для пиков. И если компилятор позволяет делать указатели на такие типы (в чем я сомневаюсь, зная систему команд пика), то поле структуры должно быть объявлено как bit volatile * pin_address; или volatile bit * pin_address;, что одно и то же.
--------------------
На любой вопрос даю любой ответ"Write code that is guaranteed to work, not code that doesn’t seem to break" ( C++ FAQ)
|
|
|
|
|
Aug 21 2008, 10:12
|

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

|
Цитата(rezident @ Aug 21 2008, 01:06)  А это как-то поможет с реализацией задуманного автором при использовании его структуры? Если бы архитектура пиков поддерживала указатели на биты (о чём Сергей и сказал), то помогло бы. Просто надо было бы так. Код unsigned char do_antiripple_pin(struct data_unit * data , char time_conversion) { ... if( data->changing_data != *data->pin_addres) // И если новые данные отличаются от старых { data->temp_pin_value = *data->pin_addres; // то сохраняем временные данные; data->diferences_counter++; // и увеличиваем счетчик несовпадений пина } Аналогично и Ваш пример Цитата(rezident @ Aug 20 2008, 18:29)  Либо попробовать симмитировать ее, как-то так Код if ((((volatile unsigned char)(*pPortM))&BIT0)==0) { ... } можно изменить так Код if ( ( (*(volatile unsigned char*)pPortM) & BIT0) ==0) { ... } Или вообще сразу объявить pPortM как volatile unsigned char * Тут достаточно адресуемости всего порта, но номер ножки придётся передавать отдельно и делать маску на месте. Так что автору вопроса надо что-то в духе Код struct data_unit { volatile unsigned char *port_address; unsigned char pin_number; // Дальше остальные поля, не факт, что все они должны быть volatile, это надо смотреть // Всю структуру делать volatile врядли надо, это как минимум сделает volatile и сам port_address, что // явно не нужно. Хотя, если сделать лишь одно обращение к полю port_address, как это сделано // ниже, квалифицирование port_address как volatile не приведёт к раздуванию кода ... };
...... unsigned char new_state = 0;
if( *data->port_address & (1 << data->pin_number) ) new_state = 1;
if( data->prev_state != new_state) { data->temp = new_state; ... // ну и дальше по вкусу, я не вникал в тонкости именно этого варианта подавления дребезга
--------------------
Ну, я пошёл… Если что – звоните…
|
|
|
|
Сообщений в этой теме
Илья_ Проблема с volatile переменной - помогите плиз. Aug 19 2008, 14:21 rezident При вызове функции do_antiripple_pin квалификатор ... Aug 19 2008, 15:25 rezident Цитата(Илья_ @ Aug 20 2008, 12:46) Но гла... Aug 20 2008, 15:29 Олег Хохлов Цитата(Илья_ @ Aug 19 2008, 17:21) В эту ... Aug 20 2008, 20:29 AHTOXA Можно ещё вот так:
Кодtypedef int (* ReadStat... Aug 21 2008, 10:56 ReAl Цитата(AHTOXA @ Aug 21 2008, 13:56) Можно... Aug 21 2008, 12:01  AHTOXA Цитата(ReAl @ Aug 21 2008, 18:01) Можно и... Aug 21 2008, 12:21 Илья_ Добрый день господа форумчане.
Ваши советы, деийс... Aug 22 2008, 07:32 AHTOXA Цитата(Илья_ @ Aug 22 2008, 13:32) Г-ну A... Aug 22 2008, 14:10
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0
|
|
|