JPCERT コーディネーションセンター

MEM34-C. 動的に割り当てられたメモリのみを解放する

MEM34-C. 動的に割り当てられたメモリのみを解放する

C 標準 [ISO/IEC 9899:2011] 附属書 J 「未定義の動作」の 179 に記載の通り、以下の場合、プログラムの動作は未定義である。

free または realloc 関数へのポインタ引数がメモリ管理関数によって以前に返されたポインタと一致しないか、もしくは、領域が free または realloc によって解放された。

動的に割り当てられていないメモリを解放すると、「MEM31-C. 動的に割り当てられたメモリは一度だけ解放する」で説明されているような深刻なエラーにつながる可能性がある。エラーの結果何が起こるかは処理系により異なる。何も起こらないかもしれないし、プログラムが異常終了するかもしれない。処理系に関係なく、malloc()calloc()realloc()aligned_alloc などの動的メモリ割り当て関数によって変えさえたポインタ以外での free() の呼び出しは避ける。

realloc() に、動的に割り当てられていないメモリへのポインタが指定された場合も同様の状況が生じる。realloc() 関数を使い、動的メモリのブロックサイズを変更できるが、malloc() などのメモリ割り当て関数によって割り当てられたものでないメモリへのポインタが指定されると、プログラムは異常終了する場合がある。

違反コード

以下のコード例では、argc の値に応じて、動的に割り当てられたメモリもしくは静的に割り当てられた文字列リテラルを参照するように str の値をセットする。どちらの場合も strfree() に引数として渡される。str が動的に割り当てられたメモリ以外を参照している場合、free(str) を実行してはならない。

enum { MAX_ALLOCATION = 1000 };

int main(int argc, const char *argv[]) {
  char *str = NULL;
  size_t len;

  if (argc == 2) {
    len = strlen(argv[1])+1;
    if (len > MAX_ALLOCATION) {
      /* エラー処理 */
    }
    str = (char *)malloc(len);
    if (str == NULL) {
      /* 割り当てエラーの処理 */
    }
    strcpy(str, argv[1]);
  }
  else {
    str = "usage: $>a.exe [string]";
    printf("%s\n", str);
  }
  /* ... */
  free(str);
  return 0;
}
適合コード

以下の解決法では、strfree() に渡されたときに動的に割り当てられていないメモリを参照する可能性を排除している。

enum { MAX_ALLOCATION = 1000 };

int main(int argc, const char *argv[]) {
  char *str = NULL;
  size_t len;

  if (argc == 2) {
    len = strlen(argv[1])+1;
    if (len > MAX_ALLOCATION) {
      /* エラー処理 */
    }
    str = (char *)malloc(len);
    if (str == NULL) {
      /* 割り当てエラーの処理 */
    }
    strcpy(str, argv[1]);
  }
  else {
    printf("%s\n", "usage: $>a.exe [string]");
    return -1;
  }
  /* ... */
  free(str);
  return 0;
}
違反コード (realloc())

以下のコード例では、realloc()buf を指すポインタパラメータは、動的に割り当てられたメモリを参照していない。

#define BUFSIZE 256
 
void f(void) {
  char buf[BUFSIZE];
  char *p;
  /* ... */
  p = (char *)realloc(buf, 2 * BUFSIZE);  /* 違反 */
  /* ... */
}
適合コード (realloc())

この適合コードでは、buf が動的に割り当てられたメモリを参照している。

#define BUFSIZE 256
 
void f(void) {
  char *buf = (char *)malloc(BUFSIZE * sizeof(char));
  char *p;
  /* ... */
  p = (char *)realloc(buf, 2 * BUFSIZE);  /* 違反 */
  /* ... */
}
例外

MEM34-EX1:一部のライブラリ処理系では、割り当てられていないメモリの解放を受け入れて無視する (そうでないと、代わりに実行時制約違反が引き起こされる)。プロジェクトで使用されているすべてのライブラリがこのように動作すると検証されている場合は、このルールを無視して構わない。

リスク評価

動的に割り当てられていないメモリの解放や再割り当てを行うと、そのメモリが malloc() によって再利用される場合に任意のコードを実行される可能性がある。

ルール

深刻度

可能性

修正コスト

優先度

レベル

MEM34-C

P18

L1

自動検出(最新の情報はこちら

ツール

バージョン

チェッカー

説明

Compass/ROSE

   

このルールの違反を検出できる。

Coverity

6.5

BAD_FREE

引数が関数または配列へのポインタの場合に、free() の呼び出しを特定する。また、Free が address-of 式で使用されている事例 (ヒープを決して割り当てることができない) を検出する。Coverity Prevent は、このルールの違反をすべて検出できるわけではないため、さらなる検証が必要である。

Klocwork

V. 9.1

FNH.MIGHT
FNH.MUST
FUM.GEN.MIGHT
FUM.GEN.MUST

 

LDRA tool suite

V. 8.5.4

483 S

実装済み
関連するガイドライン
CERT C++ Secure Coding Standard MEM34-CPP. Only free memory allocated dynamically
ISO/IEC TS 17961 (ドラフト) Reallocating or freeing memory that was not dynamically allocatied [xfree]
MITRE CWE CWE-590, Free of invalid pointer not on the heap
参考資料
[Seacord 2013] Chapter 4, "Dynamic Memory Management"
翻訳元

これは以下のページを翻訳したものです。

MEM34-C. Only free memory allocated dynamically (revision 94)

Top へ

Topへ
最新情報(RSSメーリングリストTwitter