open() : Устройство или ресурс занят (/dev/rtc(x))

Итак, у меня есть RTC, подключенный к моему бигльбону, и он отлично работает благодаря /sys/class/rtc/rtx(x)/time file, однако в моем коде C для отслеживания времени есть ошибка, которую я не могу решить.

if ((rtc_fd = open(RTC, O_RDONLY, 444)) < 0)
    REPORT_ERROR("open(RTC)");

где RTC — это путь к /dev/rtc1. REPORT_ERROR — это функция макроса для сообщения о пользовательских ошибках.

В любом случае, я запускаю этот код непосредственно перед циклом for с 10 итерациями, и он выводится в файл журнала. Я всегда получаю сообщение strerror(perror): Устройство или ресурс занят, но затем он по-прежнему выдает мне 10 правильных выводов.

Я также использую close() в конце.

Что дает?

edit: Возможно, я должен добавить, что это работает в демонизированном процессе, и я использую iocotl() с RTC_RD_TIME во время цикла.

#define REPORT_ERROR(X) do {\
        fprintf(log,"err@ "X": %s@ %s:%d\n",\
        strerror(errno), __FILE__, __LINE__ - 1);\
        exit(EXIT_FAILURE);\
        } while(0)

#define RTC "/dev/rtc1"

int main(void)
{
    int rtc_fd;
    FILE *log;
    struct rtc_time tm;

    if ((log = fopen(LOG_FILE, "a+")) == NULL)
        exit(EXIT_FAILURE);

    if ((dup2(fileno(log), STDERR_FILENO)) < 0)
        REPORT_ERROR("dup2()");

    if ((rtc_fd = open(RTC, O_RDONLY, 444)) < 0)
        REPORT_ERROR("open(RTC)");

    /* Main loop */
    for (int i = 0; i < 10; ++i)
    {
        if ((ioctl(rtc_fd, RTC_RD_TIME, &rtctime)) != 0)
            REPORT_ERROR("ioctl(rtc_fd)");

        fprintf(log, "%02d:%02d:%02.lf   %d-%d-%d\n", tm.hour, tm.minute,
                tm.second, tm.mon + 1, tm.mday, tm.year + 1900);
        sleep(1);
    }

    close(rtc_fd);
    return 0;
}

ВЫХОД:

err@ open(RTC): Device or resource busy@ ha-daemon.c:80
05:06:09   2-12-2015
05:06:10   2-12-2015
05:06:11   2-12-2015
05:06:12   2-12-2015
05:06:13   2-12-2015
05:06:14   2-12-2015
05:06:15   2-12-2015
05:06:16   2-12-2015
05:06:17   2-12-2015
05:06:18   2-12-2015

person Community    schedule 12.02.2015    source источник
comment
Создайте минимальный, полный и проверяемый пример и покажите нам.   -  person Some programmer dude    schedule 12.02.2015
comment
Вы пытались открыть свое устройство глобально?   -  person Shehryar    schedule 12.02.2015
comment
Что касается вашей реальной проблемы (открытие устройства), вы уверены, что ни один другой процесс уже не открыл точно такое же устройство?   -  person Some programmer dude    schedule 12.02.2015
comment
@JoachimPileborg, FILE * log не не определен, я просто не вырезал этот код, он есть сейчас. Я думал, что кто-то может знать причину этой проблемы, не видя так много кода, в конце концов, я сказал в своем вопросе, что вывод правильный после первоначального сообщения об ошибке.   -  person    schedule 12.02.2015
comment
@Shehryar Shehryar Я не уверен, что вы имеете в виду, открывая устройство глобально? Если я su, чем cat /sys/class/rtc/rtc1/time, я могу получить время.   -  person    schedule 12.02.2015
comment
Чтобы уточнить, вызов какой функции приводит к ошибке? Звонок open? Звонок ioctl? И вы уверены, что макрос действительно используется для печати сообщения? Потому что, если сообщение напечатано макросом, процесс должен выйти немедленно. Также предоставьте фактический вывод.   -  person Some programmer dude    schedule 12.02.2015
comment
Что касается отладки проблемы, сработает ли она, если вы не запустите ее как демон? В противном случае вы можете запустить отладчик, чтобы проверить, что происходит. Вы также можете добавить больше вывода журнала, чтобы более подробно отслеживать код, если вы не можете запустить его в отладчике.   -  person Some programmer dude    schedule 12.02.2015
comment
@JoachimPileborg, я уверен, что это макрос печатает сообщение, но это хороший момент о том, как он должен выходить немедленно. Я собираюсь попробовать это, не работая в качестве демона, и я буду публиковать обновления, хотя сначала я должен поесть. Я отредактировал вопрос с выводом. В строке 80 находится вызов open(). редактировать: я забыл спросить, как я могу проверить, обращается ли другой процесс к /dev/rtc1? Я запускаю ps aux | grep rtc и видим процесс /sbin/init fixrtc   -  person    schedule 12.02.2015
comment
Я изменил код минимально, чтобы он компилировался для меня (gcc/Ubuntu 14.04 64-бит). Он ожидает экспертной оценки. Здесь он работает так, как ожидалось: либо он запускается без сообщения об ошибке, либо печатает сообщение об ошибке и немедленно завершает работу.   -  person Edgar Bonet    schedule 12.02.2015
comment
Что касается способа проверить, в каком процессе открыты файлы, вы можете использовать lsof< /a> для списка открытых файлов, затем grep например. rtc1.   -  person Some programmer dude    schedule 12.02.2015


