llvm/clang/test/Analysis/stream-note.c

// RUN: %clang_analyze_cc1 -analyzer-checker=core,unix.Stream -analyzer-output text \
// RUN:   -analyzer-config unix.Stream:Pedantic=true \
// RUN:   -verify %s
// RUN: %clang_analyze_cc1 -analyzer-checker=core,unix.Stream,unix.StdCLibraryFunctions -analyzer-output text \
// RUN:   -analyzer-config unix.Stream:Pedantic=true \
// RUN:   -analyzer-config unix.StdCLibraryFunctions:ModelPOSIX=true -verify=expected,stdargs %s

#include "Inputs/system-header-simulator.h"

void check_note_at_correct_open(void) {
  FILE *F1 = tmpfile(); // expected-note {{Stream opened here}}
  // stdargs-note@-1 {{'tmpfile' is successful}}
  if (!F1)
    // expected-note@-1 {{'F1' is non-null}}
    // expected-note@-2 {{Taking false branch}}
    return;
  FILE *F2 = tmpfile();
  if (!F2) {
    // expected-note@-1 {{'F2' is non-null}}
    // expected-note@-2 {{Taking false branch}}
    fclose(F1);
    return;
  }
  rewind(F2);
  fclose(F2);
  rewind(F1);
}
// expected-warning@-1 {{Opened stream never closed. Potential resource leak}}
// expected-note@-2 {{Opened stream never closed. Potential resource leak}}

void check_note_fopen(void) {
  FILE *F = fopen("file", "r"); // expected-note {{Stream opened here}}
  // stdargs-note@-1 {{'fopen' is successful}}
  if (!F)
    // expected-note@-1 {{'F' is non-null}}
    // expected-note@-2 {{Taking false branch}}
    return;
}
// expected-warning@-1 {{Opened stream never closed. Potential resource leak}}
// expected-note@-2 {{Opened stream never closed. Potential resource leak}}

void check_note_freopen(void) {
  FILE *F = fopen("file", "r"); // expected-note {{Stream opened here}}
  // stdargs-note@-1 {{'fopen' is successful}}
  if (!F)
    // expected-note@-1 {{'F' is non-null}}
    // expected-note@-2 {{Taking false branch}}
    return;
  F = freopen(0, "w", F); // expected-note {{Stream reopened here}}
  // stdargs-note@-1 {{'freopen' is successful}}
  if (!F)
    // expected-note@-1 {{'F' is non-null}}
    // expected-note@-2 {{Taking false branch}}
    return;
}
// expected-warning@-1 {{Opened stream never closed. Potential resource leak}}
// expected-note@-2 {{Opened stream never closed. Potential resource leak}}

void check_note_fdopen(int fd) {
  FILE *F = fdopen(fd, "r"); // expected-note {{Stream opened here}}
  // stdargs-note@-1 {{'fdopen' is successful}}
  if (!F)
    // expected-note@-1 {{'F' is non-null}}
    // expected-note@-2 {{Taking false branch}}
    return;
}
// expected-warning@-1 {{Opened stream never closed. Potential resource leak}}
// expected-note@-2 {{Opened stream never closed. Potential resource leak}}

void check_note_leak_2(int c) {
  FILE *F1 = fopen("foo1.c", "r"); // expected-note {{Stream opened here}}
  // stdargs-note@-1 {{'fopen' is successful}}
  if (!F1)
    // expected-note@-1 {{'F1' is non-null}}
    // expected-note@-2 {{Taking false branch}}
    // expected-note@-3 {{'F1' is non-null}}
    // expected-note@-4 {{Taking false branch}}
    return;
  FILE *F2 = fopen("foo2.c", "r"); // expected-note {{Stream opened here}}
  // stdargs-note@-1 {{'fopen' is successful}}
  if (!F2) {
    // expected-note@-1 {{'F2' is non-null}}
    // expected-note@-2 {{Taking false branch}}
    // expected-note@-3 {{'F2' is non-null}}
    // expected-note@-4 {{Taking false branch}}
    fclose(F1);
    return;
  }
  if (c)
    // expected-note@-1 {{Assuming 'c' is not equal to 0}}
    // expected-note@-2 {{Taking true branch}}
    // expected-note@-3 {{Assuming 'c' is not equal to 0}}
    // expected-note@-4 {{Taking true branch}}
    return;
  // expected-warning@-1 {{Opened stream never closed. Potential resource leak}}
  // expected-note@-2 {{Opened stream never closed. Potential resource leak}}
  // expected-warning@-3 {{Opened stream never closed. Potential resource leak}}
  // expected-note@-4 {{Opened stream never closed. Potential resource leak}}
  fclose(F1);
  fclose(F2);
}

