OSDN Git Service

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