AUTOSAR C++14 Rule A27-0-3

Альтернативные операции ввода и вывода на потоке файла не должны использоваться без прошедшего сброса или располагающий вызов

Описание

Управляйте определением

Альтернативные операции ввода и вывода на потоке файла не должны использоваться без прошедшего сброса или располагающий вызов.

Объяснение

Чередование операций ввода и вывода на потоке без прошедшего сброса или расположение вызова являются неопределенным поведением.

Реализация Polyspace

Средство проверки проверяет на ситуации когда:

  • Вы не выполняете сброс или вызов расположения функции между выходной операцией и следующей входной операцией на потоке файла в режиме обновления.

    Чтобы разрешить нарушение правила, вызовите fflush() или расположение файла функционирует, такие как fseek() или fsetpos() между выходом и входными операциями на потоке обновления.

  • Вы не выполняете вызов расположения функции между входной операцией и следующей выходной операцией на потоке файла в режиме обновления.

    Чтобы разрешить нарушение правила, вызовите функцию расположения файла между операциями ввода и вывода на потоке обновления.

Поиск и устранение проблем

Если вы ожидаете нарушение правила, но не видите его, относитесь, чтобы Диагностировать, Почему Кодирующие Стандартные Нарушения Не Появляются как ожидалось.

Примеры

развернуть все

#include <stdio.h>
#define SIZE20 20

void initialize_data(char* data, size_t s) {};
const char *temp_filename = "/tmp/demo.txt";

void funcNonCompliant()
{
    char data[SIZE20];
    char append_data[SIZE20];
    FILE *file;

    file = fopen(temp_filename, "a+");
    if (file == NULL)
      {
        /* Handle error. */;
      }

    initialize_data(append_data, SIZE20);

    if (fwrite(append_data, 1, SIZE20, file) != SIZE20)
      {
        (void)fclose(file);
        /* Handle error. */;
      }
	/* Read operation after write without 
	intervening flush. */
    if (fread(data, 1, SIZE20, file) < SIZE20)  //Noncompliant
      {
          (void)fclose(file);
          /* Handle error. */;
      }

    if (fclose(file) == EOF)
      {
        /* Handle error. */;
      }
}
        

void funcCompliant()
{
    char data[SIZE20];
    char append_data[SIZE20];
    FILE *file;

    file = fopen(temp_filename, "a+");
    if (file == NULL)
      {
        /* Handle error. */;
      }

    initialize_data(append_data, SIZE20);

    if (fwrite(append_data, 1, SIZE20, file) != SIZE20)
      {
        (void)fclose(file);
        /* Handle error. */;
      }
	/* Buffer flush after write and before read */
    if (fflush(file) != 0)  
      {
        (void)fclose(file);
        /* Handle error. */;
      }
    if (fread(data, 1, SIZE20, file) < SIZE20) //Compliant
      {
        (void)fclose(file);
        /* Handle error. */;
      }

    if (fclose(file) == EOF)
      {
        /* Handle error. */;
      }
} 

В этом примере, в функциональном funcNonCompliant, файл demo.txt открыт для чтения и добавления. После вызова fwrite(), вызов fread() без прошедшего сброса операция является неопределенным поведением и нарушает правило.

Функциональный funcCompliant показывает альтернативное решение, где вызов сброса выполняется после записывания данные к файлу и прежде, чем вызвать fread().

Проверяйте информацию

Группа: библиотека ввода/вывода
Категория: необходимый, автоматизированный
Введенный в R2020b