void check_track_null(void) {
  FILE *F;
  F = fopen("foo1.c", "r"); // expected-note {{Value assigned to 'F'}} expected-note {{Assuming pointer value is null}}
  // stdargs-note@-1 {{'fopen' fails}}
  if (F != NULL) {          // expected-note {{Taking false branch}} expected-note {{'F' is equal to NULL}}
    fclose(F);
    return;
  }
  fclose(F); // expected-warning {{Stream pointer might be NULL}}
             // expected-note@-1 {{Stream pointer might be NULL}}
}

void check_eof_notes_feof_after_feof(void) {
  FILE *F;
  char Buf[10];
  F = fopen("foo1.c", "r");
  if (F == NULL) { // expected-note {{Taking false branch}} expected-note {{'F' is not equal to NULL}}
    return;
  }
  fread(Buf, 1, 1, F);
  if (feof(F)) { // expected-note {{Taking true branch}}
    clearerr(F);
    fread(Buf, 1, 1, F);   // expected-note {{Assuming stream reaches end-of-file here}}
    if (feof(F)) {         // expected-note {{Taking true branch}}
      fread(Buf, 1, 1, F); // expected-warning {{Read function called when stream is in EOF state. Function has no effect}}
      // expected-note@-1 {{Read function called when stream is in EOF state. Function has no effect}}
    }
  }
  fclose(F);
}

void check_eof_notes_feof_after_no_feof(void) {
  FILE *F;
  char Buf[10];
  F = fopen("foo1.c", "r");
  if (F == NULL) { // expected-note {{Taking false branch}} expected-note {{'F' is not equal to NULL}}
    return;
  }
  fread(Buf, 1, 1, F);
  if (feof(F)) { // expected-note {{Taking false branch}}
    fclose(F);
    return;
  } else if (ferror(F)) { // expected-note {{Taking false branch}}
    fclose(F);
    return;
  }
  fread(Buf, 1, 1, F);   // expected-note {{Assuming stream reaches end-of-file here}}
  if (feof(F)) {         // expected-note {{Taking true branch}}
    fread(Buf, 1, 1, F); // expected-warning {{Read function called when stream is in EOF state. Function has no effect}}
    // expected-note@-1 {{Read function called when stream is in EOF state. Function has no effect}}
  }
  fclose(F);
}

void check_eof_notes_feof_or_no_error(void) {
  FILE *F;
  char Buf[10];
  F = fopen("foo1.c", "r");
  if (F == NULL) // expected-note {{Taking false branch}} expected-note {{'F' is not equal to NULL}}
    return;
  int RRet = fread(Buf, 1, 1, F); // expected-note {{Assuming stream reaches end-of-file here}}
  if (ferror(F)) {                // expected-note {{Taking false branch}}
  } else {
    fread(Buf, 1, 1, F); // expected-warning {{Read function called when stream is in EOF state. Function has no effect}}
    // expected-note@-1 {{Read function called when stream is in EOF state. Function has no effect}}
  }
  fclose(F);
}