Ответы (1)


На самом деле это не ответ, но мое редактирование вопроса было отклонено с комментарием, что вместо этого я должен опубликовать его как ответ. Итак, вот оно.

Код, как указано, не компилируется. Я изменил его минимально, чтобы он компилировался без предупреждений, и я мог его протестировать. Вот список изменений. Некоторые из них могут относиться к моей среде (gcc 4.8.1/Ubuntu Linux 14.04/x86-64):

  • добавлены все недостающие #includes
  • определено LOG_FILE, определение его как /dev/tty просто упрощает тестирование
  • заменил /dev/rtc1 на /dev/rtc0 (у меня нет rtc1)
  • заменил &rtctime (последний аргумент ioctl) на &tm, иначе это не имело бы смысла и не компилировалось бы
  • исправлен формат printf: "%02d" вместо "%02.lf"
  • укажите правильные имена для полей структуры rtctime: tm_hour, tm_min и т. д.

А вот модифицированный код:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <unistd.h>
#include <time.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/ioctl.h>
#include <fcntl.h>
#include <linux/rtc.h>

#define LOG_FILE "/dev/tty"

#define REPORT_ERROR(X) do {\
        fprintf(log,"err@ "X": %s@ %s:%d\n",\
        strerror(errno), __FILE__, __LINE__ - 1);\
        exit(EXIT_FAILURE);\
        } while(0)

#define RTC "/dev/rtc0"

int main(void)
{
    int rtc_fd;
    FILE *log;
    struct rtc_time tm;

    if ((log = fopen(LOG_FILE, "a+")) == NULL)
        exit(EXIT_FAILURE);

    if ((dup2(fileno(log), STDERR_FILENO)) < 0)
        REPORT_ERROR("dup2()");

    if ((rtc_fd = open(RTC, O_RDONLY, 444)) < 0)
        REPORT_ERROR("open(RTC)");

    /* Main loop */
    for (int i = 0; i < 10; ++i)
    {
        if ((ioctl(rtc_fd, RTC_RD_TIME, &tm)) != 0)
            REPORT_ERROR("ioctl(rtc_fd)");

        fprintf(log, "%02d:%02d:%02d   %d-%d-%d\n", tm.tm_hour, tm.tm_min,
                tm.tm_sec, tm.tm_mon + 1, tm.tm_mday, tm.tm_year + 1900);
        sleep(1);
    }

    close(rtc_fd);
    return 0;
}

Результаты теста: компилируется и работает как положено. Либо он запускается без сообщения об ошибке, либо печатает сообщение об ошибке и немедленно завершает работу. Если запущен один экземпляр программы, запуск второго экземпляра выдает сообщение об ошибке «Устройство или ресурс занят», как и ожидалось.

Другими словами, «работает на меня».

person Edgar Bonet    schedule 12.02.2015