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

 
 
> Проверка инициализации указателя в С, Керниган и Ритчи молчат как партизаны...
tolik_zp
сообщение May 24 2007, 06:54
Сообщение #1


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

Группа: Участник
Сообщений: 127
Регистрация: 2-08-06
Пользователь №: 19 265



Есть код на с, называется FatFs (FatFs). В этом коде присутствуют такие выражения:

static
FATFS *FatFs[_DRIVES]; /* Pointer to the file system objects (logical drives, _DRIVES=1) */

тип FATFS определен в заголовочном файле, вроде с этим проблем нет.

далее есть функция:
f_mkfs (...)
{
FATFS *fs;
...
fs = FatFs[0];
if (!fs) return FR_NOT_ENABLED;
...

Объясните пожалуйста, почему функция возвращает FR_NOT_ENABLED и как вообще происходит проверка указателя?
Go to the top of the page
 
+Quote Post
 
Start new topic
Ответов (1 - 6)
Сергей Борщ
сообщение May 24 2007, 07:23
Сообщение #2


Гуру
******

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



Цитата(tolik_zp @ May 24 2007, 09:54) *
Объясните пожалуйста, почему функция возвращает FR_NOT_ENABLED и как вообще происходит проверка указателя?


Код
static
FATFS *FatFs[_DRIVES];  /* Pointer to the file system objects (logical drives, _DRIVES=1) */
Объявляется глобальная переменная (массив). Глобальные переменные при старте программы обнуляются - это аксиома, требование стандарта. Значит, все элементы этого массива содержат нули при старте программы. По тому же стандарту указатель, содержащий 0 никуда не указывает. Это и используется в следующей проверке
Код
  fs = FatFs[0];
  if (!fs) return FR_NOT_ENABLED;
Берется первый элемент массива. Если там указатель, который указывает на что-то реальное, он ненулевой.


--------------------
На любой вопрос даю любой ответ
"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
Edmundo
сообщение May 24 2007, 07:54
Сообщение #3


Мастер
****

Группа: Свой
Сообщений: 730
Регистрация: 18-02-06
Из: Москва
Пользователь №: 14 474



Цитата(Сергей Борщ @ May 24 2007, 11:23) *
Глобальные переменные при старте программы обнуляются - это аксиома, требование стандарта.

Что-то мне помнится, что обнуление переменных в стандарте не прописано, просто некоторые компиляторы могут это делать. Поэтому я лично всегда указатели присваиваю NULL (либо присваиванием, либо memset'ом, когда много). Могу ошибаться, но вроде бы все переменные (и глобальные, и локальные) по умолчанию undefined.

[ADD]
Все-таки ошибаюсь. Керниган-Ричи:

"Автоматические переменные, не инициализируемые яв-
но, имеют неопределенные значения, (т.е. мусор). Внешние и
статические переменные по умолчанию инициализируются нулем,
но, тем не менее, их явная инициализация является признаком
хорошего стиля."


--------------------
شامل
Go to the top of the page
 
+Quote Post
tolik_zp
сообщение May 24 2007, 09:30
Сообщение #4


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

Группа: Участник
Сообщений: 127
Регистрация: 2-08-06
Пользователь №: 19 265



Цитата(Сергей Борщ @ May 24 2007, 10:23) *
Код
static
FATFS *FatFs[_DRIVES];  /* Pointer to the file system objects (logical drives, _DRIVES=1) */
Объявляется глобальная переменная (массив). Глобальные переменные при старте программы обнуляются - это аксиома, требование стандарта. Значит, все элементы этого массива содержат нули при старте программы. По тому же стандарту указатель, содержащий 0 никуда не указывает. Это и используется в следующей проверке
Код
  fs = FatFs[0];
  if (!fs) return FR_NOT_ENABLED;
Берется первый элемент массива. Если там указатель, который указывает на что-то реальное, он ненулевой.


Хм... Почему тогда функция вылетает с ошибкой FR_NOT_ENABLED, если массив указателей на структуру я инициализировал, а также инициализировал указатель fs? Как с этим бороться?
Go to the top of the page
 
+Quote Post
zltigo
сообщение May 24 2007, 09:42
Сообщение #5


Гуру
******

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



Цитата(tolik_zp @ May 24 2007, 12:30) *
Хм... Почему тогда функция вылетает с ошибкой FR_NOT_ENABLED, если массив указателей на структуру я инициализировал, а также инициализировал указатель fs? Как с этим бороться?

Читаем Сергея. Внимательно читаем:
Цитата
Берется первый элемент массива. Если там указатель, который указывает на что-то реальное, он ненулевой.

Чудес не бывает. Массив указателей не проинициаизирован реальными значениями.


--------------------
Feci, quod potui, faciant meliora potentes
Go to the top of the page
 
+Quote Post
sergeeff
сообщение May 24 2007, 11:03
Сообщение #6


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

Группа: Свой
Сообщений: 1 481
Регистрация: 10-04-05
Пользователь №: 4 007



Неинициализированные переменные компилятор помещает в сегмент BSS. Данный сегмент заполняется нулями, как правило, в startup модуле, который вызывается до main. Советую там и посмотреть.
Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение May 24 2007, 15:27
Сообщение #7


Гуру
******

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



Цитата(tolik_zp @ May 24 2007, 12:30) *
если массив указателей на структуру я инициализировал, а также инициализировал указатель fs? Как с этим бороться?
Если вы инициализировали массив FatFs - то вы в него занесли какое-то отличное от нуля число (реальный адрес). Если функция вылетает именно в приведенном вами месте, то значит между этими двумя событиями кто-то массив "почистил". Вообще-то инициализацию этого массива делает сама FatFS при вызове f_mount, если мне не изменяет память. Т.е. перед вызовом f_mkfs вам надо сначала смонтировать носитель.


--------------------
На любой вопрос даю любой ответ
"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

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

 


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


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