* tar xzf utvpn-src-unix-v101-7101-public-2010.06.27.tar.gz
[lab.git] / utvpn / utvpn-unix-v101-7101-public / src / Mayaqua / Memory.c
diff --git a/utvpn/utvpn-unix-v101-7101-public/src/Mayaqua/Memory.c b/utvpn/utvpn-unix-v101-7101-public/src/Mayaqua/Memory.c
new file mode 100644 (file)
index 0000000..630faf3
--- /dev/null
@@ -0,0 +1,2868 @@
+// SoftEther UT-VPN SourceCode\r
+// \r
+// Copyright (C) 2004-2010 SoftEther Corporation.\r
+// Copyright (C) 2004-2010 University of Tsukuba, Japan.\r
+// Copyright (C) 2003-2010 Daiyuu Nobori.\r
+// All Rights Reserved.\r
+// \r
+// http://utvpn.tsukuba.ac.jp/\r
+// \r
+// This program is free software; you can redistribute it and/or\r
+// modify it under the terms of the GNU General Public License\r
+// version 2 as published by the Free Software Foundation.\r
+// \r
+// This program is distributed in the hope that it will be useful,\r
+// but WITHOUT ANY WARRANTY; without even the implied warranty of\r
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\r
+// GNU General Public License for more details.\r
+// \r
+// You should have received a copy of the GNU General Public License version 2\r
+// along with this program; if not, write to the Free Software\r
+// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.\r
+// \r
+// このファイルは GPL バージョン 2 ライセンスで公開されています。\r
+// 誰でもこのファイルの内容を複製、改変したり、改変したバージョンを再配布\r
+// することができます。ただし、原著作物を改変した場合は、原著作物の著作権表示\r
+// を除去することはできません。改変した著作物を配布する場合は、改変実施者の\r
+// 著作権表示を原著作物の著作権表示に付随して記載するようにしてください。\r
+// \r
+// この SoftEther UT-VPN オープンソース・プロジェクトは、日本国の\r
+// ソフトイーサ株式会社 (SoftEther Corporation, http://www.softether.co.jp/ )\r
+// および筑波大学 (University of Tsukuba, http://www.tsukuba.ac.jp/ ) によって\r
+// ホストされています。\r
+// 本プログラムの配布者は、本プログラムを、業としての利用以外のため、\r
+// および、試験または研究のために利用が行われることを想定して配布\r
+// しています。\r
+// SoftEther UT-VPN プロジェクトの Web サイトは http://utvpn.tsukuba.ac.jp/ に\r
+// あります。\r
+// 本ソフトウェアの不具合の修正、機能改良、セキュリティホールの修復などのコード\r
+// の改変を行った場合で、その成果物を SoftEther UT-VPN プロジェクトに提出して\r
+// いただける場合は、 http://utvpn.tsukuba.ac.jp/ までソースコードを送付して\r
+// ください。SoftEther UT-VPN プロジェクトの本体リリースまたはブランチリリース\r
+// に組み込みさせていただきます。\r
+// \r
+// GPL に基づいて原著作物が提供される本ソフトウェアの改良版を配布、販売する\r
+// 場合は、そのソースコードを GPL に基づいて誰にでも開示する義務が生じます。\r
+// \r
+// 本ソフトウェアに関連する著作権、特許権、商標権はソフトイーサ株式会社\r
+// (SoftEther Corporation) およびその他の著作権保持者が保有しています。\r
+// ソフトイーサ株式会社等はこれらの権利を放棄していません。本ソフトウェアの\r
+// 二次著作物を配布、販売する場合は、これらの権利を侵害しないようにご注意\r
+// ください。\r
+// \r
+// お願い: どのような通信ソフトウェアにも通常は必ず未発見の\r
+// セキュリティホールが潜んでいます。本ソースコードをご覧いただいた結果、\r
+// UT-VPN にセキュリティホールを発見された場合は、当該セキュリティホールの\r
+// 情報を不特定多数に開示される前に、必ず、ソフトイーサ株式会社\r
+// および脆弱性情報の届出を受け付ける公的機関まで通報いただき、\r
+// 公益保護にご協力いただきますようお願い申し上げます。\r
+// \r
+// ソフトイーサ株式会社は、当該セキュリティホールについて迅速に対処を\r
+// 行い、UT-VPN および UT-VPN に関連するソフトウェアのユーザー・顧客\r
+// を保護するための努力を行います。\r
+// \r
+// ソフトイーサへの届出先: http://www.softether.co.jp/jp/contact/\r
+// 日本国内の脆弱性情報届出受付公的機関:\r
+//         独立行政法人 情報処理推進機構\r
+//         http://www.ipa.go.jp/security/vuln/report/\r
+// \r
+// 上記各事項について不明な点は、ソフトイーサ株式会社までご連絡ください。\r
+// 連絡先: http://www.softether.co.jp/jp/contact/\r
+\r
+// -----------------------------------------------\r
+// [ChangeLog]\r
+// 2010.05.20\r
+//  新規リリース by SoftEther\r
+// -----------------------------------------------\r
+\r
+// Memory.c\r
+// メモリ管理プログラム\r
+\r
+#include <stdio.h>\r
+#include <stdlib.h>\r
+#include <string.h>\r
+#include <wchar.h>\r
+#include <stdarg.h>\r
+#include <time.h>\r
+#include <errno.h>\r
+#include <zlib/zlib.h>\r
+#include <Mayaqua/Mayaqua.h>\r
+\r
+#define        MEMORY_SLEEP_TIME               150\r
+#define        MEMORY_MAX_RETRY                30\r
+#define        INIT_BUF_SIZE                   10240\r
+\r
+#define        FIFO_INIT_MEM_SIZE              4096\r
+#define        FIFO_REALLOC_MEM_SIZE   (65536 * 10)    // 絶妙な値\r
+#define FIFO_REALLOC_MEM_SIZE_SMALL    65536\r
+\r
+#define        INIT_NUM_RESERVED               32\r
+static UINT fifo_default_realloc_mem_size = FIFO_REALLOC_MEM_SIZE;\r
+\r
+// バイナリを検索\r
+UINT SearchBin(void *data, UINT data_start, UINT data_size, void *key, UINT key_size)\r
+{\r
+       UINT i;\r
+       // 引数チェック\r
+       if (data == NULL || key == NULL || key_size == 0 || data_size == 0 ||\r
+               (data_start >= data_size) || (data_start + key_size > data_size))\r
+       {\r
+               return INFINITE;\r
+       }\r
+\r
+       for (i = data_start;i < (data_size - key_size + 1);i++)\r
+       {\r
+               UCHAR *p = ((UCHAR *)data) + i;\r
+\r
+               if (Cmp(p, key, key_size) == 0)\r
+               {\r
+                       return i;\r
+               }\r
+       }\r
+\r
+       return INFINITE;\r
+}\r
+\r
+// すぐにクラッシュする\r
+void CrashNow()\r
+{\r
+       // これでとりあえずどのような OS 上でもプロセスは落ちるはずである\r
+       while (true)\r
+       {\r
+               UINT r = Rand32();\r
+               UCHAR *c = (UCHAR *)r;\r
+\r
+               *c = Rand8();\r
+       }\r
+}\r
+\r
+// バッファを候補に変換\r
+LIST *BufToCandidate(BUF *b)\r
+{\r
+       LIST *o;\r
+       UINT i;\r
+       UINT num;\r
+       // 引数チェック\r
+       if (b == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       num = ReadBufInt(b);\r
+       o = NewCandidateList();\r
+\r
+       for (i = 0;i < num;i++)\r
+       {\r
+               CANDIDATE *c;\r
+               wchar_t *s;\r
+               UINT64 sec64;\r
+               UINT len, size;\r
+               sec64 = ReadBufInt64(b);\r
+               len = ReadBufInt(b);\r
+               if (len >= 65536)\r
+               {\r
+                       break;\r
+               }\r
+               size = (len + 1) * 2;\r
+               s = ZeroMalloc(size);\r
+               if (ReadBuf(b, s, size) != size)\r
+               {\r
+                       Free(s);\r
+                       break;\r
+               }\r
+               else\r
+               {\r
+                       c = ZeroMalloc(sizeof(CANDIDATE));\r
+                       c->LastSelectedTime = sec64;\r
+                       c->Str = s;\r
+                       Add(o, c);\r
+               }\r
+       }\r
+\r
+       Sort(o);\r
+       return o;\r
+}\r
+\r
+// 候補をバッファに変換\r
+BUF *CandidateToBuf(LIST *o)\r
+{\r
+       BUF *b;\r
+       UINT i;\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       b = NewBuf();\r
+       WriteBufInt(b, LIST_NUM(o));\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               CANDIDATE *c = LIST_DATA(o, i);\r
+               WriteBufInt64(b, c->LastSelectedTime);\r
+               WriteBufInt(b, UniStrLen(c->Str));\r
+               WriteBuf(b, c->Str, UniStrSize(c->Str));\r
+       }\r
+\r
+       SeekBuf(b, 0, 0);\r
+\r
+       return b;\r
+}\r
+\r
+// 候補の追加\r
+void AddCandidate(LIST *o, wchar_t *str, UINT num_max)\r
+{\r
+       UINT i;\r
+       bool exists;\r
+       // 引数チェック\r
+       if (o == NULL || str == NULL)\r
+       {\r
+               return;\r
+       }\r
+       if (num_max == 0)\r
+       {\r
+               num_max = 0x7fffffff;\r
+       }\r
+\r
+       // 文字列コピー\r
+       str = UniCopyStr(str);\r
+       UniTrim(str);\r
+\r
+       exists = false;\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               CANDIDATE *c = LIST_DATA(o, i);\r
+               if (UniStrCmpi(c->Str, str) == 0)\r
+               {\r
+                       // 既存のものを発見したので時刻を更新する\r
+                       c->LastSelectedTime = SystemTime64();\r
+                       exists = true;\r
+                       break;\r
+               }\r
+       }\r
+\r
+       if (exists == false)\r
+       {\r
+               // 新しく挿入する\r
+               CANDIDATE *c = ZeroMalloc(sizeof(CANDIDATE));\r
+               c->LastSelectedTime = SystemTime64();\r
+               c->Str = UniCopyStr(str);\r
+               Insert(o, c);\r
+       }\r
+\r
+       // 文字列解放\r
+       Free(str);\r
+\r
+       // 現在の候補数を調べて、num_max より多ければ\r
+       // 古いものから順に削除する\r
+       if (LIST_NUM(o) > num_max)\r
+       {\r
+               while (LIST_NUM(o) > num_max)\r
+               {\r
+                       UINT index = LIST_NUM(o) - 1;\r
+                       CANDIDATE *c = LIST_DATA(o, index);\r
+                       Delete(o, c);\r
+                       Free(c->Str);\r
+                       Free(c);\r
+               }\r
+       }\r
+}\r
+\r
+// 候補の比較\r
+int ComapreCandidate(void *p1, void *p2)\r
+{\r
+       CANDIDATE *c1, *c2;\r
+       if (p1 == NULL || p2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+       c1 = *(CANDIDATE **)p1;\r
+       c2 = *(CANDIDATE **)p2;\r
+       if (c1 == NULL || c2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+       if (c1->LastSelectedTime > c2->LastSelectedTime)\r
+       {\r
+               return -1;\r
+       }\r
+       else if (c1->LastSelectedTime < c2->LastSelectedTime)\r
+       {\r
+               return 1;\r
+       }\r
+       else\r
+       {\r
+               return UniStrCmpi(c1->Str, c2->Str);\r
+       }\r
+}\r
+\r
+// 候補リストの解放\r
+void FreeCandidateList(LIST *o)\r
+{\r
+       UINT i;\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               CANDIDATE *c = LIST_DATA(o, i);\r
+               Free(c->Str);\r
+               Free(c);\r
+       }\r
+\r
+       ReleaseList(o);\r
+}\r
+\r
+// 新しい候補リストの作成\r
+LIST *NewCandidateList()\r
+{\r
+       return NewList(ComapreCandidate);\r
+}\r
+\r
+// 指定したアドレスがすべてゼロかどうか調べる\r
+bool IsZero(void *data, UINT size)\r
+{\r
+       UINT i;\r
+       UCHAR *c = (UCHAR *)data;\r
+       // 引数チェック\r
+       if (data == NULL || size == 0)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       for (i = 0;i < size;i++)\r
+       {\r
+               if (c[i] != 0)\r
+               {\r
+                       return false;\r
+               }\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// データを展開する\r
+UINT Uncompress(void *dst, UINT dst_size, void *src, UINT src_size)\r
+{\r
+       unsigned long dst_size_long = dst_size;\r
+       // 引数チェック\r
+       if (dst == NULL || dst_size_long == 0 || src == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       if (uncompress(dst, &dst_size_long, src, src_size) != Z_OK)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       return (UINT)dst_size_long;\r
+}\r
+\r
+// データを圧縮する\r
+UINT Compress(void *dst, UINT dst_size, void *src, UINT src_size)\r
+{\r
+       return CompressEx(dst, dst_size, src, src_size, Z_DEFAULT_COMPRESSION);\r
+}\r
+\r
+// データをオプション付きで圧縮する\r
+UINT CompressEx(void *dst, UINT dst_size, void *src, UINT src_size, UINT level)\r
+{\r
+       unsigned long dst_size_long = dst_size;\r
+       // 引数チェック\r
+       if (dst == NULL || dst_size_long == 0 || src == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       if (compress2(dst, &dst_size_long, src, src_size, (int)level) != Z_OK)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       return dst_size_long;\r
+}\r
+\r
+// src_size データを圧縮した場合の最大サイズを取得する\r
+UINT CalcCompress(UINT src_size)\r
+{\r
+       // あっ これは いい加減!\r
+       return src_size * 2 + 100;\r
+}\r
+\r
+// スタックの作成\r
+SK *NewSk()\r
+{\r
+       return NewSkEx(false);\r
+}\r
+SK *NewSkEx(bool no_compact)\r
+{\r
+       SK *s;\r
+\r
+       s = Malloc(sizeof(SK));\r
+       s->lock = NewLock();\r
+       s->ref = NewRef();\r
+       s->num_item = 0;\r
+       s->num_reserved = INIT_NUM_RESERVED;\r
+       s->p = Malloc(sizeof(void *) * s->num_reserved);\r
+       s->no_compact = no_compact;\r
+\r
+#ifndef        DONT_USE_KERNEL_STATUS\r
+       TrackNewObj(POINTER_TO_UINT64(s), "SK", 0);\r
+#endif // DONT_USE_KERNEL_STATUS\r
+\r
+       // KS\r
+       KS_INC(KS_NEWSK_COUNT);\r
+\r
+       return s;\r
+}\r
+\r
+// スタックの解放\r
+void ReleaseSk(SK *s)\r
+{\r
+       // 引数チェック\r
+       if (s == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (Release(s->ref) == 0)\r
+       {\r
+               CleanupSk(s);\r
+       }\r
+}\r
+\r
+// スタックのクリーンアップ\r
+void CleanupSk(SK *s)\r
+{\r
+       // 引数チェック\r
+       if (s == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // メモリ解放\r
+       Free(s->p);\r
+       DeleteLock(s->lock);\r
+       Free(s);\r
+\r
+#ifndef        DONT_USE_KERNEL_STATUS\r
+       TrackDeleteObj(POINTER_TO_UINT64(s));\r
+#endif // DONT_USE_KERNEL_STATUS\r
+\r
+       // KS\r
+       KS_INC(KS_FREESK_COUNT);\r
+}\r
+\r
+// スタックのロック\r
+void LockSk(SK *s)\r
+{\r
+       // 引数チェック\r
+       if (s == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Lock(s->lock);\r
+}\r
+\r
+// スタックのロック解除\r
+void UnlockSk(SK *s)\r
+{\r
+       // 引数チェック\r
+       if (s == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Unlock(s->lock);\r
+}\r
+\r
+// スタックの Push\r
+void Push(SK *s, void *p)\r
+{\r
+       UINT i;\r
+       // 引数チェック\r
+       if (s == NULL || p == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       i = s->num_item;\r
+       s->num_item++;\r
+\r
+       // サイズ拡大\r
+       if (s->num_item > s->num_reserved)\r
+       {\r
+               s->num_reserved = s->num_reserved * 2;\r
+               s->p = ReAlloc(s->p, sizeof(void *) * s->num_reserved);\r
+       }\r
+       s->p[i] = p;\r
+\r
+       // KS\r
+       KS_INC(KS_PUSH_COUNT);\r
+}\r
+\r
+// スタックの Pop\r
+void *Pop(SK *s)\r
+{\r
+       void *ret;\r
+       // 引数チェック\r
+       if (s == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+       if (s->num_item == 0)\r
+       {\r
+               return NULL;\r
+       }\r
+       ret = s->p[s->num_item - 1];\r
+       s->num_item--;\r
+\r
+       // サイズ縮小\r
+       if (s->no_compact == false)\r
+       {\r
+               // no_compact が true の場合は縮小しない\r
+               if ((s->num_item * 2) <= s->num_reserved)\r
+               {\r
+                       if (s->num_reserved >= (INIT_NUM_RESERVED * 2))\r
+                       {\r
+                               s->num_reserved = s->num_reserved / 2;\r
+                               s->p = ReAlloc(s->p, sizeof(void *) * s->num_reserved);\r
+                       }\r
+               }\r
+       }\r
+\r
+       // KS\r
+       KS_INC(KS_POP_COUNT);\r
+\r
+       return ret;\r
+}\r
+\r
+// 1 つ取得\r
+void *GetNext(QUEUE *q)\r
+{\r
+       void *p = NULL;\r
+       // 引数チェック\r
+       if (q == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       if (q->num_item == 0)\r
+       {\r
+               // アイテム無し\r
+               return NULL;\r
+       }\r
+\r
+       // FIFO から読み込む\r
+       ReadFifo(q->fifo, &p, sizeof(void *));\r
+       q->num_item--;\r
+\r
+       // KS\r
+       KS_INC(KS_GETNEXT_COUNT);\r
+\r
+       return p;\r
+}\r
+\r
+// キューに Int 型を挿入\r
+void InsertQueueInt(QUEUE *q, UINT value)\r
+{\r
+       UINT *p;\r
+       // 引数チェック\r
+       if (q == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       p = Clone(&value, sizeof(UINT));\r
+\r
+       InsertQueue(q, p);\r
+}\r
+\r
+// キューに挿入\r
+void InsertQueue(QUEUE *q, void *p)\r
+{\r
+       // 引数チェック\r
+       if (q == NULL || p == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // FIFO に書き込む\r
+       WriteFifo(q->fifo, &p, sizeof(void *));\r
+\r
+       q->num_item++;\r
+\r
+       // KS\r
+       KS_INC(KS_INSERT_QUEUE_COUNT);\r
+}\r
+\r
+// キューのロック\r
+void LockQueue(QUEUE *q)\r
+{\r
+       // 引数チェック\r
+       if (q == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Lock(q->lock);\r
+}\r
+\r
+// キューのロック解除\r
+void UnlockQueue(QUEUE *q)\r
+{\r
+       // 引数チェック\r
+       if (q == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Unlock(q->lock);\r
+}\r
+\r
+// キューの解放\r
+void ReleaseQueue(QUEUE *q)\r
+{\r
+       // 引数チェック\r
+       if (q == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (q->ref == NULL || Release(q->ref) == 0)\r
+       {\r
+               CleanupQueue(q);\r
+       }\r
+}\r
+\r
+// キューのクリーンアップ\r
+void CleanupQueue(QUEUE *q)\r
+{\r
+       // 引数チェック\r
+       if (q == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // メモリ解放\r
+       ReleaseFifo(q->fifo);\r
+       DeleteLock(q->lock);\r
+       Free(q);\r
+\r
+#ifndef        DONT_USE_KERNEL_STATUS\r
+       TrackDeleteObj(POINTER_TO_UINT64(q));\r
+#endif // DONT_USE_KERNEL_STATUS\r
+\r
+       // KS\r
+       KS_INC(KS_FREEQUEUE_COUNT);\r
+}\r
+\r
+// キューの作成\r
+QUEUE *NewQueue()\r
+{\r
+       QUEUE *q;\r
+\r
+       q = ZeroMalloc(sizeof(QUEUE));\r
+       q->lock = NewLock();\r
+       q->ref = NewRef();\r
+       q->num_item = 0;\r
+       q->fifo = NewFifo();\r
+\r
+#ifndef        DONT_USE_KERNEL_STATUS\r
+       TrackNewObj(POINTER_TO_UINT64(q), "QUEUE", 0);\r
+#endif // DONT_USE_KERNEL_STATUS\r
+\r
+       // KS\r
+       KS_INC(KS_NEWQUEUE_COUNT);\r
+\r
+       return q;\r
+}\r
+QUEUE *NewQueueFast()\r
+{\r
+       QUEUE *q;\r
+\r
+       q = ZeroMalloc(sizeof(QUEUE));\r
+       q->lock = NULL;\r
+       q->ref = NULL;\r
+       q->num_item = 0;\r
+       q->fifo = NewFifoFast();\r
+\r
+#ifndef        DONT_USE_KERNEL_STATUS\r
+       TrackNewObj(POINTER_TO_UINT64(q), "QUEUE", 0);\r
+#endif // DONT_USE_KERNEL_STATUS\r
+\r
+       // KS\r
+       KS_INC(KS_NEWQUEUE_COUNT);\r
+\r
+       return q;\r
+}\r
+\r
+// リストに比較関数をセットする\r
+void SetCmp(LIST *o, COMPARE *cmp)\r
+{\r
+       // 引数チェック\r
+       if (o == NULL || cmp == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (o->cmp != cmp)\r
+       {\r
+               o->cmp = cmp;\r
+               o->sorted = false;\r
+       }\r
+}\r
+\r
+// リストのクローン\r
+LIST *CloneList(LIST *o)\r
+{\r
+       LIST *n = NewList(o->cmp);\r
+\r
+       // メモリ再確保\r
+       Free(n->p);\r
+       n->p = ToArray(o);\r
+       n->num_item = n->num_reserved = LIST_NUM(o);\r
+       n->sorted = o->sorted;\r
+\r
+       return n;\r
+}\r
+\r
+// リストを配列にコピー\r
+void CopyToArray(LIST *o, void *p)\r
+{\r
+       // 引数チェック\r
+       if (o == NULL || p == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // KS\r
+       KS_INC(KS_TOARRAY_COUNT);\r
+\r
+       Copy(p, o->p, sizeof(void *) * o->num_item);\r
+}\r
+\r
+// リストを配列化する\r
+void *ToArray(LIST *o)\r
+{\r
+       return ToArrayEx(o, false);\r
+}\r
+void *ToArrayEx(LIST *o, bool fast)\r
+{\r
+       void *p;\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // メモリ確保\r
+       if (fast == false)\r
+       {\r
+               p = Malloc(sizeof(void *) * LIST_NUM(o));\r
+       }\r
+       else\r
+       {\r
+               p = MallocFast(sizeof(void *) * LIST_NUM(o));\r
+       }\r
+       // コピー\r
+       CopyToArray(o, p);\r
+\r
+       return p;\r
+}\r
+\r
+// リストのサーチ\r
+void *Search(LIST *o, void *target)\r
+{\r
+       void **ret;\r
+       // 引数チェック\r
+       if (o == NULL || target == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+       if (o->cmp == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // ソートのチェック\r
+       if (o->sorted == false)\r
+       {\r
+               // 未ソートなのでソートを行う\r
+               Sort(o);\r
+       }\r
+\r
+       // なんだ C ライブラリのバイナリサーチ関数を呼んでいるだけか\r
+       ret = (void **)bsearch(&target, o->p, o->num_item, sizeof(void *),\r
+               (int(*)(const void *, const void *))o->cmp);\r
+\r
+       // KS\r
+       KS_INC(KS_SEARCH_COUNT);\r
+\r
+       if (ret != NULL)\r
+       {\r
+               return *ret;\r
+       }\r
+       else\r
+       {\r
+               return NULL;\r
+       }\r
+}\r
+\r
+// リストに項目を挿入\r
+// 本当はもうちょっとましなデータ構造 & アルゴリズムにするべき\r
+void Insert(LIST *o, void *p)\r
+{\r
+       int low, high, middle;\r
+       UINT pos;\r
+       int i;\r
+       // 引数チェック\r
+       if (o == NULL || p == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (o->cmp == NULL)\r
+       {\r
+               // ソート関数が無い場合は単純に追加する\r
+               Add(o, p);\r
+               return;\r
+       }\r
+\r
+       // ソートされていない場合は直ちにソートする\r
+       if (o->sorted == false)\r
+       {\r
+               Sort(o);\r
+       }\r
+\r
+       low = 0;\r
+       high = LIST_NUM(o) - 1;\r
+\r
+       pos = INFINITE;\r
+\r
+       while (low <= high)\r
+       {\r
+               int ret;\r
+\r
+               middle = (low + high) / 2;\r
+               ret = o->cmp(&(o->p[middle]), &p);\r
+\r
+               if (ret == 0)\r
+               {\r
+                       pos = middle;\r
+                       break;\r
+               }\r
+               else if (ret > 0)\r
+               {\r
+                       high = middle - 1;\r
+               }\r
+               else\r
+               {\r
+                       low = middle + 1;\r
+               }\r
+       }\r
+\r
+       if (pos == INFINITE)\r
+       {\r
+               pos = low;\r
+       }\r
+\r
+       o->num_item++;\r
+       if (o->num_item > o->num_reserved)\r
+       {\r
+               o->num_reserved *= 2;\r
+               o->p = ReAlloc(o->p, sizeof(void *) * o->num_reserved);\r
+       }\r
+\r
+       if (LIST_NUM(o) >= 2)\r
+       {\r
+               for (i = (LIST_NUM(o) - 2);i >= (int)pos;i--)\r
+               {\r
+                       o->p[i + 1] = o->p[i];\r
+               }\r
+       }\r
+\r
+       o->p[pos] = p;\r
+\r
+       // KS\r
+       KS_INC(KS_INSERT_COUNT);\r
+}\r
+\r
+// ソートフラグの設定\r
+void SetSortFlag(LIST *o, bool sorted)\r
+{\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       o->sorted = sorted;\r
+}\r
+\r
+// リストのソート\r
+void Sort(LIST *o)\r
+{\r
+       // 引数チェック\r
+       if (o == NULL || o->cmp == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       qsort(o->p, o->num_item, sizeof(void *), (int(*)(const void *, const void *))o->cmp);\r
+       o->sorted = true;\r
+\r
+       // KS\r
+       KS_INC(KS_SORT_COUNT);\r
+}\r
+\r
+// ある文字列項目がリスト内に存在しているかどうか調べる (Unicode 版)\r
+bool IsInListUniStr(LIST *o, wchar_t *str)\r
+{\r
+       UINT i;\r
+       // 引数チェック\r
+       if (o == NULL || str == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               wchar_t *s = LIST_DATA(o, i);\r
+\r
+               if (UniStrCmpi(s, str) == 0)\r
+               {\r
+                       return true;\r
+               }\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// リスト内のポインタを置換する\r
+bool ReplaceListPointer(LIST *o, void *oldptr, void *newptr)\r
+{\r
+       UINT i;\r
+       // 引数チェック\r
+       if (o == NULL || oldptr == NULL || newptr == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               void *p = LIST_DATA(o, i);\r
+\r
+               if (p == oldptr)\r
+               {\r
+                       o->p[i] = newptr;\r
+                       return true;\r
+               }\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// ある文字列項目がリスト内に存在しているかどうか調べる\r
+bool IsInListStr(LIST *o, char *str)\r
+{\r
+       UINT i;\r
+       // 引数チェック\r
+       if (o == NULL || str == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               char *s = LIST_DATA(o, i);\r
+\r
+               if (StrCmpi(s, str) == 0)\r
+               {\r
+                       return true;\r
+               }\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// リスト内を UINT 形式のポインタで走査してポインタを取得する\r
+void *ListKeyToPointer(LIST *o, UINT key)\r
+{\r
+       UINT i;\r
+       // 引数チェック\r
+       if (o == NULL || key == 0)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               void *p = LIST_DATA(o, i);\r
+\r
+               if (POINTER_TO_KEY(p) == key)\r
+               {\r
+                       return p;\r
+               }\r
+       }\r
+\r
+       return NULL;\r
+}\r
+\r
+// あるキーがリスト内に存在するかどうか調べる\r
+bool IsInListKey(LIST *o, UINT key)\r
+{\r
+       void *p;\r
+       // 引数チェック\r
+       if (o == NULL || key == 0)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       p = ListKeyToPointer(o, key);\r
+       if (p == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// ある項目がリスト内に存在するかどうか調べる\r
+bool IsInList(LIST *o, void *p)\r
+{\r
+       UINT i;\r
+       // 引数チェック\r
+       if (o == NULL || p == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               void *q = LIST_DATA(o, i);\r
+               if (p == q)\r
+               {\r
+                       return true;\r
+               }\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// リストへの要素の追加\r
+void Add(LIST *o, void *p)\r
+{\r
+       UINT i;\r
+       // 引数チェック\r
+       if (o == NULL || p == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       i = o->num_item;\r
+       o->num_item++;\r
+\r
+       if (o->num_item > o->num_reserved)\r
+       {\r
+               o->num_reserved = o->num_reserved * 2;\r
+               o->p = ReAlloc(o->p, sizeof(void *) * o->num_reserved);\r
+       }\r
+\r
+       o->p[i] = p;\r
+       o->sorted = false;\r
+\r
+       // KS\r
+       KS_INC(KS_INSERT_COUNT);\r
+}\r
+\r
+// リストからキーで指定した要素の削除\r
+bool DeleteKey(LIST *o, UINT key)\r
+{\r
+       void *p;\r
+       // 引数チェック\r
+       if (o == NULL || key == 0)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       p = ListKeyToPointer(o, key);\r
+       if (p == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return Delete(o, p);\r
+}\r
+\r
+// リストから要素の削除\r
+bool Delete(LIST *o, void *p)\r
+{\r
+       UINT i, n;\r
+       // 引数チェック\r
+       if (o == NULL || p == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       for (i = 0;i < o->num_item;i++)\r
+       {\r
+               if (o->p[i] == p)\r
+               {\r
+                       break;\r
+               }\r
+       }\r
+       if (i == o->num_item)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       n = i;\r
+       for (i = n;i < (o->num_item - 1);i++)\r
+       {\r
+               o->p[i] = o->p[i + 1];\r
+       }\r
+       o->num_item--;\r
+       if ((o->num_item * 2) <= o->num_reserved)\r
+       {\r
+               if (o->num_reserved > (INIT_NUM_RESERVED * 2))\r
+               {\r
+                       o->num_reserved = o->num_reserved / 2;\r
+                       o->p = ReAlloc(o->p, sizeof(void *) * o->num_reserved);\r
+               }\r
+       }\r
+\r
+       // KS\r
+       KS_INC(KS_DELETE_COUNT);\r
+\r
+       return true;\r
+}\r
+\r
+// リストからすべての要素の削除\r
+void DeleteAll(LIST *o)\r
+{\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       o->num_item = 0;\r
+       o->num_reserved = INIT_NUM_RESERVED;\r
+       o->p = ReAlloc(o->p, sizeof(void *) * INIT_NUM_RESERVED);\r
+}\r
+\r
+// リストのロック\r
+void LockList(LIST *o)\r
+{\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Lock(o->lock);\r
+}\r
+\r
+// リストのロック解除\r
+void UnlockList(LIST *o)\r
+{\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Unlock(o->lock);\r
+}\r
+\r
+// リストの解放\r
+void ReleaseList(LIST *o)\r
+{\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (o->ref == NULL || Release(o->ref) == 0)\r
+       {\r
+               CleanupList(o);\r
+       }\r
+}\r
+\r
+// リストのクリーンアップ\r
+void CleanupList(LIST *o)\r
+{\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Free(o->p);\r
+       if (o->lock != NULL)\r
+       {\r
+               DeleteLock(o->lock);\r
+       }\r
+       Free(o);\r
+\r
+       // KS\r
+       KS_INC(KS_FREELIST_COUNT);\r
+\r
+#ifndef        DONT_USE_KERNEL_STATUS\r
+       TrackDeleteObj(POINTER_TO_UINT64(o));\r
+#endif // DONT_USE_KERNEL_STATUS\r
+}\r
+\r
+// 文字列比較関数 (Unicode)\r
+int CompareUniStr(void *p1, void *p2)\r
+{\r
+       wchar_t *s1, *s2;\r
+       if (p1 == NULL || p2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+       s1 = *(wchar_t **)p1;\r
+       s2 = *(wchar_t **)p2;\r
+\r
+       return UniStrCmp(s1, s2);\r
+}\r
+\r
+// 文字列をリストに挿入する\r
+bool InsertStr(LIST *o, char *str)\r
+{\r
+       // 引数チェック\r
+       if (o == NULL || str == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (Search(o, str) == NULL)\r
+       {\r
+               Insert(o, str);\r
+\r
+               return true;\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// 文字列比較関数\r
+int CompareStr(void *p1, void *p2)\r
+{\r
+       char *s1, *s2;\r
+       if (p1 == NULL || p2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+       s1 = *(char **)p1;\r
+       s2 = *(char **)p2;\r
+\r
+       return StrCmpi(s1, s2);\r
+}\r
+\r
+// 高速リスト (ロック無し) の作成\r
+LIST *NewListFast(COMPARE *cmp)\r
+{\r
+       return NewListEx(cmp, true);\r
+}\r
+\r
+// リストの作成\r
+LIST *NewList(COMPARE *cmp)\r
+{\r
+       return NewListEx(cmp, false);\r
+}\r
+LIST *NewListEx(COMPARE *cmp, bool fast)\r
+{\r
+       return NewListEx2(cmp, fast, false);\r
+}\r
+LIST *NewListEx2(COMPARE *cmp, bool fast, bool fast_malloc)\r
+{\r
+       LIST *o;\r
+\r
+       if (fast_malloc == false)\r
+       {\r
+               o = Malloc(sizeof(LIST));\r
+       }\r
+       else\r
+       {\r
+               o = MallocFast(sizeof(LIST));\r
+       }\r
+\r
+       if (fast == false)\r
+       {\r
+               o->lock = NewLock();\r
+               o->ref = NewRef();\r
+       }\r
+       else\r
+       {\r
+               o->lock = NULL;\r
+               o->ref = NULL;\r
+       }\r
+       o->num_item = 0;\r
+       o->num_reserved = INIT_NUM_RESERVED;\r
+\r
+       if (fast_malloc == false)\r
+       {\r
+               o->p = Malloc(sizeof(void *) * o->num_reserved);\r
+       }\r
+       else\r
+       {\r
+               o->p = MallocFast(sizeof(void *) * o->num_reserved);\r
+       }\r
+\r
+       o->cmp = cmp;\r
+       o->sorted = true;\r
+\r
+#ifndef        DONT_USE_KERNEL_STATUS\r
+       TrackNewObj(POINTER_TO_UINT64(o), "LIST", 0);\r
+#endif //DONT_USE_KERNEL_STATUS\r
+\r
+       // KS\r
+       KS_INC(KS_NEWLIST_COUNT);\r
+\r
+       return o;\r
+}\r
+\r
+// FIFO から peek する\r
+UINT PeekFifo(FIFO *f, void *p, UINT size)\r
+{\r
+       UINT read_size;\r
+       if (f == NULL || size == 0)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       // KS\r
+       KS_INC(KS_PEEK_FIFO_COUNT);\r
+\r
+       read_size = MIN(size, f->size);\r
+       if (read_size == 0)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       if (p != NULL)\r
+       {\r
+               Copy(p, (UCHAR *)f->p + f->pos, read_size);\r
+       }\r
+\r
+       return read_size;\r
+}\r
+\r
+// FIFO から読み取る\r
+UINT ReadFifo(FIFO *f, void *p, UINT size)\r
+{\r
+       UINT read_size;\r
+       // 引数チェック\r
+       if (f == NULL || size == 0)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       read_size = MIN(size, f->size);\r
+       if (read_size == 0)\r
+       {\r
+               return 0;\r
+       }\r
+       if (p != NULL)\r
+       {\r
+               Copy(p, (UCHAR *)f->p + f->pos, read_size);\r
+       }\r
+       f->pos += read_size;\r
+       f->size -= read_size;\r
+\r
+       if (f->size == 0)\r
+       {\r
+               f->pos = 0;\r
+       }\r
+\r
+       // メモリの詰め直し\r
+       if (f->pos >= FIFO_INIT_MEM_SIZE &&\r
+               f->memsize >= f->realloc_mem_size &&\r
+               (f->memsize / 2) > f->size)\r
+       {\r
+               void *new_p;\r
+               UINT new_size;\r
+\r
+               new_size = MAX(f->memsize / 2, FIFO_INIT_MEM_SIZE);\r
+               new_p = Malloc(new_size);\r
+               Copy(new_p, (UCHAR *)f->p + f->pos, f->size);\r
+\r
+               Free(f->p);\r
+\r
+               f->memsize = new_size;\r
+               f->p = new_p;\r
+               f->pos = 0;\r
+       }\r
+\r
+       // KS\r
+       KS_INC(KS_READ_FIFO_COUNT);\r
+\r
+       return read_size;\r
+}\r
+\r
+// FIFO に書き込む\r
+void WriteFifo(FIFO *f, void *p, UINT size)\r
+{\r
+       UINT i, need_size;\r
+       bool realloc_flag;\r
+       // 引数チェック\r
+       if (f == NULL || size == 0)\r
+       {\r
+               return;\r
+       }\r
+\r
+       i = f->size;\r
+       f->size += size;\r
+       need_size = f->pos + f->size;\r
+       realloc_flag = false;\r
+\r
+       // メモリ拡張\r
+       while (need_size > f->memsize)\r
+       {\r
+               f->memsize = MAX(f->memsize, FIFO_INIT_MEM_SIZE) * 3;\r
+               realloc_flag = true;\r
+       }\r
+\r
+       if (realloc_flag)\r
+       {\r
+               f->p = ReAlloc(f->p, f->memsize);\r
+       }\r
+\r
+       // データ書き込み\r
+       if (p != NULL)\r
+       {\r
+               Copy((UCHAR *)f->p + f->pos + i, p, size);\r
+       }\r
+\r
+       // KS\r
+       KS_INC(KS_WRITE_FIFO_COUNT);\r
+}\r
+\r
+// FIFO のクリア\r
+void ClearFifo(FIFO *f)\r
+{\r
+       // 引数チェック\r
+       if (f == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       f->size = f->pos = 0;\r
+       f->memsize = FIFO_INIT_MEM_SIZE;\r
+       f->p = ReAlloc(f->p, f->memsize);\r
+}\r
+\r
+// FIFO のサイズ取得\r
+UINT FifoSize(FIFO *f)\r
+{\r
+       // 引数チェック\r
+       if (f == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       return f->size;\r
+}\r
+\r
+// FIFO のロック\r
+void LockFifo(FIFO *f)\r
+{\r
+       // 引数チェック\r
+       if (f == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Lock(f->lock);\r
+}\r
+\r
+// FIFO のロック解除\r
+void UnlockFifo(FIFO *f)\r
+{\r
+       // 引数チェック\r
+       if (f == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Unlock(f->lock);\r
+}\r
+\r
+// FIFO の解放\r
+void ReleaseFifo(FIFO *f)\r
+{\r
+       // 引数チェック\r
+       if (f == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (f->ref == NULL || Release(f->ref) == 0)\r
+       {\r
+               CleanupFifo(f);\r
+       }\r
+}\r
+\r
+// FIFO のクリーンアップ\r
+void CleanupFifo(FIFO *f)\r
+{\r
+       // 引数チェック\r
+       if (f == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       DeleteLock(f->lock);\r
+       Free(f->p);\r
+       Free(f);\r
+\r
+#ifndef        DONT_USE_KERNEL_STATUS\r
+       TrackDeleteObj(POINTER_TO_UINT64(f));\r
+#endif //DONT_USE_KERNEL_STATUS\r
+\r
+       // KS\r
+       KS_INC(KS_FREEFIFO_COUNT);\r
+}\r
+\r
+// FIFO システムの初期化\r
+void InitFifo()\r
+{\r
+       fifo_default_realloc_mem_size = FIFO_REALLOC_MEM_SIZE;\r
+}\r
+\r
+// FIFO の作成\r
+FIFO *NewFifo()\r
+{\r
+       return NewFifoEx(0, false);\r
+}\r
+FIFO *NewFifoFast()\r
+{\r
+       return NewFifoEx(0, true);\r
+}\r
+FIFO *NewFifoEx(UINT realloc_mem_size, bool fast)\r
+{\r
+       FIFO *f;\r
+\r
+       // メモリ確保\r
+       f = Malloc(sizeof(FIFO));\r
+\r
+       if (fast == false)\r
+       {\r
+               f->lock = NewLock();\r
+               f->ref = NewRef();\r
+       }\r
+       else\r
+       {\r
+               f->lock = NULL;\r
+               f->ref = NULL;\r
+       }\r
+\r
+       f->size = f->pos = 0;\r
+       f->memsize = FIFO_INIT_MEM_SIZE;\r
+       f->p = Malloc(FIFO_INIT_MEM_SIZE);\r
+\r
+       if (realloc_mem_size == 0)\r
+       {\r
+               realloc_mem_size = fifo_default_realloc_mem_size;\r
+       }\r
+\r
+       f->realloc_mem_size = realloc_mem_size;\r
+\r
+#ifndef        DONT_USE_KERNEL_STATUS\r
+       TrackNewObj(POINTER_TO_UINT64(f), "FIFO", 0);\r
+#endif // DONT_USE_KERNEL_STATUS\r
+\r
+       // KS\r
+       KS_INC(KS_NEWFIFO_COUNT);\r
+\r
+       return f;\r
+}\r
+\r
+// FIFO のデフォルトのメモリ再確保サイズを取得する\r
+UINT GetFifoDefaultReallocMemSize()\r
+{\r
+       return fifo_default_realloc_mem_size;\r
+}\r
+\r
+// FIFO のデフォルトのメモリ再確保サイズを設定する\r
+void SetFifoDefaultReallocMemSize(UINT size)\r
+{\r
+       if (size == 0)\r
+       {\r
+               size = FIFO_REALLOC_MEM_SIZE;\r
+       }\r
+\r
+       fifo_default_realloc_mem_size = size;\r
+}\r
+\r
+// バッファをファイルから読み込む\r
+BUF *FileToBuf(IO *o)\r
+{\r
+       UCHAR hash1[MD5_SIZE], hash2[MD5_SIZE];\r
+       UINT size;\r
+       void *buf;\r
+       BUF *b;\r
+\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // サイズを読み込む\r
+       if (FileRead(o, &size, sizeof(size)) == false)\r
+       {\r
+               return NULL;\r
+       }\r
+       size = Endian32(size);\r
+\r
+       if (size > FileSize(o))\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // ハッシュを読み込む\r
+       if (FileRead(o, hash1, sizeof(hash1)) == false)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // バッファを読み込む\r
+       buf = Malloc(size);\r
+       if (FileRead(o, buf, size) == false)\r
+       {\r
+               Free(buf);\r
+               return NULL;\r
+       }\r
+\r
+       // ハッシュをとる\r
+       Hash(hash2, buf, size, false);\r
+\r
+       // ハッシュを比較する\r
+       if (Cmp(hash1, hash2, sizeof(hash1)) != 0)\r
+       {\r
+               // ハッシュが異なる\r
+               Free(buf);\r
+               return NULL;\r
+       }\r
+\r
+       // バッファを作成する\r
+       b = NewBuf();\r
+       WriteBuf(b, buf, size);\r
+       Free(buf);\r
+       b->Current = 0;\r
+\r
+       return b;\r
+}\r
+\r
+// ダンプファイルをバッファに読み込む\r
+BUF *ReadDump(char *filename)\r
+{\r
+       IO *o;\r
+       BUF *b;\r
+       UINT size;\r
+       void *data;\r
+       // 引数チェック\r
+       if (filename == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       o = FileOpen(filename, false);\r
+       if (o == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       size = FileSize(o);\r
+       data = Malloc(size);\r
+       FileRead(o, data, size);\r
+       FileClose(o);\r
+\r
+       b = NewBuf();\r
+       WriteBuf(b, data, size);\r
+       b->Current = 0;\r
+       Free(data);\r
+\r
+       return b;\r
+}\r
+BUF *ReadDumpW(wchar_t *filename)\r
+{\r
+       IO *o;\r
+       BUF *b;\r
+       UINT size;\r
+       void *data;\r
+       // 引数チェック\r
+       if (filename == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       o = FileOpenW(filename, false);\r
+       if (o == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       size = FileSize(o);\r
+       data = Malloc(size);\r
+       FileRead(o, data, size);\r
+       FileClose(o);\r
+\r
+       b = NewBuf();\r
+       WriteBuf(b, data, size);\r
+       b->Current = 0;\r
+       Free(data);\r
+\r
+       return b;\r
+}\r
+\r
+// バッファ内容をファイルにダンプする\r
+bool DumpBuf(BUF *b, char *filename)\r
+{\r
+       IO *o;\r
+       // 引数チェック\r
+       if (b == NULL || filename == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       o = FileCreate(filename);\r
+       if (o == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       FileWrite(o, b->Buf, b->Size);\r
+       FileClose(o);\r
+\r
+       return true;\r
+}\r
+bool DumpBufW(BUF *b, wchar_t *filename)\r
+{\r
+       IO *o;\r
+       // 引数チェック\r
+       if (b == NULL || filename == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       o = FileCreateW(filename);\r
+       if (o == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       FileWrite(o, b->Buf, b->Size);\r
+       FileClose(o);\r
+\r
+       return true;\r
+}\r
+\r
+// バッファをファイルに書き込む\r
+bool BufToFile(IO *o, BUF *b)\r
+{\r
+       UCHAR hash[MD5_SIZE];\r
+       UINT size;\r
+\r
+       // 引数チェック\r
+       if (o == NULL || b == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // データをハッシュする\r
+       Hash(hash, b->Buf, b->Size, false);\r
+\r
+       size = Endian32(b->Size);\r
+\r
+       // サイズを書き込む\r
+       if (FileWrite(o, &size, sizeof(size)) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // ハッシュを書き込む\r
+       if (FileWrite(o, hash, sizeof(hash)) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // データを書き込む\r
+       if (FileWrite(o, b->Buf, b->Size) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// バッファの作成\r
+BUF *NewBuf()\r
+{\r
+       BUF *b;\r
+\r
+       // メモリ確保\r
+       b = Malloc(sizeof(BUF));\r
+       b->Buf = Malloc(INIT_BUF_SIZE);\r
+       b->Size = 0;\r
+       b->Current = 0;\r
+       b->SizeReserved = INIT_BUF_SIZE;\r
+\r
+#ifndef        DONT_USE_KERNEL_STATUS\r
+       TrackNewObj(POINTER_TO_UINT64(b), "BUF", 0);\r
+#endif // DONT_USE_KERNEL_STATUS\r
+\r
+       // KS\r
+       KS_INC(KS_NEWBUF_COUNT);\r
+       KS_INC(KS_CURRENT_BUF_COUNT);\r
+\r
+       return b;\r
+}\r
+\r
+// バッファのクリア\r
+void ClearBuf(BUF *b)\r
+{\r
+       // 引数チェック\r
+       if (b == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       b->Size = 0;\r
+       b->Current = 0;\r
+}\r
+\r
+// バッファへ書き込み\r
+void WriteBuf(BUF *b, void *buf, UINT size)\r
+{\r
+       UINT new_size;\r
+       // 引数チェック\r
+       if (b == NULL || buf == NULL || size == 0)\r
+       {\r
+               return;\r
+       }\r
+\r
+       new_size = b->Current + size;\r
+       if (new_size > b->Size)\r
+       {\r
+               // サイズを調整する\r
+               AdjustBufSize(b, new_size);\r
+       }\r
+       if (b->Buf != NULL)\r
+       {\r
+               Copy((UCHAR *)b->Buf + b->Current, buf, size);\r
+       }\r
+       b->Current += size;\r
+       b->Size = new_size;\r
+\r
+       // KS\r
+       KS_INC(KS_WRITE_BUF_COUNT);\r
+}\r
+\r
+// バッファに文字列を追記\r
+void AddBufStr(BUF *b, char *str)\r
+{\r
+       // 引数チェック\r
+       if (b == NULL || str == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       WriteBuf(b, str, StrLen(str));\r
+}\r
+\r
+// バッファに 1 行書き込む\r
+void WriteBufLine(BUF *b, char *str)\r
+{\r
+       char *crlf = "\r\n";\r
+       // 引数チェック\r
+       if (b == NULL || str == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       WriteBuf(b, str, StrLen(str));\r
+       WriteBuf(b, crlf, StrLen(crlf));\r
+}\r
+\r
+// バッファに文字列を書き込む\r
+bool WriteBufStr(BUF *b, char *str)\r
+{\r
+       UINT len;\r
+       // 引数チェック\r
+       if (b == NULL || str == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // 文字列長\r
+       len = StrLen(str);\r
+       if (WriteBufInt(b, len + 1) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // 文字列本体\r
+       WriteBuf(b, str, len);\r
+\r
+       return true;\r
+}\r
+\r
+// バッファから文字列を読み込む\r
+bool ReadBufStr(BUF *b, char *str, UINT size)\r
+{\r
+       UINT len;\r
+       UINT read_size;\r
+       // 引数チェック\r
+       if (b == NULL || str == NULL || size == 0)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // 文字列長を読み込む\r
+       len = ReadBufInt(b);\r
+       if (len == 0)\r
+       {\r
+               return false;\r
+       }\r
+       len--;\r
+       if (len <= (size - 1))\r
+       {\r
+               size = len + 1;\r
+       }\r
+\r
+       read_size = MIN(len, (size - 1));\r
+\r
+       // 文字列本体を読み込む\r
+       if (ReadBuf(b, str, read_size) != read_size)\r
+       {\r
+               return false;\r
+       }\r
+       if (read_size < len)\r
+       {\r
+               ReadBuf(b, NULL, len - read_size);\r
+       }\r
+       str[read_size] = 0;\r
+\r
+       return true;\r
+}\r
+\r
+// バッファに 64 bit 整数を書き込む\r
+bool WriteBufInt64(BUF *b, UINT64 value)\r
+{\r
+       // 引数チェック\r
+       if (b == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       value = Endian64(value);\r
+\r
+       WriteBuf(b, &value, sizeof(UINT64));\r
+       return true;\r
+}\r
+\r
+// バッファに整数を書き込む\r
+bool WriteBufInt(BUF *b, UINT value)\r
+{\r
+       // 引数チェック\r
+       if (b == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       value = Endian32(value);\r
+\r
+       WriteBuf(b, &value, sizeof(UINT));\r
+       return true;\r
+}\r
+\r
+// バッファから 64bit 整数を読み込む\r
+UINT64 ReadBufInt64(BUF *b)\r
+{\r
+       UINT64 value;\r
+       // 引数チェック\r
+       if (b == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       if (ReadBuf(b, &value, sizeof(UINT64)) != sizeof(UINT64))\r
+       {\r
+               return 0;\r
+       }\r
+       return Endian64(value);\r
+}\r
+\r
+// バッファから整数を読み込む\r
+UINT ReadBufInt(BUF *b)\r
+{\r
+       UINT value;\r
+       // 引数チェック\r
+       if (b == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       if (ReadBuf(b, &value, sizeof(UINT)) != sizeof(UINT))\r
+       {\r
+               return 0;\r
+       }\r
+       return Endian32(value);\r
+}\r
+\r
+// バッファにバッファを書き込み\r
+void WriteBufBuf(BUF *b, BUF *bb)\r
+{\r
+       // 引数チェック\r
+       if (b == NULL || bb == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       WriteBuf(b, bb->Buf, bb->Size);\r
+}\r
+\r
+// バッファからバッファを読み込み\r
+BUF *ReadBufFromBuf(BUF *b, UINT size)\r
+{\r
+       BUF *ret;\r
+       UCHAR *data;\r
+       // 引数チェック\r
+       if (b == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       data = Malloc(size);\r
+       if (ReadBuf(b, data, size) != size)\r
+       {\r
+               Free(data);\r
+               return NULL;\r
+       }\r
+\r
+       ret = NewBuf();\r
+       WriteBuf(ret, data, size);\r
+       SeekBuf(ret, 0, 0);\r
+\r
+       Free(data);\r
+\r
+       return ret;\r
+}\r
+\r
+// バッファから読み込み\r
+UINT ReadBuf(BUF *b, void *buf, UINT size)\r
+{\r
+       UINT size_read;\r
+       // 引数チェック\r
+       if (b == NULL || size == 0)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       if (b->Buf == NULL)\r
+       {\r
+               Zero(buf, size);\r
+               return 0;\r
+       }\r
+       size_read = size;\r
+       if ((b->Current + size) >= b->Size)\r
+       {\r
+               size_read = b->Size - b->Current;\r
+               if (buf != NULL)\r
+               {\r
+                       Zero((UCHAR *)buf + size_read, size - size_read);\r
+               }\r
+       }\r
+\r
+       if (buf != NULL)\r
+       {\r
+               Copy(buf, (UCHAR *)b->Buf + b->Current, size_read);\r
+       }\r
+\r
+       b->Current += size_read;\r
+\r
+       // KS\r
+       KS_INC(KS_READ_BUF_COUNT);\r
+\r
+       return size_read;\r
+}\r
+\r
+// バッファサイズの調整\r
+void AdjustBufSize(BUF *b, UINT new_size)\r
+{\r
+       // 引数チェック\r
+       if (b == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (b->SizeReserved >= new_size)\r
+       {\r
+               return;\r
+       }\r
+\r
+       while (b->SizeReserved < new_size)\r
+       {\r
+               b->SizeReserved = b->SizeReserved * 2;\r
+       }\r
+       b->Buf = ReAlloc(b->Buf, b->SizeReserved);\r
+\r
+       // KS\r
+       KS_INC(KS_ADJUST_BUFSIZE_COUNT);\r
+}\r
+\r
+// バッファのシーク\r
+void SeekBuf(BUF *b, UINT offset, int mode)\r
+{\r
+       UINT new_pos;\r
+       // 引数チェック\r
+       if (b == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (mode == 0)\r
+       {\r
+               // 絶対位置\r
+               new_pos = offset;\r
+       }\r
+       else\r
+       {\r
+               if (mode > 0)\r
+               {\r
+                       // 右へ移動\r
+                       new_pos = b->Current + offset;\r
+               }\r
+               else\r
+               {\r
+                       // 左へ移動\r
+                       if (b->Current >= offset)\r
+                       {\r
+                               new_pos = b->Current - offset;\r
+                       }\r
+                       else\r
+                       {\r
+                               new_pos = 0;\r
+                       }\r
+               }\r
+       }\r
+       b->Current = MAKESURE(new_pos, 0, b->Size);\r
+\r
+       KS_INC(KS_SEEK_BUF_COUNT);\r
+}\r
+\r
+// バッファの解放\r
+void FreeBuf(BUF *b)\r
+{\r
+       // 引数チェック\r
+       if (b == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // メモリ解放\r
+       Free(b->Buf);\r
+       Free(b);\r
+\r
+       // KS\r
+       KS_INC(KS_FREEBUF_COUNT);\r
+       KS_DEC(KS_CURRENT_BUF_COUNT);\r
+\r
+#ifndef        DONT_USE_KERNEL_STATUS\r
+       TrackDeleteObj(POINTER_TO_UINT64(b));\r
+#endif // DONT_USE_KERNEL_STATUS\r
+}\r
+\r
+// Unicode 文字列のエンディアン変換\r
+void EndianUnicode(wchar_t *str)\r
+{\r
+       UINT i, len;\r
+       // 引数チェック\r
+       if (str == NULL)\r
+       {\r
+               return;\r
+       }\r
+       len = UniStrLen(str);\r
+\r
+       for (i = 0;i < len;i++)\r
+       {\r
+               str[i] = Endian16(str[i]);\r
+       }\r
+}\r
+\r
+// エンディアン変換 16bit\r
+USHORT Endian16(USHORT src)\r
+{\r
+       int x = 1;\r
+       if (*((char *)&x))\r
+       {\r
+               return Swap16(src);\r
+       }\r
+       else\r
+       {\r
+               return src;\r
+       }\r
+}\r
+\r
+// エンディアン変換 32bit\r
+UINT Endian32(UINT src)\r
+{\r
+       int x = 1;\r
+       if (*((char *)&x))\r
+       {\r
+               return Swap32(src);\r
+       }\r
+       else\r
+       {\r
+               return src;\r
+       }\r
+}\r
+\r
+// エンディアン変換 64bit\r
+UINT64 Endian64(UINT64 src)\r
+{\r
+       int x = 1;\r
+       if (*((char *)&x))\r
+       {\r
+               return Swap64(src);\r
+       }\r
+       else\r
+       {\r
+               return src;\r
+       }\r
+}\r
+\r
+// 任意のデータのスワップ\r
+void Swap(void *buf, UINT size)\r
+{\r
+       UCHAR *tmp, *src;\r
+       UINT i;\r
+       // 引数チェック\r
+       if (buf == NULL || size == 0)\r
+       {\r
+               return;\r
+       }\r
+\r
+       src = (UCHAR *)buf;\r
+       tmp = Malloc(size);\r
+       for (i = 0;i < size;i++)\r
+       {\r
+               tmp[size - i - 1] = src[i];\r
+       }\r
+\r
+       Copy(buf, tmp, size);\r
+       Free(buf);\r
+}\r
+\r
+// 16bit スワップ\r
+USHORT Swap16(USHORT value)\r
+{\r
+       USHORT r;\r
+       // 汚いコード\r
+       ((BYTE *)&r)[0] = ((BYTE *)&value)[1];\r
+       ((BYTE *)&r)[1] = ((BYTE *)&value)[0];\r
+       return r;\r
+}\r
+\r
+// 32bit スワップ\r
+UINT Swap32(UINT value)\r
+{\r
+       UINT r;\r
+       // 汚いコード\r
+       ((BYTE *)&r)[0] = ((BYTE *)&value)[3];\r
+       ((BYTE *)&r)[1] = ((BYTE *)&value)[2];\r
+       ((BYTE *)&r)[2] = ((BYTE *)&value)[1];\r
+       ((BYTE *)&r)[3] = ((BYTE *)&value)[0];\r
+       return r;\r
+}\r
+\r
+// 64bit スワップ\r
+UINT64 Swap64(UINT64 value)\r
+{\r
+       UINT64 r;\r
+       // 汚いコード\r
+       ((BYTE *)&r)[0] = ((BYTE *)&value)[7];\r
+       ((BYTE *)&r)[1] = ((BYTE *)&value)[6];\r
+       ((BYTE *)&r)[2] = ((BYTE *)&value)[5];\r
+       ((BYTE *)&r)[3] = ((BYTE *)&value)[4];\r
+       ((BYTE *)&r)[4] = ((BYTE *)&value)[3];\r
+       ((BYTE *)&r)[5] = ((BYTE *)&value)[2];\r
+       ((BYTE *)&r)[6] = ((BYTE *)&value)[1];\r
+       ((BYTE *)&r)[7] = ((BYTE *)&value)[0];\r
+       return r;\r
+}\r
+\r
+// Base64 エンコード\r
+UINT Encode64(char *dst, char *src)\r
+{\r
+       // 引数チェック\r
+       if (dst == NULL || src == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       return B64_Encode(dst, src, StrLen(src));\r
+}\r
+\r
+// Base64 デコード\r
+UINT Decode64(char *dst, char *src)\r
+{\r
+       // 引数チェック\r
+       if (dst == NULL || src == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       return B64_Decode(dst, src, StrLen(src));\r
+}\r
+\r
+// Base64 エンコード\r
+int B64_Encode(char *set, char *source, int len)\r
+{\r
+       BYTE *src;\r
+       int i,j;\r
+       src = (BYTE *)source;\r
+       j = 0;\r
+       i = 0;\r
+       if (!len)\r
+       {\r
+               return 0;\r
+       }\r
+       while (TRUE)\r
+       {\r
+               if (i >= len)\r
+               {\r
+                       return j;\r
+               }\r
+               if (set)\r
+               {\r
+                       set[j] = B64_CodeToChar((src[i]) >> 2);\r
+               }\r
+               if (i + 1 >= len)\r
+               {\r
+                       if (set)\r
+                       {\r
+                               set[j + 1] = B64_CodeToChar((src[i] & 0x03) << 4);\r
+                               set[j + 2] = '=';\r
+                               set[j + 3] = '=';\r
+                       }\r
+                       return j + 4;\r
+               }\r
+               if (set)\r
+               {\r
+                       set[j + 1] = B64_CodeToChar(((src[i] & 0x03) << 4) + ((src[i + 1] >> 4)));\r
+               }\r
+               if (i + 2 >= len)\r
+               {\r
+                       if (set)\r
+                       {\r
+                               set[j + 2] = B64_CodeToChar((src[i + 1] & 0x0f) << 2);\r
+                               set[j + 3] = '=';\r
+                       }\r
+                       return j + 4;\r
+               }\r
+               if (set)\r
+               {\r
+                       set[j + 2] = B64_CodeToChar(((src[i + 1] & 0x0f) << 2) + ((src[i + 2] >> 6)));\r
+                       set[j + 3] = B64_CodeToChar(src[i + 2] & 0x3f);\r
+               }\r
+               i += 3;\r
+               j += 4;\r
+       }\r
+}\r
+\r
+// Base64 デコード\r
+int B64_Decode(char *set, char *source, int len)\r
+{\r
+       int i,j;\r
+       char a1,a2,a3,a4;\r
+       char *src;\r
+       int f1,f2,f3,f4;\r
+       src = source;\r
+       i = 0;\r
+       j = 0;\r
+       while (TRUE)\r
+       {\r
+               f1 = f2 = f3 = f4 = 0;\r
+               if (i >= len)\r
+               {\r
+                       break;\r
+               }\r
+               f1 = 1;\r
+               a1 = B64_CharToCode(src[i]);\r
+               if (a1 == -1)\r
+               {\r
+                       f1 = 0;\r
+               }\r
+               if (i >= len + 1)\r
+               {\r
+                       a2 = 0;\r
+               }\r
+               else\r
+               {\r
+                       a2 = B64_CharToCode(src[i + 1]);\r
+                       f2 = 1;\r
+                       if (a2 == -1)\r
+                       {\r
+                               f2 = 0;\r
+                       }\r
+               }\r
+               if (i >= len + 2)\r
+               {\r
+                       a3 = 0;\r
+               }\r
+               else\r
+               {\r
+                       a3 = B64_CharToCode(src[i + 2]);\r
+                       f3 = 1;\r
+                       if (a3 == -1)\r
+                       {\r
+                               f3 = 0;\r
+                       }\r
+               }\r
+               if (i >= len + 3)\r
+               {\r
+                       a4 = 0;\r
+               }\r
+               else\r
+               {\r
+                       a4 = B64_CharToCode(src[i + 3]);\r
+                       f4 = 1;\r
+                       if (a4 == -1)\r
+                       {\r
+                               f4 = 0;\r
+                       }\r
+               }\r
+               if (f1 && f2)\r
+               {\r
+                       if (set)\r
+                       {\r
+                               set[j] = (a1 << 2) + (a2 >> 4);\r
+                       }\r
+                       j++;\r
+               }\r
+               if (f2 && f3)\r
+               {\r
+                       if (set)\r
+                       {\r
+                               set[j] = (a2 << 4) + (a3 >> 2);\r
+                       }\r
+                       j++;\r
+               }\r
+               if (f3 && f4)\r
+               {\r
+                       if (set)\r
+                       {\r
+                               set[j] = (a3 << 6) + a4;\r
+                       }\r
+                       j++;\r
+               }\r
+               i += 4;\r
+       }\r
+       return j;\r
+}\r
+\r
+// Base64 - コードを文字に変換\r
+char B64_CodeToChar(BYTE c)\r
+{\r
+       BYTE r;\r
+       r = '=';\r
+       if (c <= 0x19)\r
+       {\r
+               r = c + 'A';\r
+       }\r
+       if (c >= 0x1a && c <= 0x33)\r
+       {\r
+               r = c - 0x1a + 'a';\r
+       }\r
+       if (c >= 0x34 && c <= 0x3d)\r
+       {\r
+               r = c - 0x34 + '0';\r
+       }\r
+       if (c == 0x3e)\r
+       {\r
+               r = '+';\r
+       }\r
+       if (c == 0x3f)\r
+       {\r
+               r = '/';\r
+       }\r
+       return r;\r
+}\r
+\r
+// Base64 - 文字をコードに変換\r
+char B64_CharToCode(char c)\r
+{\r
+       if (c >= 'A' && c <= 'Z')\r
+       {\r
+               return c - 'A';\r
+       }\r
+       if (c >= 'a' && c <= 'z')\r
+       {\r
+               return c - 'a' + 0x1a;\r
+       }\r
+       if (c >= '0' && c <= '9')\r
+       {\r
+               return c - '0' + 0x34;\r
+       }\r
+       if (c == '+')\r
+       {\r
+               return 0x3e;\r
+       }\r
+       if (c == '/')\r
+       {\r
+               return 0x3f;\r
+       }\r
+       if (c == '=')\r
+       {\r
+               return -1;\r
+       }\r
+       return 0;\r
+}\r
+\r
+// 高速な Malloc (現在未実装)\r
+// 実は小さなバッファをたくさんまとめておいてそれを動的に割り当てるコードを昔\r
+// 書いたのだが、Windows, Linux, Solaris で試しても普通の malloc() と比べて\r
+// ほとんど速度に影響がなかったので、やめにした。\r
+void *MallocFast(UINT size)\r
+{\r
+       return Malloc(size);\r
+}\r
+\r
+// Malloc\r
+void *Malloc(UINT size)\r
+{\r
+       return MallocEx(size, false);\r
+}\r
+void *MallocEx(UINT size, bool zero_clear_when_free)\r
+{\r
+       MEMTAG *tag;\r
+       UINT real_size;\r
+\r
+       real_size = CALC_MALLOCSIZE(size);\r
+\r
+       tag = InternalMalloc(real_size);\r
+\r
+       Zero(tag, sizeof(MEMTAG));\r
+       tag->Magic = MEMTAG_MAGIC;\r
+       tag->Size = size;\r
+       tag->ZeroFree = zero_clear_when_free;\r
+\r
+       return MEMTAG_TO_POINTER(tag);\r
+}\r
+\r
+// ReAlloc\r
+void *ReAlloc(void *addr, UINT size)\r
+{\r
+       MEMTAG *tag;\r
+       bool zerofree;\r
+       // 引数チェック\r
+       if (IS_NULL_POINTER(addr))\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       tag = POINTER_TO_MEMTAG(addr);\r
+       CheckMemTag(tag);\r
+\r
+       zerofree = tag->ZeroFree;\r
+\r
+       if (tag->Size == size)\r
+       {\r
+               // サイズ変更無し\r
+               return addr;\r
+       }\r
+       else\r
+       {\r
+               if (zerofree)\r
+               {\r
+                       // サイズ変更有り (ゼロクリア必須)\r
+                       void *new_p = MallocEx(size, true);\r
+\r
+                       if (tag->Size <= size)\r
+                       {\r
+                               // サイズ拡大\r
+                               Copy(new_p, addr, tag->Size);\r
+                       }\r
+                       else\r
+                       {\r
+                               // サイズ縮小\r
+                               Copy(new_p, addr, size);\r
+                       }\r
+\r
+                       // 古いブロックの解放\r
+                       Free(addr);\r
+\r
+                       return new_p;\r
+               }\r
+               else\r
+               {\r
+                       // サイズ変更有り\r
+                       MEMTAG *tag2 = InternalReAlloc(tag, CALC_MALLOCSIZE(size));\r
+\r
+                       Zero(tag2, sizeof(MEMTAG));\r
+                       tag2->Magic = MEMTAG_MAGIC;\r
+                       tag2->Size = size;\r
+\r
+                       return MEMTAG_TO_POINTER(tag2);\r
+               }\r
+       }\r
+}\r
+\r
+// Free\r
+void Free(void *addr)\r
+{\r
+       MEMTAG *tag;\r
+       // 引数チェック\r
+       if (IS_NULL_POINTER(addr))\r
+       {\r
+               return;\r
+       }\r
+\r
+       tag = POINTER_TO_MEMTAG(addr);\r
+       CheckMemTag(tag);\r
+\r
+       if (tag->ZeroFree)\r
+       {\r
+               // ゼロクリア\r
+               Zero(addr, tag->Size);\r
+       }\r
+\r
+       // メモリ解放\r
+       tag->Magic = 0;\r
+       InternalFree(tag);\r
+}\r
+\r
+// memtag をチェック\r
+void CheckMemTag(MEMTAG *tag)\r
+{\r
+#ifndef        DONT_CHECK_HEAP\r
+       // 引数チェック\r
+       if (tag == NULL)\r
+       {\r
+               AbortExitEx("CheckMemTag: tag == NULL");\r
+               return;\r
+       }\r
+\r
+       if (tag->Magic != MEMTAG_MAGIC)\r
+       {\r
+               AbortExitEx("CheckMemTag: tag->Magic != MEMTAG_MAGIC");\r
+               return;\r
+       }\r
+#endif // DONT_CHECK_HEAP\r
+}\r
+\r
+// ZeroMalloc\r
+void *ZeroMalloc(UINT size)\r
+{\r
+       return ZeroMallocEx(size, false);\r
+}\r
+void *ZeroMallocEx(UINT size, bool zero_clear_when_free)\r
+{\r
+       void *p = MallocEx(size, zero_clear_when_free);\r
+       Zero(p, size);\r
+       return p;\r
+}\r
+void *ZeroMallocFast(UINT size)\r
+{\r
+       void *p = MallocFast(size);\r
+       Zero(p, size);\r
+       return p;\r
+}\r
+\r
+// メモリ確保\r
+void *InternalMalloc(UINT size)\r
+{\r
+       void *addr;\r
+       UINT retry = 0;\r
+       size = MORE(size, 1);\r
+\r
+       // KS\r
+       KS_INC(KS_MALLOC_COUNT);\r
+       KS_INC(KS_TOTAL_MEM_COUNT);\r
+       KS_ADD(KS_TOTAL_MEM_SIZE, size);\r
+       KS_INC(KS_CURRENT_MEM_COUNT);\r
+\r
+       // メモリが確保されるまで試行する\r
+       while (true)\r
+       {\r
+               if ((retry++) > MEMORY_MAX_RETRY)\r
+               {\r
+                       AbortExitEx("InternalMalloc: error: malloc() failed.\n\n");\r
+               }\r
+               addr = OSMemoryAlloc(size);\r
+               if (addr != NULL)\r
+               {\r
+                       break;\r
+               }\r
+\r
+               OSSleep(MEMORY_SLEEP_TIME);\r
+       }\r
+\r
+#ifndef        DONT_USE_KERNEL_STATUS\r
+       TrackNewObj(POINTER_TO_UINT64(addr), "MEM", size);\r
+#endif //DONT_USE_KERNEL_STATUS\r
+\r
+       return addr;\r
+}\r
+\r
+// メモリ解放\r
+void InternalFree(void *addr)\r
+{\r
+       // 引数チェック\r
+       if (addr == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // KS\r
+       KS_DEC(KS_CURRENT_MEM_COUNT);\r
+       KS_INC(KS_FREE_COUNT);\r
+\r
+#ifndef        DONT_USE_KERNEL_STATUS\r
+       TrackDeleteObj(POINTER_TO_UINT64(addr));\r
+#endif // DONT_USE_KERNEL_STATUS\r
+\r
+       // メモリ解放\r
+       OSMemoryFree(addr);\r
+}\r
+\r
+// メモリ再確保\r
+void *InternalReAlloc(void *addr, UINT size)\r
+{\r
+       void *new_addr;\r
+       UINT retry = 0;\r
+       size = MORE(size, 1);\r
+\r
+       // KS\r
+       KS_INC(KS_REALLOC_COUNT);\r
+       KS_ADD(KS_TOTAL_MEM_SIZE, size);\r
+\r
+       // メモリが確保されるまで試行する\r
+       while (true)\r
+       {\r
+               if ((retry++) > MEMORY_MAX_RETRY)\r
+               {\r
+                       AbortExitEx("InternalReAlloc: error: realloc() failed.\n\n");\r
+               }\r
+               new_addr = OSMemoryReAlloc(addr, size);\r
+               if (new_addr != NULL)\r
+               {\r
+                       break;\r
+               }\r
+\r
+               OSSleep(MEMORY_SLEEP_TIME);\r
+       }\r
+\r
+#ifndef        DONT_USE_KERNEL_STATUS\r
+       TrackChangeObjSize((DWORD)addr, size, (DWORD)new_addr);\r
+#endif // DONT_USE_KERNEL_STATUS\r
+\r
+       return new_addr;\r
+}\r
+\r
+// メモリ領域のクローン\r
+void *Clone(void *addr, UINT size)\r
+{\r
+       void *ret;\r
+       // 引数チェック\r
+       if (addr == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       ret = Malloc(size);\r
+       Copy(ret, addr, size);\r
+\r
+       return ret;\r
+}\r
+\r
+// メモリコピー\r
+void Copy(void *dst, void *src, UINT size)\r
+{\r
+       // 引数チェック\r
+       if (dst == NULL || src == NULL || size == 0 || dst == src)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // KS\r
+       KS_INC(KS_COPY_COUNT);\r
+\r
+       memcpy(dst, src, size);\r
+}\r
+\r
+// メモリ比較\r
+int Cmp(void *p1, void *p2, UINT size)\r
+{\r
+       // 引数チェック\r
+       if (p1 == NULL || p2 == NULL || size == 0)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       return memcmp(p1, p2, (size_t)size);\r
+}\r
+\r
+// メモリのゼロクリア\r
+void Zero(void *addr, UINT size)\r
+{\r
+       ZeroMem(addr, size);\r
+}\r
+void ZeroMem(void *addr, UINT size)\r
+{\r
+       // 引数チェック\r
+       if (addr == NULL || size == 0)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // KS\r
+       KS_INC(KS_ZERO_COUNT);\r
+\r
+       memset(addr, 0, size);\r
+}\r
+\r
+// 文字列マップエントリの比較\r
+int StrMapCmp(void *p1, void *p2)\r
+{\r
+       STRMAP_ENTRY *s1, *s2;\r
+       if (p1 == NULL || p2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+       s1 = *(STRMAP_ENTRY **)p1;\r
+       s2 = *(STRMAP_ENTRY **)p2;\r
+       if (s1 == NULL || s2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+       return StrCmpi(s1->Name, s2->Name);\r
+}\r
+\r
+// 文字列マップ(文字列で検索できるデータ)の作成\r
+LIST *NewStrMap()\r
+{\r
+       return NewList(StrMapCmp);\r
+}\r
+\r
+// 文字列マップの検索\r
+void *StrMapSearch(LIST *map, char *key)\r
+{\r
+       STRMAP_ENTRY tmp, *result;\r
+       tmp.Name = key;\r
+       result = (STRMAP_ENTRY*)Search(map, &tmp);\r
+       if(result != NULL)\r
+       {\r
+               return result->Value;\r
+       }\r
+       return NULL;\r
+}\r