void check_indeterminate_notes(void) {
  FILE *F;
  F = fopen("foo1.c", "r");
  if (F == NULL)     // expected-note {{Taking false branch}} \
                     // expected-note {{'F' is not equal to NULL}}
    return;
  int R = fgetc(F);  // no note
  if (R >= 0) {      // expected-note {{Taking true branch}} \
                     // expected-note {{'R' is >= 0}}
    fgetc(F);        // expected-note {{Assuming this stream operation fails}}
    if (ferror(F))   // expected-note {{Taking true branch}}
      fgetc(F);      // expected-warning {{File position of the stream might be 'indeterminate' after a failed operation. Can cause undefined behavior}} \
                     // expected-note {{File position of the stream might be 'indeterminate' after a failed operation. Can cause undefined behavior}}
  }
  fclose(F);
}

void check_indeterminate_after_clearerr(void) {
  FILE *F;
  char Buf[10];
  F = fopen("foo1.c", "r");
  if (F == NULL)          // expected-note {{Taking false branch}} \
                          // expected-note {{'F' is not equal to NULL}}
    return;
  fread(Buf, 1, 1, F);    // expected-note {{Assuming this stream operation fails}}
  if (ferror(F)) {        // expected-note {{Taking true branch}}
    clearerr(F);
    fread(Buf, 1, 1, F);  // expected-warning {{might be 'indeterminate' after a failed operation}} \
                          // expected-note {{might be 'indeterminate' after a failed operation}}
  }
  fclose(F);
}

void check_indeterminate_eof(void) {
  FILE *F;
  char Buf[2];
  F = fopen("foo1.c", "r");
  if (F == NULL)               // expected-note {{Taking false branch}} \
                               // expected-note {{'F' is not equal to NULL}} \
                               // expected-note {{Taking false branch}} \
                               // expected-note {{'F' is not equal to NULL}}
    return;
  fgets(Buf, sizeof(Buf), F);  // expected-note {{Assuming this stream operation fails}} \
                               // expected-note {{Assuming stream reaches end-of-file here}}

  fgets(Buf, sizeof(Buf), F);  // expected-warning {{might be 'indeterminate'}} \
                               // expected-note {{might be 'indeterminate'}} \
                               // expected-warning {{stream is in EOF state}} \
                               // expected-note {{stream is in EOF state}}
  fclose(F);
}

void check_indeterminate_fseek(void) {
  FILE *F = fopen("file", "r");
  if (!F)                           // expected-note {{Taking false branch}} \
                                    // expected-note {{'F' is non-null}}
    return;
  int Ret = fseek(F, 1, SEEK_SET);  // expected-note {{Assuming this stream operation fails}}
  if (Ret) {                        // expected-note {{Taking true branch}} \
                                    // expected-note {{'Ret' is -1}}
    char Buf[2];
    fwrite(Buf, 1, 2, F);           // expected-warning {{might be 'indeterminate'}} \
                                    // expected-note {{might be 'indeterminate'}}
  }
  fclose(F);
}

void error_fseek_ftell(void) {
  FILE *F = fopen("file", "r");
  if (!F)                 // expected-note {{Taking false branch}} \
                          // expected-note {{'F' is non-null}}
    return;
  fseek(F, 0, SEEK_END);  // expected-note {{Assuming this stream operation fails}}
  long size = ftell(F);   // expected-warning {{might be 'indeterminate'}} \
                          // expected-note {{might be 'indeterminate'}}
  if (size == -1) {
    fclose(F);
    return;
  }
  if (size == 1)
    fprintf(F, "abcd");
  fclose(F);
}

void error_fseek_read_eof(void) {
  FILE *F = fopen("file", "r");
  if (!F)
    return;
  if (fseek(F, 22, SEEK_SET) == -1) {
    fclose(F);
    return;
  }
  fgetc(F); // no warning
  fclose(F);
}

void check_note_at_use_after_close(void) {
  FILE *F = tmpfile();
  if (!F) // expected-note {{'F' is non-null}} expected-note {{Taking false branch}}
    return;
  fclose(F); // expected-note {{Stream is closed here}}
  rewind(F); // expected-warning {{Use of a stream that might be already closed}}
  // expected-note@-1 {{Use of a stream that might be already closed}}
}