OSDN Git Service

replace 'SUCCESS' and 'FAIL' macro to 'FFFTP_SUCCESS' and 'FFFTP_FAIL' to suppress...
[ffftp/ffftp.git] / registory.c
1 /*=============================================================================\r
2 *                                                               レジストリ関係\r
3 *\r
4 *\r
5 ===============================================================================\r
6 / Copyright (C) 1997-2007 Sota. All rights reserved.\r
7 /\r
8 / Redistribution and use in source and binary forms, with or without \r
9 / modification, are permitted provided that the following conditions \r
10 / are met:\r
11 /\r
12 /  1. Redistributions of source code must retain the above copyright \r
13 /     notice, this list of conditions and the following disclaimer.\r
14 /  2. Redistributions in binary form must reproduce the above copyright \r
15 /     notice, this list of conditions and the following disclaimer in the \r
16 /     documentation and/or other materials provided with the distribution.\r
17 /\r
18 / THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR \r
19 / IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES \r
20 / OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. \r
21 / IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, \r
22 / INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, \r
23 / BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF \r
24 / USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON \r
25 / ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT \r
26 / (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF \r
27 / THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.\r
28 /============================================================================*/\r
29 \r
30 // VC 8.0(2005)以上でのみ rand_s を利用可能\r
31 #if 1400 <= _MSC_VER\r
32 //#define _CRT_RAND_S\r
33 #endif\r
34 \r
35 #define STRICT\r
36 #include <windows.h>\r
37 #include <stdio.h>\r
38 #include <stdlib.h>\r
39 #include <time.h>\r
40 #include <string.h>\r
41 #include <windowsx.h>\r
42 #include <process.h>\r
43 \r
44 \r
45 #include "common.h"\r
46 #include "resource.h"\r
47 #include "sample.h"\r
48 #include "sha.h"\r
49 #include "aes.h"\r
50 \r
51 \r
52 /*===== プロトタイプ =====*/\r
53 \r
54 static void SaveStr(HKEY hKey, char *Key, char *Str, char *DefaultStr);\r
55 static void SaveIntNum(HKEY hKey, char *Key, int Num, int DefaultNum);\r
56 static void MakeFontData(LOGFONT Font, HFONT hFont, char *Buf);\r
57 static int RestoreFontData(char *Str, LOGFONT *Font);\r
58 \r
59 static void EncodePassword(char *Str, char *Buf);\r
60 static void EncodePasswordOriginal(char *Str, char *Buf);\r
61 static void EncodePassword2(char *Str, char *Buf, const char *Key);\r
62 static void EncodePassword3(char *Str, char *Buf, const char *Key);\r
63 \r
64 static void DecodePassword(char *Str, char *Buf);\r
65 static void DecodePasswordOriginal(char *Str, char *Buf);\r
66 static void DecodePassword2(char *Str, char *Buf, const char *Key);\r
67 static void DecodePassword3(char *Str, char *Buf, const char *Key);\r
68 static int CreateAesKey(unsigned char *AesKey, const char* Key);\r
69 \r
70 static void SetRegType(int Type);\r
71 static int OpenReg(char *Name, void **Handle);\r
72 static int CreateReg(char *Name, void **Handle);\r
73 static int CloseReg(void *Handle);\r
74 static int OpenSubKey(void *Parent, char *Name, void **Handle);\r
75 static int CreateSubKey(void *Parent, char *Name, void **Handle);\r
76 static int CloseSubKey(void *Handle);\r
77 static int DeleteSubKey(void *Handle, char *Name);\r
78 static int DeleteValue(void *Handle, char *Name);\r
79 static int ReadIntValueFromReg(void *Handle, char *Name, int *Value);\r
80 static int WriteIntValueToReg(void *Handle, char *Name, int Value);\r
81 static int ReadStringFromReg(void *Handle, char *Name, char *Str, DWORD Size);\r
82 static int WriteStringToReg(void *Handle, char *Name, char *Str);\r
83 static int ReadMultiStringFromReg(void *Handle, char *Name, char *Str, DWORD Size);\r
84 static int WriteMultiStringToReg(void *Handle, char *Name, char *Str);\r
85 static int ReadBinaryFromReg(void *Handle, char *Name, void *Bin, DWORD Size);\r
86 static int WriteBinaryToReg(void *Handle, char *Name, void *Bin, int Len);\r
87 \r
88 int CheckPasswordValidity( char* Password, int length, const char* HashStr );\r
89 void CreatePasswordHash( char* Password, int length, char* HashStr );\r
90 void SetHashSalt( DWORD salt );\r
91 \r
92 DWORD GetRandamDWRODValue(void);\r
93 \r
94 /* 2010.01.30 genta 追加 */\r
95 static char SecretKey[FMAX_PATH+1];\r
96 static int SecretKeyLength;\r
97 static int IsMasterPasswordError = PASSWORD_OK;\r
98 \r
99 static int IsRndSourceInit = 0;\r
100 static ulong RndSource[9];\r
101 \r
102 /*===== 外部参照 =====*/\r
103 \r
104 /* 設定値 */\r
105 extern int WinPosX;\r
106 extern int WinPosY;\r
107 extern int WinWidth;\r
108 extern int WinHeight;\r
109 extern int LocalWidth;\r
110 extern int TaskHeight;\r
111 extern int LocalTabWidth[4];\r
112 extern int RemoteTabWidth[6];\r
113 extern char UserMailAdrs[USER_MAIL_LEN+1];\r
114 extern char ViewerName[VIEWERS][FMAX_PATH+1];\r
115 extern HFONT ListFont;\r
116 extern LOGFONT ListLogFont;\r
117 extern int LocalFileSort;\r
118 extern int LocalDirSort;\r
119 extern int RemoteFileSort;\r
120 extern int RemoteDirSort;\r
121 extern int TransMode;\r
122 extern int ConnectOnStart;\r
123 extern int DebugConsole;\r
124 extern int SaveWinPos;\r
125 extern char AsciiExt[ASCII_EXT_LEN+1];\r
126 extern int RecvMode;\r
127 extern int SendMode;\r
128 extern int MoveMode;\r
129 extern int ListType;\r
130 extern int CacheEntry;\r
131 extern int CacheSave;\r
132 extern char DefaultLocalPath[FMAX_PATH+1];\r
133 extern int SaveTimeStamp;\r
134 extern int FindMode;\r
135 extern int DotFile;\r
136 extern int DclickOpen;\r
137 extern SOUNDFILE Sound[SOUND_TYPES];\r
138 extern int FnameCnv;\r
139 extern int ConnectAndSet;\r
140 extern int TimeOut;\r
141 extern int RmEOF;\r
142 extern int RegType;\r
143 extern char FwallHost[HOST_ADRS_LEN+1];\r
144 extern char FwallUser[USER_NAME_LEN+1];\r
145 extern char FwallPass[PASSWORD_LEN+1];\r
146 extern int FwallPort;\r
147 extern int FwallType;\r
148 extern int FwallDefault;\r
149 extern int FwallSecurity;\r
150 extern int FwallResolv;\r
151 extern int FwallLower;\r
152 extern int FwallDelimiter;\r
153 extern int PasvDefault;\r
154 extern char MirrorNoTrn[MIRROR_LEN+1];\r
155 extern char MirrorNoDel[MIRROR_LEN+1];\r
156 extern int MirrorFnameCnv;\r
157 //extern int MirrorFolderCnv;\r
158 extern int RasClose;\r
159 extern int RasCloseNotify;\r
160 extern int FileHist;\r
161 extern char DefAttrList[DEFATTRLIST_LEN+1];\r
162 extern SIZE HostDlgSize;\r
163 extern SIZE BmarkDlgSize;\r
164 extern SIZE MirrorDlgSize;\r
165 extern int Sizing;\r
166 extern int SortSave;\r
167 extern char TmpPath[FMAX_PATH+1];\r
168 extern int QuickAnonymous;\r
169 extern int PassToHist;\r
170 extern int VaxSemicolon;\r
171 extern int SendQuit;\r
172 extern int NoRasControl;\r
173 extern int SuppressSave;\r
174 \r
175 extern int UpExistMode;\r
176 extern int ExistMode;\r
177 extern int DispIgnoreHide;\r
178 extern int DispDrives;\r
179 extern int MirUpDelNotify;\r
180 extern int MirDownDelNotify;\r
181 \r
182 extern int FolderAttr;\r
183 extern int FolderAttrNum;\r
184 \r
185 /*----- マスタパスワードの設定 ----------------------------------------------\r
186 *\r
187 *       Parameter\r
188 *               const char* Password : マスターパスワード\r
189 *\r
190 *       Return Value\r
191 *               なし\r
192 *----------------------------------------------------------------------------*/\r
193 void SetMasterPassword( const char* Password )\r
194 {\r
195         ZeroMemory( SecretKey, MAX_PASSWORD_LEN + 12 );\r
196         if( Password != NULL ){\r
197                 strncpy( SecretKey, Password, MAX_PASSWORD_LEN );\r
198         }\r
199         else {\r
200                 strcpy( SecretKey, DEFAULT_PASSWORD );\r
201         }\r
202         SecretKeyLength = strlen( SecretKey );\r
203         \r
204         /* 未検証なので,初期状態はOKにする (強制再設定→保存にを可能にする)*/\r
205         IsMasterPasswordError = PASSWORD_OK;\r
206 }\r
207 \r
208 /*----- マスタパスワードの状態取得 ----------------------------------------------\r
209 *\r
210 *       Parameter\r
211 *               なし\r
212 *\r
213 *       Return Value\r
214 *               PASSWORD_OK : OK\r
215 *               PASSWORD_UNMATCH : パスワード不一致\r
216 *               BAD_PASSWORD_HASH : パスワード確認失敗\r
217 *----------------------------------------------------------------------------*/\r
218 int GetMasterPasswordStatus(void)\r
219 {\r
220         return IsMasterPasswordError;\r
221 }\r
222 \r
223 /*----- レジストリ/INIファイルのマスターパスワードの検証を行う ------------\r
224 *\r
225 *       Parameter\r
226 *               なし\r
227 *\r
228 *       Return Value\r
229 *               \r
230 *----------------------------------------------------------------------------*/\r
231 \r
232 int ValidateMasterPassword(void)\r
233 {\r
234         void *hKey3;\r
235         int i;\r
236 \r
237         SetRegType(REGTYPE_INI);\r
238         if((i = OpenReg("FFFTP", &hKey3)) != FFFTP_SUCCESS)\r
239         {\r
240                 if(AskForceIni() == NO)\r
241                 {\r
242                         SetRegType(REGTYPE_REG);\r
243                         i = OpenReg("FFFTP", &hKey3);\r
244                 }\r
245         }\r
246         if(i == FFFTP_SUCCESS){\r
247                 char checkbuf[48];\r
248                 int salt = 0;\r
249 \r
250                 if( ReadIntValueFromReg(hKey3, "CredentialSalt", &salt)){\r
251                         SetHashSalt( salt );\r
252                 }\r
253                 if( ReadStringFromReg(hKey3, "CredentialCheck", checkbuf, sizeof( checkbuf )) == FFFTP_SUCCESS ){\r
254                         switch( CheckPasswordValidity( SecretKey, SecretKeyLength, checkbuf ) ){\r
255                         case 0: /* not match */\r
256                                 IsMasterPasswordError = PASSWORD_UNMATCH;\r
257                                 break;\r
258                         case 1: /* match */\r
259                                 IsMasterPasswordError = PASSWORD_OK;\r
260                                 break;\r
261                         case 2: /* invalid hash */\r
262                         default:\r
263                                 IsMasterPasswordError = BAD_PASSWORD_HASH;\r
264                                 break;\r
265                         }\r
266                 }\r
267                 CloseReg(hKey3);\r
268                 return YES;\r
269         }\r
270         return NO;\r
271 }\r
272 \r
273 /*----- レジストリ/INIファイルに設定値を保存 ---------------------------------\r
274 *\r
275 *       Parameter\r
276 *               なし\r
277 *\r
278 *       Return Value\r
279 *               なし\r
280 *----------------------------------------------------------------------------*/\r
281 \r
282 void SaveRegistory(void)\r
283 {\r
284         void *hKey3;\r
285         void *hKey4;\r
286         void *hKey5;\r
287         char Str[FMAX_PATH+1];\r
288         int i;\r
289         int n;\r
290         HOSTDATA DefaultHost;\r
291         HOSTDATA Host;\r
292         HISTORYDATA Hist;\r
293         HISTORYDATA DefaultHist;\r
294         \r
295         if( GetMasterPasswordStatus() == PASSWORD_UNMATCH ){\r
296                 /* 2010.01.30 genta: マスターパスワードが不一致の場合は不用意に上書きしない */\r
297                 return;\r
298         }\r
299 \r
300         SetRegType(RegType);\r
301         if(CreateReg("FFFTP", &hKey3) == FFFTP_SUCCESS)\r
302         {\r
303                 char buf[48];\r
304                 int salt = GetTickCount();\r
305         \r
306                 WriteIntValueToReg(hKey3, "Version", VER_NUM);\r
307                 WriteIntValueToReg(hKey3, "CredentialSalt", salt);\r
308                 \r
309                 SetHashSalt( salt );\r
310                 /* save password hash */\r
311                 CreatePasswordHash( SecretKey, SecretKeyLength, buf );\r
312                 WriteStringToReg(hKey3, "CredentialCheck", buf);\r
313 \r
314                 if(CreateSubKey(hKey3, "Options", &hKey4) == FFFTP_SUCCESS)\r
315                 {\r
316                         WriteIntValueToReg(hKey4, "NoSave", SuppressSave);\r
317 \r
318                         if(SuppressSave != YES)\r
319                         {\r
320                                 WriteIntValueToReg(hKey4, "WinPosX", WinPosX);\r
321                                 WriteIntValueToReg(hKey4, "WinPosY", WinPosY);\r
322                                 WriteIntValueToReg(hKey4, "WinWidth", WinWidth);\r
323                                 WriteIntValueToReg(hKey4, "WinHeight", WinHeight);\r
324                                 WriteIntValueToReg(hKey4, "LocalWidth", LocalWidth);\r
325                                 WriteIntValueToReg(hKey4, "TaskHeight", TaskHeight);\r
326                                 WriteBinaryToReg(hKey4, "LocalColm", LocalTabWidth, sizeof(LocalTabWidth));\r
327                                 WriteBinaryToReg(hKey4, "RemoteColm", RemoteTabWidth, sizeof(RemoteTabWidth));\r
328                                 WriteIntValueToReg(hKey4, "SwCmd", Sizing);\r
329 \r
330                                 WriteStringToReg(hKey4, "UserMail", UserMailAdrs);\r
331                                 WriteStringToReg(hKey4, "Viewer", ViewerName[0]);\r
332                                 WriteStringToReg(hKey4, "Viewer2", ViewerName[1]);\r
333                                 WriteStringToReg(hKey4, "Viewer3", ViewerName[2]);\r
334 \r
335                                 WriteIntValueToReg(hKey4, "TrType", TransMode);\r
336                                 WriteIntValueToReg(hKey4, "Recv", RecvMode);\r
337                                 WriteIntValueToReg(hKey4, "Send", SendMode);\r
338                                 WriteIntValueToReg(hKey4, "Move", MoveMode);\r
339                                 WriteStringToReg(hKey4, "Path", DefaultLocalPath);\r
340                                 WriteIntValueToReg(hKey4, "Time", SaveTimeStamp);\r
341                                 WriteIntValueToReg(hKey4, "EOF", RmEOF);\r
342                                 WriteIntValueToReg(hKey4, "Scolon", VaxSemicolon);\r
343 \r
344                                 WriteIntValueToReg(hKey4, "RecvEx", ExistMode);\r
345                                 WriteIntValueToReg(hKey4, "SendEx", UpExistMode);\r
346 \r
347                                 WriteIntValueToReg(hKey4, "LFsort", LocalFileSort);\r
348                                 WriteIntValueToReg(hKey4, "LDsort", LocalDirSort);\r
349                                 WriteIntValueToReg(hKey4, "RFsort", RemoteFileSort);\r
350                                 WriteIntValueToReg(hKey4, "RDsort", RemoteDirSort);\r
351                                 WriteIntValueToReg(hKey4, "SortSave", SortSave);\r
352 \r
353                                 WriteIntValueToReg(hKey4, "ListType", ListType);\r
354                                 WriteIntValueToReg(hKey4, "Cache", CacheEntry);\r
355                                 WriteIntValueToReg(hKey4, "CacheSave", CacheSave);\r
356                                 WriteIntValueToReg(hKey4, "DotFile", DotFile);\r
357                                 WriteIntValueToReg(hKey4, "Dclick", DclickOpen);\r
358 \r
359                                 WriteIntValueToReg(hKey4, "ConS", ConnectOnStart);\r
360                                 WriteIntValueToReg(hKey4, "OldDlg", ConnectAndSet);\r
361                                 WriteIntValueToReg(hKey4, "RasClose", RasClose);\r
362                                 WriteIntValueToReg(hKey4, "RasNotify", RasCloseNotify);\r
363                                 WriteIntValueToReg(hKey4, "Qanony", QuickAnonymous);\r
364                                 WriteIntValueToReg(hKey4, "PassHist", PassToHist);\r
365                                 WriteIntValueToReg(hKey4, "SendQuit", SendQuit);\r
366                                 WriteIntValueToReg(hKey4, "NoRas", NoRasControl);\r
367 \r
368                                 WriteIntValueToReg(hKey4, "Debug", DebugConsole);\r
369                                 WriteIntValueToReg(hKey4, "WinPos", SaveWinPos);\r
370                                 WriteIntValueToReg(hKey4, "RegExp", FindMode);\r
371                                 WriteIntValueToReg(hKey4, "Reg", RegType);\r
372 \r
373                                 WriteMultiStringToReg(hKey4, "AsciiFile", AsciiExt);\r
374                                 WriteIntValueToReg(hKey4, "LowUp", FnameCnv);\r
375                                 WriteIntValueToReg(hKey4, "Tout", TimeOut);\r
376 \r
377                                 WriteMultiStringToReg(hKey4, "NoTrn", MirrorNoTrn);\r
378                                 WriteMultiStringToReg(hKey4, "NoDel", MirrorNoDel);\r
379                                 WriteIntValueToReg(hKey4, "MirFile", MirrorFnameCnv);\r
380                                 WriteIntValueToReg(hKey4, "MirUNot", MirUpDelNotify);\r
381                                 WriteIntValueToReg(hKey4, "MirDNot", MirDownDelNotify);\r
382 \r
383                                 MakeFontData(ListLogFont, ListFont, Str);\r
384                                 WriteStringToReg(hKey4, "ListFont", Str);\r
385                                 WriteIntValueToReg(hKey4, "ListHide", DispIgnoreHide);\r
386                                 WriteIntValueToReg(hKey4, "ListDrv", DispDrives);\r
387 \r
388                                 WriteStringToReg(hKey4, "FwallHost", FwallHost);\r
389                                 WriteStringToReg(hKey4, "FwallUser", FwallUser);\r
390                                 EncodePassword(FwallPass, Str);\r
391                                 WriteStringToReg(hKey4, "FwallPass", Str);\r
392                                 WriteIntValueToReg(hKey4, "FwallPort", FwallPort);\r
393                                 WriteIntValueToReg(hKey4, "FwallType", FwallType);\r
394                                 WriteIntValueToReg(hKey4, "FwallDef", FwallDefault);\r
395                                 WriteIntValueToReg(hKey4, "FwallSec", FwallSecurity);\r
396                                 WriteIntValueToReg(hKey4, "PasvDef", PasvDefault);\r
397                                 WriteIntValueToReg(hKey4, "FwallRes", FwallResolv);\r
398                                 WriteIntValueToReg(hKey4, "FwallLow", FwallLower);\r
399                                 WriteIntValueToReg(hKey4, "FwallDel", FwallDelimiter);\r
400 \r
401                                 WriteIntValueToReg(hKey4, "SndConSw", Sound[SND_CONNECT].On);\r
402                                 WriteIntValueToReg(hKey4, "SndTrnSw", Sound[SND_TRANS].On);\r
403                                 WriteIntValueToReg(hKey4, "SndErrSw", Sound[SND_ERROR].On);\r
404                                 WriteStringToReg(hKey4, "SndCon", Sound[SND_CONNECT].Fname);\r
405                                 WriteStringToReg(hKey4, "SndTrn", Sound[SND_TRANS].Fname);\r
406                                 WriteStringToReg(hKey4, "SndErr", Sound[SND_ERROR].Fname);\r
407 \r
408                                 WriteMultiStringToReg(hKey4, "DefAttr", DefAttrList);\r
409 \r
410                                 GetTempPath(FMAX_PATH, Str);\r
411                                 SaveStr(hKey4, "Tmp", TmpPath, Str);\r
412 \r
413                                 WriteBinaryToReg(hKey4, "Hdlg", &HostDlgSize, sizeof(SIZE));\r
414                                 WriteBinaryToReg(hKey4, "Bdlg", &BmarkDlgSize, sizeof(SIZE));\r
415                                 WriteBinaryToReg(hKey4, "Mdlg", &MirrorDlgSize, sizeof(SIZE));\r
416 \r
417                                 WriteIntValueToReg(hKey4, "FAttrSw", FolderAttr);\r
418                                 WriteIntValueToReg(hKey4, "FAttr", FolderAttrNum);\r
419 \r
420                                 WriteIntValueToReg(hKey4, "HistNum", FileHist);\r
421 \r
422                                 /* Ver1.54a以前の形式のヒストリデータは削除 */\r
423                                 DeleteValue(hKey4, "Hist");\r
424 \r
425                                 /* ヒストリの設定を保存 */\r
426                                 CopyDefaultHistory(&DefaultHist);\r
427                                 n = 0;\r
428                                 for(i = AskHistoryNum(); i > 0; i--)\r
429                                 {\r
430                                         if(GetHistoryByNum(i-1, &Hist) == FFFTP_SUCCESS)\r
431                                         {\r
432                                                 sprintf(Str, "History%d", n);\r
433                                                 if(CreateSubKey(hKey4, Str, &hKey5) == FFFTP_SUCCESS)\r
434                                                 {\r
435                                                         SaveStr(hKey5, "HostAdrs", Hist.HostAdrs, DefaultHist.HostAdrs);\r
436                                                         SaveStr(hKey5, "UserName", Hist.UserName, DefaultHist.UserName);\r
437                                                         SaveStr(hKey5, "Account", Hist.Account, DefaultHist.Account);\r
438                                                         SaveStr(hKey5, "LocalDir", Hist.LocalInitDir, NULL);\r
439                                                         SaveStr(hKey5, "RemoteDir", Hist.RemoteInitDir, DefaultHist.RemoteInitDir);\r
440                                                         SaveStr(hKey5, "Chmod", Hist.ChmodCmd, DefaultHist.ChmodCmd);\r
441                                                         SaveStr(hKey5, "Nlst", Hist.LsName, DefaultHist.LsName);\r
442                                                         SaveStr(hKey5, "Init", Hist.InitCmd, DefaultHist.InitCmd);\r
443                                                         EncodePassword(Hist.PassWord, Str);\r
444                                                         SaveStr(hKey5, "Password", Str, DefaultHist.PassWord);\r
445                                                         SaveIntNum(hKey5, "Port", Hist.Port, DefaultHist.Port);\r
446                                                         SaveIntNum(hKey5, "Kanji", Hist.KanjiCode, DefaultHist.KanjiCode);\r
447                                                         SaveIntNum(hKey5, "KanaCnv", Hist.KanaCnv, DefaultHist.KanaCnv);\r
448                                                         SaveIntNum(hKey5, "NameKanji", Hist.NameKanjiCode, DefaultHist.NameKanjiCode);\r
449                                                         SaveIntNum(hKey5, "NameKana", Hist.NameKanaCnv, DefaultHist.NameKanaCnv);\r
450                                                         SaveIntNum(hKey5, "Pasv", Hist.Pasv, DefaultHist.Pasv);\r
451                                                         SaveIntNum(hKey5, "Fwall", Hist.FireWall, DefaultHist.FireWall);\r
452                                                         SaveIntNum(hKey5, "List", Hist.ListCmdOnly, DefaultHist.ListCmdOnly);\r
453                                                         SaveIntNum(hKey5, "NLST-R", Hist.UseNLST_R, DefaultHist.UseNLST_R);\r
454                                                         SaveIntNum(hKey5, "Tzone", Hist.TimeZone, DefaultHist.TimeZone);\r
455                                                         SaveIntNum(hKey5, "Type", Hist.HostType, DefaultHist.HostType);\r
456                                                         SaveIntNum(hKey5, "Sync", Hist.SyncMove, DefaultHist.SyncMove);\r
457                                                         SaveIntNum(hKey5, "Fpath", Hist.NoFullPath, DefaultHist.NoFullPath);\r
458                                                         WriteBinaryToReg(hKey5, "Sort", &Hist.Sort, sizeof(Hist.Sort));\r
459                                                         SaveIntNum(hKey5, "Secu", Hist.Security, DefaultHist.Security);\r
460                                                         WriteIntValueToReg(hKey5, "TrType", Hist.Type);\r
461                                                         SaveIntNum(hKey5, "Dial", Hist.Dialup, DefaultHist.Dialup);\r
462                                                         SaveIntNum(hKey5, "UseIt", Hist.DialupAlways, DefaultHist.DialupAlways);\r
463                                                         SaveIntNum(hKey5, "Notify", Hist.DialupNotify, DefaultHist.DialupNotify);\r
464                                                         SaveStr(hKey5, "DialTo", Hist.DialEntry, DefaultHist.DialEntry);\r
465 \r
466                                                         CloseSubKey(hKey5);\r
467                                                         n++;\r
468                                                 }\r
469                                         }\r
470                                 }\r
471                                 WriteIntValueToReg(hKey4, "SavedHist", n);\r
472 \r
473                                 /* 余分なヒストリがあったら削除 */\r
474                                 for(; n < 999; n++)\r
475                                 {\r
476                                         sprintf(Str, "History%d", n);\r
477                                         if(DeleteSubKey(hKey4, Str) != FFFTP_SUCCESS)\r
478                                                 break;\r
479                                 }\r
480 \r
481                                 /* ホストの設定を保存 */\r
482                                 CopyDefaultHost(&DefaultHost);\r
483                                 i = 0;\r
484                                 while(CopyHostFromList(i, &Host) == FFFTP_SUCCESS)\r
485                                 {\r
486                                         sprintf(Str, "Host%d", i);\r
487                                         if(CreateSubKey(hKey4, Str, &hKey5) == FFFTP_SUCCESS)\r
488                                         {\r
489 //                                              SaveIntNum(hKey5, "Set", Host.Level, DefaultHost.Level);\r
490                                                 WriteIntValueToReg(hKey5, "Set", Host.Level);\r
491                                                 SaveStr(hKey5, "HostName", Host.HostName, DefaultHost.HostName);\r
492                                                 if((Host.Level & SET_LEVEL_GROUP) == 0)\r
493                                                 {\r
494                                                         SaveStr(hKey5, "HostAdrs", Host.HostAdrs, DefaultHost.HostAdrs);\r
495                                                         SaveStr(hKey5, "UserName", Host.UserName, DefaultHost.UserName);\r
496                                                         SaveStr(hKey5, "Account", Host.Account, DefaultHost.Account);\r
497                                                         SaveStr(hKey5, "LocalDir", Host.LocalInitDir, NULL);\r
498                                                         SaveStr(hKey5, "RemoteDir", Host.RemoteInitDir, DefaultHost.RemoteInitDir);\r
499                                                         SaveStr(hKey5, "Chmod", Host.ChmodCmd, DefaultHost.ChmodCmd);\r
500                                                         SaveStr(hKey5, "Nlst", Host.LsName, DefaultHost.LsName);\r
501                                                         SaveStr(hKey5, "Init", Host.InitCmd, DefaultHost.InitCmd);\r
502 \r
503                                                         if(Host.Anonymous == NO)\r
504                                                                 EncodePassword(Host.PassWord, Str);\r
505                                                         else\r
506                                                                 strcpy(Str, DefaultHost.PassWord);\r
507                                                         SaveStr(hKey5, "Password", Str, DefaultHost.PassWord);\r
508 \r
509                                                         SaveIntNum(hKey5, "Port", Host.Port, DefaultHost.Port);\r
510                                                         SaveIntNum(hKey5, "Anonymous", Host.Anonymous, DefaultHost.Anonymous);\r
511                                                         SaveIntNum(hKey5, "Kanji", Host.KanjiCode, DefaultHost.KanjiCode);\r
512                                                         SaveIntNum(hKey5, "KanaCnv", Host.KanaCnv, DefaultHost.KanaCnv);\r
513                                                         SaveIntNum(hKey5, "NameKanji", Host.NameKanjiCode, DefaultHost.NameKanjiCode);\r
514                                                         SaveIntNum(hKey5, "NameKana", Host.NameKanaCnv, DefaultHost.NameKanaCnv);\r
515                                                         SaveIntNum(hKey5, "Pasv", Host.Pasv, DefaultHost.Pasv);\r
516                                                         SaveIntNum(hKey5, "Fwall", Host.FireWall, DefaultHost.FireWall);\r
517                                                         SaveIntNum(hKey5, "List", Host.ListCmdOnly, DefaultHost.ListCmdOnly);\r
518                                                         SaveIntNum(hKey5, "NLST-R", Host.UseNLST_R, DefaultHost.UseNLST_R);\r
519                                                         SaveIntNum(hKey5, "Last", Host.LastDir, DefaultHost.LastDir);\r
520                                                         SaveIntNum(hKey5, "Tzone", Host.TimeZone, DefaultHost.TimeZone);\r
521                                                         SaveIntNum(hKey5, "Type", Host.HostType, DefaultHost.HostType);\r
522                                                         SaveIntNum(hKey5, "Sync", Host.SyncMove, DefaultHost.SyncMove);\r
523                                                         SaveIntNum(hKey5, "Fpath", Host.NoFullPath, DefaultHost.NoFullPath);\r
524                                                         WriteBinaryToReg(hKey5, "Sort", &Host.Sort, sizeof(Host.Sort));\r
525                                                         SaveIntNum(hKey5, "Secu", Host.Security, DefaultHost.Security);\r
526 \r
527                                                         WriteMultiStringToReg(hKey5, "Bmarks", Host.BookMark);\r
528 \r
529                                                         SaveIntNum(hKey5, "Dial", Host.Dialup, DefaultHost.Dialup);\r
530                                                         SaveIntNum(hKey5, "UseIt", Host.DialupAlways, DefaultHost.DialupAlways);\r
531                                                         SaveIntNum(hKey5, "Notify", Host.DialupNotify, DefaultHost.DialupNotify);\r
532                                                         SaveStr(hKey5, "DialTo", Host.DialEntry, DefaultHost.DialEntry);\r
533                                                 }\r
534                                                 CloseSubKey(hKey5);\r
535                                         }\r
536                                         i++;\r
537                                 }\r
538                                 WriteIntValueToReg(hKey4, "SetNum", i);\r
539 \r
540                                 /* 余分なホストの設定があったら削除 */\r
541                                 for(; i < 998; i++)\r
542                                 {\r
543                                         sprintf(Str, "Host%d", i);\r
544                                         if(DeleteSubKey(hKey4, Str) != FFFTP_SUCCESS)\r
545                                                 break;\r
546                                 }\r
547 \r
548                                 if((i = AskCurrentHost()) == HOSTNUM_NOENTRY)\r
549                                         i = 0;\r
550                                 WriteIntValueToReg(hKey4, "CurSet", i);\r
551                         }\r
552                         CloseSubKey(hKey4);\r
553                 }\r
554                 CloseReg(hKey3);\r
555         }\r
556         return;\r
557 }\r
558 \r
559 /*----- レジストリ/INIファイルから設定値を呼び出す ---------------------------\r
560 *\r
561 *       この関数を複数回呼び出すと,ホスト設定が追加される.\r
562 *\r
563 *       Parameter\r
564 *               なし\r
565 *\r
566 *       Return Value\r
567 *               YES: 読み出し成功\r
568 *               NO:  読み出し失敗(設定無し)\r
569 *----------------------------------------------------------------------------*/\r
570 \r
571 int LoadRegistory(void)\r
572 {\r
573         void *hKey3;\r
574         void *hKey4;\r
575         void *hKey5;\r
576         int i;\r
577         int Sets;\r
578         char Str[256];  /* ASCII_EXT_LENより大きい事 */\r
579         char *Pos;\r
580         char *Pos2;\r
581         HOSTDATA Host;\r
582         HISTORYDATA Hist;\r
583         int Sts;\r
584         int Version;\r
585 \r
586         Sts = NO;\r
587 \r
588         SetRegType(REGTYPE_INI);\r
589         if((i = OpenReg("FFFTP", &hKey3)) != FFFTP_SUCCESS)\r
590         {\r
591                 if(AskForceIni() == NO)\r
592                 {\r
593                         SetRegType(REGTYPE_REG);\r
594                         i = OpenReg("FFFTP", &hKey3);\r
595                 }\r
596         }\r
597 \r
598         if(i == FFFTP_SUCCESS)\r
599         {\r
600                 /* char checkbuf[48]; */\r
601                 int salt = 0;\r
602                 Sts = YES;\r
603 \r
604                 ReadIntValueFromReg(hKey3, "Version", &Version);\r
605 \r
606                 if(OpenSubKey(hKey3, "Options", &hKey4) == FFFTP_SUCCESS)\r
607                 {\r
608                         ReadIntValueFromReg(hKey4, "WinPosX", &WinPosX);\r
609                         ReadIntValueFromReg(hKey4, "WinPosY", &WinPosY);\r
610                         ReadIntValueFromReg(hKey4, "WinWidth", &WinWidth);\r
611                         ReadIntValueFromReg(hKey4, "WinHeight", &WinHeight);\r
612                         ReadIntValueFromReg(hKey4, "LocalWidth", &LocalWidth);\r
613                         /* ↓旧バージョンのバグ対策 */\r
614                         LocalWidth = max1(0, LocalWidth);\r
615                         ReadIntValueFromReg(hKey4, "TaskHeight", &TaskHeight);\r
616                         /* ↓旧バージョンのバグ対策 */\r
617                         TaskHeight = max1(0, TaskHeight);\r
618                         ReadBinaryFromReg(hKey4, "LocalColm", &LocalTabWidth, sizeof(LocalTabWidth));\r
619                         ReadBinaryFromReg(hKey4, "RemoteColm", &RemoteTabWidth, sizeof(RemoteTabWidth));\r
620                         ReadIntValueFromReg(hKey4, "SwCmd", &Sizing);\r
621 \r
622                         ReadStringFromReg(hKey4, "UserMail", UserMailAdrs, USER_MAIL_LEN+1);\r
623                         ReadStringFromReg(hKey4, "Viewer", ViewerName[0], FMAX_PATH+1);\r
624                         ReadStringFromReg(hKey4, "Viewer2", ViewerName[1], FMAX_PATH+1);\r
625                         ReadStringFromReg(hKey4, "Viewer3", ViewerName[2], FMAX_PATH+1);\r
626 \r
627                         ReadIntValueFromReg(hKey4, "TrType", &TransMode);\r
628                         ReadIntValueFromReg(hKey4, "Recv", &RecvMode);\r
629                         ReadIntValueFromReg(hKey4, "Send", &SendMode);\r
630                         ReadIntValueFromReg(hKey4, "Move", &MoveMode);\r
631                         ReadStringFromReg(hKey4, "Path", DefaultLocalPath, FMAX_PATH+1);\r
632                         ReadIntValueFromReg(hKey4, "Time", &SaveTimeStamp);\r
633                         ReadIntValueFromReg(hKey4, "EOF", &RmEOF);\r
634                         ReadIntValueFromReg(hKey4, "Scolon", &VaxSemicolon);\r
635 \r
636                         ReadIntValueFromReg(hKey4, "RecvEx", &ExistMode);\r
637                         ReadIntValueFromReg(hKey4, "SendEx", &UpExistMode);\r
638 \r
639                         ReadIntValueFromReg(hKey4, "LFsort", &LocalFileSort);\r
640                         ReadIntValueFromReg(hKey4, "LDsort", &LocalDirSort);\r
641                         ReadIntValueFromReg(hKey4, "RFsort", &RemoteFileSort);\r
642                         ReadIntValueFromReg(hKey4, "RDsort", &RemoteDirSort);\r
643                         ReadIntValueFromReg(hKey4, "SortSave", &SortSave);\r
644 \r
645                         ReadIntValueFromReg(hKey4, "ListType", &ListType);\r
646                         ReadIntValueFromReg(hKey4, "Cache", &CacheEntry);\r
647                         ReadIntValueFromReg(hKey4, "CacheSave", &CacheSave);\r
648                         ReadIntValueFromReg(hKey4, "DotFile", &DotFile);\r
649                         ReadIntValueFromReg(hKey4, "Dclick", &DclickOpen);\r
650 \r
651                         ReadIntValueFromReg(hKey4, "ConS", &ConnectOnStart);\r
652                         ReadIntValueFromReg(hKey4, "OldDlg", &ConnectAndSet);\r
653                         ReadIntValueFromReg(hKey4, "RasClose", &RasClose);\r
654                         ReadIntValueFromReg(hKey4, "RasNotify", &RasCloseNotify);\r
655                         ReadIntValueFromReg(hKey4, "Qanony", &QuickAnonymous);\r
656                         ReadIntValueFromReg(hKey4, "PassHist", &PassToHist);\r
657                         ReadIntValueFromReg(hKey4, "SendQuit", &SendQuit);\r
658                         ReadIntValueFromReg(hKey4, "NoRas", &NoRasControl);\r
659 \r
660                         ReadIntValueFromReg(hKey4, "Debug", &DebugConsole);\r
661                         ReadIntValueFromReg(hKey4, "WinPos", &SaveWinPos);\r
662                         ReadIntValueFromReg(hKey4, "RegExp", &FindMode);\r
663                         ReadIntValueFromReg(hKey4, "Reg", &RegType);\r
664 \r
665                         if(ReadMultiStringFromReg(hKey4, "AsciiFile", AsciiExt, ASCII_EXT_LEN+1) == FFFTP_FAIL)\r
666                         {\r
667                                 /* 旧ASCIIモードの拡張子の設定を新しいものに変換 */\r
668                                 ReadStringFromReg(hKey4, "Ascii", Str, ASCII_EXT_LEN+1);\r
669                                 memset(AsciiExt, NUL, ASCII_EXT_LEN+1);\r
670                                 Pos = Str;\r
671                                 while(*Pos != NUL)\r
672                                 {\r
673                                         if((Pos2 = strchr(Pos, ';')) == NULL)\r
674                                                 Pos2 = strchr(Pos, NUL);\r
675                                         if((Pos2 - Pos) > 0)\r
676                                         {\r
677                                                 if((StrMultiLen(AsciiExt) + (Pos2 - Pos) + 2) >= ASCII_EXT_LEN)\r
678                                                         break;\r
679                                                 strcpy(AsciiExt + StrMultiLen(AsciiExt), "*.");\r
680                                                 strncpy(AsciiExt + StrMultiLen(AsciiExt) - 1, Pos, (Pos2 - Pos));\r
681                                         }\r
682                                         Pos = Pos2;\r
683                                         if(*Pos == ';')\r
684                                                 Pos++;\r
685                                 }\r
686                         }\r
687 \r
688                         ReadIntValueFromReg(hKey4, "LowUp", &FnameCnv);\r
689                         ReadIntValueFromReg(hKey4, "Tout", &TimeOut);\r
690 \r
691                         ReadMultiStringFromReg(hKey4, "NoTrn", MirrorNoTrn, MIRROR_LEN+1);\r
692                         ReadMultiStringFromReg(hKey4, "NoDel", MirrorNoDel, MIRROR_LEN+1);\r
693                         ReadIntValueFromReg(hKey4, "MirFile", &MirrorFnameCnv);\r
694                         ReadIntValueFromReg(hKey4, "MirUNot", &MirUpDelNotify);\r
695                         ReadIntValueFromReg(hKey4, "MirDNot", &MirDownDelNotify);\r
696 \r
697                         if(ReadStringFromReg(hKey4, "ListFont", Str, 256) == FFFTP_SUCCESS)\r
698                         {\r
699                                 if(RestoreFontData(Str, &ListLogFont) == FFFTP_SUCCESS)\r
700                                         ListFont = CreateFontIndirect(&ListLogFont);\r
701                         }\r
702                         ReadIntValueFromReg(hKey4, "ListHide", &DispIgnoreHide);\r
703                         ReadIntValueFromReg(hKey4, "ListDrv", &DispDrives);\r
704 \r
705                         ReadStringFromReg(hKey4, "FwallHost", FwallHost, HOST_ADRS_LEN+1);\r
706                         ReadStringFromReg(hKey4, "FwallUser", FwallUser, USER_NAME_LEN+1);\r
707                         ReadStringFromReg(hKey4, "FwallPass", Str, 255);\r
708                         DecodePassword(Str, FwallPass);\r
709                         ReadIntValueFromReg(hKey4, "FwallPort", &FwallPort);\r
710                         ReadIntValueFromReg(hKey4, "FwallType", &FwallType);\r
711                         ReadIntValueFromReg(hKey4, "FwallDef", &FwallDefault);\r
712                         ReadIntValueFromReg(hKey4, "FwallSec", &FwallSecurity);\r
713                         ReadIntValueFromReg(hKey4, "PasvDef", &PasvDefault);\r
714                         ReadIntValueFromReg(hKey4, "FwallRes", &FwallResolv);\r
715                         ReadIntValueFromReg(hKey4, "FwallLow", &FwallLower);\r
716                         ReadIntValueFromReg(hKey4, "FwallDel", &FwallDelimiter);\r
717 \r
718                         ReadIntValueFromReg(hKey4, "SndConSw", &Sound[SND_CONNECT].On);\r
719                         ReadIntValueFromReg(hKey4, "SndTrnSw", &Sound[SND_TRANS].On);\r
720                         ReadIntValueFromReg(hKey4, "SndErrSw", &Sound[SND_ERROR].On);\r
721                         ReadStringFromReg(hKey4, "SndCon", Sound[SND_CONNECT].Fname, FMAX_PATH+1);\r
722                         ReadStringFromReg(hKey4, "SndTrn", Sound[SND_TRANS].Fname, FMAX_PATH+1);\r
723                         ReadStringFromReg(hKey4, "SndErr", Sound[SND_ERROR].Fname, FMAX_PATH+1);\r
724 \r
725                         ReadMultiStringFromReg(hKey4, "DefAttr", DefAttrList, DEFATTRLIST_LEN+1);\r
726 \r
727                         ReadStringFromReg(hKey4, "Tmp", TmpPath, FMAX_PATH+1);\r
728 \r
729                         ReadBinaryFromReg(hKey4, "Hdlg", &HostDlgSize, sizeof(SIZE));\r
730                         ReadBinaryFromReg(hKey4, "Bdlg", &BmarkDlgSize, sizeof(SIZE));\r
731                         ReadBinaryFromReg(hKey4, "Mdlg", &MirrorDlgSize, sizeof(SIZE));\r
732 \r
733                         ReadIntValueFromReg(hKey4, "FAttrSw", &FolderAttr);\r
734                         ReadIntValueFromReg(hKey4, "FAttr", &FolderAttrNum);\r
735 \r
736                         ReadIntValueFromReg(hKey4, "NoSave", &SuppressSave);\r
737 \r
738                         ReadIntValueFromReg(hKey4, "HistNum", &FileHist);\r
739 //                      ReadMultiStringFromReg(hKey4, "Hist", Hist, (FMAX_PATH+1)*HISTORY_MAX+1);\r
740 \r
741                         /* ヒストリの設定を読み込む */\r
742                         Sets = 0;\r
743                         ReadIntValueFromReg(hKey4, "SavedHist", &Sets);\r
744 \r
745                         for(i = 0; i < Sets; i++)\r
746                         {\r
747                                 sprintf(Str, "History%d", i);\r
748                                 if(OpenSubKey(hKey4, Str, &hKey5) == FFFTP_SUCCESS)\r
749                                 {\r
750                                         CopyDefaultHistory(&Hist);\r
751 \r
752                                         ReadStringFromReg(hKey5, "HostAdrs", Hist.HostAdrs, HOST_ADRS_LEN+1);\r
753                                         ReadStringFromReg(hKey5, "UserName", Hist.UserName, USER_NAME_LEN+1);\r
754                                         ReadStringFromReg(hKey5, "Account", Hist.Account, ACCOUNT_LEN+1);\r
755                                         ReadStringFromReg(hKey5, "LocalDir", Hist.LocalInitDir, INIT_DIR_LEN+1);\r
756                                         ReadStringFromReg(hKey5, "RemoteDir", Hist.RemoteInitDir, INIT_DIR_LEN+1);\r
757                                         ReadStringFromReg(hKey5, "Chmod", Hist.ChmodCmd, CHMOD_CMD_LEN+1);\r
758                                         ReadStringFromReg(hKey5, "Nlst", Hist.LsName, NLST_NAME_LEN+1);\r
759                                         ReadStringFromReg(hKey5, "Init", Hist.InitCmd, INITCMD_LEN+1);\r
760                                         ReadIntValueFromReg(hKey5, "Port", &Hist.Port);\r
761                                         ReadIntValueFromReg(hKey5, "Kanji", &Hist.KanjiCode);\r
762                                         ReadIntValueFromReg(hKey5, "KanaCnv", &Hist.KanaCnv);\r
763                                         ReadIntValueFromReg(hKey5, "NameKanji", &Hist.NameKanjiCode);\r
764                                         ReadIntValueFromReg(hKey5, "NameKana", &Hist.NameKanaCnv);\r
765                                         ReadIntValueFromReg(hKey5, "Pasv", &Hist.Pasv);\r
766                                         ReadIntValueFromReg(hKey5, "Fwall", &Hist.FireWall);\r
767                                         ReadIntValueFromReg(hKey5, "List", &Hist.ListCmdOnly);\r
768                                         ReadIntValueFromReg(hKey5, "NLST-R", &Hist.UseNLST_R);\r
769                                         ReadIntValueFromReg(hKey5, "Tzone", &Hist.TimeZone);\r
770                                         ReadIntValueFromReg(hKey5, "Type", &Hist.HostType);\r
771                                         ReadIntValueFromReg(hKey5, "Sync", &Hist.SyncMove);\r
772                                         ReadIntValueFromReg(hKey5, "Fpath", &Hist.NoFullPath);\r
773                                         ReadBinaryFromReg(hKey5, "Sort", &Hist.Sort, sizeof(Hist.Sort));\r
774                                         ReadIntValueFromReg(hKey5, "Secu", &Hist.Security);\r
775                                         ReadIntValueFromReg(hKey5, "TrType", &Hist.Type);\r
776                                         strcpy(Str, "");\r
777                                         ReadStringFromReg(hKey5, "Password", Str, 255);\r
778                                         DecodePassword(Str, Hist.PassWord);\r
779                                         ReadIntValueFromReg(hKey5, "Dial", &Hist.Dialup);\r
780                                         ReadIntValueFromReg(hKey5, "UseIt", &Hist.DialupAlways);\r
781                                         ReadIntValueFromReg(hKey5, "Notify", &Hist.DialupNotify);\r
782                                         ReadStringFromReg(hKey5, "DialTo", Hist.DialEntry, RAS_NAME_LEN+1);\r
783 \r
784                                         CloseSubKey(hKey5);\r
785                                         AddHistoryToHistory(&Hist);\r
786                                 }\r
787                         }\r
788 \r
789                         /* ホストの設定を読み込む */\r
790                         Sets = 0;\r
791                         ReadIntValueFromReg(hKey4, "SetNum", &Sets);\r
792 \r
793                         for(i = 0; i < Sets; i++)\r
794                         {\r
795                                 sprintf(Str, "Host%d", i);\r
796                                 if(OpenSubKey(hKey4, Str, &hKey5) == FFFTP_SUCCESS)\r
797                                 {\r
798                                         CopyDefaultHost(&Host);\r
799                                         /* 下位互換性のため */\r
800                                         if(Version < VER_NUM)\r
801                                         {\r
802                                                 Host.Pasv = NO;\r
803                                                 Host.ListCmdOnly = NO;\r
804                                         }\r
805                                         ReadIntValueFromReg(hKey5, "Set", &Host.Level);\r
806 \r
807                                         ReadStringFromReg(hKey5, "HostName", Host.HostName, HOST_NAME_LEN+1);\r
808                                         ReadStringFromReg(hKey5, "HostAdrs", Host.HostAdrs, HOST_ADRS_LEN+1);\r
809                                         ReadStringFromReg(hKey5, "UserName", Host.UserName, USER_NAME_LEN+1);\r
810                                         ReadStringFromReg(hKey5, "Account", Host.Account, ACCOUNT_LEN+1);\r
811                                         ReadStringFromReg(hKey5, "LocalDir", Host.LocalInitDir, INIT_DIR_LEN+1);\r
812                                         ReadStringFromReg(hKey5, "RemoteDir", Host.RemoteInitDir, INIT_DIR_LEN+1);\r
813                                         ReadStringFromReg(hKey5, "Chmod", Host.ChmodCmd, CHMOD_CMD_LEN+1);\r
814                                         ReadStringFromReg(hKey5, "Nlst", Host.LsName, NLST_NAME_LEN+1);\r
815                                         ReadStringFromReg(hKey5, "Init", Host.InitCmd, INITCMD_LEN+1);\r
816                                         ReadIntValueFromReg(hKey5, "Port", &Host.Port);\r
817                                         ReadIntValueFromReg(hKey5, "Anonymous", &Host.Anonymous);\r
818                                         ReadIntValueFromReg(hKey5, "Kanji", &Host.KanjiCode);\r
819                                         ReadIntValueFromReg(hKey5, "KanaCnv", &Host.KanaCnv);\r
820                                         ReadIntValueFromReg(hKey5, "NameKanji", &Host.NameKanjiCode);\r
821                                         ReadIntValueFromReg(hKey5, "NameKana", &Host.NameKanaCnv);\r
822                                         ReadIntValueFromReg(hKey5, "Pasv", &Host.Pasv);\r
823                                         ReadIntValueFromReg(hKey5, "Fwall", &Host.FireWall);\r
824                                         ReadIntValueFromReg(hKey5, "List", &Host.ListCmdOnly);\r
825                                         ReadIntValueFromReg(hKey5, "NLST-R", &Host.UseNLST_R);\r
826                                         ReadIntValueFromReg(hKey5, "Last", &Host.LastDir);\r
827                                         ReadIntValueFromReg(hKey5, "Tzone", &Host.TimeZone);\r
828                                         ReadIntValueFromReg(hKey5, "Type", &Host.HostType);\r
829                                         ReadIntValueFromReg(hKey5, "Sync", &Host.SyncMove);\r
830                                         ReadIntValueFromReg(hKey5, "Fpath", &Host.NoFullPath);\r
831                                         ReadBinaryFromReg(hKey5, "Sort", &Host.Sort, sizeof(Host.Sort));\r
832                                         ReadIntValueFromReg(hKey5, "Secu", &Host.Security);\r
833                                         if(Host.Anonymous != YES)\r
834                                         {\r
835                                                 strcpy(Str, "");\r
836                                                 ReadStringFromReg(hKey5, "Password", Str, 255);\r
837                                                 DecodePassword(Str, Host.PassWord);\r
838                                         }\r
839                                         else\r
840                                                 strcpy(Host.PassWord, UserMailAdrs);\r
841 \r
842                                         ReadMultiStringFromReg(hKey5, "Bmarks", Host.BookMark, BOOKMARK_SIZE);\r
843 \r
844                                         ReadIntValueFromReg(hKey5, "Dial", &Host.Dialup);\r
845                                         ReadIntValueFromReg(hKey5, "UseIt", &Host.DialupAlways);\r
846                                         ReadIntValueFromReg(hKey5, "Notify", &Host.DialupNotify);\r
847                                         ReadStringFromReg(hKey5, "DialTo", Host.DialEntry, RAS_NAME_LEN+1);\r
848 \r
849                                         CloseSubKey(hKey5);\r
850 \r
851                                         AddHostToList(&Host, -1, Host.Level);\r
852                                 }\r
853                         }\r
854 \r
855                         ReadIntValueFromReg(hKey4, "CurSet", &Sets);\r
856                         SetCurrentHost(Sets);\r
857 \r
858                         CloseSubKey(hKey4);\r
859                 }\r
860                 CloseReg(hKey3);\r
861         }\r
862         else\r
863         {\r
864                 /*===== 最初の起動時(設定が無い) =====*/\r
865 \r
866 #if 0\r
867                 strcpy(UserMailAdrs, "");\r
868                 strcpy(Str, "");\r
869                 if(InputDialogBox(mailadrs_dlg, HWND_DESKTOP, NULL, Str, USER_MAIL_LEN+1, &i, IDH_HELP_TOPIC_0000001) == YES)\r
870                         strcpy(UserMailAdrs, Str);\r
871 \r
872                 for(i = 0; i < SAMPLE_HOSTS; i++)\r
873                 {\r
874                         CopyDefaultHost(&Host);\r
875                         Host.Level = Sample[i].Level;\r
876                         strcpy(Host.PassWord, UserMailAdrs);\r
877                         strcpy(Host.HostName, Sample[i].HostName);\r
878                         strcpy(Host.HostAdrs, Sample[i].HostAdrs);\r
879                         strcpy(Host.UserName, "anonymous");\r
880                         AddHostToList(&Host, -1, Host.Level);\r
881                 }\r
882 #endif\r
883         }\r
884         return(Sts);\r
885 }\r
886 \r
887 \r
888 /*----- 隠しドライブ情報を取得 ------------------------------------------------\r
889 *\r
890 *       Parameter\r
891 *               なし\r
892 *\r
893 *       Return Value\r
894 *               DWORD \r
895 *                       YES/NO=設定無し\r
896 *----------------------------------------------------------------------------*/\r
897 \r
898 DWORD LoadHideDriveListRegistory(void)\r
899 {\r
900         HKEY hKey1;\r
901         HKEY hKey2;\r
902         HKEY hKey3;\r
903         HKEY hKey4;\r
904         HKEY hKey5;\r
905         HKEY hKey6;\r
906         DWORD Size;\r
907         DWORD Type;\r
908         DWORD Ret;\r
909 \r
910         Ret = 0;\r
911         if(RegOpenKeyEx(HKEY_CURRENT_USER, "Software", 0, KEY_READ, &hKey1) == ERROR_SUCCESS)\r
912         {\r
913                 if(RegOpenKeyEx(hKey1, "Microsoft", 0, KEY_READ, &hKey2) == ERROR_SUCCESS)\r
914                 {\r
915                         if(RegOpenKeyEx(hKey2, "Windows", 0, KEY_READ, &hKey3) == ERROR_SUCCESS)\r
916                         {\r
917                                 if(RegOpenKeyEx(hKey3, "CurrentVersion", 0, KEY_READ, &hKey4) == ERROR_SUCCESS)\r
918                                 {\r
919                                         if(RegOpenKeyEx(hKey4, "Policies", 0, KEY_READ, &hKey5) == ERROR_SUCCESS)\r
920                                         {\r
921                                                 if(RegOpenKeyEx(hKey5, "Explorer", 0, KEY_READ, &hKey6) == ERROR_SUCCESS)\r
922                                                 {\r
923                                                         Size = sizeof(DWORD);\r
924                                                         RegQueryValueEx(hKey6, "NoDrives", NULL, &Type, (BYTE *)&Ret, &Size);\r
925                                                         RegCloseKey(hKey6);\r
926                                                 }\r
927                                                 RegCloseKey(hKey5);\r
928                                         }\r
929                                         RegCloseKey(hKey4);\r
930                                 }\r
931                                 RegCloseKey(hKey3);\r
932                         }\r
933                         RegCloseKey(hKey2);\r
934                 }\r
935                 RegCloseKey(hKey1);\r
936         }\r
937         return(Ret);\r
938 }\r
939 \r
940 \r
941 /*----- レジストリの設定値をクリア --------------------------------------------\r
942 *\r
943 *       Parameter\r
944 *               なし\r
945 *\r
946 *       Return Value\r
947 *               なし\r
948 *----------------------------------------------------------------------------*/\r
949 \r
950 void ClearRegistory(void)\r
951 {\r
952         HKEY hKey2;\r
953         HKEY hKey3;\r
954         HKEY hKey4;\r
955         DWORD Dispos;\r
956         char Str[20];\r
957         int i;\r
958 \r
959         if(RegCreateKeyEx(HKEY_CURRENT_USER, "Software\\Sota", 0, "", REG_OPTION_NON_VOLATILE, KEY_CREATE_SUB_KEY, NULL, &hKey2, &Dispos) == ERROR_SUCCESS)\r
960         {\r
961                 if(RegCreateKeyEx(hKey2, "FFFTP", 0, "", REG_OPTION_NON_VOLATILE, KEY_WRITE, NULL, &hKey3, &Dispos) == ERROR_SUCCESS)\r
962                 {\r
963                         if(RegCreateKeyEx(hKey3, "Options", 0, "", REG_OPTION_NON_VOLATILE, KEY_WRITE, NULL, &hKey4, &Dispos) == ERROR_SUCCESS)\r
964                         {\r
965                                 for(i = 0; ; i++)\r
966                                 {\r
967                                         sprintf(Str, "Host%d", i);\r
968                                         if(RegDeleteKey(hKey4, Str) != ERROR_SUCCESS)\r
969                                                 break;\r
970                                 }\r
971                                 for(i = 0; ; i++)\r
972                                 {\r
973                                         sprintf(Str, "History%d", i);\r
974                                         if(RegDeleteKey(hKey4, Str) != ERROR_SUCCESS)\r
975                                                 break;\r
976                                 }\r
977                                 RegCloseKey(hKey4);\r
978                         }\r
979                         RegDeleteKey(hKey3, "Options");\r
980                         RegCloseKey(hKey3);\r
981                 }\r
982                 RegDeleteKey(hKey2, "FFFTP");\r
983                 RegCloseKey(hKey2);\r
984         }\r
985         return;\r
986 }\r
987 \r
988 \r
989 /*----- 設定をファイルに保存 --------------------------------------------------\r
990 *\r
991 *       Parameter\r
992 *               なし\r
993 *\r
994 *       Return Value\r
995 *               なし\r
996 *----------------------------------------------------------------------------*/\r
997 \r
998 void SaveSettingsToFile(void)\r
999 {\r
1000         char Tmp[FMAX_PATH*2];\r
1001         char Fname[FMAX_PATH+1];\r
1002 \r
1003         if(RegType == REGTYPE_REG)\r
1004         {\r
1005                 strcpy(Fname, "FFFTP.reg");\r
1006                 if(SelectFile(GetMainHwnd(), Fname, MSGJPN286, MSGJPN287, "reg", OFN_EXTENSIONDIFFERENT | OFN_OVERWRITEPROMPT, 1) == TRUE)\r
1007                 {\r
1008                         sprintf(Tmp, "/e \x22%s\x22 HKEY_CURRENT_USER\\Software\\sota\\FFFTP", Fname);\r
1009                         if(ShellExecute(NULL, "open", "regedit", Tmp, ".", SW_SHOW) <= (HINSTANCE)32)\r
1010                         {\r
1011                                 MessageBox(NULL, MSGJPN285, "FFFTP", MB_OK);\r
1012                         }\r
1013                 }\r
1014         }\r
1015         else\r
1016         {\r
1017                 strcpy(Fname, "FFFTP-Backup.ini");\r
1018                 if(SelectFile(GetMainHwnd(), Fname, MSGJPN286, MSGJPN288, "ini", OFN_EXTENSIONDIFFERENT | OFN_OVERWRITEPROMPT, 1) == TRUE)\r
1019                 {\r
1020                         CopyFile(AskIniFilePath(), Fname, FALSE);\r
1021                 }\r
1022         }\r
1023         return;\r
1024 }\r
1025 \r
1026 \r
1027 /*----- 設定をファイルから復元 ------------------------------------------------\r
1028 *\r
1029 *       Parameter\r
1030 *               なし\r
1031 *\r
1032 *       Return Value\r
1033 *               int     ロードしたかどうか (YES/NO)\r
1034 *----------------------------------------------------------------------------*/\r
1035 \r
1036 int LoadSettingsFromFile(void)\r
1037 {\r
1038         int Ret;\r
1039         char Tmp[FMAX_PATH*2];\r
1040         char Fname[FMAX_PATH+1];\r
1041 \r
1042         Ret = NO;\r
1043         strcpy(Fname, "");\r
1044         if(SelectFile(GetMainHwnd(), Fname, MSGJPN291, MSGJPN290, "", OFN_FILEMUSTEXIST, 0) == TRUE)\r
1045         {\r
1046                 if((strlen(Fname) >= 5) && (_stricmp(&Fname[strlen(Fname)-4], ".reg") == 0))\r
1047                 {\r
1048                         sprintf(Tmp, "\x22%s\x22", Fname);\r
1049                         if(ShellExecute(NULL, "open", "regedit", Tmp, ".", SW_SHOW) <= (HINSTANCE)32)\r
1050                         {\r
1051                                 MessageBox(NULL, MSGJPN285, "FFFTP", MB_OK);\r
1052                         }\r
1053                         else\r
1054                         {\r
1055                                 Ret = YES;\r
1056                                 /* レジストリエディタが終了するのを待つ */\r
1057 //                              WaitForSingleObject(Info.hProcess, INFINITE);\r
1058                         }\r
1059                 }\r
1060                 else if((strlen(Fname) >= 5) && (_stricmp(&Fname[strlen(Fname)-4], ".ini") == 0))\r
1061                 {\r
1062                         CopyFile(Fname, AskIniFilePath(), FALSE);\r
1063                         Ret = YES;\r
1064                 }\r
1065                 else\r
1066                         MessageBox(NULL, MSGJPN293, "FFFTP", MB_OK);\r
1067         }\r
1068         return(Ret);\r
1069 }\r
1070 \r
1071 \r
1072 \r
1073 \r
1074 /*----- レジストリ/INIファイルに文字列をセーブ --------------------------------\r
1075 *\r
1076 *       Parameter\r
1077 *               HKEY hKey : レジストリキー\r
1078 *               char *Key : キー名\r
1079 *               char *Str : セーブする文字列\r
1080 *               char *DefaultStr : デフォルトの文字列\r
1081 *\r
1082 *       Return Value\r
1083 *               なし\r
1084 *\r
1085 *       Note\r
1086 *               文字列がデフォルトの文字列と同じならセーブしない\r
1087 *----------------------------------------------------------------------------*/\r
1088 \r
1089 static void SaveStr(HKEY hKey, char *Key, char *Str, char *DefaultStr)\r
1090 {\r
1091         if((DefaultStr != NULL) && (strcmp(Str, DefaultStr) == 0))\r
1092                 DeleteValue(hKey, Key);\r
1093         else\r
1094                 WriteStringToReg(hKey, Key, Str);\r
1095 \r
1096         return;\r
1097 }\r
1098 \r
1099 \r
1100 /*----- レジストリ/INIファイルに数値(INT)をセーブ -----------------------------\r
1101 *\r
1102 *       Parameter\r
1103 *               HKEY hKey : レジストリキー\r
1104 *               char *Key : キー名\r
1105 *               int Num : セーブする値\r
1106 *               int DefaultNum : デフォルトの値\r
1107 *\r
1108 *       Return Value\r
1109 *               なし\r
1110 *\r
1111 *       Note\r
1112 *               数値がデフォルトの値と同じならセーブしない\r
1113 *----------------------------------------------------------------------------*/\r
1114 \r
1115 static void SaveIntNum(HKEY hKey, char *Key, int Num, int DefaultNum)\r
1116 {\r
1117         if(Num == DefaultNum)\r
1118                 DeleteValue(hKey, Key);\r
1119         else\r
1120                 WriteIntValueToReg(hKey, Key, Num);\r
1121 \r
1122         return;\r
1123 }\r
1124 \r
1125 \r
1126 /*----- LOGFONTデータを文字列に変換する ---------------------------------------\r
1127 *\r
1128 *       Parameter\r
1129 *               LOGFONT Font : フォントデータ\r
1130 *               HFONT hFont : フォントのハンドル\r
1131 *                       NULL = デフォルトのフォント\r
1132 *               char *Buf : バッファ\r
1133 *\r
1134 *       Return Value\r
1135 *               なし\r
1136 *----------------------------------------------------------------------------*/\r
1137 \r
1138 static void MakeFontData(LOGFONT Font, HFONT hFont, char *Buf)\r
1139 {\r
1140         *Buf = NUL;\r
1141         if(hFont != NULL)\r
1142                 sprintf(Buf, "%d %d %d %d %d %d %d %d %d %d %d %d %d %s",\r
1143                         Font.lfHeight, Font.lfWidth, Font.lfEscapement, Font.lfOrientation,\r
1144                         Font.lfWeight, Font.lfItalic, Font.lfUnderline, Font.lfStrikeOut,\r
1145                         Font.lfCharSet, Font.lfOutPrecision, Font.lfClipPrecision,\r
1146                         Font.lfQuality, Font.lfPitchAndFamily, Font.lfFaceName);\r
1147         return;\r
1148 }\r
1149 \r
1150 \r
1151 /*----- 文字列をLOGFONTデータに変換する ---------------------------------------\r
1152 *\r
1153 *       Parameter\r
1154 *               char *Str : 文字列\r
1155 *               LOGFONT *Font : フォントデータ\r
1156 *\r
1157 *       Return Value\r
1158 *               int ステータス\r
1159 *                       FFFTP_SUCCESS/FFFTP_FAIL=変換できない\r
1160 *----------------------------------------------------------------------------*/\r
1161 \r
1162 static int RestoreFontData(char *Str, LOGFONT *Font)\r
1163 {\r
1164         int i;\r
1165         int Sts;\r
1166 \r
1167         Sts = FFFTP_FAIL;\r
1168         if(sscanf(Str, "%d %d %d %d %d %d %d %d %d %d %d %d %d",\r
1169                         &(Font->lfHeight), &(Font->lfWidth), &(Font->lfEscapement), &(Font->lfOrientation),\r
1170                         &(Font->lfWeight), &(Font->lfItalic), &(Font->lfUnderline), &(Font->lfStrikeOut),\r
1171                         &(Font->lfCharSet), &(Font->lfOutPrecision), &(Font->lfClipPrecision),\r
1172                         &(Font->lfQuality), &(Font->lfPitchAndFamily)) == 13)\r
1173         {\r
1174                 for(i = 13; i > 0; i--)\r
1175                 {\r
1176                         if((Str = strchr(Str, ' ')) == NULL)\r
1177                                 break;\r
1178                         Str++;\r
1179                 }\r
1180                 if(i == 0)\r
1181                 {\r
1182                         strcpy(Font->lfFaceName, Str);\r
1183                         Sts = FFFTP_SUCCESS;\r
1184                 }\r
1185         }\r
1186 \r
1187         if(Sts == FFFTP_FAIL)\r
1188                 memset(Font, NUL, sizeof(LOGFONT));\r
1189 \r
1190         return(Sts);\r
1191 }\r
1192 \r
1193 /*----- パスワードを暗号化する ------------------------------------------------\r
1194 *\r
1195 *       Parameter\r
1196 *               char *Str : パスワード\r
1197 *               char *Buf : 暗号化したパスワードを格納するバッファ\r
1198 *\r
1199 *       Return Value\r
1200 *               なし\r
1201 *----------------------------------------------------------------------------*/\r
1202 static void EncodePassword(char *Str, char *Buf)\r
1203 {\r
1204         EncodePassword3( Str, Buf, SecretKey );\r
1205 }\r
1206 \r
1207 /*----- パスワードを暗号化する(オリジナルアルゴリズム)  ------------------\r
1208 *\r
1209 *       Parameter\r
1210 *               char *Str : パスワード\r
1211 *               char *Buf : 暗号化したパスワードを格納するバッファ\r
1212 *\r
1213 *       Return Value\r
1214 *               なし\r
1215 *----------------------------------------------------------------------------*/\r
1216 \r
1217 static void EncodePasswordOriginal(char *Str, char *Buf)\r
1218 {\r
1219         unsigned char *Get;\r
1220         unsigned char *Put;\r
1221         int Rnd;\r
1222         int Ch;\r
1223 \r
1224         srand((unsigned)time(NULL));\r
1225 \r
1226         Get = (unsigned char *)Str;\r
1227         Put = (unsigned char *)Buf;\r
1228         \r
1229         if( *Get == NUL ){\r
1230                 *Put = NUL;\r
1231                 return;\r
1232         }\r
1233 \r
1234         /* 識別子を先頭に置く */\r
1235         Put[0] = '0';\r
1236         Put[1] = 'A';\r
1237         Put += 2;\r
1238 \r
1239         while(*Get != NUL)\r
1240         {\r
1241                 Rnd = rand() % 3;\r
1242                 Ch = ((int)*Get++) << Rnd;\r
1243                 Ch = (unsigned char)Ch | (unsigned char)(Ch >> 8);\r
1244                 *Put++ = 0x40 | ((Rnd & 0x3) << 4) | (Ch & 0xF);\r
1245                 *Put++ = 0x40 | ((Ch >> 4) & 0xF);\r
1246                 if((*(Put-2) & 0x1) != 0)\r
1247                         *Put++ = (rand() % 62) + 0x40;\r
1248         }\r
1249         *Put = NUL;\r
1250         return;\r
1251 }\r
1252 \r
1253 /*----- パスワードを暗号化する(オリジナルアルゴリズム^Key)  ----------------\r
1254 *\r
1255 *       Parameter\r
1256 *               char *Str : パスワード\r
1257 *               char *Buf : 暗号化したパスワードを格納するバッファ\r
1258 *               const char *Key : 暗号化キー\r
1259 *\r
1260 *       Return Value\r
1261 *               なし\r
1262 *----------------------------------------------------------------------------*/\r
1263 \r
1264 static void EncodePassword2(char *Str, char *Buf, const char* Key)\r
1265 {\r
1266         unsigned char *Get;\r
1267         unsigned char *Put;\r
1268         int Rnd;\r
1269         int Ch;\r
1270 \r
1271         /* 2010.01.31 genta Key */\r
1272         unsigned char *KeyHead = (unsigned char *)Key;\r
1273         unsigned char *KeyEnd = KeyHead + strlen(KeyHead);\r
1274         unsigned char *KeyCurrent = KeyHead;\r
1275 \r
1276         srand((unsigned)time(NULL));\r
1277 \r
1278         Get = (unsigned char *)Str;\r
1279         Put = (unsigned char *)Buf;\r
1280         \r
1281         if( *Get == NUL ){\r
1282                 *Put = NUL;\r
1283                 return;\r
1284         }\r
1285 \r
1286         /* 識別子を先頭に置く */\r
1287         Put[0] = '0';\r
1288         Put[1] = 'B';\r
1289         Put += 2;\r
1290 \r
1291         while(*Get != NUL)\r
1292         {\r
1293                 Rnd = rand() % 3;\r
1294                 Ch = ((int)(*Get++ ^ *KeyCurrent)) << Rnd;\r
1295                 Ch = (unsigned char)Ch | (unsigned char)(Ch >> 8);\r
1296                 *Put++ = 0x40 | ((Rnd & 0x3) << 4) | (Ch & 0xF);\r
1297                 *Put++ = 0x40 | ((Ch >> 4) & 0xF);\r
1298                 if((*(Put-2) & 0x1) != 0)\r
1299                         *Put++ = (rand() % 62) + 0x40;\r
1300 \r
1301                 /* 2010.01.31 genta Key */\r
1302                 if( ++KeyCurrent == KeyEnd ){\r
1303                         KeyCurrent = KeyHead;\r
1304                 }\r
1305         }\r
1306         *Put = NUL;\r
1307         return;\r
1308 }\r
1309 \r
1310 /*----- パスワードを暗号化する(AES) ------------------------------------------\r
1311 *\r
1312 *       Parameter\r
1313 *               char *Str : パスワード\r
1314 *               char *Buf : 暗号化したパスワードを格納するバッファ\r
1315 *               const char *Key : 暗号化キー\r
1316 *\r
1317 *       Return Value\r
1318 *               なし\r
1319 *----------------------------------------------------------------------------*/\r
1320 \r
1321 static void EncodePassword3(char *Str, char *Buf, const char *Key)\r
1322 {\r
1323         unsigned char *Put;\r
1324         unsigned char *AesEncBuf;\r
1325         unsigned char *StrPadBuf;\r
1326         size_t StrLen;\r
1327         size_t StrPadLen;\r
1328         size_t StrPadIndex;\r
1329         size_t IvIndex;\r
1330         size_t EncBufIndex;\r
1331         DWORD RandValue;\r
1332         int RandByteCount;\r
1333         unsigned char AesKey[32];\r
1334         unsigned char AesCbcIv[AES_BLOCK_SIZE];\r
1335         aes_encrypt_ctx Ctx;\r
1336 \r
1337         Buf[0] = NUL;\r
1338 \r
1339         Put = (unsigned char *)Buf;\r
1340         StrLen = strlen(Str);\r
1341         StrPadLen = ((StrLen + AES_BLOCK_SIZE - 1) / AES_BLOCK_SIZE) * AES_BLOCK_SIZE;\r
1342 \r
1343         /* 最低長を32文字とする */\r
1344 //      StrPadLen = min1(StrPadLen, AES_BLOCK_SIZE * 2);\r
1345         StrPadLen = max1(StrPadLen, AES_BLOCK_SIZE * 2);\r
1346 \r
1347         if((StrPadBuf = malloc(StrPadLen)) != NULL)\r
1348         {\r
1349                 if((AesEncBuf = malloc(StrPadLen)) != NULL)\r
1350                 {\r
1351                         BOOL PutState;\r
1352 \r
1353                         PutState = FALSE;\r
1354                         strncpy(StrPadBuf, Str, StrPadLen);\r
1355                         \r
1356                         /* PAD部分を乱数で埋める StrPad[StrLen](が有効な場合) は NUL */\r
1357                         for(StrPadIndex = StrLen + 1; StrPadIndex < StrPadLen;)\r
1358                         {\r
1359                                 RandValue = GetRandamDWRODValue();\r
1360                                 for(RandByteCount = 0; RandByteCount < 4; RandByteCount++)\r
1361                                 {\r
1362                                         if(StrPadIndex < StrPadLen)\r
1363                                         {\r
1364                                                 StrPadBuf[StrPadIndex++] = (unsigned char)(RandValue & 0xff);\r
1365                                                 RandValue >>= 8;\r
1366                                         }\r
1367                                 }\r
1368                         }\r
1369                         \r
1370                         // IVの初期化\r
1371                         for(IvIndex = 0; IvIndex < AES_BLOCK_SIZE;)\r
1372                         {\r
1373                                 RandValue = GetRandamDWRODValue();\r
1374                                 for(RandByteCount = 0; RandByteCount < 4; RandByteCount++)\r
1375                                 {\r
1376                                         if(IvIndex < AES_BLOCK_SIZE)\r
1377                                         {\r
1378                                                 AesCbcIv[IvIndex++] = (unsigned char)(RandValue & 0xff);\r
1379                                                 RandValue >>= 8;\r
1380                                         }\r
1381                                 }\r
1382                         }\r
1383                         Put[0] = '0';\r
1384                         Put[1] = 'C';\r
1385                         Put += 2;\r
1386                         for(IvIndex = 0; IvIndex < AES_BLOCK_SIZE; IvIndex++)\r
1387                         {\r
1388                                 sprintf(Put, "%02x", AesCbcIv[IvIndex]);\r
1389                                 Put += 2;\r
1390                         }\r
1391                         *Put++ = ':';\r
1392 \r
1393                         if(CreateAesKey(AesKey, Key) == FFFTP_SUCCESS)\r
1394                         {\r
1395                                 aes_encrypt_key(AesKey, 32, &Ctx);\r
1396 \r
1397                                 if(aes_cbc_encrypt(StrPadBuf, AesEncBuf, StrPadLen, AesCbcIv, &Ctx) == EXIT_SUCCESS)\r
1398                                 {\r
1399                                         for(EncBufIndex = 0; EncBufIndex < StrPadLen; EncBufIndex++)\r
1400                                         {\r
1401                                                 sprintf(Put, "%02x", AesEncBuf[EncBufIndex]);\r
1402                                                 Put += 2;\r
1403                                         }\r
1404                                         *Put = NUL;\r
1405                                         PutState = TRUE;\r
1406                                 }\r
1407                         }\r
1408                         if(FALSE == PutState)\r
1409                         {\r
1410                                 Buf[0] = NUL;\r
1411                         }\r
1412                         free(StrPadBuf);\r
1413                 }\r
1414                 free(AesEncBuf);\r
1415         }\r
1416         return;\r
1417 }\r
1418 \r
1419 \r
1420 /*----- パスワードの暗号化を解く ----------------------------------------------\r
1421 *\r
1422 *       Parameter\r
1423 *               char *Str : 暗号化したパスワード\r
1424 *               char *Buf : パスワードを格納するバッファ\r
1425 *\r
1426 *       Return Value\r
1427 *               なし\r
1428 *----------------------------------------------------------------------------*/\r
1429 \r
1430 static void DecodePassword(char *Str, char *Buf)\r
1431 {\r
1432         unsigned char *Get;\r
1433         unsigned char *Put;\r
1434 \r
1435         Get = (unsigned char *)Str;\r
1436         Put = (unsigned char *)Buf;\r
1437         \r
1438         if( *Get == NUL ){\r
1439                 *Put = NUL;\r
1440         }\r
1441         else if( 0x40 <= *Get && *Get < 0x80 ){\r
1442                 /* Original algorithm */\r
1443                 DecodePasswordOriginal( Str, Buf );\r
1444         }\r
1445         else if( strncmp( Get, "0A", 2 ) == 0 ){\r
1446                 DecodePasswordOriginal( Str + 2, Buf );\r
1447         }\r
1448         else if( strncmp( Get, "0B", 2 ) == 0 ){\r
1449                 DecodePassword2( Str + 2, Buf, SecretKey );\r
1450         }\r
1451         else if( strncmp( Get, "0C", 2 ) == 0 ){\r
1452                 DecodePassword3( Str + 2, Buf, SecretKey );\r
1453         }\r
1454         else {\r
1455                 //      unknown encoding\r
1456                 *Put = NUL;\r
1457                 return;\r
1458         }\r
1459 }\r
1460 \r
1461 /*----- パスワードの暗号化を解く(オリジナルアルゴリズム) -------------------\r
1462 *\r
1463 *       Parameter\r
1464 *               char *Str : 暗号化したパスワード\r
1465 *               char *Buf : パスワードを格納するバッファ\r
1466 *\r
1467 *       Return Value\r
1468 *               なし\r
1469 *----------------------------------------------------------------------------*/\r
1470 static void DecodePasswordOriginal(char *Str, char *Buf)\r
1471 {\r
1472         unsigned char *Get;\r
1473         unsigned char *Put;\r
1474         int Rnd;\r
1475         int Ch;\r
1476 \r
1477         Get = (unsigned char *)Str;\r
1478         Put = (unsigned char *)Buf;\r
1479 \r
1480         while(*Get != NUL)\r
1481         {\r
1482                 Rnd = ((unsigned int)*Get >> 4) & 0x3;\r
1483                 Ch = (*Get & 0xF) | ((*(Get+1) & 0xF) << 4);\r
1484                 Ch <<= 8;\r
1485                 if((*Get & 0x1) != 0)\r
1486                         Get++;\r
1487                 Get += 2;\r
1488                 Ch >>= Rnd;\r
1489                 Ch = (Ch & 0xFF) | ((Ch >> 8) & 0xFF);\r
1490                 *Put++ = Ch;\r
1491         }\r
1492         *Put = NUL;\r
1493         return;\r
1494 }\r
1495 \r
1496 /*----- パスワードの暗号化を解く(オリジナルアルゴリズム^Key) -------------------\r
1497 *\r
1498 *       Parameter\r
1499 *               char *Str : 暗号化したパスワード\r
1500 *               char *Buf : パスワードを格納するバッファ\r
1501 *               const char *Key : 暗号化キー\r
1502 *\r
1503 *       Return Value\r
1504 *               なし\r
1505 *----------------------------------------------------------------------------*/\r
1506 static void DecodePassword2(char *Str, char *Buf, const char* Key)\r
1507 {\r
1508         int Rnd;\r
1509         int Ch;\r
1510         unsigned char *Get = (unsigned char *)Str;\r
1511         unsigned char *Put = (unsigned char *)Buf;\r
1512 \r
1513         /* 2010.01.31 genta Key */\r
1514         unsigned char *KeyHead = (unsigned char *)Key;\r
1515         unsigned char *KeyEnd = KeyHead + strlen(KeyHead);\r
1516         unsigned char *KeyCurrent = KeyHead;\r
1517 \r
1518         while(*Get != NUL)\r
1519         {\r
1520                 Rnd = ((unsigned int)*Get >> 4) & 0x3;\r
1521                 Ch = (*Get & 0xF) | ((*(Get+1) & 0xF) << 4);\r
1522                 Ch <<= 8;\r
1523                 if((*Get & 0x1) != 0)\r
1524                         Get++;\r
1525                 Get += 2;\r
1526                 Ch >>= Rnd;\r
1527                 Ch = (Ch & 0xFF) | ((Ch >> 8) & 0xFF);\r
1528                 *Put++ = Ch ^ *KeyCurrent;\r
1529                 \r
1530                 /* 2010.01.31 genta Key */\r
1531                 if( ++KeyCurrent == KeyEnd ){\r
1532                         KeyCurrent = KeyHead;\r
1533                 }\r
1534         }\r
1535         *Put = NUL;\r
1536         return;\r
1537 }\r
1538 \r
1539 /*----- パスワードの暗号化を解く(AES) ---------------------------------------\r
1540 *\r
1541 *       Parameter\r
1542 *               char *Str : 暗号化したパスワード\r
1543 *               char *Buf : パスワードを格納するバッファ\r
1544 *               const char *Key : 暗号化キー\r
1545 *\r
1546 *       Return Value\r
1547 *               なし\r
1548 *----------------------------------------------------------------------------*/\r
1549 \r
1550 static void DecodePassword3(char *Str, char *Buf, const char *Key)\r
1551 {\r
1552         char *Get;\r
1553         unsigned char *EncBuf;\r
1554         size_t StrLen;\r
1555         size_t IvIndex;\r
1556         size_t EncBufIndex;\r
1557         size_t EncBufLen;\r
1558         unsigned char AesKey[32];\r
1559         unsigned char AesCbcIv[AES_BLOCK_SIZE];\r
1560         aes_decrypt_ctx Ctx;\r
1561 \r
1562         Buf[0] = NUL;\r
1563 \r
1564         Get = Str;\r
1565         StrLen = strlen(Str);\r
1566 \r
1567         if(AES_BLOCK_SIZE * 2 + 1 < StrLen)\r
1568         {\r
1569 \r
1570                 EncBufLen = (StrLen - 1 ) / 2 - AES_BLOCK_SIZE;\r
1571                 if((EncBuf = malloc(EncBufLen)) != NULL)\r
1572                 {\r
1573                         for(IvIndex = 0; IvIndex < AES_BLOCK_SIZE; IvIndex++)\r
1574                         {\r
1575                                 AesCbcIv[IvIndex]  = hex2bin(*Get++) << 4;\r
1576                                 AesCbcIv[IvIndex] |= hex2bin(*Get++);\r
1577                         }\r
1578 \r
1579                         if(*Get++ == ':')\r
1580                         {\r
1581                                 if(CreateAesKey(AesKey, Key) == FFFTP_SUCCESS)\r
1582                                 {\r
1583                                         aes_decrypt_key(AesKey, 32, &Ctx);\r
1584 \r
1585                                         for(EncBufIndex = 0; EncBufIndex < EncBufLen; EncBufIndex++)\r
1586                                         {\r
1587                                                 EncBuf[EncBufIndex]  = hex2bin(*Get++) << 4;\r
1588                                                 EncBuf[EncBufIndex] |= hex2bin(*Get++);\r
1589                                         }\r
1590                                         if(aes_cbc_decrypt(EncBuf, Buf, EncBufLen, AesCbcIv, &Ctx) == EXIT_SUCCESS)\r
1591                                         {\r
1592                                                 Buf[EncBufLen] = NUL;\r
1593                                         }\r
1594                                 }\r
1595                         }\r
1596                         free(EncBuf);\r
1597                 }\r
1598         }\r
1599         return;\r
1600 }\r
1601 \r
1602 /*----- AES用固定長キーを作成 ----------------------------------------------\r
1603 *\r
1604 *       Parameter\r
1605 *               unsigned char *AesKey : AES暗号鍵\r
1606 *               const char *Key : 暗号化キー\r
1607 *\r
1608 *       Return Value\r
1609 *               int ステータス (FFFTP_SUCCESS/FFFTP_FAIL)\r
1610 *       Note\r
1611 *               SHA-1をもちいて32Byte鍵を生成する\r
1612 *----------------------------------------------------------------------------*/\r
1613 \r
1614 static int CreateAesKey(unsigned char *AesKey, const char* Key)\r
1615 {\r
1616         char* HashKey;\r
1617         uint32 HashKeyLen;\r
1618         uint32 results[10];\r
1619         int ByteOffset;\r
1620         int KeyIndex;\r
1621         int ResIndex;\r
1622 \r
1623         HashKeyLen = strlen(Key) + 16;\r
1624         if((HashKey = malloc(HashKeyLen + 1)) == NULL){\r
1625                 return (FFFTP_FAIL);\r
1626         }\r
1627 \r
1628         strcpy(HashKey, Key);\r
1629         strcat(HashKey, ">g^r=@N7=//z<[`:");\r
1630     sha_memory((uchar *)HashKey, HashKeyLen, results);\r
1631 \r
1632         strcpy(HashKey, Key);\r
1633         strcat(HashKey, "VG77dO1#EyC]$|C@");\r
1634     sha_memory((uchar *)HashKey, HashKeyLen, results + 5);\r
1635 \r
1636         KeyIndex = 0;\r
1637         ResIndex = 0;\r
1638         while(ResIndex < 8){\r
1639                 for(ByteOffset = 0; ByteOffset < 4; ByteOffset++){\r
1640                         AesKey[KeyIndex++] = (results[ResIndex] >> ByteOffset * 8) & 0xff;\r
1641                 }\r
1642                 ResIndex++;\r
1643         }\r
1644         free(HashKey);\r
1645 \r
1646         return (FFFTP_SUCCESS);\r
1647 }\r
1648 \r
1649 \r
1650 /*===== レジストリとINIファイルのアクセス処理 ============*/\r
1651 \r
1652 \r
1653 /*===== INIファイル用のレジストリデータ =====*/\r
1654 \r
1655 typedef struct regdatatbl {\r
1656         char KeyName[80+1];                     /* キー名 */\r
1657         char ValTbl[REG_SECT_MAX];      /* 値のテーブル */\r
1658         int ValLen;                                     /* 値データのバイト数 */\r
1659         int Mode;                                       /* キーのモード */\r
1660         struct regdatatbl *Next;\r
1661 } REGDATATBL;\r
1662 \r
1663 /*===== プロトタイプ =====*/\r
1664 \r
1665 static BOOL WriteOutRegToFile(REGDATATBL *Pos);\r
1666 static int ReadInReg(char *Name, REGDATATBL **Handle);\r
1667 static int StrCatOut(char *Src, int Len, char *Dst);\r
1668 static int StrReadIn(char *Src, int Max, char *Dst);\r
1669 static char *ScanValue(void *Handle, char *Name);\r
1670 \r
1671 \r
1672 /*===== ローカルなワーク =====*/\r
1673 \r
1674 static int TmpRegType;\r
1675 \r
1676 \r
1677 \r
1678 /*----- レジストリのタイプを設定する ------------------------------------------\r
1679 *\r
1680 *       Parameter\r
1681 *               int Type : タイプ (REGTYPE_xxx)\r
1682 *\r
1683 *       Return Value\r
1684 *               int ステータス\r
1685 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
1686 *----------------------------------------------------------------------------*/\r
1687 \r
1688 static void SetRegType(int Type)\r
1689 {\r
1690         TmpRegType = Type;\r
1691         return;\r
1692 }\r
1693 \r
1694 \r
1695 /*----- レジストリ/INIファイルをオープンする(読み込み)-----------------------\r
1696 *\r
1697 *       Parameter\r
1698 *               char *Name : レジストリ名\r
1699 *               void **Handle : ハンドルを返すワーク\r
1700 *\r
1701 *       Return Value\r
1702 *               int ステータス\r
1703 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
1704 *----------------------------------------------------------------------------*/\r
1705 \r
1706 static int OpenReg(char *Name, void **Handle)\r
1707 {\r
1708         int Sts;\r
1709         char Tmp[FMAX_PATH+1];\r
1710 \r
1711         Sts = FFFTP_FAIL;\r
1712         if(TmpRegType == REGTYPE_REG)\r
1713         {\r
1714                 strcpy(Tmp, "Software\\Sota\\");\r
1715                 strcat(Tmp, Name);\r
1716                 if(RegOpenKeyEx(HKEY_CURRENT_USER, Tmp, 0, KEY_READ, (HKEY *)Handle) == ERROR_SUCCESS)\r
1717                         Sts = FFFTP_SUCCESS;\r
1718         }\r
1719         else\r
1720         {\r
1721                 if((Sts = ReadInReg(Name, (REGDATATBL **)Handle)) == FFFTP_SUCCESS)\r
1722                         ((REGDATATBL *)(*Handle))->Mode = 0;\r
1723         }\r
1724         return(Sts);\r
1725 }\r
1726 \r
1727 \r
1728 /*----- レジストリ/INIファイルを作成する(書き込み)---------------------------\r
1729 *\r
1730 *       Parameter\r
1731 *               char *Name : レジストリ名\r
1732 *               void **Handle : ハンドルを返すワーク\r
1733 *\r
1734 *       Return Value\r
1735 *               int ステータス\r
1736 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
1737 *----------------------------------------------------------------------------*/\r
1738 \r
1739 static int CreateReg(char *Name, void **Handle)\r
1740 {\r
1741         int Sts;\r
1742         char Tmp[FMAX_PATH+1];\r
1743         DWORD Dispos;\r
1744 \r
1745         Sts = FFFTP_FAIL;\r
1746         if(TmpRegType == REGTYPE_REG)\r
1747         {\r
1748                 strcpy(Tmp, "Software\\Sota\\");\r
1749                 strcat(Tmp, Name);\r
1750                 if(RegCreateKeyEx(HKEY_CURRENT_USER, Tmp, 0, "", REG_OPTION_NON_VOLATILE, KEY_CREATE_SUB_KEY | KEY_SET_VALUE, NULL, (HKEY *)Handle, &Dispos) == ERROR_SUCCESS)\r
1751                         Sts = FFFTP_SUCCESS;\r
1752         }\r
1753         else\r
1754         {\r
1755                 if((*Handle = malloc(sizeof(REGDATATBL))) != NULL)\r
1756                 {\r
1757                         strcpy(((REGDATATBL *)(*Handle))->KeyName, Name);\r
1758                         ((REGDATATBL *)(*Handle))->ValLen = 0;\r
1759                         ((REGDATATBL *)(*Handle))->Next = NULL;\r
1760                         ((REGDATATBL *)(*Handle))->Mode = 1;\r
1761                         Sts = FFFTP_SUCCESS;\r
1762                 }\r
1763         }\r
1764         return(Sts);\r
1765 }\r
1766 \r
1767 \r
1768 /*----- レジストリ/INIファイルをクローズする ----------------------------------\r
1769 *\r
1770 *       Parameter\r
1771 *               void *Handle : ハンドル\r
1772 *\r
1773 *       Return Value\r
1774 *               int ステータス\r
1775 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
1776 *----------------------------------------------------------------------------*/\r
1777 \r
1778 static int CloseReg(void *Handle)\r
1779 {\r
1780         REGDATATBL *Pos;\r
1781         REGDATATBL *Next;\r
1782         FILE *Strm;\r
1783 \r
1784         if(TmpRegType == REGTYPE_REG)\r
1785         {\r
1786                 RegCloseKey(Handle);\r
1787 \r
1788                 /* INIファイルを削除 */\r
1789                 if((Strm = fopen(AskIniFilePath(), "rt")) != NULL)\r
1790                 {\r
1791                         fclose(Strm);\r
1792                         MoveFileToTrashCan(AskIniFilePath());\r
1793                 }\r
1794         }\r
1795         else\r
1796         {\r
1797                 if(((REGDATATBL *)Handle)->Mode == 1)\r
1798                 {\r
1799                         if(WriteOutRegToFile(Handle) == TRUE)\r
1800                         {\r
1801 //                              /* レジストリをクリア */\r
1802 //                              ClearRegistory();\r
1803                         }\r
1804                 }\r
1805                 /* テーブルを削除 */\r
1806                 Pos = Handle;\r
1807                 while(Pos != NULL)\r
1808                 {\r
1809                         Next = Pos->Next;\r
1810                         free(Pos);\r
1811                         Pos = Next;\r
1812                 }\r
1813         }\r
1814         return(FFFTP_SUCCESS);\r
1815 }\r
1816 \r
1817 \r
1818 /*----- レジストリ情報をINIファイルに書き込む ---------------------------------\r
1819 *\r
1820 *       Parameter\r
1821 *               REGDATATBL *Pos : レジストリデータ\r
1822 *\r
1823 *       Return Value\r
1824 *               なし\r
1825 *----------------------------------------------------------------------------*/\r
1826 \r
1827 static BOOL WriteOutRegToFile(REGDATATBL *Pos)\r
1828 {\r
1829         FILE *Strm;\r
1830         char *Disp;\r
1831         BOOL Ret;\r
1832 \r
1833         Ret = FALSE;\r
1834         if((Strm = fopen(AskIniFilePath(), "wt")) != NULL)\r
1835         {\r
1836                 fprintf(Strm, MSGJPN239);\r
1837                 while(Pos != NULL)\r
1838                 {\r
1839                         fprintf(Strm, "\n[%s]\n", Pos->KeyName);\r
1840 \r
1841                         Disp = Pos->ValTbl;\r
1842                         while(Disp < (Pos->ValTbl + Pos->ValLen))\r
1843                         {\r
1844                                 fprintf(Strm, "%s\n", Disp);\r
1845                                 Disp = Disp + strlen(Disp) + 1;\r
1846                         }\r
1847                         Pos = Pos->Next;\r
1848                 }\r
1849                 fclose(Strm);\r
1850                 Ret = TRUE;\r
1851         }\r
1852         else\r
1853                 MessageBox(NULL, MSGJPN240, "FFFTP", MB_OK);\r
1854 \r
1855         return(Ret);\r
1856 }\r
1857 \r
1858 \r
1859 /*----- INIファイルからレジストリ情報を読み込む -------------------------------\r
1860 *\r
1861 *       Parameter\r
1862 *               char *Name : 名前\r
1863 *               void *Handle : ハンドル\r
1864 *\r
1865 *       Return Value\r
1866 *               int ステータス\r
1867 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
1868 *----------------------------------------------------------------------------*/\r
1869 \r
1870 static int ReadInReg(char *Name, REGDATATBL **Handle)\r
1871 {\r
1872         FILE *Strm;\r
1873         char *Buf;\r
1874         char *Tmp;\r
1875         char *Data;\r
1876         REGDATATBL *New;\r
1877         REGDATATBL *Pos;\r
1878         int Sts;\r
1879 \r
1880         Sts = FFFTP_FAIL;\r
1881         *Handle = NULL;\r
1882 \r
1883         if((Strm = fopen(AskIniFilePath(), "rt")) != NULL)\r
1884         {\r
1885                 if((Buf = malloc(REG_SECT_MAX)) != NULL)\r
1886                 {\r
1887                         while(fgets(Buf, REG_SECT_MAX, Strm) != NULL)\r
1888                         {\r
1889                                 if(*Buf != '#')\r
1890                                 {\r
1891                                         if((Tmp = strchr(Buf, '\n')) != NULL)\r
1892                                                 *Tmp = NUL;\r
1893 \r
1894                                         if(*Buf == '[')\r
1895                                         {\r
1896                                                 if((New = malloc(sizeof(REGDATATBL))) != NULL)\r
1897                                                 {\r
1898                                                         if((Tmp = strchr(Buf, ']')) != NULL)\r
1899                                                                 *Tmp = NUL;\r
1900                                                         strcpy(New->KeyName, Buf+1);\r
1901                                                         New->ValLen = 0;\r
1902                                                         New->Next = NULL;\r
1903                                                         Data = New->ValTbl;\r
1904                                                 }\r
1905                                                 if(*Handle == NULL)\r
1906                                                         *Handle = New;\r
1907                                                 else\r
1908                                                 {\r
1909                                                         Pos = *Handle;\r
1910                                                         while(Pos->Next != NULL)\r
1911                                                                 Pos = Pos->Next;\r
1912                                                         Pos->Next = New;\r
1913                                                 }\r
1914                                         }\r
1915                                         else if(strlen(Buf) > 0)\r
1916                                         {\r
1917                                                 strcpy(Data, Buf);\r
1918                                                 Data += strlen(Buf) + 1;\r
1919                                                 New->ValLen += strlen(Buf) + 1;\r
1920                                         }\r
1921                                 }\r
1922                         }\r
1923                         Sts = FFFTP_SUCCESS;\r
1924                         free(Buf);\r
1925                 }\r
1926                 fclose(Strm);\r
1927         }\r
1928         return(Sts);\r
1929 }\r
1930 \r
1931 \r
1932 /*----- サブキーをオープンする ------------------------------------------------\r
1933 *\r
1934 *       Parameter\r
1935 *               void *Parent : 親のハンドル\r
1936 *               char *Name : 名前\r
1937 *               void **Handle : ハンドルを返すワーク\r
1938 *\r
1939 *       Return Value\r
1940 *               int ステータス\r
1941 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
1942 *----------------------------------------------------------------------------*/\r
1943 \r
1944 static int OpenSubKey(void *Parent, char *Name, void **Handle)\r
1945 {\r
1946         int Sts;\r
1947         char Key[80];\r
1948         REGDATATBL *Pos;\r
1949 \r
1950         Sts = FFFTP_FAIL;\r
1951         if(TmpRegType == REGTYPE_REG)\r
1952         {\r
1953                 if(RegOpenKeyEx(Parent, Name, 0, KEY_READ, (HKEY *)Handle) == ERROR_SUCCESS)\r
1954                         Sts = FFFTP_SUCCESS;\r
1955         }\r
1956         else\r
1957         {\r
1958                 strcpy(Key, ((REGDATATBL *)Parent)->KeyName);\r
1959                 strcat(Key, "\\");\r
1960                 strcat(Key, Name);\r
1961                 Pos = Parent;\r
1962                 while(Pos != NULL)\r
1963                 {\r
1964                         if(strcmp(Pos->KeyName, Key) == 0)\r
1965                         {\r
1966                                 *Handle = Pos;\r
1967                                 Sts = FFFTP_SUCCESS;\r
1968                                 break;\r
1969                         }\r
1970                         Pos = Pos->Next;\r
1971                 }\r
1972         }\r
1973         return(Sts);\r
1974 }\r
1975 \r
1976 \r
1977 /*----- サブキーを作成する ----------------------------------------------------\r
1978 *\r
1979 *       Parameter\r
1980 *               void *Parent : 親のハンドル\r
1981 *               char *Name : 名前\r
1982 *               void **Handle : ハンドルを返すワーク\r
1983 *\r
1984 *       Return Value\r
1985 *               int ステータス\r
1986 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
1987 *----------------------------------------------------------------------------*/\r
1988 \r
1989 static int CreateSubKey(void *Parent, char *Name, void **Handle)\r
1990 {\r
1991         int Sts;\r
1992         DWORD Dispos;\r
1993         REGDATATBL *Pos;\r
1994 \r
1995         Sts = FFFTP_FAIL;\r
1996         if(TmpRegType == REGTYPE_REG)\r
1997         {\r
1998                 if(RegCreateKeyEx(Parent, Name, 0, "", REG_OPTION_NON_VOLATILE, KEY_SET_VALUE, NULL, (HKEY *)Handle, &Dispos) == ERROR_SUCCESS)\r
1999                         Sts = FFFTP_SUCCESS;\r
2000         }\r
2001         else\r
2002         {\r
2003                 if((*Handle = malloc(sizeof(REGDATATBL))) != NULL)\r
2004                 {\r
2005                         strcpy(((REGDATATBL *)(*Handle))->KeyName, ((REGDATATBL *)Parent)->KeyName);\r
2006                         strcat(((REGDATATBL *)(*Handle))->KeyName, "\\");\r
2007                         strcat(((REGDATATBL *)(*Handle))->KeyName, Name);\r
2008 \r
2009                         ((REGDATATBL *)(*Handle))->ValLen = 0;\r
2010                         ((REGDATATBL *)(*Handle))->Next = NULL;\r
2011 \r
2012                         Pos = (REGDATATBL *)Parent;\r
2013                         while(Pos->Next != NULL)\r
2014                                 Pos = Pos->Next;\r
2015                         Pos->Next = *Handle;\r
2016                         Sts = FFFTP_SUCCESS;\r
2017                 }\r
2018         }\r
2019         return(Sts);\r
2020 }\r
2021 \r
2022 \r
2023 /*----- サブキーをクローズする ------------------------------------------------\r
2024 *\r
2025 *       Parameter\r
2026 *               void *Handle : ハンドル\r
2027 *\r
2028 *       Return Value\r
2029 *               int ステータス\r
2030 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2031 *----------------------------------------------------------------------------*/\r
2032 \r
2033 static int CloseSubKey(void *Handle)\r
2034 {\r
2035         if(TmpRegType == REGTYPE_REG)\r
2036                 RegCloseKey(Handle);\r
2037         else\r
2038         {\r
2039                 /* Nothing */\r
2040         }\r
2041         return(FFFTP_SUCCESS);\r
2042 }\r
2043 \r
2044 \r
2045 /*----- サブキーを削除する ----------------------------------------------------\r
2046 *\r
2047 *       Parameter\r
2048 *               void *Handle : ハンドル\r
2049 *               char *Name : 名前\r
2050 *\r
2051 *       Return Value\r
2052 *               int ステータス\r
2053 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2054 *----------------------------------------------------------------------------*/\r
2055 \r
2056 static int DeleteSubKey(void *Handle, char *Name)\r
2057 {\r
2058         int Sts;\r
2059 \r
2060         Sts = FFFTP_FAIL;\r
2061         if(TmpRegType == REGTYPE_REG)\r
2062         {\r
2063                 if(RegDeleteKey(Handle, Name) == ERROR_SUCCESS)\r
2064                         Sts = FFFTP_SUCCESS;\r
2065         }\r
2066         else\r
2067         {\r
2068                 Sts = FFFTP_FAIL;\r
2069         }\r
2070         return(Sts);\r
2071 }\r
2072 \r
2073 \r
2074 /*----- 値を削除する ----------------------------------------------------------\r
2075 *\r
2076 *       Parameter\r
2077 *               void *Handle : ハンドル\r
2078 *               char *Name : 名前\r
2079 *\r
2080 *       Return Value\r
2081 *               int ステータス\r
2082 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2083 *----------------------------------------------------------------------------*/\r
2084 \r
2085 static int DeleteValue(void *Handle, char *Name)\r
2086 {\r
2087         int Sts;\r
2088 \r
2089         Sts = FFFTP_FAIL;\r
2090         if(TmpRegType == REGTYPE_REG)\r
2091         {\r
2092                 if(RegDeleteValue(Handle, Name) == ERROR_SUCCESS)\r
2093                         Sts = FFFTP_SUCCESS;\r
2094         }\r
2095         else\r
2096         {\r
2097                 Sts = FFFTP_FAIL;\r
2098         }\r
2099         return(Sts);\r
2100 }\r
2101 \r
2102 \r
2103 /*----- INT値を読み込む -------------------------------------------------------\r
2104 *\r
2105 *       Parameter\r
2106 *               void *Handle : ハンドル\r
2107 *               char *Name : 名前\r
2108 *               int *Value : INT値を返すワーク\r
2109 *\r
2110 *       Return Value\r
2111 *               int ステータス\r
2112 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2113 *----------------------------------------------------------------------------*/\r
2114 \r
2115 static int ReadIntValueFromReg(void *Handle, char *Name, int *Value)\r
2116 {\r
2117         int Sts;\r
2118         DWORD Size;\r
2119         char *Pos;\r
2120 \r
2121         Sts = FFFTP_FAIL;\r
2122         if(TmpRegType == REGTYPE_REG)\r
2123         {\r
2124                 Size = sizeof(int);\r
2125                 if(RegQueryValueEx(Handle, Name, NULL, NULL, (BYTE *)Value, &Size) == ERROR_SUCCESS)\r
2126                         Sts = FFFTP_SUCCESS;\r
2127         }\r
2128         else\r
2129         {\r
2130                 if((Pos = ScanValue(Handle, Name)) != NULL)\r
2131                 {\r
2132                         *Value = atoi(Pos);\r
2133                         Sts = FFFTP_SUCCESS;\r
2134                 }\r
2135         }\r
2136         return(Sts);\r
2137 }\r
2138 \r
2139 \r
2140 /*----- INT値を書き込む -------------------------------------------------------\r
2141 *\r
2142 *       Parameter\r
2143 *               void *Handle : ハンドル\r
2144 *               char *Name : 名前\r
2145 *               int Value : INT値\r
2146 *\r
2147 *       Return Value\r
2148 *               int ステータス\r
2149 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2150 *----------------------------------------------------------------------------*/\r
2151 \r
2152 static int WriteIntValueToReg(void *Handle, char *Name, int Value)\r
2153 {\r
2154         REGDATATBL *Pos;\r
2155         char *Data;\r
2156         char Tmp[20];\r
2157 \r
2158         if(TmpRegType == REGTYPE_REG)\r
2159                 RegSetValueEx(Handle, Name, 0, REG_DWORD, (CONST BYTE *)&Value, sizeof(int));\r
2160         else\r
2161         {\r
2162                 Pos = (REGDATATBL *)Handle;\r
2163                 Data = Pos->ValTbl + Pos->ValLen;\r
2164                 strcpy(Data, Name);\r
2165                 strcat(Data, "=");\r
2166                 sprintf(Tmp, "%d", Value);\r
2167                 strcat(Data, Tmp);\r
2168                 Pos->ValLen += strlen(Data) + 1;\r
2169         }\r
2170         return(FFFTP_SUCCESS);\r
2171 }\r
2172 \r
2173 \r
2174 /*----- 文字列を読み込む ------------------------------------------------------\r
2175 *\r
2176 *       Parameter\r
2177 *               void *Handle : ハンドル\r
2178 *               char *Name : 名前\r
2179 *               char *Str : 文字列を返すワーク\r
2180 *               DWORD Size : 最大サイズ\r
2181 *\r
2182 *       Return Value\r
2183 *               int ステータス\r
2184 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2185 *----------------------------------------------------------------------------*/\r
2186 \r
2187 static int ReadStringFromReg(void *Handle, char *Name, char *Str, DWORD Size)\r
2188 {\r
2189         int Sts;\r
2190         char *Pos;\r
2191 \r
2192         Sts = FFFTP_FAIL;\r
2193         if(TmpRegType == REGTYPE_REG)\r
2194         {\r
2195                 if(RegQueryValueEx(Handle, Name, NULL, NULL, (BYTE *)Str, &Size) == ERROR_SUCCESS)\r
2196                 {\r
2197                         if(*(Str + Size - 1) != NUL)\r
2198                                 *(Str + Size) = NUL;\r
2199                         Sts = FFFTP_SUCCESS;\r
2200                 }\r
2201         }\r
2202         else\r
2203         {\r
2204                 if((Pos = ScanValue(Handle, Name)) != NULL)\r
2205                 {\r
2206                         Size = min1(Size-1, strlen(Pos));\r
2207                         Size = StrReadIn(Pos, Size, Str);\r
2208                         *(Str + Size) = NUL;\r
2209                         Sts = FFFTP_SUCCESS;\r
2210                 }\r
2211         }\r
2212         return(Sts);\r
2213 }\r
2214 \r
2215 \r
2216 /*----- 文字列を書き込む ------------------------------------------------------\r
2217 *\r
2218 *       Parameter\r
2219 *               void *Handle : ハンドル\r
2220 *               char *Name : 名前\r
2221 *               char *Str :文字列\r
2222 *\r
2223 *       Return Value\r
2224 *               int ステータス\r
2225 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2226 *----------------------------------------------------------------------------*/\r
2227 \r
2228 static int WriteStringToReg(void *Handle, char *Name, char *Str)\r
2229 {\r
2230         REGDATATBL *Pos;\r
2231         char *Data;\r
2232 \r
2233         if(TmpRegType == REGTYPE_REG)\r
2234                 RegSetValueEx(Handle, Name, 0, REG_SZ, (CONST BYTE *)Str, strlen(Str)+1);\r
2235         else\r
2236         {\r
2237                 Pos = (REGDATATBL *)Handle;\r
2238                 Data = Pos->ValTbl + Pos->ValLen;\r
2239                 strcpy(Data, Name);\r
2240                 strcat(Data, "=");\r
2241                 Pos->ValLen += strlen(Data);\r
2242                 Data = Pos->ValTbl + Pos->ValLen;\r
2243                 Pos->ValLen += StrCatOut(Str, strlen(Str), Data) + 1;\r
2244         }\r
2245         return(FFFTP_SUCCESS);\r
2246 }\r
2247 \r
2248 \r
2249 /*----- マルチ文字列を読み込む ------------------------------------------------\r
2250 *\r
2251 *       Parameter\r
2252 *               void *Handle : ハンドル\r
2253 *               char *Name : 名前\r
2254 *               char *Str : 文字列を返すワーク\r
2255 *               DWORD Size : 最大サイズ\r
2256 *\r
2257 *       Return Value\r
2258 *               int ステータス\r
2259 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2260 *----------------------------------------------------------------------------*/\r
2261 \r
2262 static int ReadMultiStringFromReg(void *Handle, char *Name, char *Str, DWORD Size)\r
2263 {\r
2264         int Sts;\r
2265         char *Pos;\r
2266 \r
2267         Sts = FFFTP_FAIL;\r
2268         if(TmpRegType == REGTYPE_REG)\r
2269         {\r
2270                 if(RegQueryValueEx(Handle, Name, NULL, NULL, (BYTE *)Str, &Size) == ERROR_SUCCESS)\r
2271                 {\r
2272                         if(*(Str + Size - 1) != NUL)\r
2273                                 *(Str + Size) = NUL;\r
2274                         Sts = FFFTP_SUCCESS;\r
2275                 }\r
2276         }\r
2277         else\r
2278         {\r
2279                 if((Pos = ScanValue(Handle, Name)) != NULL)\r
2280                 {\r
2281                         Size = min1(Size-1, strlen(Pos));\r
2282                         Size = StrReadIn(Pos, Size, Str);\r
2283                         *(Str + Size) = NUL;\r
2284                         Sts = FFFTP_SUCCESS;\r
2285                 }\r
2286         }\r
2287         return(Sts);\r
2288 }\r
2289 \r
2290 \r
2291 /*----- マルチ文字列を書き込む ------------------------------------------------\r
2292 *\r
2293 *       Parameter\r
2294 *               void *Handle : ハンドル\r
2295 *               char *Name : 名前\r
2296 *               char *Str : 文字列\r
2297 *\r
2298 *       Return Value\r
2299 *               int ステータス\r
2300 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2301 *----------------------------------------------------------------------------*/\r
2302 \r
2303 static int WriteMultiStringToReg(void *Handle, char *Name, char *Str)\r
2304 {\r
2305         REGDATATBL *Pos;\r
2306         char *Data;\r
2307 \r
2308         if(TmpRegType == REGTYPE_REG)\r
2309                 RegSetValueEx(Handle, Name, 0, REG_MULTI_SZ, (CONST BYTE *)Str, StrMultiLen(Str)+1);\r
2310         else\r
2311         {\r
2312                 Pos = (REGDATATBL *)Handle;\r
2313                 Data = Pos->ValTbl + Pos->ValLen;\r
2314                 strcpy(Data, Name);\r
2315                 strcat(Data, "=");\r
2316                 Pos->ValLen += strlen(Data);\r
2317                 Data = Pos->ValTbl + Pos->ValLen;\r
2318                 Pos->ValLen += StrCatOut(Str, StrMultiLen(Str), Data) + 1;\r
2319         }\r
2320         return(FFFTP_SUCCESS);\r
2321 }\r
2322 \r
2323 \r
2324 /*----- バイナリを読み込む-----------------------------------------------------\r
2325 *\r
2326 *       Parameter\r
2327 *               void *Handle : ハンドル\r
2328 *               char *Name : 名前\r
2329 *               void *Bin : バイナリを返すワーク\r
2330 *               DWORD Size : 最大サイズ\r
2331 *\r
2332 *       Return Value\r
2333 *               int ステータス\r
2334 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2335 *----------------------------------------------------------------------------*/\r
2336 \r
2337 static int ReadBinaryFromReg(void *Handle, char *Name, void *Bin, DWORD Size)\r
2338 {\r
2339         int Sts;\r
2340         char *Pos;\r
2341 \r
2342         Sts = FFFTP_FAIL;\r
2343         if(TmpRegType == REGTYPE_REG)\r
2344         {\r
2345                 if(RegQueryValueEx(Handle, Name, NULL, NULL, (BYTE *)Bin, &Size) == ERROR_SUCCESS)\r
2346                         Sts = FFFTP_SUCCESS;\r
2347         }\r
2348         else\r
2349         {\r
2350                 if((Pos = ScanValue(Handle, Name)) != NULL)\r
2351                 {\r
2352                         Size = min1(Size, strlen(Pos));\r
2353                         Size = StrReadIn(Pos, Size, Bin);\r
2354                         Sts = FFFTP_SUCCESS;\r
2355                 }\r
2356         }\r
2357         return(Sts);\r
2358 }\r
2359 \r
2360 \r
2361 /*----- バイナリを書き込む ----------------------------------------------------\r
2362 *\r
2363 *       Parameter\r
2364 *               void *Handle : ハンドル\r
2365 *               char *Name : 名前\r
2366 *               void *Bin : バイナリ\r
2367 *               int Len : 長さ\r
2368 *\r
2369 *       Return Value\r
2370 *               int ステータス\r
2371 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2372 *----------------------------------------------------------------------------*/\r
2373 \r
2374 static int WriteBinaryToReg(void *Handle, char *Name, void *Bin, int Len)\r
2375 {\r
2376         REGDATATBL *Pos;\r
2377         char *Data;\r
2378 \r
2379         if(TmpRegType == REGTYPE_REG)\r
2380                 RegSetValueEx(Handle, Name, 0, REG_BINARY, (CONST BYTE *)Bin, Len);\r
2381         else\r
2382         {\r
2383                 Pos = (REGDATATBL *)Handle;\r
2384                 Data = Pos->ValTbl + Pos->ValLen;\r
2385                 strcpy(Data, Name);\r
2386                 strcat(Data, "=");\r
2387                 Pos->ValLen += strlen(Data);\r
2388                 Data = Pos->ValTbl + Pos->ValLen;\r
2389                 Pos->ValLen += StrCatOut(Bin, Len, Data) + 1;\r
2390         }\r
2391         return(FFFTP_SUCCESS);\r
2392 }\r
2393 \r
2394 \r
2395 /*----- 文字列をバッファに追加書き込みする ------------------------------------\r
2396 *\r
2397 *       Parameter\r
2398 *               char *Src : 文字列\r
2399 *               int len : 文字列の長さ\r
2400 *               char *Dst : 書き込みするバッファ\r
2401 *\r
2402 *       Return Value\r
2403 *               int 追加したバイト数\r
2404 *----------------------------------------------------------------------------*/\r
2405 \r
2406 static int StrCatOut(char *Src, int Len, char *Dst)\r
2407 {\r
2408         int Count;\r
2409 \r
2410         Dst += strlen(Dst);\r
2411         Count = 0;\r
2412         for(; Len > 0; Len--)\r
2413         {\r
2414                 if(*Src == '\\')\r
2415                 {\r
2416                         *Dst++ = '\\';\r
2417                         *Dst++ = '\\';\r
2418                         Count += 2;\r
2419                 }\r
2420                 else if((*Src >= 0x20) && (*Src <= 0x7E))\r
2421                 {\r
2422                         *Dst++ = *Src;\r
2423                         Count++;\r
2424                 }\r
2425                 else\r
2426                 {\r
2427                         sprintf(Dst, "\\%02X", *(unsigned char *)Src);\r
2428                         Dst += 3;\r
2429                         Count += 3;\r
2430                 }\r
2431                 Src++;\r
2432         }\r
2433         *Dst = NUL;\r
2434         return(Count);\r
2435 }\r
2436 \r
2437 \r
2438 /*----- 文字列をバッファに読み込む --------------------------------------------\r
2439 *\r
2440 *       Parameter\r
2441 *               char *Src : 文字列\r
2442 *               int Max : 最大サイズ\r
2443 *               char *Dst : 書き込みするバッファ\r
2444 *\r
2445 *       Return Value\r
2446 *               int 読み込んだバイト数\r
2447 *----------------------------------------------------------------------------*/\r
2448 \r
2449 static int StrReadIn(char *Src, int Max, char *Dst)\r
2450 {\r
2451         int Count;\r
2452         int Tmp;\r
2453 \r
2454         Count = 0;\r
2455         while(*Src != NUL)\r
2456         {\r
2457                 if(Count >= Max)\r
2458                         break;\r
2459 \r
2460                 if(*Src == '\\')\r
2461                 {\r
2462                         Src++;\r
2463                         if(*Src == '\\')\r
2464                                 *Dst = '\\';\r
2465                         else\r
2466                         {\r
2467                                 sscanf(Src, "%02x", &Tmp);\r
2468                                 *Dst = Tmp;\r
2469                                 Src++;\r
2470                         }\r
2471                 }\r
2472                 else\r
2473                         *Dst = *Src;\r
2474 \r
2475                 Count++;\r
2476                 Dst++;\r
2477                 Src++;\r
2478         }\r
2479         return(Count);\r
2480 }\r
2481 \r
2482 \r
2483 /*----- 値を検索する ----------------------------------------------------------\r
2484 *\r
2485 *       Parameter\r
2486 *               char *Handle : ハンドル\r
2487 *               char *Name : 名前\r
2488 *\r
2489 *       Return Value\r
2490 *               char *値データの先頭\r
2491 *                       NULL=指定の名前の値が見つからない\r
2492 *----------------------------------------------------------------------------*/\r
2493 \r
2494 static char *ScanValue(void *Handle, char *Name)\r
2495 {\r
2496         REGDATATBL *Cur;\r
2497         char *Pos;\r
2498         char *Ret;\r
2499 \r
2500         Ret = NULL;\r
2501         Cur = Handle;\r
2502         Pos = Cur->ValTbl;\r
2503         while(Pos < (Cur->ValTbl + Cur->ValLen))\r
2504         {\r
2505                 if((strncmp(Name, Pos, strlen(Name)) == 0) &&\r
2506                    (*(Pos + strlen(Name)) == '='))\r
2507                 {\r
2508                         Ret = Pos + strlen(Name) + 1;\r
2509                         break;\r
2510                 }\r
2511                 Pos += strlen(Pos) + 1;\r
2512         }\r
2513         return(Ret);\r
2514 }\r
2515 \r
2516 \r
2517 /*----------- パスワードの妥当性を確認する ------------------------------------\r
2518 *\r
2519 *       Parameter\r
2520 *               char *Password: パスワード文字列\r
2521 *               char *HashStr: SHA-1ハッシュ文字列\r
2522 *\r
2523 *       Return Value\r
2524 *               int 0 不一致\r
2525 *                       1: 一致\r
2526 *                       2: 異常\r
2527 *----------------------------------------------------------------------------*/\r
2528 int CheckPasswordValidity( char* Password, int length, const char* HashStr )\r
2529 {\r
2530         ulong hash1[5];\r
2531         ulong hash2[5];\r
2532         \r
2533         int i, j;\r
2534         \r
2535         const char* p = HashStr;\r
2536         \r
2537         /* 空文字列は一致したことにする */\r
2538         if( HashStr[0] == NUL ) return 1;\r
2539 \r
2540         /* Hashをチェックするする*/\r
2541         if( strlen(HashStr) != 40 )     return 2;\r
2542 \r
2543         /* Hashをデコードする*/\r
2544         for( i = 0; i < 5; i++ ){\r
2545                 ulong decode = 0;\r
2546                 for( j = 0; j < 8; j++ ){\r
2547                         if( *p < 0x40 || 0x40 + 15 < *p ){\r
2548                                 return 2;\r
2549                         }\r
2550                         decode = (decode << 4 ) + (*p - 0x40);\r
2551                         ++p;\r
2552                 }\r
2553                 hash1[i] = decode;\r
2554         }\r
2555         \r
2556         /* Password をハッシュする */\r
2557         sha_memory( Password, length, hash2 );\r
2558         \r
2559         if( memcmp( (char*)hash1, (char*)hash2, sizeof( hash1 )) == 0 ){\r
2560                 return 1;\r
2561         }\r
2562         return 0;\r
2563 }\r
2564 \r
2565 /*----------- パスワードの妥当性チェックのための文字列を作成する ------------\r
2566 *\r
2567 *       Parameter\r
2568 *               char *Password: パスワード文字列\r
2569 *               char *Str: SHA-1ハッシュ文字列格納場所 (41bytes以上)\r
2570 *\r
2571 *       Return Value\r
2572 *               None\r
2573 *----------------------------------------------------------------------------*/\r
2574 void CreatePasswordHash( char* Password, int length, char* HashStr )\r
2575 {\r
2576         ulong hash[5];\r
2577         int i, j;\r
2578         unsigned char *p = (unsigned char *)HashStr;\r
2579 \r
2580         sha_memory( Password, length, hash );\r
2581 \r
2582         for( i = 0; i < 5; i++ ){\r
2583                 ulong rest = hash[i];\r
2584                 for( j = 0; j < 8; j++ ){\r
2585                         *p++ = (unsigned char)((rest & 0xf0000000) >> 28) + '@';\r
2586                         rest <<= 4;\r
2587                 }\r
2588         }\r
2589         *p = NUL;\r
2590 }\r
2591 \r
2592 void SetHashSalt( DWORD salt )\r
2593 {\r
2594         unsigned char* pos = &SecretKey[strlen(SecretKey) + 1];\r
2595         *pos++ = ( salt >> 24 ) & 0xff;\r
2596         *pos++ = ( salt >> 16 ) & 0xff;\r
2597         *pos++ = ( salt >>  8 ) & 0xff;\r
2598         *pos++ = ( salt       ) & 0xff;\r
2599         \r
2600         SecretKeyLength = strlen( SecretKey ) + 5;\r
2601 }\r
2602 \r
2603 /*----------- 乱数生成をする -------------------------------------------------\r
2604 *\r
2605 *       Parameter\r
2606 *\r
2607 *       Return Value\r
2608 *               ランダムな値:コンパイラVS2005/動作環境WinXP以上では rand_s から取得する\r
2609 *----------------------------------------------------------------------------*/\r
2610 DWORD GetRandamDWRODValue(void)\r
2611 {\r
2612         DWORD rndValue;\r
2613         int errorCode;\r
2614 #ifdef _CRT_RAND_S\r
2615         errno_t errnoRand_s;\r
2616         errnoRand_s = rand_s(&rndValue);\r
2617         errorCode = (0 != errnoRand_s ? 1 : 0);\r
2618 #else\r
2619         errorCode = 1;\r
2620 #endif\r
2621         if(0 != errorCode){\r
2622 #ifdef USE_RANDAM_C_RAND\r
2623                 rndValue = rand() | (rand() << 16);\r
2624 #else\r
2625                 /* rand() のかわりに、SHA-1とパフォーマンスカウンタを用いる */\r
2626                 ulong shaValue[5];\r
2627                 LARGE_INTEGER Counter;\r
2628                 \r
2629                 if(0 == IsRndSourceInit){\r
2630                         /* 初回取得時 */\r
2631                         HANDLE CurProcHandle;\r
2632                         HANDLE CurThreadHandle;\r
2633                         \r
2634                         if(DuplicateHandle(GetCurrentProcess(), GetCurrentProcess(), GetCurrentProcess(),\r
2635                                 &CurProcHandle, 0, FALSE, DUPLICATE_SAME_ACCESS))\r
2636                         {\r
2637                                 CloseHandle(CurProcHandle);\r
2638                         }\r
2639                         if(DuplicateHandle(GetCurrentProcess(), GetCurrentThread(), GetCurrentProcess(),\r
2640                                 &CurThreadHandle, 0, FALSE, DUPLICATE_SAME_ACCESS))\r
2641                         {\r
2642                                 CloseHandle(CurThreadHandle);\r
2643                         }\r
2644                         \r
2645                         /* _WIN64 では64bitだが、その場合はrand_sが大抵利用可能なのでここでは32bitのみ用いる */\r
2646                         RndSource[0] = (ulong)CurProcHandle;\r
2647                         RndSource[1] = (ulong)CurThreadHandle;\r
2648                         RndSource[2] = (ulong)GetTickCount();\r
2649                         RndSource[3] = (ulong)timeGetTime();\r
2650                         RndSource[4] = 0; /* カウントアップ */\r
2651                         RndSource[5] = RndSource[3] + 1;\r
2652                         IsRndSourceInit = 1;\r
2653                 }\r
2654                 RndSource[4]++;\r
2655                 RndSource[5] += 0x00010010;\r
2656                 if(QueryPerformanceCounter(&Counter)){\r
2657                         RndSource[6] = Counter.LowPart;\r
2658                         RndSource[7] = Counter.HighPart;\r
2659                         RndSource[8] = (ulong)rand();\r
2660                 }else{\r
2661                         RndSource[6] = (ulong)timeGetTime();\r
2662                         RndSource[7] = (ulong)rand();\r
2663                         RndSource[8] = (ulong)rand();\r
2664                 }\r
2665                 \r
2666                 sha_memory((char *)RndSource, sizeof(RndSource), shaValue);\r
2667                 rndValue = shaValue[0] ^ shaValue[1] ^ shaValue[2] ^ shaValue[3] ^ shaValue[4];\r
2668 #endif\r
2669         }\r
2670         return rndValue;\r
2671 }\r