OSDN Git Service

Add options for removing old logs.
[ffftp/ffftp.git] / registry.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 // IPv6対応\r
37 #include <winsock2.h>\r
38 #include <windows.h>\r
39 #include <stdio.h>\r
40 #include <stdlib.h>\r
41 #include <time.h>\r
42 #include <string.h>\r
43 #include <windowsx.h>\r
44 #include <process.h>\r
45 \r
46 \r
47 #include "common.h"\r
48 #include "resource.h"\r
49 #include "sample.h"\r
50 #include "sha.h"\r
51 #include "aes.h"\r
52 \r
53 \r
54 /*===== プロトタイプ =====*/\r
55 \r
56 static void SaveStr(HKEY hKey, char *Key, char *Str, char *DefaultStr);\r
57 static void SaveIntNum(HKEY hKey, char *Key, int Num, int DefaultNum);\r
58 static void MakeFontData(LOGFONT Font, HFONT hFont, char *Buf);\r
59 static int RestoreFontData(char *Str, LOGFONT *Font);\r
60 \r
61 static void EncodePassword(char *Str, char *Buf);\r
62 static void EncodePasswordOriginal(char *Str, char *Buf);\r
63 static void EncodePassword2(char *Str, char *Buf, const char *Key);\r
64 static void EncodePassword3(char *Str, char *Buf, const char *Key);\r
65 \r
66 static void DecodePassword(char *Str, char *Buf);\r
67 static void DecodePasswordOriginal(char *Str, char *Buf);\r
68 static void DecodePassword2(char *Str, char *Buf, const char *Key);\r
69 static void DecodePassword3(char *Str, char *Buf, const char *Key);\r
70 static int CreateAesKey(unsigned char *AesKey, const char* Key);\r
71 \r
72 static void SetRegType(int Type);\r
73 static int OpenReg(char *Name, void **Handle);\r
74 static int CreateReg(char *Name, void **Handle);\r
75 static int CloseReg(void *Handle);\r
76 static int OpenSubKey(void *Parent, char *Name, void **Handle);\r
77 static int CreateSubKey(void *Parent, char *Name, void **Handle);\r
78 static int CloseSubKey(void *Handle);\r
79 static int DeleteSubKey(void *Handle, char *Name);\r
80 static int DeleteValue(void *Handle, char *Name);\r
81 static int ReadIntValueFromReg(void *Handle, char *Name, int *Value);\r
82 static int WriteIntValueToReg(void *Handle, char *Name, int Value);\r
83 static int ReadStringFromReg(void *Handle, char *Name, char *Str, DWORD Size);\r
84 static int WriteStringToReg(void *Handle, char *Name, char *Str);\r
85 static int ReadMultiStringFromReg(void *Handle, char *Name, char *Str, DWORD Size);\r
86 static int WriteMultiStringToReg(void *Handle, char *Name, char *Str);\r
87 static int ReadBinaryFromReg(void *Handle, char *Name, void *Bin, DWORD Size);\r
88 static int WriteBinaryToReg(void *Handle, char *Name, void *Bin, int Len);\r
89 // 暗号化通信対応\r
90 static int StrCatOut(char *Src, int Len, char *Dst);\r
91 static int StrReadIn(char *Src, int Max, char *Dst);\r
92 \r
93 int CheckPasswordValidity( char* Password, int length, const char* HashStr );\r
94 void CreatePasswordHash( char* Password, int length, char* HashStr );\r
95 void SetHashSalt( DWORD salt );\r
96 \r
97 DWORD GetRandamDWRODValue(void);\r
98 \r
99 // 全設定暗号化対応\r
100 void MaskSettingsData(const char* Salt, int SaltLength, void* Data, DWORD Size, int EscapeZero);\r
101 void UnmaskSettingsData(const char* Salt, int SaltLength, void* Data, DWORD Size, int EscapeZero);\r
102 void CalculateSettingsDataChecksum(void* Data, DWORD Size);\r
103 \r
104 /* 2010.01.30 genta 追加 */\r
105 static char SecretKey[FMAX_PATH+1];\r
106 static int SecretKeyLength;\r
107 static int IsMasterPasswordError = PASSWORD_OK;\r
108 \r
109 static int IsRndSourceInit = 0;\r
110 static ulong RndSource[9];\r
111 \r
112 // UTF-8対応\r
113 static int IniKanjiCode = KANJI_NOCNV;\r
114 \r
115 // 全設定暗号化対応\r
116 static int EncryptSettings = NO;\r
117 static BYTE EncryptSettingsChecksum[20];\r
118 static int EncryptSettingsError = NO;\r
119 \r
120 /*===== 外部参照 =====*/\r
121 \r
122 /* 設定値 */\r
123 extern int WinPosX;\r
124 extern int WinPosY;\r
125 extern int WinWidth;\r
126 extern int WinHeight;\r
127 extern int LocalWidth;\r
128 extern int TaskHeight;\r
129 extern int LocalTabWidth[4];\r
130 extern int RemoteTabWidth[6];\r
131 extern char UserMailAdrs[USER_MAIL_LEN+1];\r
132 extern char ViewerName[VIEWERS][FMAX_PATH+1];\r
133 extern HFONT ListFont;\r
134 extern LOGFONT ListLogFont;\r
135 extern int LocalFileSort;\r
136 extern int LocalDirSort;\r
137 extern int RemoteFileSort;\r
138 extern int RemoteDirSort;\r
139 extern int TransMode;\r
140 extern int ConnectOnStart;\r
141 extern int DebugConsole;\r
142 extern int SaveWinPos;\r
143 extern char AsciiExt[ASCII_EXT_LEN+1];\r
144 extern int RecvMode;\r
145 extern int SendMode;\r
146 extern int MoveMode;\r
147 extern int ListType;\r
148 extern int CacheEntry;\r
149 extern int CacheSave;\r
150 extern char DefaultLocalPath[FMAX_PATH+1];\r
151 extern int SaveTimeStamp;\r
152 extern int FindMode;\r
153 extern int DotFile;\r
154 extern int DclickOpen;\r
155 extern SOUNDFILE Sound[SOUND_TYPES];\r
156 extern int FnameCnv;\r
157 extern int ConnectAndSet;\r
158 extern int TimeOut;\r
159 extern int RmEOF;\r
160 extern int RegType;\r
161 extern char FwallHost[HOST_ADRS_LEN+1];\r
162 extern char FwallUser[USER_NAME_LEN+1];\r
163 extern char FwallPass[PASSWORD_LEN+1];\r
164 extern int FwallPort;\r
165 extern int FwallType;\r
166 extern int FwallDefault;\r
167 extern int FwallSecurity;\r
168 extern int FwallResolv;\r
169 extern int FwallLower;\r
170 extern int FwallDelimiter;\r
171 extern int PasvDefault;\r
172 extern char MirrorNoTrn[MIRROR_LEN+1];\r
173 extern char MirrorNoDel[MIRROR_LEN+1];\r
174 extern int MirrorFnameCnv;\r
175 //extern int MirrorFolderCnv;\r
176 extern int RasClose;\r
177 extern int RasCloseNotify;\r
178 extern int FileHist;\r
179 extern char DefAttrList[DEFATTRLIST_LEN+1];\r
180 extern SIZE HostDlgSize;\r
181 extern SIZE BmarkDlgSize;\r
182 extern SIZE MirrorDlgSize;\r
183 extern int Sizing;\r
184 extern int SortSave;\r
185 extern char TmpPath[FMAX_PATH+1];\r
186 extern int QuickAnonymous;\r
187 extern int PassToHist;\r
188 extern int VaxSemicolon;\r
189 extern int SendQuit;\r
190 extern int NoRasControl;\r
191 extern int SuppressSave;\r
192 \r
193 extern int UpExistMode;\r
194 extern int ExistMode;\r
195 extern int DispIgnoreHide;\r
196 extern int DispDrives;\r
197 extern int MirUpDelNotify;\r
198 extern int MirDownDelNotify;\r
199 \r
200 extern int FolderAttr;\r
201 extern int FolderAttrNum;\r
202 \r
203 // 暗号化通信対応\r
204 extern BYTE CertificateCacheHash[MAX_CERT_CACHE_HASH][20];\r
205 extern BYTE SSLRootCAFileHash[20];\r
206 // ファイルアイコン表示対応\r
207 extern int DispFileIcon;\r
208 // タイムスタンプのバグ修正\r
209 extern int DispTimeSeconds;\r
210 // ファイルの属性を数字で表示\r
211 extern int DispPermissionsNumber;\r
212 // ディレクトリ自動作成\r
213 extern int MakeAllDir;\r
214 // UTF-8対応\r
215 extern int LocalKanjiCode;\r
216 // UPnP対応\r
217 extern int UPnPEnabled;\r
218 // 全設定暗号化対応\r
219 extern int EncryptAllSettings;\r
220 // ローカル側自動更新\r
221 extern int AutoRefreshFileList;\r
222 // 古い処理内容を消去\r
223 extern int RemoveOldLog;\r
224 \r
225 /*----- マスタパスワードの設定 ----------------------------------------------\r
226 *\r
227 *       Parameter\r
228 *               const char* Password : マスターパスワード\r
229 *\r
230 *       Return Value\r
231 *               なし\r
232 *----------------------------------------------------------------------------*/\r
233 void SetMasterPassword( const char* Password )\r
234 {\r
235         ZeroMemory( SecretKey, MAX_PASSWORD_LEN + 12 );\r
236         if( Password != NULL ){\r
237                 strncpy( SecretKey, Password, MAX_PASSWORD_LEN );\r
238         }\r
239         else {\r
240                 strcpy( SecretKey, DEFAULT_PASSWORD );\r
241         }\r
242         SecretKeyLength = strlen( SecretKey );\r
243         \r
244         /* 未検証なので,初期状態はOKにする (強制再設定→保存にを可能にする)*/\r
245         IsMasterPasswordError = PASSWORD_OK;\r
246 }\r
247 \r
248 // セキュリティ強化\r
249 void GetMasterPassword(char* Password)\r
250 {\r
251         strcpy(Password, SecretKey);\r
252 }\r
253 \r
254 /*----- マスタパスワードの状態取得 ----------------------------------------------\r
255 *\r
256 *       Parameter\r
257 *               なし\r
258 *\r
259 *       Return Value\r
260 *               PASSWORD_OK : OK\r
261 *               PASSWORD_UNMATCH : パスワード不一致\r
262 *               BAD_PASSWORD_HASH : パスワード確認失敗\r
263 *----------------------------------------------------------------------------*/\r
264 int GetMasterPasswordStatus(void)\r
265 {\r
266         return IsMasterPasswordError;\r
267 }\r
268 \r
269 /*----- レジストリ/INIファイルのマスターパスワードの検証を行う ------------\r
270 *\r
271 *       Parameter\r
272 *               なし\r
273 *\r
274 *       Return Value\r
275 *               \r
276 *----------------------------------------------------------------------------*/\r
277 \r
278 int ValidateMasterPassword(void)\r
279 {\r
280         void *hKey3;\r
281         int i;\r
282 \r
283         SetRegType(REGTYPE_INI);\r
284         if((i = OpenReg("FFFTP", &hKey3)) != FFFTP_SUCCESS)\r
285         {\r
286                 if(AskForceIni() == NO)\r
287                 {\r
288                         SetRegType(REGTYPE_REG);\r
289                         i = OpenReg("FFFTP", &hKey3);\r
290                 }\r
291         }\r
292         if(i == FFFTP_SUCCESS){\r
293                 char checkbuf[48];\r
294                 int salt = 0;\r
295 \r
296                 if( ReadIntValueFromReg(hKey3, "CredentialSalt", &salt)){\r
297                         SetHashSalt( salt );\r
298                 }\r
299                 if( ReadStringFromReg(hKey3, "CredentialCheck", checkbuf, sizeof( checkbuf )) == FFFTP_SUCCESS ){\r
300                         switch( CheckPasswordValidity( SecretKey, SecretKeyLength, checkbuf ) ){\r
301                         case 0: /* not match */\r
302                                 IsMasterPasswordError = PASSWORD_UNMATCH;\r
303                                 break;\r
304                         case 1: /* match */\r
305                                 IsMasterPasswordError = PASSWORD_OK;\r
306                                 break;\r
307                         case 2: /* invalid hash */\r
308                         default:\r
309                                 IsMasterPasswordError = BAD_PASSWORD_HASH;\r
310                                 break;\r
311                         }\r
312                 }\r
313                 CloseReg(hKey3);\r
314                 return YES;\r
315         }\r
316         return NO;\r
317 }\r
318 \r
319 /*----- レジストリ/INIファイルに設定値を保存 ---------------------------------\r
320 *\r
321 *       Parameter\r
322 *               なし\r
323 *\r
324 *       Return Value\r
325 *               なし\r
326 *----------------------------------------------------------------------------*/\r
327 \r
328 void SaveRegistry(void)\r
329 {\r
330         void *hKey3;\r
331         void *hKey4;\r
332         void *hKey5;\r
333         // 暗号化通信対応\r
334 //      char Str[FMAX_PATH+1];\r
335         char Str[PRIVATE_KEY_LEN*4+1];\r
336         char Buf[FMAX_PATH+1];\r
337         int i;\r
338         int n;\r
339         HOSTDATA DefaultHost;\r
340         HOSTDATA Host;\r
341         HISTORYDATA Hist;\r
342         HISTORYDATA DefaultHist;\r
343         \r
344         if( GetMasterPasswordStatus() == PASSWORD_UNMATCH ){\r
345                 /* 2010.01.30 genta: マスターパスワードが不一致の場合は不用意に上書きしない */\r
346                 return;\r
347         }\r
348 \r
349         // 全設定暗号化対応\r
350         if(EncryptSettingsError == YES)\r
351                 return;\r
352 \r
353         SetRegType(RegType);\r
354         if(CreateReg("FFFTP", &hKey3) == FFFTP_SUCCESS)\r
355         {\r
356                 char buf[48];\r
357                 int salt = GetTickCount();\r
358         \r
359                 WriteIntValueToReg(hKey3, "Version", VER_NUM);\r
360                 WriteIntValueToReg(hKey3, "CredentialSalt", salt);\r
361                 \r
362                 SetHashSalt( salt );\r
363                 /* save password hash */\r
364                 CreatePasswordHash( SecretKey, SecretKeyLength, buf );\r
365                 WriteStringToReg(hKey3, "CredentialCheck", buf);\r
366 \r
367                 // 全設定暗号化対応\r
368                 WriteIntValueToReg(hKey3, "EncryptAll", EncryptAllSettings);\r
369                 sprintf(Buf, "%d", EncryptAllSettings);\r
370                 EncodePassword(Buf, Str);\r
371                 WriteStringToReg(hKey3, "EncryptAllDetector", Str);\r
372                 EncryptSettings = EncryptAllSettings;\r
373                 memset(&EncryptSettingsChecksum, 0, 20);\r
374 \r
375                 if(CreateSubKey(hKey3, "Options", &hKey4) == FFFTP_SUCCESS)\r
376                 {\r
377                         WriteIntValueToReg(hKey4, "NoSave", SuppressSave);\r
378 \r
379                         if(SuppressSave != YES)\r
380                         {\r
381                                 WriteIntValueToReg(hKey4, "WinPosX", WinPosX);\r
382                                 WriteIntValueToReg(hKey4, "WinPosY", WinPosY);\r
383                                 WriteIntValueToReg(hKey4, "WinWidth", WinWidth);\r
384                                 WriteIntValueToReg(hKey4, "WinHeight", WinHeight);\r
385                                 WriteIntValueToReg(hKey4, "LocalWidth", LocalWidth);\r
386                                 WriteIntValueToReg(hKey4, "TaskHeight", TaskHeight);\r
387                                 WriteBinaryToReg(hKey4, "LocalColm", LocalTabWidth, sizeof(LocalTabWidth));\r
388                                 WriteBinaryToReg(hKey4, "RemoteColm", RemoteTabWidth, sizeof(RemoteTabWidth));\r
389                                 WriteIntValueToReg(hKey4, "SwCmd", Sizing);\r
390 \r
391                                 WriteStringToReg(hKey4, "UserMail", UserMailAdrs);\r
392                                 WriteStringToReg(hKey4, "Viewer", ViewerName[0]);\r
393                                 WriteStringToReg(hKey4, "Viewer2", ViewerName[1]);\r
394                                 WriteStringToReg(hKey4, "Viewer3", ViewerName[2]);\r
395 \r
396                                 WriteIntValueToReg(hKey4, "TrType", TransMode);\r
397                                 WriteIntValueToReg(hKey4, "Recv", RecvMode);\r
398                                 WriteIntValueToReg(hKey4, "Send", SendMode);\r
399                                 WriteIntValueToReg(hKey4, "Move", MoveMode);\r
400                                 WriteStringToReg(hKey4, "Path", DefaultLocalPath);\r
401                                 WriteIntValueToReg(hKey4, "Time", SaveTimeStamp);\r
402                                 WriteIntValueToReg(hKey4, "EOF", RmEOF);\r
403                                 WriteIntValueToReg(hKey4, "Scolon", VaxSemicolon);\r
404 \r
405                                 WriteIntValueToReg(hKey4, "RecvEx", ExistMode);\r
406                                 WriteIntValueToReg(hKey4, "SendEx", UpExistMode);\r
407 \r
408                                 WriteIntValueToReg(hKey4, "LFsort", LocalFileSort);\r
409                                 WriteIntValueToReg(hKey4, "LDsort", LocalDirSort);\r
410                                 WriteIntValueToReg(hKey4, "RFsort", RemoteFileSort);\r
411                                 WriteIntValueToReg(hKey4, "RDsort", RemoteDirSort);\r
412                                 WriteIntValueToReg(hKey4, "SortSave", SortSave);\r
413 \r
414                                 WriteIntValueToReg(hKey4, "ListType", ListType);\r
415                                 WriteIntValueToReg(hKey4, "Cache", CacheEntry);\r
416                                 WriteIntValueToReg(hKey4, "CacheSave", CacheSave);\r
417                                 WriteIntValueToReg(hKey4, "DotFile", DotFile);\r
418                                 WriteIntValueToReg(hKey4, "Dclick", DclickOpen);\r
419 \r
420                                 WriteIntValueToReg(hKey4, "ConS", ConnectOnStart);\r
421                                 WriteIntValueToReg(hKey4, "OldDlg", ConnectAndSet);\r
422                                 WriteIntValueToReg(hKey4, "RasClose", RasClose);\r
423                                 WriteIntValueToReg(hKey4, "RasNotify", RasCloseNotify);\r
424                                 WriteIntValueToReg(hKey4, "Qanony", QuickAnonymous);\r
425                                 WriteIntValueToReg(hKey4, "PassHist", PassToHist);\r
426                                 WriteIntValueToReg(hKey4, "SendQuit", SendQuit);\r
427                                 WriteIntValueToReg(hKey4, "NoRas", NoRasControl);\r
428 \r
429                                 WriteIntValueToReg(hKey4, "Debug", DebugConsole);\r
430                                 WriteIntValueToReg(hKey4, "WinPos", SaveWinPos);\r
431                                 WriteIntValueToReg(hKey4, "RegExp", FindMode);\r
432                                 WriteIntValueToReg(hKey4, "Reg", RegType);\r
433 \r
434                                 WriteMultiStringToReg(hKey4, "AsciiFile", AsciiExt);\r
435                                 WriteIntValueToReg(hKey4, "LowUp", FnameCnv);\r
436                                 WriteIntValueToReg(hKey4, "Tout", TimeOut);\r
437 \r
438                                 WriteMultiStringToReg(hKey4, "NoTrn", MirrorNoTrn);\r
439                                 WriteMultiStringToReg(hKey4, "NoDel", MirrorNoDel);\r
440                                 WriteIntValueToReg(hKey4, "MirFile", MirrorFnameCnv);\r
441                                 WriteIntValueToReg(hKey4, "MirUNot", MirUpDelNotify);\r
442                                 WriteIntValueToReg(hKey4, "MirDNot", MirDownDelNotify);\r
443 \r
444                                 MakeFontData(ListLogFont, ListFont, Str);\r
445                                 WriteStringToReg(hKey4, "ListFont", Str);\r
446                                 WriteIntValueToReg(hKey4, "ListHide", DispIgnoreHide);\r
447                                 WriteIntValueToReg(hKey4, "ListDrv", DispDrives);\r
448 \r
449                                 WriteStringToReg(hKey4, "FwallHost", FwallHost);\r
450                                 WriteStringToReg(hKey4, "FwallUser", FwallUser);\r
451                                 EncodePassword(FwallPass, Str);\r
452                                 WriteStringToReg(hKey4, "FwallPass", Str);\r
453                                 WriteIntValueToReg(hKey4, "FwallPort", FwallPort);\r
454                                 WriteIntValueToReg(hKey4, "FwallType", FwallType);\r
455                                 WriteIntValueToReg(hKey4, "FwallDef", FwallDefault);\r
456                                 WriteIntValueToReg(hKey4, "FwallSec", FwallSecurity);\r
457                                 WriteIntValueToReg(hKey4, "PasvDef", PasvDefault);\r
458                                 WriteIntValueToReg(hKey4, "FwallRes", FwallResolv);\r
459                                 WriteIntValueToReg(hKey4, "FwallLow", FwallLower);\r
460                                 WriteIntValueToReg(hKey4, "FwallDel", FwallDelimiter);\r
461 \r
462                                 WriteIntValueToReg(hKey4, "SndConSw", Sound[SND_CONNECT].On);\r
463                                 WriteIntValueToReg(hKey4, "SndTrnSw", Sound[SND_TRANS].On);\r
464                                 WriteIntValueToReg(hKey4, "SndErrSw", Sound[SND_ERROR].On);\r
465                                 WriteStringToReg(hKey4, "SndCon", Sound[SND_CONNECT].Fname);\r
466                                 WriteStringToReg(hKey4, "SndTrn", Sound[SND_TRANS].Fname);\r
467                                 WriteStringToReg(hKey4, "SndErr", Sound[SND_ERROR].Fname);\r
468 \r
469                                 WriteMultiStringToReg(hKey4, "DefAttr", DefAttrList);\r
470 \r
471                                 // 環境依存の不具合対策\r
472 //                              GetTempPath(FMAX_PATH, Str);\r
473                                 GetAppTempPath(Str);\r
474                                 SetYenTail(Str);\r
475                                 SaveStr(hKey4, "Tmp", TmpPath, Str);\r
476 \r
477                                 WriteBinaryToReg(hKey4, "Hdlg", &HostDlgSize, sizeof(SIZE));\r
478                                 WriteBinaryToReg(hKey4, "Bdlg", &BmarkDlgSize, sizeof(SIZE));\r
479                                 WriteBinaryToReg(hKey4, "Mdlg", &MirrorDlgSize, sizeof(SIZE));\r
480 \r
481                                 WriteIntValueToReg(hKey4, "FAttrSw", FolderAttr);\r
482                                 WriteIntValueToReg(hKey4, "FAttr", FolderAttrNum);\r
483 \r
484                                 WriteIntValueToReg(hKey4, "HistNum", FileHist);\r
485 \r
486                                 /* Ver1.54a以前の形式のヒストリデータは削除 */\r
487                                 DeleteValue(hKey4, "Hist");\r
488 \r
489                                 /* ヒストリの設定を保存 */\r
490                                 CopyDefaultHistory(&DefaultHist);\r
491                                 n = 0;\r
492                                 for(i = AskHistoryNum(); i > 0; i--)\r
493                                 {\r
494                                         if(GetHistoryByNum(i-1, &Hist) == FFFTP_SUCCESS)\r
495                                         {\r
496                                                 sprintf(Str, "History%d", n);\r
497                                                 if(CreateSubKey(hKey4, Str, &hKey5) == FFFTP_SUCCESS)\r
498                                                 {\r
499                                                         SaveStr(hKey5, "HostAdrs", Hist.HostAdrs, DefaultHist.HostAdrs);\r
500                                                         SaveStr(hKey5, "UserName", Hist.UserName, DefaultHist.UserName);\r
501                                                         SaveStr(hKey5, "Account", Hist.Account, DefaultHist.Account);\r
502                                                         SaveStr(hKey5, "LocalDir", Hist.LocalInitDir, NULL);\r
503                                                         SaveStr(hKey5, "RemoteDir", Hist.RemoteInitDir, DefaultHist.RemoteInitDir);\r
504                                                         SaveStr(hKey5, "Chmod", Hist.ChmodCmd, DefaultHist.ChmodCmd);\r
505                                                         SaveStr(hKey5, "Nlst", Hist.LsName, DefaultHist.LsName);\r
506                                                         SaveStr(hKey5, "Init", Hist.InitCmd, DefaultHist.InitCmd);\r
507                                                         EncodePassword(Hist.PassWord, Str);\r
508                                                         SaveStr(hKey5, "Password", Str, DefaultHist.PassWord);\r
509                                                         SaveIntNum(hKey5, "Port", Hist.Port, DefaultHist.Port);\r
510                                                         SaveIntNum(hKey5, "Kanji", Hist.KanjiCode, DefaultHist.KanjiCode);\r
511                                                         SaveIntNum(hKey5, "KanaCnv", Hist.KanaCnv, DefaultHist.KanaCnv);\r
512                                                         SaveIntNum(hKey5, "NameKanji", Hist.NameKanjiCode, DefaultHist.NameKanjiCode);\r
513                                                         SaveIntNum(hKey5, "NameKana", Hist.NameKanaCnv, DefaultHist.NameKanaCnv);\r
514                                                         SaveIntNum(hKey5, "Pasv", Hist.Pasv, DefaultHist.Pasv);\r
515                                                         SaveIntNum(hKey5, "Fwall", Hist.FireWall, DefaultHist.FireWall);\r
516                                                         SaveIntNum(hKey5, "List", Hist.ListCmdOnly, DefaultHist.ListCmdOnly);\r
517                                                         SaveIntNum(hKey5, "NLST-R", Hist.UseNLST_R, DefaultHist.UseNLST_R);\r
518                                                         SaveIntNum(hKey5, "Tzone", Hist.TimeZone, DefaultHist.TimeZone);\r
519                                                         SaveIntNum(hKey5, "Type", Hist.HostType, DefaultHist.HostType);\r
520                                                         SaveIntNum(hKey5, "Sync", Hist.SyncMove, DefaultHist.SyncMove);\r
521                                                         SaveIntNum(hKey5, "Fpath", Hist.NoFullPath, DefaultHist.NoFullPath);\r
522                                                         WriteBinaryToReg(hKey5, "Sort", &Hist.Sort, sizeof(Hist.Sort));\r
523                                                         SaveIntNum(hKey5, "Secu", Hist.Security, DefaultHist.Security);\r
524                                                         WriteIntValueToReg(hKey5, "TrType", Hist.Type);\r
525                                                         SaveIntNum(hKey5, "Dial", Hist.Dialup, DefaultHist.Dialup);\r
526                                                         SaveIntNum(hKey5, "UseIt", Hist.DialupAlways, DefaultHist.DialupAlways);\r
527                                                         SaveIntNum(hKey5, "Notify", Hist.DialupNotify, DefaultHist.DialupNotify);\r
528                                                         SaveStr(hKey5, "DialTo", Hist.DialEntry, DefaultHist.DialEntry);\r
529                                                         // 暗号化通信対応\r
530                                                         SaveIntNum(hKey5, "NoEncryption", Hist.UseNoEncryption, DefaultHist.UseNoEncryption);\r
531                                                         SaveIntNum(hKey5, "FTPES", Hist.UseFTPES, DefaultHist.UseFTPES);\r
532                                                         SaveIntNum(hKey5, "FTPIS", Hist.UseFTPIS, DefaultHist.UseFTPIS);\r
533                                                         SaveIntNum(hKey5, "SFTP", Hist.UseSFTP, DefaultHist.UseSFTP);\r
534                                                         EncodePassword(Hist.PrivateKey, Str);\r
535                                                         SaveStr(hKey5, "PKey", Str, DefaultHist.PrivateKey);\r
536                                                         // 同時接続対応\r
537                                                         SaveIntNum(hKey5, "ThreadCount", Hist.MaxThreadCount, DefaultHist.MaxThreadCount);\r
538                                                         SaveIntNum(hKey5, "ReuseCmdSkt", Hist.ReuseCmdSkt, DefaultHist.ReuseCmdSkt);\r
539                                                         // MLSD対応\r
540                                                         SaveIntNum(hKey5, "MLSD", Hist.UseMLSD, DefaultHist.UseMLSD);\r
541                                                         // IPv6対応\r
542                                                         SaveIntNum(hKey5, "NetType", Hist.NetType, DefaultHist.NetType);\r
543                                                         // 自動切断対策\r
544                                                         SaveIntNum(hKey5, "Noop", Hist.NoopInterval, DefaultHist.NoopInterval);\r
545                                                         // 再転送対応\r
546                                                         SaveIntNum(hKey5, "ErrMode", Hist.TransferErrorMode, DefaultHist.TransferErrorMode);\r
547                                                         SaveIntNum(hKey5, "ErrNotify", Hist.TransferErrorNotify, DefaultHist.TransferErrorNotify);\r
548                                                         // セッションあたりの転送量制限対策\r
549                                                         SaveIntNum(hKey5, "ErrReconnect", Hist.TransferErrorReconnect, DefaultHist.TransferErrorReconnect);\r
550 \r
551                                                         CloseSubKey(hKey5);\r
552                                                         n++;\r
553                                                 }\r
554                                         }\r
555                                 }\r
556                                 WriteIntValueToReg(hKey4, "SavedHist", n);\r
557 \r
558                                 /* 余分なヒストリがあったら削除 */\r
559                                 for(; n < 999; n++)\r
560                                 {\r
561                                         sprintf(Str, "History%d", n);\r
562                                         if(DeleteSubKey(hKey4, Str) != FFFTP_SUCCESS)\r
563                                                 break;\r
564                                 }\r
565 \r
566                                 /* ホストの設定を保存 */\r
567                                 CopyDefaultHost(&DefaultHost);\r
568                                 i = 0;\r
569                                 while(CopyHostFromList(i, &Host) == FFFTP_SUCCESS)\r
570                                 {\r
571                                         sprintf(Str, "Host%d", i);\r
572                                         if(CreateSubKey(hKey4, Str, &hKey5) == FFFTP_SUCCESS)\r
573                                         {\r
574 //                                              SaveIntNum(hKey5, "Set", Host.Level, DefaultHost.Level);\r
575                                                 WriteIntValueToReg(hKey5, "Set", Host.Level);\r
576                                                 SaveStr(hKey5, "HostName", Host.HostName, DefaultHost.HostName);\r
577                                                 if((Host.Level & SET_LEVEL_GROUP) == 0)\r
578                                                 {\r
579                                                         SaveStr(hKey5, "HostAdrs", Host.HostAdrs, DefaultHost.HostAdrs);\r
580                                                         SaveStr(hKey5, "UserName", Host.UserName, DefaultHost.UserName);\r
581                                                         SaveStr(hKey5, "Account", Host.Account, DefaultHost.Account);\r
582                                                         SaveStr(hKey5, "LocalDir", Host.LocalInitDir, NULL);\r
583                                                         SaveStr(hKey5, "RemoteDir", Host.RemoteInitDir, DefaultHost.RemoteInitDir);\r
584                                                         SaveStr(hKey5, "Chmod", Host.ChmodCmd, DefaultHost.ChmodCmd);\r
585                                                         SaveStr(hKey5, "Nlst", Host.LsName, DefaultHost.LsName);\r
586                                                         SaveStr(hKey5, "Init", Host.InitCmd, DefaultHost.InitCmd);\r
587 \r
588                                                         if(Host.Anonymous == NO)\r
589                                                                 EncodePassword(Host.PassWord, Str);\r
590                                                         else\r
591                                                                 strcpy(Str, DefaultHost.PassWord);\r
592                                                         SaveStr(hKey5, "Password", Str, DefaultHost.PassWord);\r
593 \r
594                                                         SaveIntNum(hKey5, "Port", Host.Port, DefaultHost.Port);\r
595                                                         SaveIntNum(hKey5, "Anonymous", Host.Anonymous, DefaultHost.Anonymous);\r
596                                                         SaveIntNum(hKey5, "Kanji", Host.KanjiCode, DefaultHost.KanjiCode);\r
597                                                         SaveIntNum(hKey5, "KanaCnv", Host.KanaCnv, DefaultHost.KanaCnv);\r
598                                                         SaveIntNum(hKey5, "NameKanji", Host.NameKanjiCode, DefaultHost.NameKanjiCode);\r
599                                                         SaveIntNum(hKey5, "NameKana", Host.NameKanaCnv, DefaultHost.NameKanaCnv);\r
600                                                         SaveIntNum(hKey5, "Pasv", Host.Pasv, DefaultHost.Pasv);\r
601                                                         SaveIntNum(hKey5, "Fwall", Host.FireWall, DefaultHost.FireWall);\r
602                                                         SaveIntNum(hKey5, "List", Host.ListCmdOnly, DefaultHost.ListCmdOnly);\r
603                                                         SaveIntNum(hKey5, "NLST-R", Host.UseNLST_R, DefaultHost.UseNLST_R);\r
604                                                         SaveIntNum(hKey5, "Last", Host.LastDir, DefaultHost.LastDir);\r
605                                                         SaveIntNum(hKey5, "Tzone", Host.TimeZone, DefaultHost.TimeZone);\r
606                                                         SaveIntNum(hKey5, "Type", Host.HostType, DefaultHost.HostType);\r
607                                                         SaveIntNum(hKey5, "Sync", Host.SyncMove, DefaultHost.SyncMove);\r
608                                                         SaveIntNum(hKey5, "Fpath", Host.NoFullPath, DefaultHost.NoFullPath);\r
609                                                         WriteBinaryToReg(hKey5, "Sort", &Host.Sort, sizeof(Host.Sort));\r
610                                                         SaveIntNum(hKey5, "Secu", Host.Security, DefaultHost.Security);\r
611 \r
612                                                         WriteMultiStringToReg(hKey5, "Bmarks", Host.BookMark);\r
613 \r
614                                                         SaveIntNum(hKey5, "Dial", Host.Dialup, DefaultHost.Dialup);\r
615                                                         SaveIntNum(hKey5, "UseIt", Host.DialupAlways, DefaultHost.DialupAlways);\r
616                                                         SaveIntNum(hKey5, "Notify", Host.DialupNotify, DefaultHost.DialupNotify);\r
617                                                         SaveStr(hKey5, "DialTo", Host.DialEntry, DefaultHost.DialEntry);\r
618                                                         // 暗号化通信対応\r
619                                                         SaveIntNum(hKey5, "NoEncryption", Host.UseNoEncryption, DefaultHost.UseNoEncryption);\r
620                                                         SaveIntNum(hKey5, "FTPES", Host.UseFTPES, DefaultHost.UseFTPES);\r
621                                                         SaveIntNum(hKey5, "FTPIS", Host.UseFTPIS, DefaultHost.UseFTPIS);\r
622                                                         SaveIntNum(hKey5, "SFTP", Host.UseSFTP, DefaultHost.UseSFTP);\r
623                                                         EncodePassword(Host.PrivateKey, Str);\r
624                                                         SaveStr(hKey5, "PKey", Str, DefaultHost.PrivateKey);\r
625                                                         // 同時接続対応\r
626                                                         SaveIntNum(hKey5, "ThreadCount", Host.MaxThreadCount, DefaultHost.MaxThreadCount);\r
627                                                         SaveIntNum(hKey5, "ReuseCmdSkt", Host.ReuseCmdSkt, DefaultHost.ReuseCmdSkt);\r
628                                                         // MLSD対応\r
629                                                         SaveIntNum(hKey5, "MLSD", Host.UseMLSD, DefaultHost.UseMLSD);\r
630                                                         // IPv6対応\r
631                                                         SaveIntNum(hKey5, "NetType", Host.NetType, DefaultHost.NetType);\r
632                                                         // 自動切断対策\r
633                                                         SaveIntNum(hKey5, "Noop", Host.NoopInterval, DefaultHost.NoopInterval);\r
634                                                         // 再転送対応\r
635                                                         SaveIntNum(hKey5, "ErrMode", Host.TransferErrorMode, DefaultHost.TransferErrorMode);\r
636                                                         SaveIntNum(hKey5, "ErrNotify", Host.TransferErrorNotify, DefaultHost.TransferErrorNotify);\r
637                                                         // セッションあたりの転送量制限対策\r
638                                                         SaveIntNum(hKey5, "ErrReconnect", Host.TransferErrorReconnect, DefaultHost.TransferErrorReconnect);\r
639                                                 }\r
640                                                 CloseSubKey(hKey5);\r
641                                         }\r
642                                         i++;\r
643                                 }\r
644                                 WriteIntValueToReg(hKey4, "SetNum", i);\r
645 \r
646                                 /* 余分なホストの設定があったら削除 */\r
647                                 for(; i < 998; i++)\r
648                                 {\r
649                                         sprintf(Str, "Host%d", i);\r
650                                         if(DeleteSubKey(hKey4, Str) != FFFTP_SUCCESS)\r
651                                                 break;\r
652                                 }\r
653 \r
654                                 if((i = AskCurrentHost()) == HOSTNUM_NOENTRY)\r
655                                         i = 0;\r
656                                 WriteIntValueToReg(hKey4, "CurSet", i);\r
657 \r
658                                 // 暗号化通信対応\r
659                                 WriteBinaryToReg(hKey4, "CertCacheHash", &CertificateCacheHash, sizeof(CertificateCacheHash));\r
660                                 strcpy(Buf, "");\r
661                                 StrCatOut((char*)&SSLRootCAFileHash, sizeof(SSLRootCAFileHash), Buf);\r
662                                 EncodePassword(Buf, Str);\r
663                                 WriteStringToReg(hKey4, "RootCertHash", Str);\r
664                                 // ファイルアイコン表示対応\r
665                                 WriteIntValueToReg(hKey4, "ListIcon", DispFileIcon);\r
666                                 // タイムスタンプのバグ修正\r
667                                 WriteIntValueToReg(hKey4, "ListSecond", DispTimeSeconds);\r
668                                 // ファイルの属性を数字で表示\r
669                                 WriteIntValueToReg(hKey4, "ListPermitNum", DispPermissionsNumber);\r
670                                 // ディレクトリ自動作成\r
671                                 WriteIntValueToReg(hKey4, "MakeDir", MakeAllDir);\r
672                                 // UTF-8対応\r
673                                 WriteIntValueToReg(hKey4, "Kanji", LocalKanjiCode);\r
674                                 // UPnP対応\r
675                                 WriteIntValueToReg(hKey4, "UPnP", UPnPEnabled);\r
676                                 // ローカル側自動更新\r
677                                 WriteIntValueToReg(hKey4, "ListRefresh", AutoRefreshFileList);\r
678                                 // 古い処理内容を消去\r
679                                 WriteIntValueToReg(hKey4, "OldLog", RemoveOldLog);\r
680                         }\r
681                         CloseSubKey(hKey4);\r
682                 }\r
683                 // 全設定暗号化対応\r
684                 EncryptSettings = NO;\r
685                 WriteBinaryToReg(hKey3, "EncryptAllChecksum", &EncryptSettingsChecksum, 20);\r
686                 CloseReg(hKey3);\r
687         }\r
688         return;\r
689 }\r
690 \r
691 /*----- レジストリ/INIファイルから設定値を呼び出す ---------------------------\r
692 *\r
693 *       この関数を複数回呼び出すと,ホスト設定が追加される.\r
694 *\r
695 *       Parameter\r
696 *               なし\r
697 *\r
698 *       Return Value\r
699 *               YES: 読み出し成功\r
700 *               NO:  読み出し失敗(設定無し)\r
701 *----------------------------------------------------------------------------*/\r
702 \r
703 int LoadRegistry(void)\r
704 {\r
705         void *hKey3;\r
706         void *hKey4;\r
707         void *hKey5;\r
708         int i;\r
709         int Sets;\r
710         // 暗号化通信対応\r
711 //      char Str[256];  /* ASCII_EXT_LENより大きい事 */\r
712         char Str[PRIVATE_KEY_LEN*4+1];\r
713         char Buf[FMAX_PATH+1];\r
714         // 全設定暗号化対応\r
715         char Buf2[FMAX_PATH+1];\r
716         char *Pos;\r
717         char *Pos2;\r
718         HOSTDATA Host;\r
719         HISTORYDATA Hist;\r
720         int Sts;\r
721         int Version;\r
722         // 全設定暗号化対応\r
723         BYTE Checksum[20];\r
724 \r
725         Sts = NO;\r
726 \r
727         SetRegType(REGTYPE_INI);\r
728         if((i = OpenReg("FFFTP", &hKey3)) != FFFTP_SUCCESS)\r
729         {\r
730                 if(AskForceIni() == NO)\r
731                 {\r
732                         SetRegType(REGTYPE_REG);\r
733                         i = OpenReg("FFFTP", &hKey3);\r
734                 }\r
735         }\r
736 \r
737         if(i == FFFTP_SUCCESS)\r
738         {\r
739 //              char checkbuf[48];\r
740                 int salt = 0;\r
741                 Sts = YES;\r
742 \r
743                 ReadIntValueFromReg(hKey3, "Version", &Version);\r
744                 // UTF-8対応\r
745                 if(Version < 1980)\r
746                         IniKanjiCode = KANJI_SJIS;\r
747 \r
748                 // 全設定暗号化対応\r
749                 if(Version >= 1990)\r
750                 {\r
751                         ReadIntValueFromReg(hKey3, "EncryptAll", &EncryptAllSettings);\r
752                         sprintf(Buf, "%d", EncryptAllSettings);\r
753                         ReadStringFromReg(hKey3, "EncryptAllDetector", Str, 255);\r
754                         DecodePassword(Str, Buf2);\r
755                         EncryptSettings = EncryptAllSettings;\r
756                         memset(&EncryptSettingsChecksum, 0, 20);\r
757                         if(strcmp(Buf, Buf2) != 0)\r
758                         {\r
759                                 switch(MessageBox(GetMainHwnd(), MSGJPN343, "FFFTP", MB_ABORTRETRYIGNORE | MB_DEFBUTTON2))\r
760                                 {\r
761                                 case IDABORT:\r
762                                         CloseReg(hKey3);\r
763                                         ClearRegistry();\r
764                                         ClearIni();\r
765                                         RestartAndTerminate();\r
766                                         break;\r
767                                 case IDRETRY:\r
768                                         EncryptSettingsError = YES;\r
769                                         break;\r
770                                 case IDIGNORE:\r
771                                         break;\r
772                                 }\r
773                         }\r
774                 }\r
775 \r
776                 if(OpenSubKey(hKey3, "Options", &hKey4) == FFFTP_SUCCESS)\r
777                 {\r
778                         ReadIntValueFromReg(hKey4, "WinPosX", &WinPosX);\r
779                         ReadIntValueFromReg(hKey4, "WinPosY", &WinPosY);\r
780                         ReadIntValueFromReg(hKey4, "WinWidth", &WinWidth);\r
781                         ReadIntValueFromReg(hKey4, "WinHeight", &WinHeight);\r
782                         ReadIntValueFromReg(hKey4, "LocalWidth", &LocalWidth);\r
783                         /* ↓旧バージョンのバグ対策 */\r
784                         LocalWidth = max1(0, LocalWidth);\r
785                         ReadIntValueFromReg(hKey4, "TaskHeight", &TaskHeight);\r
786                         /* ↓旧バージョンのバグ対策 */\r
787                         TaskHeight = max1(0, TaskHeight);\r
788                         ReadBinaryFromReg(hKey4, "LocalColm", &LocalTabWidth, sizeof(LocalTabWidth));\r
789                         ReadBinaryFromReg(hKey4, "RemoteColm", &RemoteTabWidth, sizeof(RemoteTabWidth));\r
790                         ReadIntValueFromReg(hKey4, "SwCmd", &Sizing);\r
791 \r
792                         ReadStringFromReg(hKey4, "UserMail", UserMailAdrs, USER_MAIL_LEN+1);\r
793                         ReadStringFromReg(hKey4, "Viewer", ViewerName[0], FMAX_PATH+1);\r
794                         ReadStringFromReg(hKey4, "Viewer2", ViewerName[1], FMAX_PATH+1);\r
795                         ReadStringFromReg(hKey4, "Viewer3", ViewerName[2], FMAX_PATH+1);\r
796 \r
797                         ReadIntValueFromReg(hKey4, "TrType", &TransMode);\r
798                         ReadIntValueFromReg(hKey4, "Recv", &RecvMode);\r
799                         ReadIntValueFromReg(hKey4, "Send", &SendMode);\r
800                         ReadIntValueFromReg(hKey4, "Move", &MoveMode);\r
801                         ReadStringFromReg(hKey4, "Path", DefaultLocalPath, FMAX_PATH+1);\r
802                         ReadIntValueFromReg(hKey4, "Time", &SaveTimeStamp);\r
803                         ReadIntValueFromReg(hKey4, "EOF", &RmEOF);\r
804                         ReadIntValueFromReg(hKey4, "Scolon", &VaxSemicolon);\r
805 \r
806                         ReadIntValueFromReg(hKey4, "RecvEx", &ExistMode);\r
807                         ReadIntValueFromReg(hKey4, "SendEx", &UpExistMode);\r
808 \r
809                         ReadIntValueFromReg(hKey4, "LFsort", &LocalFileSort);\r
810                         ReadIntValueFromReg(hKey4, "LDsort", &LocalDirSort);\r
811                         ReadIntValueFromReg(hKey4, "RFsort", &RemoteFileSort);\r
812                         ReadIntValueFromReg(hKey4, "RDsort", &RemoteDirSort);\r
813                         ReadIntValueFromReg(hKey4, "SortSave", &SortSave);\r
814 \r
815                         ReadIntValueFromReg(hKey4, "ListType", &ListType);\r
816                         ReadIntValueFromReg(hKey4, "Cache", &CacheEntry);\r
817                         ReadIntValueFromReg(hKey4, "CacheSave", &CacheSave);\r
818                         ReadIntValueFromReg(hKey4, "DotFile", &DotFile);\r
819                         ReadIntValueFromReg(hKey4, "Dclick", &DclickOpen);\r
820 \r
821                         ReadIntValueFromReg(hKey4, "ConS", &ConnectOnStart);\r
822                         ReadIntValueFromReg(hKey4, "OldDlg", &ConnectAndSet);\r
823                         ReadIntValueFromReg(hKey4, "RasClose", &RasClose);\r
824                         ReadIntValueFromReg(hKey4, "RasNotify", &RasCloseNotify);\r
825                         ReadIntValueFromReg(hKey4, "Qanony", &QuickAnonymous);\r
826                         ReadIntValueFromReg(hKey4, "PassHist", &PassToHist);\r
827                         ReadIntValueFromReg(hKey4, "SendQuit", &SendQuit);\r
828                         ReadIntValueFromReg(hKey4, "NoRas", &NoRasControl);\r
829 \r
830                         ReadIntValueFromReg(hKey4, "Debug", &DebugConsole);\r
831                         ReadIntValueFromReg(hKey4, "WinPos", &SaveWinPos);\r
832                         ReadIntValueFromReg(hKey4, "RegExp", &FindMode);\r
833                         ReadIntValueFromReg(hKey4, "Reg", &RegType);\r
834 \r
835                         if(ReadMultiStringFromReg(hKey4, "AsciiFile", AsciiExt, ASCII_EXT_LEN+1) == FFFTP_FAIL)\r
836                         {\r
837                                 /* 旧ASCIIモードの拡張子の設定を新しいものに変換 */\r
838                                 // アスキーモード判別の改良\r
839 //                              ReadStringFromReg(hKey4, "Ascii", Str, ASCII_EXT_LEN+1);\r
840 //                              memset(AsciiExt, NUL, ASCII_EXT_LEN+1);\r
841                                 Str[0] = NUL;\r
842                                 if(ReadStringFromReg(hKey4, "Ascii", Str, ASCII_EXT_LEN+1) == FFFTP_SUCCESS)\r
843                                         memset(AsciiExt, NUL, ASCII_EXT_LEN+1);\r
844                                 Pos = Str;\r
845                                 while(*Pos != NUL)\r
846                                 {\r
847                                         if((Pos2 = strchr(Pos, ';')) == NULL)\r
848                                                 Pos2 = strchr(Pos, NUL);\r
849                                         if((Pos2 - Pos) > 0)\r
850                                         {\r
851                                                 if((StrMultiLen(AsciiExt) + (Pos2 - Pos) + 2) >= ASCII_EXT_LEN)\r
852                                                         break;\r
853                                                 strcpy(AsciiExt + StrMultiLen(AsciiExt), "*.");\r
854                                                 strncpy(AsciiExt + StrMultiLen(AsciiExt) - 1, Pos, (Pos2 - Pos));\r
855                                         }\r
856                                         Pos = Pos2;\r
857                                         if(*Pos == ';')\r
858                                                 Pos++;\r
859                                 }\r
860                         }\r
861                         // アスキーモード判別の改良\r
862                         if(Version < 1986)\r
863                         {\r
864                                 Pos = "*.js\0*.vbs\0*.css\0*.rss\0*.rdf\0*.xml\0*.xhtml\0*.xht\0*.shtml\0*.shtm\0*.sh\0*.py\0*.rb\0*.properties\0*.sql\0*.asp\0*.aspx\0*.php\0*.htaccess\0";\r
865                                 while(*Pos != NUL)\r
866                                 {\r
867                                         Pos2 = AsciiExt;\r
868                                         while(*Pos2 != NUL)\r
869                                         {\r
870                                                 if(_stricmp(Pos2, Pos) == 0)\r
871                                                         break;\r
872                                                 Pos2 = strchr(Pos2, NUL) + 1;\r
873                                         }\r
874                                         if(*Pos2 == NUL)\r
875                                         {\r
876                                                 if((StrMultiLen(AsciiExt) + strlen(Pos) + 2) < ASCII_EXT_LEN)\r
877                                                         strncpy(AsciiExt + StrMultiLen(AsciiExt), Pos, strlen(Pos) + 2);\r
878                                         }\r
879                                         Pos = strchr(Pos, NUL) + 1;\r
880                                 }\r
881                         }\r
882 \r
883                         ReadIntValueFromReg(hKey4, "LowUp", &FnameCnv);\r
884                         ReadIntValueFromReg(hKey4, "Tout", &TimeOut);\r
885 \r
886                         ReadMultiStringFromReg(hKey4, "NoTrn", MirrorNoTrn, MIRROR_LEN+1);\r
887                         ReadMultiStringFromReg(hKey4, "NoDel", MirrorNoDel, MIRROR_LEN+1);\r
888                         ReadIntValueFromReg(hKey4, "MirFile", &MirrorFnameCnv);\r
889                         ReadIntValueFromReg(hKey4, "MirUNot", &MirUpDelNotify);\r
890                         ReadIntValueFromReg(hKey4, "MirDNot", &MirDownDelNotify);\r
891 \r
892                         if(ReadStringFromReg(hKey4, "ListFont", Str, 256) == FFFTP_SUCCESS)\r
893                         {\r
894                                 if(RestoreFontData(Str, &ListLogFont) == FFFTP_SUCCESS)\r
895                                         ListFont = CreateFontIndirect(&ListLogFont);\r
896                         }\r
897                         ReadIntValueFromReg(hKey4, "ListHide", &DispIgnoreHide);\r
898                         ReadIntValueFromReg(hKey4, "ListDrv", &DispDrives);\r
899 \r
900                         ReadStringFromReg(hKey4, "FwallHost", FwallHost, HOST_ADRS_LEN+1);\r
901                         ReadStringFromReg(hKey4, "FwallUser", FwallUser, USER_NAME_LEN+1);\r
902                         ReadStringFromReg(hKey4, "FwallPass", Str, 255);\r
903                         DecodePassword(Str, FwallPass);\r
904                         ReadIntValueFromReg(hKey4, "FwallPort", &FwallPort);\r
905                         ReadIntValueFromReg(hKey4, "FwallType", &FwallType);\r
906                         ReadIntValueFromReg(hKey4, "FwallDef", &FwallDefault);\r
907                         ReadIntValueFromReg(hKey4, "FwallSec", &FwallSecurity);\r
908                         ReadIntValueFromReg(hKey4, "PasvDef", &PasvDefault);\r
909                         ReadIntValueFromReg(hKey4, "FwallRes", &FwallResolv);\r
910                         ReadIntValueFromReg(hKey4, "FwallLow", &FwallLower);\r
911                         ReadIntValueFromReg(hKey4, "FwallDel", &FwallDelimiter);\r
912 \r
913                         ReadIntValueFromReg(hKey4, "SndConSw", &Sound[SND_CONNECT].On);\r
914                         ReadIntValueFromReg(hKey4, "SndTrnSw", &Sound[SND_TRANS].On);\r
915                         ReadIntValueFromReg(hKey4, "SndErrSw", &Sound[SND_ERROR].On);\r
916                         ReadStringFromReg(hKey4, "SndCon", Sound[SND_CONNECT].Fname, FMAX_PATH+1);\r
917                         ReadStringFromReg(hKey4, "SndTrn", Sound[SND_TRANS].Fname, FMAX_PATH+1);\r
918                         ReadStringFromReg(hKey4, "SndErr", Sound[SND_ERROR].Fname, FMAX_PATH+1);\r
919 \r
920                         ReadMultiStringFromReg(hKey4, "DefAttr", DefAttrList, DEFATTRLIST_LEN+1);\r
921 \r
922                         ReadStringFromReg(hKey4, "Tmp", TmpPath, FMAX_PATH+1);\r
923 \r
924                         ReadBinaryFromReg(hKey4, "Hdlg", &HostDlgSize, sizeof(SIZE));\r
925                         ReadBinaryFromReg(hKey4, "Bdlg", &BmarkDlgSize, sizeof(SIZE));\r
926                         ReadBinaryFromReg(hKey4, "Mdlg", &MirrorDlgSize, sizeof(SIZE));\r
927 \r
928                         ReadIntValueFromReg(hKey4, "FAttrSw", &FolderAttr);\r
929                         ReadIntValueFromReg(hKey4, "FAttr", &FolderAttrNum);\r
930 \r
931                         ReadIntValueFromReg(hKey4, "NoSave", &SuppressSave);\r
932 \r
933                         ReadIntValueFromReg(hKey4, "HistNum", &FileHist);\r
934 //                      ReadMultiStringFromReg(hKey4, "Hist", Hist, (FMAX_PATH+1)*HISTORY_MAX+1);\r
935 \r
936                         /* ヒストリの設定を読み込む */\r
937                         Sets = 0;\r
938                         ReadIntValueFromReg(hKey4, "SavedHist", &Sets);\r
939 \r
940                         for(i = 0; i < Sets; i++)\r
941                         {\r
942                                 sprintf(Str, "History%d", i);\r
943                                 if(OpenSubKey(hKey4, Str, &hKey5) == FFFTP_SUCCESS)\r
944                                 {\r
945                                         CopyDefaultHistory(&Hist);\r
946 \r
947                                         ReadStringFromReg(hKey5, "HostAdrs", Hist.HostAdrs, HOST_ADRS_LEN+1);\r
948                                         ReadStringFromReg(hKey5, "UserName", Hist.UserName, USER_NAME_LEN+1);\r
949                                         ReadStringFromReg(hKey5, "Account", Hist.Account, ACCOUNT_LEN+1);\r
950                                         ReadStringFromReg(hKey5, "LocalDir", Hist.LocalInitDir, INIT_DIR_LEN+1);\r
951                                         ReadStringFromReg(hKey5, "RemoteDir", Hist.RemoteInitDir, INIT_DIR_LEN+1);\r
952                                         ReadStringFromReg(hKey5, "Chmod", Hist.ChmodCmd, CHMOD_CMD_LEN+1);\r
953                                         ReadStringFromReg(hKey5, "Nlst", Hist.LsName, NLST_NAME_LEN+1);\r
954                                         ReadStringFromReg(hKey5, "Init", Hist.InitCmd, INITCMD_LEN+1);\r
955                                         ReadIntValueFromReg(hKey5, "Port", &Hist.Port);\r
956                                         ReadIntValueFromReg(hKey5, "Kanji", &Hist.KanjiCode);\r
957                                         ReadIntValueFromReg(hKey5, "KanaCnv", &Hist.KanaCnv);\r
958                                         ReadIntValueFromReg(hKey5, "NameKanji", &Hist.NameKanjiCode);\r
959                                         ReadIntValueFromReg(hKey5, "NameKana", &Hist.NameKanaCnv);\r
960                                         ReadIntValueFromReg(hKey5, "Pasv", &Hist.Pasv);\r
961                                         ReadIntValueFromReg(hKey5, "Fwall", &Hist.FireWall);\r
962                                         ReadIntValueFromReg(hKey5, "List", &Hist.ListCmdOnly);\r
963                                         ReadIntValueFromReg(hKey5, "NLST-R", &Hist.UseNLST_R);\r
964                                         ReadIntValueFromReg(hKey5, "Tzone", &Hist.TimeZone);\r
965                                         ReadIntValueFromReg(hKey5, "Type", &Hist.HostType);\r
966                                         ReadIntValueFromReg(hKey5, "Sync", &Hist.SyncMove);\r
967                                         ReadIntValueFromReg(hKey5, "Fpath", &Hist.NoFullPath);\r
968                                         ReadBinaryFromReg(hKey5, "Sort", &Hist.Sort, sizeof(Hist.Sort));\r
969                                         ReadIntValueFromReg(hKey5, "Secu", &Hist.Security);\r
970                                         ReadIntValueFromReg(hKey5, "TrType", &Hist.Type);\r
971                                         strcpy(Str, "");\r
972                                         ReadStringFromReg(hKey5, "Password", Str, 255);\r
973                                         DecodePassword(Str, Hist.PassWord);\r
974                                         ReadIntValueFromReg(hKey5, "Dial", &Hist.Dialup);\r
975                                         ReadIntValueFromReg(hKey5, "UseIt", &Hist.DialupAlways);\r
976                                         ReadIntValueFromReg(hKey5, "Notify", &Hist.DialupNotify);\r
977                                         ReadStringFromReg(hKey5, "DialTo", Hist.DialEntry, RAS_NAME_LEN+1);\r
978                                         // 暗号化通信対応\r
979                                         ReadIntValueFromReg(hKey5, "NoEncryption", &Hist.UseNoEncryption);\r
980                                         ReadIntValueFromReg(hKey5, "FTPES", &Hist.UseFTPES);\r
981                                         ReadIntValueFromReg(hKey5, "FTPIS", &Hist.UseFTPIS);\r
982                                         ReadIntValueFromReg(hKey5, "SFTP", &Hist.UseSFTP);\r
983                                         strcpy(Str, "");\r
984                                         ReadStringFromReg(hKey5, "PKey", Str, PRIVATE_KEY_LEN*4+1);\r
985                                         DecodePassword(Str, Hist.PrivateKey);\r
986                                         // 同時接続対応\r
987                                         ReadIntValueFromReg(hKey5, "ThreadCount", &Hist.MaxThreadCount);\r
988                                         ReadIntValueFromReg(hKey5, "ReuseCmdSkt", &Hist.ReuseCmdSkt);\r
989                                         // MLSD対応\r
990                                         ReadIntValueFromReg(hKey5, "MLSD", &Hist.UseMLSD);\r
991                                         // IPv6対応\r
992                                         ReadIntValueFromReg(hKey5, "NetType", &Hist.NetType);\r
993                                         // 自動切断対策\r
994                                         ReadIntValueFromReg(hKey5, "Noop", &Hist.NoopInterval);\r
995                                         // 再転送対応\r
996                                         ReadIntValueFromReg(hKey5, "ErrMode", &Hist.TransferErrorMode);\r
997                                         ReadIntValueFromReg(hKey5, "ErrNotify", &Hist.TransferErrorNotify);\r
998 \r
999                                         CloseSubKey(hKey5);\r
1000                                         AddHistoryToHistory(&Hist);\r
1001                                 }\r
1002                         }\r
1003 \r
1004                         /* ホストの設定を読み込む */\r
1005                         Sets = 0;\r
1006                         ReadIntValueFromReg(hKey4, "SetNum", &Sets);\r
1007 \r
1008                         for(i = 0; i < Sets; i++)\r
1009                         {\r
1010                                 sprintf(Str, "Host%d", i);\r
1011                                 if(OpenSubKey(hKey4, Str, &hKey5) == FFFTP_SUCCESS)\r
1012                                 {\r
1013                                         CopyDefaultHost(&Host);\r
1014                                         /* 下位互換性のため */\r
1015                                         // SourceForge.JPによるフォーク\r
1016 //                                      if(Version < VER_NUM)\r
1017                                         if(Version < 1921)\r
1018                                         {\r
1019                                                 Host.Pasv = NO;\r
1020                                                 Host.ListCmdOnly = NO;\r
1021                                         }\r
1022                                         // 1.97b以前はデフォルトでShift_JIS\r
1023                                         if(Version < 1980)\r
1024                                                 Host.NameKanjiCode = KANJI_SJIS;\r
1025                                         ReadIntValueFromReg(hKey5, "Set", &Host.Level);\r
1026 \r
1027                                         ReadStringFromReg(hKey5, "HostName", Host.HostName, HOST_NAME_LEN+1);\r
1028                                         ReadStringFromReg(hKey5, "HostAdrs", Host.HostAdrs, HOST_ADRS_LEN+1);\r
1029                                         ReadStringFromReg(hKey5, "UserName", Host.UserName, USER_NAME_LEN+1);\r
1030                                         ReadStringFromReg(hKey5, "Account", Host.Account, ACCOUNT_LEN+1);\r
1031                                         ReadStringFromReg(hKey5, "LocalDir", Host.LocalInitDir, INIT_DIR_LEN+1);\r
1032                                         ReadStringFromReg(hKey5, "RemoteDir", Host.RemoteInitDir, INIT_DIR_LEN+1);\r
1033                                         ReadStringFromReg(hKey5, "Chmod", Host.ChmodCmd, CHMOD_CMD_LEN+1);\r
1034                                         ReadStringFromReg(hKey5, "Nlst", Host.LsName, NLST_NAME_LEN+1);\r
1035                                         ReadStringFromReg(hKey5, "Init", Host.InitCmd, INITCMD_LEN+1);\r
1036                                         ReadIntValueFromReg(hKey5, "Port", &Host.Port);\r
1037                                         ReadIntValueFromReg(hKey5, "Anonymous", &Host.Anonymous);\r
1038                                         ReadIntValueFromReg(hKey5, "Kanji", &Host.KanjiCode);\r
1039                                         // 1.98b以前のUTF-8はBOMあり\r
1040                                         if(Version < 1983)\r
1041                                         {\r
1042                                                 if(Host.KanjiCode == KANJI_UTF8N)\r
1043                                                         Host.KanjiCode = KANJI_UTF8BOM;\r
1044                                         }\r
1045                                         ReadIntValueFromReg(hKey5, "KanaCnv", &Host.KanaCnv);\r
1046                                         ReadIntValueFromReg(hKey5, "NameKanji", &Host.NameKanjiCode);\r
1047                                         ReadIntValueFromReg(hKey5, "NameKana", &Host.NameKanaCnv);\r
1048                                         ReadIntValueFromReg(hKey5, "Pasv", &Host.Pasv);\r
1049                                         ReadIntValueFromReg(hKey5, "Fwall", &Host.FireWall);\r
1050                                         ReadIntValueFromReg(hKey5, "List", &Host.ListCmdOnly);\r
1051                                         ReadIntValueFromReg(hKey5, "NLST-R", &Host.UseNLST_R);\r
1052                                         ReadIntValueFromReg(hKey5, "Last", &Host.LastDir);\r
1053                                         ReadIntValueFromReg(hKey5, "Tzone", &Host.TimeZone);\r
1054                                         ReadIntValueFromReg(hKey5, "Type", &Host.HostType);\r
1055                                         ReadIntValueFromReg(hKey5, "Sync", &Host.SyncMove);\r
1056                                         ReadIntValueFromReg(hKey5, "Fpath", &Host.NoFullPath);\r
1057                                         ReadBinaryFromReg(hKey5, "Sort", &Host.Sort, sizeof(Host.Sort));\r
1058                                         ReadIntValueFromReg(hKey5, "Secu", &Host.Security);\r
1059                                         if(Host.Anonymous != YES)\r
1060                                         {\r
1061                                                 strcpy(Str, "");\r
1062                                                 ReadStringFromReg(hKey5, "Password", Str, 255);\r
1063                                                 DecodePassword(Str, Host.PassWord);\r
1064                                         }\r
1065                                         else\r
1066                                                 strcpy(Host.PassWord, UserMailAdrs);\r
1067 \r
1068                                         ReadMultiStringFromReg(hKey5, "Bmarks", Host.BookMark, BOOKMARK_SIZE);\r
1069 \r
1070                                         ReadIntValueFromReg(hKey5, "Dial", &Host.Dialup);\r
1071                                         ReadIntValueFromReg(hKey5, "UseIt", &Host.DialupAlways);\r
1072                                         ReadIntValueFromReg(hKey5, "Notify", &Host.DialupNotify);\r
1073                                         ReadStringFromReg(hKey5, "DialTo", Host.DialEntry, RAS_NAME_LEN+1);\r
1074                                         // 暗号化通信対応\r
1075                                         ReadIntValueFromReg(hKey5, "NoEncryption", &Host.UseNoEncryption);\r
1076                                         ReadIntValueFromReg(hKey5, "FTPES", &Host.UseFTPES);\r
1077                                         ReadIntValueFromReg(hKey5, "FTPIS", &Host.UseFTPIS);\r
1078                                         ReadIntValueFromReg(hKey5, "SFTP", &Host.UseSFTP);\r
1079                                         strcpy(Str, "");\r
1080                                         ReadStringFromReg(hKey5, "PKey", Str, PRIVATE_KEY_LEN*4+1);\r
1081                                         DecodePassword(Str, Host.PrivateKey);\r
1082                                         // 同時接続対応\r
1083                                         ReadIntValueFromReg(hKey5, "ThreadCount", &Host.MaxThreadCount);\r
1084                                         ReadIntValueFromReg(hKey5, "ReuseCmdSkt", &Host.ReuseCmdSkt);\r
1085                                         // 1.98d以前で同時接続数が1より大きい場合はソケットの再利用なし\r
1086                                         if(Version < 1985)\r
1087                                         {\r
1088                                                 if(Host.MaxThreadCount > 1)\r
1089                                                         Host.ReuseCmdSkt = NO;\r
1090                                         }\r
1091                                         // MLSD対応\r
1092                                         ReadIntValueFromReg(hKey5, "MLSD", &Host.UseMLSD);\r
1093                                         // IPv6対応\r
1094                                         ReadIntValueFromReg(hKey5, "NetType", &Host.NetType);\r
1095                                         // 自動切断対策\r
1096                                         ReadIntValueFromReg(hKey5, "Noop", &Host.NoopInterval);\r
1097                                         // 再転送対応\r
1098                                         ReadIntValueFromReg(hKey5, "ErrMode", &Host.TransferErrorMode);\r
1099                                         ReadIntValueFromReg(hKey5, "ErrNotify", &Host.TransferErrorNotify);\r
1100                                         // セッションあたりの転送量制限対策\r
1101                                         ReadIntValueFromReg(hKey5, "ErrReconnect", &Host.TransferErrorReconnect);\r
1102 \r
1103                                         CloseSubKey(hKey5);\r
1104 \r
1105                                         AddHostToList(&Host, -1, Host.Level);\r
1106                                 }\r
1107                         }\r
1108 \r
1109                         ReadIntValueFromReg(hKey4, "CurSet", &Sets);\r
1110                         SetCurrentHost(Sets);\r
1111 \r
1112                         // 暗号化通信対応\r
1113                         ReadBinaryFromReg(hKey4, "CertCacheHash", &CertificateCacheHash, sizeof(CertificateCacheHash));\r
1114                         ReadStringFromReg(hKey4, "RootCertHash", Str, PRIVATE_KEY_LEN*4+1);\r
1115                         DecodePassword(Str, Buf);\r
1116                         StrReadIn(Buf, sizeof(SSLRootCAFileHash), (char*)&SSLRootCAFileHash);\r
1117                         // ファイルアイコン表示対応\r
1118                         ReadIntValueFromReg(hKey4, "ListIcon", &DispFileIcon);\r
1119                         // タイムスタンプのバグ修正\r
1120                         ReadIntValueFromReg(hKey4, "ListSecond", &DispTimeSeconds);\r
1121                         // ファイルの属性を数字で表示\r
1122                         ReadIntValueFromReg(hKey4, "ListPermitNum", &DispPermissionsNumber);\r
1123                         // ディレクトリ自動作成\r
1124                         ReadIntValueFromReg(hKey4, "MakeDir", &MakeAllDir);\r
1125                         // UTF-8対応\r
1126                         ReadIntValueFromReg(hKey4, "Kanji", &LocalKanjiCode);\r
1127                         // UPnP対応\r
1128                         ReadIntValueFromReg(hKey4, "UPnP", &UPnPEnabled);\r
1129                         // ローカル側自動更新\r
1130                         ReadIntValueFromReg(hKey4, "ListRefresh", &AutoRefreshFileList);\r
1131                         // 古い処理内容を消去\r
1132                         ReadIntValueFromReg(hKey4, "OldLog", &RemoveOldLog);\r
1133 \r
1134                         CloseSubKey(hKey4);\r
1135                 }\r
1136                 // 全設定暗号化対応\r
1137                 EncryptSettings = NO;\r
1138                 if(Version >= 1990)\r
1139                 {\r
1140                         memset(&Checksum, 0, 20);\r
1141                         ReadBinaryFromReg(hKey3, "EncryptAllChecksum", &Checksum, 20);\r
1142                         if(memcmp(&Checksum, &EncryptSettingsChecksum, 20) != 0)\r
1143                         {\r
1144                                 switch(MessageBox(GetMainHwnd(), MSGJPN343, "FFFTP", MB_ABORTRETRYIGNORE | MB_DEFBUTTON2))\r
1145                                 {\r
1146                                 case IDABORT:\r
1147                                         CloseReg(hKey3);\r
1148                                         ClearRegistry();\r
1149                                         ClearIni();\r
1150                                         RestartAndTerminate();\r
1151                                         break;\r
1152                                 case IDRETRY:\r
1153                                         EncryptSettingsError = YES;\r
1154                                         break;\r
1155                                 case IDIGNORE:\r
1156                                         break;\r
1157                                 }\r
1158                         }\r
1159                 }\r
1160                 CloseReg(hKey3);\r
1161         }\r
1162         else\r
1163         {\r
1164                 /*===== 最初の起動時(設定が無い) =====*/\r
1165 \r
1166 #if 0\r
1167                 strcpy(UserMailAdrs, "");\r
1168                 strcpy(Str, "");\r
1169                 if(InputDialogBox(mailadrs_dlg, HWND_DESKTOP, NULL, Str, USER_MAIL_LEN+1, &i, IDH_HELP_TOPIC_0000001) == YES)\r
1170                         strcpy(UserMailAdrs, Str);\r
1171 \r
1172                 for(i = 0; i < SAMPLE_HOSTS; i++)\r
1173                 {\r
1174                         CopyDefaultHost(&Host);\r
1175                         Host.Level = Sample[i].Level;\r
1176                         strcpy(Host.PassWord, UserMailAdrs);\r
1177                         strcpy(Host.HostName, Sample[i].HostName);\r
1178                         strcpy(Host.HostAdrs, Sample[i].HostAdrs);\r
1179                         strcpy(Host.UserName, "anonymous");\r
1180                         AddHostToList(&Host, -1, Host.Level);\r
1181                 }\r
1182 #endif\r
1183         }\r
1184         return(Sts);\r
1185 }\r
1186 \r
1187 \r
1188 /*----- 隠しドライブ情報を取得 ------------------------------------------------\r
1189 *\r
1190 *       Parameter\r
1191 *               なし\r
1192 *\r
1193 *       Return Value\r
1194 *               DWORD \r
1195 *                       YES/NO=設定無し\r
1196 *----------------------------------------------------------------------------*/\r
1197 \r
1198 DWORD LoadHideDriveListRegistry(void)\r
1199 {\r
1200         HKEY hKey1;\r
1201         HKEY hKey2;\r
1202         HKEY hKey3;\r
1203         HKEY hKey4;\r
1204         HKEY hKey5;\r
1205         HKEY hKey6;\r
1206         DWORD Size;\r
1207         DWORD Type;\r
1208         DWORD Ret;\r
1209 \r
1210         Ret = 0;\r
1211         if(RegOpenKeyEx(HKEY_CURRENT_USER, "Software", 0, KEY_READ, &hKey1) == ERROR_SUCCESS)\r
1212         {\r
1213                 if(RegOpenKeyEx(hKey1, "Microsoft", 0, KEY_READ, &hKey2) == ERROR_SUCCESS)\r
1214                 {\r
1215                         if(RegOpenKeyEx(hKey2, "Windows", 0, KEY_READ, &hKey3) == ERROR_SUCCESS)\r
1216                         {\r
1217                                 if(RegOpenKeyEx(hKey3, "CurrentVersion", 0, KEY_READ, &hKey4) == ERROR_SUCCESS)\r
1218                                 {\r
1219                                         if(RegOpenKeyEx(hKey4, "Policies", 0, KEY_READ, &hKey5) == ERROR_SUCCESS)\r
1220                                         {\r
1221                                                 if(RegOpenKeyEx(hKey5, "Explorer", 0, KEY_READ, &hKey6) == ERROR_SUCCESS)\r
1222                                                 {\r
1223                                                         Size = sizeof(DWORD);\r
1224                                                         RegQueryValueEx(hKey6, "NoDrives", NULL, &Type, (BYTE *)&Ret, &Size);\r
1225                                                         RegCloseKey(hKey6);\r
1226                                                 }\r
1227                                                 RegCloseKey(hKey5);\r
1228                                         }\r
1229                                         RegCloseKey(hKey4);\r
1230                                 }\r
1231                                 RegCloseKey(hKey3);\r
1232                         }\r
1233                         RegCloseKey(hKey2);\r
1234                 }\r
1235                 RegCloseKey(hKey1);\r
1236         }\r
1237         return(Ret);\r
1238 }\r
1239 \r
1240 \r
1241 /*----- レジストリの設定値をクリア --------------------------------------------\r
1242 *\r
1243 *       Parameter\r
1244 *               なし\r
1245 *\r
1246 *       Return Value\r
1247 *               なし\r
1248 *----------------------------------------------------------------------------*/\r
1249 \r
1250 void ClearRegistry(void)\r
1251 {\r
1252         HKEY hKey2;\r
1253         HKEY hKey3;\r
1254         HKEY hKey4;\r
1255         DWORD Dispos;\r
1256         char Str[20];\r
1257         int i;\r
1258 \r
1259         if(RegCreateKeyEx(HKEY_CURRENT_USER, "Software\\Sota", 0, "", REG_OPTION_NON_VOLATILE, KEY_CREATE_SUB_KEY, NULL, &hKey2, &Dispos) == ERROR_SUCCESS)\r
1260         {\r
1261                 if(RegCreateKeyEx(hKey2, "FFFTP", 0, "", REG_OPTION_NON_VOLATILE, KEY_WRITE, NULL, &hKey3, &Dispos) == ERROR_SUCCESS)\r
1262                 {\r
1263                         if(RegCreateKeyEx(hKey3, "Options", 0, "", REG_OPTION_NON_VOLATILE, KEY_WRITE, NULL, &hKey4, &Dispos) == ERROR_SUCCESS)\r
1264                         {\r
1265                                 for(i = 0; ; i++)\r
1266                                 {\r
1267                                         sprintf(Str, "Host%d", i);\r
1268                                         if(RegDeleteKey(hKey4, Str) != ERROR_SUCCESS)\r
1269                                                 break;\r
1270                                 }\r
1271                                 for(i = 0; ; i++)\r
1272                                 {\r
1273                                         sprintf(Str, "History%d", i);\r
1274                                         if(RegDeleteKey(hKey4, Str) != ERROR_SUCCESS)\r
1275                                                 break;\r
1276                                 }\r
1277                                 RegCloseKey(hKey4);\r
1278                         }\r
1279                         RegDeleteKey(hKey3, "Options");\r
1280                         RegCloseKey(hKey3);\r
1281                 }\r
1282                 RegDeleteKey(hKey2, "FFFTP");\r
1283                 RegCloseKey(hKey2);\r
1284         }\r
1285         return;\r
1286 }\r
1287 \r
1288 \r
1289 // ポータブル版判定\r
1290 void ClearIni(void)\r
1291 {\r
1292 //      FILE *Strm;\r
1293 //      if((Strm = fopen(AskIniFilePath(), "rt")) != NULL)\r
1294 //      {\r
1295 //              fclose(Strm);\r
1296 //              MoveFileToTrashCan(AskIniFilePath());\r
1297 //      }\r
1298         DeleteFile(AskIniFilePath());\r
1299         return;\r
1300 }\r
1301 \r
1302 \r
1303 /*----- 設定をファイルに保存 --------------------------------------------------\r
1304 *\r
1305 *       Parameter\r
1306 *               なし\r
1307 *\r
1308 *       Return Value\r
1309 *               なし\r
1310 *----------------------------------------------------------------------------*/\r
1311 \r
1312 void SaveSettingsToFile(void)\r
1313 {\r
1314         char Tmp[FMAX_PATH*2];\r
1315         char Fname[FMAX_PATH+1];\r
1316         // 任意のコードが実行されるバグ修正\r
1317         char CurDir[FMAX_PATH+1];\r
1318         char SysDir[FMAX_PATH+1];\r
1319 \r
1320         if(RegType == REGTYPE_REG)\r
1321         {\r
1322                 strcpy(Fname, "FFFTP.reg");\r
1323                 if(SelectFile(GetMainHwnd(), Fname, MSGJPN286, MSGJPN287, "reg", OFN_EXTENSIONDIFFERENT | OFN_OVERWRITEPROMPT, 1) == TRUE)\r
1324                 {\r
1325                         sprintf(Tmp, "/e \x22%s\x22 HKEY_CURRENT_USER\\Software\\sota\\FFFTP", Fname);\r
1326                         // 任意のコードが実行されるバグ修正\r
1327 //                      if(ShellExecute(NULL, "open", "regedit", Tmp, ".", SW_SHOW) <= (HINSTANCE)32)\r
1328 //                      {\r
1329 //                              MessageBox(NULL, MSGJPN285, "FFFTP", MB_OK);\r
1330 //                      }\r
1331                         if(GetCurrentDirectory(FMAX_PATH, CurDir) > 0)\r
1332                         {\r
1333                                 if(GetSystemDirectory(SysDir, FMAX_PATH) > 0)\r
1334                                 {\r
1335                                         if(SetCurrentDirectory(SysDir))\r
1336                                         {\r
1337                                                 if(ShellExecute(NULL, "open", "regedit", Tmp, NULL, SW_SHOW) <= (HINSTANCE)32)\r
1338                                                 {\r
1339                                                         MessageBox(NULL, MSGJPN285, "FFFTP", MB_OK);\r
1340                                                 }\r
1341                                                 SetCurrentDirectory(CurDir);\r
1342                                         }\r
1343                                 }\r
1344                         }\r
1345                 }\r
1346         }\r
1347         else\r
1348         {\r
1349                 strcpy(Fname, "FFFTP-Backup.ini");\r
1350                 if(SelectFile(GetMainHwnd(), Fname, MSGJPN286, MSGJPN288, "ini", OFN_EXTENSIONDIFFERENT | OFN_OVERWRITEPROMPT, 1) == TRUE)\r
1351                 {\r
1352                         CopyFile(AskIniFilePath(), Fname, FALSE);\r
1353                 }\r
1354         }\r
1355         return;\r
1356 }\r
1357 \r
1358 \r
1359 /*----- 設定をファイルから復元 ------------------------------------------------\r
1360 *\r
1361 *       Parameter\r
1362 *               なし\r
1363 *\r
1364 *       Return Value\r
1365 *               int     ロードしたかどうか (YES/NO)\r
1366 *----------------------------------------------------------------------------*/\r
1367 \r
1368 int LoadSettingsFromFile(void)\r
1369 {\r
1370         int Ret;\r
1371         char Tmp[FMAX_PATH*2];\r
1372         char Fname[FMAX_PATH+1];\r
1373         // 任意のコードが実行されるバグ修正\r
1374         char CurDir[FMAX_PATH+1];\r
1375         char SysDir[FMAX_PATH+1];\r
1376 \r
1377         Ret = NO;\r
1378         strcpy(Fname, "");\r
1379         if(SelectFile(GetMainHwnd(), Fname, MSGJPN291, MSGJPN290, "", OFN_FILEMUSTEXIST, 0) == TRUE)\r
1380         {\r
1381                 if((strlen(Fname) >= 5) && (_stricmp(&Fname[strlen(Fname)-4], ".reg") == 0))\r
1382                 {\r
1383                         sprintf(Tmp, "\x22%s\x22", Fname);\r
1384                         // 任意のコードが実行されるバグ修正\r
1385 //                      if(ShellExecute(NULL, "open", "regedit", Tmp, ".", SW_SHOW) <= (HINSTANCE)32)\r
1386 //                      {\r
1387 //                              MessageBox(NULL, MSGJPN285, "FFFTP", MB_OK);\r
1388 //                      }\r
1389 //                      else\r
1390 //                      {\r
1391 //                              Ret = YES;\r
1392 //                              /* レジストリエディタが終了するのを待つ */\r
1393 //                              WaitForSingleObject(Info.hProcess, INFINITE);\r
1394 //                      }\r
1395                         if(GetCurrentDirectory(FMAX_PATH, CurDir) > 0)\r
1396                         {\r
1397                                 if(GetSystemDirectory(SysDir, FMAX_PATH) > 0)\r
1398                                 {\r
1399                                         if(SetCurrentDirectory(SysDir))\r
1400                                         {\r
1401                                                 if(ShellExecute(NULL, "open", "regedit", Tmp, NULL, SW_SHOW) <= (HINSTANCE)32)\r
1402                                                 {\r
1403                                                         MessageBox(NULL, MSGJPN285, "FFFTP", MB_OK);\r
1404                                                 }\r
1405                                                 else\r
1406                                                 {\r
1407                                                         Ret = YES;\r
1408                                                         /* レジストリエディタが終了するのを待つ */\r
1409 //                                                      WaitForSingleObject(Info.hProcess, INFINITE);\r
1410                                                 }\r
1411                                                 SetCurrentDirectory(CurDir);\r
1412                                         }\r
1413                                 }\r
1414                         }\r
1415                 }\r
1416                 else if((strlen(Fname) >= 5) && (_stricmp(&Fname[strlen(Fname)-4], ".ini") == 0))\r
1417                 {\r
1418                         CopyFile(Fname, AskIniFilePath(), FALSE);\r
1419                         Ret = YES;\r
1420                 }\r
1421                 else\r
1422                         MessageBox(NULL, MSGJPN293, "FFFTP", MB_OK);\r
1423         }\r
1424         return(Ret);\r
1425 }\r
1426 \r
1427 \r
1428 \r
1429 \r
1430 /*----- レジストリ/INIファイルに文字列をセーブ --------------------------------\r
1431 *\r
1432 *       Parameter\r
1433 *               HKEY hKey : レジストリキー\r
1434 *               char *Key : キー名\r
1435 *               char *Str : セーブする文字列\r
1436 *               char *DefaultStr : デフォルトの文字列\r
1437 *\r
1438 *       Return Value\r
1439 *               なし\r
1440 *\r
1441 *       Note\r
1442 *               文字列がデフォルトの文字列と同じならセーブしない\r
1443 *----------------------------------------------------------------------------*/\r
1444 \r
1445 static void SaveStr(HKEY hKey, char *Key, char *Str, char *DefaultStr)\r
1446 {\r
1447         if((DefaultStr != NULL) && (strcmp(Str, DefaultStr) == 0))\r
1448                 DeleteValue(hKey, Key);\r
1449         else\r
1450                 WriteStringToReg(hKey, Key, Str);\r
1451 \r
1452         return;\r
1453 }\r
1454 \r
1455 \r
1456 /*----- レジストリ/INIファイルに数値(INT)をセーブ -----------------------------\r
1457 *\r
1458 *       Parameter\r
1459 *               HKEY hKey : レジストリキー\r
1460 *               char *Key : キー名\r
1461 *               int Num : セーブする値\r
1462 *               int DefaultNum : デフォルトの値\r
1463 *\r
1464 *       Return Value\r
1465 *               なし\r
1466 *\r
1467 *       Note\r
1468 *               数値がデフォルトの値と同じならセーブしない\r
1469 *----------------------------------------------------------------------------*/\r
1470 \r
1471 static void SaveIntNum(HKEY hKey, char *Key, int Num, int DefaultNum)\r
1472 {\r
1473         if(Num == DefaultNum)\r
1474                 DeleteValue(hKey, Key);\r
1475         else\r
1476                 WriteIntValueToReg(hKey, Key, Num);\r
1477 \r
1478         return;\r
1479 }\r
1480 \r
1481 \r
1482 /*----- LOGFONTデータを文字列に変換する ---------------------------------------\r
1483 *\r
1484 *       Parameter\r
1485 *               LOGFONT Font : フォントデータ\r
1486 *               HFONT hFont : フォントのハンドル\r
1487 *                       NULL = デフォルトのフォント\r
1488 *               char *Buf : バッファ\r
1489 *\r
1490 *       Return Value\r
1491 *               なし\r
1492 *----------------------------------------------------------------------------*/\r
1493 \r
1494 static void MakeFontData(LOGFONT Font, HFONT hFont, char *Buf)\r
1495 {\r
1496         *Buf = NUL;\r
1497         if(hFont != NULL)\r
1498                 sprintf(Buf, "%d %d %d %d %d %d %d %d %d %d %d %d %d %s",\r
1499                         Font.lfHeight, Font.lfWidth, Font.lfEscapement, Font.lfOrientation,\r
1500                         Font.lfWeight, Font.lfItalic, Font.lfUnderline, Font.lfStrikeOut,\r
1501                         Font.lfCharSet, Font.lfOutPrecision, Font.lfClipPrecision,\r
1502                         Font.lfQuality, Font.lfPitchAndFamily, Font.lfFaceName);\r
1503         return;\r
1504 }\r
1505 \r
1506 \r
1507 /*----- 文字列をLOGFONTデータに変換する ---------------------------------------\r
1508 *\r
1509 *       Parameter\r
1510 *               char *Str : 文字列\r
1511 *               LOGFONT *Font : フォントデータ\r
1512 *\r
1513 *       Return Value\r
1514 *               int ステータス\r
1515 *                       FFFTP_SUCCESS/FFFTP_FAIL=変換できない\r
1516 *----------------------------------------------------------------------------*/\r
1517 \r
1518 static int RestoreFontData(char *Str, LOGFONT *Font)\r
1519 {\r
1520         int i;\r
1521         int Sts;\r
1522 \r
1523         Sts = FFFTP_FAIL;\r
1524         if(sscanf(Str, "%d %d %d %d %d %d %d %d %d %d %d %d %d",\r
1525                         &(Font->lfHeight), &(Font->lfWidth), &(Font->lfEscapement), &(Font->lfOrientation),\r
1526                         &(Font->lfWeight), &(Font->lfItalic), &(Font->lfUnderline), &(Font->lfStrikeOut),\r
1527                         &(Font->lfCharSet), &(Font->lfOutPrecision), &(Font->lfClipPrecision),\r
1528                         &(Font->lfQuality), &(Font->lfPitchAndFamily)) == 13)\r
1529         {\r
1530                 for(i = 13; i > 0; i--)\r
1531                 {\r
1532                         if((Str = strchr(Str, ' ')) == NULL)\r
1533                                 break;\r
1534                         Str++;\r
1535                 }\r
1536                 if(i == 0)\r
1537                 {\r
1538                         strcpy(Font->lfFaceName, Str);\r
1539                         Sts = FFFTP_SUCCESS;\r
1540                 }\r
1541         }\r
1542 \r
1543         if(Sts == FFFTP_FAIL)\r
1544                 memset(Font, NUL, sizeof(LOGFONT));\r
1545 \r
1546         return(Sts);\r
1547 }\r
1548 \r
1549 /*----- パスワードを暗号化する ------------------------------------------------\r
1550 *\r
1551 *       Parameter\r
1552 *               char *Str : パスワード\r
1553 *               char *Buf : 暗号化したパスワードを格納するバッファ\r
1554 *\r
1555 *       Return Value\r
1556 *               なし\r
1557 *----------------------------------------------------------------------------*/\r
1558 static void EncodePassword(char *Str, char *Buf)\r
1559 {\r
1560         EncodePassword3( Str, Buf, SecretKey );\r
1561 }\r
1562 \r
1563 /*----- パスワードを暗号化する(オリジナルアルゴリズム)  ------------------\r
1564 *\r
1565 *       Parameter\r
1566 *               char *Str : パスワード\r
1567 *               char *Buf : 暗号化したパスワードを格納するバッファ\r
1568 *\r
1569 *       Return Value\r
1570 *               なし\r
1571 *----------------------------------------------------------------------------*/\r
1572 \r
1573 static void EncodePasswordOriginal(char *Str, char *Buf)\r
1574 {\r
1575         unsigned char *Get;\r
1576         unsigned char *Put;\r
1577         int Rnd;\r
1578         int Ch;\r
1579 \r
1580         srand((unsigned)time(NULL));\r
1581 \r
1582         Get = (unsigned char *)Str;\r
1583         Put = (unsigned char *)Buf;\r
1584         \r
1585         if( *Get == NUL ){\r
1586                 *Put = NUL;\r
1587                 return;\r
1588         }\r
1589 \r
1590         /* 識別子を先頭に置く */\r
1591         Put[0] = '0';\r
1592         Put[1] = 'A';\r
1593         Put += 2;\r
1594 \r
1595         while(*Get != NUL)\r
1596         {\r
1597                 Rnd = rand() % 3;\r
1598                 Ch = ((int)*Get++) << Rnd;\r
1599                 Ch = (unsigned char)Ch | (unsigned char)(Ch >> 8);\r
1600                 *Put++ = 0x40 | ((Rnd & 0x3) << 4) | (Ch & 0xF);\r
1601                 *Put++ = 0x40 | ((Ch >> 4) & 0xF);\r
1602                 if((*(Put-2) & 0x1) != 0)\r
1603                         *Put++ = (rand() % 62) + 0x40;\r
1604         }\r
1605         *Put = NUL;\r
1606         return;\r
1607 }\r
1608 \r
1609 /*----- パスワードを暗号化する(オリジナルアルゴリズム^Key)  ----------------\r
1610 *\r
1611 *       Parameter\r
1612 *               char *Str : パスワード\r
1613 *               char *Buf : 暗号化したパスワードを格納するバッファ\r
1614 *               const char *Key : 暗号化キー\r
1615 *\r
1616 *       Return Value\r
1617 *               なし\r
1618 *----------------------------------------------------------------------------*/\r
1619 \r
1620 static void EncodePassword2(char *Str, char *Buf, const char* Key)\r
1621 {\r
1622         unsigned char *Get;\r
1623         unsigned char *Put;\r
1624         int Rnd;\r
1625         int Ch;\r
1626 \r
1627         /* 2010.01.31 genta Key */\r
1628         unsigned char *KeyHead = (unsigned char *)Key;\r
1629         unsigned char *KeyEnd = KeyHead + strlen(KeyHead);\r
1630         unsigned char *KeyCurrent = KeyHead;\r
1631 \r
1632         srand((unsigned)time(NULL));\r
1633 \r
1634         Get = (unsigned char *)Str;\r
1635         Put = (unsigned char *)Buf;\r
1636         \r
1637         if( *Get == NUL ){\r
1638                 *Put = NUL;\r
1639                 return;\r
1640         }\r
1641 \r
1642         /* 識別子を先頭に置く */\r
1643         Put[0] = '0';\r
1644         Put[1] = 'B';\r
1645         Put += 2;\r
1646 \r
1647         while(*Get != NUL)\r
1648         {\r
1649                 Rnd = rand() % 3;\r
1650                 Ch = ((int)(*Get++ ^ *KeyCurrent)) << Rnd;\r
1651                 Ch = (unsigned char)Ch | (unsigned char)(Ch >> 8);\r
1652                 *Put++ = 0x40 | ((Rnd & 0x3) << 4) | (Ch & 0xF);\r
1653                 *Put++ = 0x40 | ((Ch >> 4) & 0xF);\r
1654                 if((*(Put-2) & 0x1) != 0)\r
1655                         *Put++ = (rand() % 62) + 0x40;\r
1656 \r
1657                 /* 2010.01.31 genta Key */\r
1658                 if( ++KeyCurrent == KeyEnd ){\r
1659                         KeyCurrent = KeyHead;\r
1660                 }\r
1661         }\r
1662         *Put = NUL;\r
1663         return;\r
1664 }\r
1665 \r
1666 /*----- パスワードを暗号化する(AES) ------------------------------------------\r
1667 *\r
1668 *       Parameter\r
1669 *               char *Str : パスワード\r
1670 *               char *Buf : 暗号化したパスワードを格納するバッファ\r
1671 *               const char *Key : 暗号化キー\r
1672 *\r
1673 *       Return Value\r
1674 *               なし\r
1675 *----------------------------------------------------------------------------*/\r
1676 \r
1677 static void EncodePassword3(char *Str, char *Buf, const char *Key)\r
1678 {\r
1679         unsigned char *Put;\r
1680         unsigned char *AesEncBuf;\r
1681         unsigned char *StrPadBuf;\r
1682         size_t StrLen;\r
1683         size_t StrPadLen;\r
1684         size_t StrPadIndex;\r
1685         size_t IvIndex;\r
1686         size_t EncBufIndex;\r
1687         DWORD RandValue;\r
1688         int RandByteCount;\r
1689         unsigned char AesKey[32];\r
1690         unsigned char AesCbcIv[AES_BLOCK_SIZE];\r
1691         aes_encrypt_ctx Ctx;\r
1692 \r
1693         Buf[0] = NUL;\r
1694 \r
1695         Put = (unsigned char *)Buf;\r
1696         StrLen = strlen(Str);\r
1697         StrPadLen = ((StrLen + AES_BLOCK_SIZE - 1) / AES_BLOCK_SIZE) * AES_BLOCK_SIZE;\r
1698 \r
1699         /* 最低長を32文字とする */\r
1700 //      StrPadLen = min1(StrPadLen, AES_BLOCK_SIZE * 2);\r
1701         StrPadLen = max1(StrPadLen, AES_BLOCK_SIZE * 2);\r
1702 \r
1703         if((StrPadBuf = malloc(StrPadLen)) != NULL)\r
1704         {\r
1705                 if((AesEncBuf = malloc(StrPadLen)) != NULL)\r
1706                 {\r
1707                         BOOL PutState;\r
1708 \r
1709                         PutState = FALSE;\r
1710                         strncpy(StrPadBuf, Str, StrPadLen);\r
1711                         \r
1712                         /* PAD部分を乱数で埋める StrPad[StrLen](が有効な場合) は NUL */\r
1713                         for(StrPadIndex = StrLen + 1; StrPadIndex < StrPadLen;)\r
1714                         {\r
1715                                 RandValue = GetRandamDWRODValue();\r
1716                                 for(RandByteCount = 0; RandByteCount < 4; RandByteCount++)\r
1717                                 {\r
1718                                         if(StrPadIndex < StrPadLen)\r
1719                                         {\r
1720                                                 StrPadBuf[StrPadIndex++] = (unsigned char)(RandValue & 0xff);\r
1721                                                 RandValue >>= 8;\r
1722                                         }\r
1723                                 }\r
1724                         }\r
1725                         \r
1726                         // IVの初期化\r
1727                         for(IvIndex = 0; IvIndex < AES_BLOCK_SIZE;)\r
1728                         {\r
1729                                 RandValue = GetRandamDWRODValue();\r
1730                                 for(RandByteCount = 0; RandByteCount < 4; RandByteCount++)\r
1731                                 {\r
1732                                         if(IvIndex < AES_BLOCK_SIZE)\r
1733                                         {\r
1734                                                 AesCbcIv[IvIndex++] = (unsigned char)(RandValue & 0xff);\r
1735                                                 RandValue >>= 8;\r
1736                                         }\r
1737                                 }\r
1738                         }\r
1739                         Put[0] = '0';\r
1740                         Put[1] = 'C';\r
1741                         Put += 2;\r
1742                         for(IvIndex = 0; IvIndex < AES_BLOCK_SIZE; IvIndex++)\r
1743                         {\r
1744                                 sprintf(Put, "%02x", AesCbcIv[IvIndex]);\r
1745                                 Put += 2;\r
1746                         }\r
1747                         *Put++ = ':';\r
1748 \r
1749                         if(CreateAesKey(AesKey, Key) == FFFTP_SUCCESS)\r
1750                         {\r
1751                                 aes_encrypt_key(AesKey, 32, &Ctx);\r
1752 \r
1753                                 if(aes_cbc_encrypt(StrPadBuf, AesEncBuf, StrPadLen, AesCbcIv, &Ctx) == EXIT_SUCCESS)\r
1754                                 {\r
1755                                         for(EncBufIndex = 0; EncBufIndex < StrPadLen; EncBufIndex++)\r
1756                                         {\r
1757                                                 sprintf(Put, "%02x", AesEncBuf[EncBufIndex]);\r
1758                                                 Put += 2;\r
1759                                         }\r
1760                                         *Put = NUL;\r
1761                                         PutState = TRUE;\r
1762                                 }\r
1763                         }\r
1764                         if(FALSE == PutState)\r
1765                         {\r
1766                                 Buf[0] = NUL;\r
1767                         }\r
1768                         free(StrPadBuf);\r
1769                 }\r
1770                 free(AesEncBuf);\r
1771         }\r
1772         return;\r
1773 }\r
1774 \r
1775 \r
1776 /*----- パスワードの暗号化を解く ----------------------------------------------\r
1777 *\r
1778 *       Parameter\r
1779 *               char *Str : 暗号化したパスワード\r
1780 *               char *Buf : パスワードを格納するバッファ\r
1781 *\r
1782 *       Return Value\r
1783 *               なし\r
1784 *----------------------------------------------------------------------------*/\r
1785 \r
1786 static void DecodePassword(char *Str, char *Buf)\r
1787 {\r
1788         unsigned char *Get;\r
1789         unsigned char *Put;\r
1790 \r
1791         Get = (unsigned char *)Str;\r
1792         Put = (unsigned char *)Buf;\r
1793         \r
1794         if( *Get == NUL ){\r
1795                 *Put = NUL;\r
1796         }\r
1797         else if( 0x40 <= *Get && *Get < 0x80 ){\r
1798                 /* Original algorithm */\r
1799                 DecodePasswordOriginal( Str, Buf );\r
1800         }\r
1801         else if( strncmp( Get, "0A", 2 ) == 0 ){\r
1802                 DecodePasswordOriginal( Str + 2, Buf );\r
1803         }\r
1804         else if( strncmp( Get, "0B", 2 ) == 0 ){\r
1805                 DecodePassword2( Str + 2, Buf, SecretKey );\r
1806         }\r
1807         else if( strncmp( Get, "0C", 2 ) == 0 ){\r
1808                 DecodePassword3( Str + 2, Buf, SecretKey );\r
1809         }\r
1810         else {\r
1811                 //      unknown encoding\r
1812                 *Put = NUL;\r
1813                 return;\r
1814         }\r
1815 }\r
1816 \r
1817 /*----- パスワードの暗号化を解く(オリジナルアルゴリズム) -------------------\r
1818 *\r
1819 *       Parameter\r
1820 *               char *Str : 暗号化したパスワード\r
1821 *               char *Buf : パスワードを格納するバッファ\r
1822 *\r
1823 *       Return Value\r
1824 *               なし\r
1825 *----------------------------------------------------------------------------*/\r
1826 static void DecodePasswordOriginal(char *Str, char *Buf)\r
1827 {\r
1828         unsigned char *Get;\r
1829         unsigned char *Put;\r
1830         int Rnd;\r
1831         int Ch;\r
1832 \r
1833         Get = (unsigned char *)Str;\r
1834         Put = (unsigned char *)Buf;\r
1835 \r
1836         while(*Get != NUL)\r
1837         {\r
1838                 Rnd = ((unsigned int)*Get >> 4) & 0x3;\r
1839                 Ch = (*Get & 0xF) | ((*(Get+1) & 0xF) << 4);\r
1840                 Ch <<= 8;\r
1841                 if((*Get & 0x1) != 0)\r
1842                         Get++;\r
1843                 Get += 2;\r
1844                 Ch >>= Rnd;\r
1845                 Ch = (Ch & 0xFF) | ((Ch >> 8) & 0xFF);\r
1846                 *Put++ = Ch;\r
1847         }\r
1848         *Put = NUL;\r
1849         return;\r
1850 }\r
1851 \r
1852 /*----- パスワードの暗号化を解く(オリジナルアルゴリズム^Key) -------------------\r
1853 *\r
1854 *       Parameter\r
1855 *               char *Str : 暗号化したパスワード\r
1856 *               char *Buf : パスワードを格納するバッファ\r
1857 *               const char *Key : 暗号化キー\r
1858 *\r
1859 *       Return Value\r
1860 *               なし\r
1861 *----------------------------------------------------------------------------*/\r
1862 static void DecodePassword2(char *Str, char *Buf, const char* Key)\r
1863 {\r
1864         int Rnd;\r
1865         int Ch;\r
1866         unsigned char *Get = (unsigned char *)Str;\r
1867         unsigned char *Put = (unsigned char *)Buf;\r
1868 \r
1869         /* 2010.01.31 genta Key */\r
1870         unsigned char *KeyHead = (unsigned char *)Key;\r
1871         unsigned char *KeyEnd = KeyHead + strlen(KeyHead);\r
1872         unsigned char *KeyCurrent = KeyHead;\r
1873 \r
1874         while(*Get != NUL)\r
1875         {\r
1876                 Rnd = ((unsigned int)*Get >> 4) & 0x3;\r
1877                 Ch = (*Get & 0xF) | ((*(Get+1) & 0xF) << 4);\r
1878                 Ch <<= 8;\r
1879                 if((*Get & 0x1) != 0)\r
1880                         Get++;\r
1881                 Get += 2;\r
1882                 Ch >>= Rnd;\r
1883                 Ch = (Ch & 0xFF) | ((Ch >> 8) & 0xFF);\r
1884                 *Put++ = Ch ^ *KeyCurrent;\r
1885                 \r
1886                 /* 2010.01.31 genta Key */\r
1887                 if( ++KeyCurrent == KeyEnd ){\r
1888                         KeyCurrent = KeyHead;\r
1889                 }\r
1890         }\r
1891         *Put = NUL;\r
1892         return;\r
1893 }\r
1894 \r
1895 /*----- パスワードの暗号化を解く(AES) ---------------------------------------\r
1896 *\r
1897 *       Parameter\r
1898 *               char *Str : 暗号化したパスワード\r
1899 *               char *Buf : パスワードを格納するバッファ\r
1900 *               const char *Key : 暗号化キー\r
1901 *\r
1902 *       Return Value\r
1903 *               なし\r
1904 *----------------------------------------------------------------------------*/\r
1905 \r
1906 static void DecodePassword3(char *Str, char *Buf, const char *Key)\r
1907 {\r
1908         char *Get;\r
1909         unsigned char *EncBuf;\r
1910         size_t StrLen;\r
1911         size_t IvIndex;\r
1912         size_t EncBufIndex;\r
1913         size_t EncBufLen;\r
1914         unsigned char AesKey[32];\r
1915         unsigned char AesCbcIv[AES_BLOCK_SIZE];\r
1916         aes_decrypt_ctx Ctx;\r
1917 \r
1918         Buf[0] = NUL;\r
1919 \r
1920         Get = Str;\r
1921         StrLen = strlen(Str);\r
1922 \r
1923         if(AES_BLOCK_SIZE * 2 + 1 < StrLen)\r
1924         {\r
1925 \r
1926                 EncBufLen = (StrLen - 1 ) / 2 - AES_BLOCK_SIZE;\r
1927                 if((EncBuf = malloc(EncBufLen)) != NULL)\r
1928                 {\r
1929                         for(IvIndex = 0; IvIndex < AES_BLOCK_SIZE; IvIndex++)\r
1930                         {\r
1931                                 AesCbcIv[IvIndex]  = hex2bin(*Get++) << 4;\r
1932                                 AesCbcIv[IvIndex] |= hex2bin(*Get++);\r
1933                         }\r
1934 \r
1935                         if(*Get++ == ':')\r
1936                         {\r
1937                                 if(CreateAesKey(AesKey, Key) == FFFTP_SUCCESS)\r
1938                                 {\r
1939                                         aes_decrypt_key(AesKey, 32, &Ctx);\r
1940 \r
1941                                         for(EncBufIndex = 0; EncBufIndex < EncBufLen; EncBufIndex++)\r
1942                                         {\r
1943                                                 EncBuf[EncBufIndex]  = hex2bin(*Get++) << 4;\r
1944                                                 EncBuf[EncBufIndex] |= hex2bin(*Get++);\r
1945                                         }\r
1946                                         if(aes_cbc_decrypt(EncBuf, Buf, EncBufLen, AesCbcIv, &Ctx) == EXIT_SUCCESS)\r
1947                                         {\r
1948                                                 Buf[EncBufLen] = NUL;\r
1949                                         }\r
1950                                 }\r
1951                         }\r
1952                         free(EncBuf);\r
1953                 }\r
1954         }\r
1955         return;\r
1956 }\r
1957 \r
1958 /*----- AES用固定長キーを作成 ----------------------------------------------\r
1959 *\r
1960 *       Parameter\r
1961 *               unsigned char *AesKey : AES暗号鍵\r
1962 *               const char *Key : 暗号化キー\r
1963 *\r
1964 *       Return Value\r
1965 *               int ステータス (FFFTP_SUCCESS/FFFTP_FAIL)\r
1966 *       Note\r
1967 *               SHA-1をもちいて32Byte鍵を生成する\r
1968 *----------------------------------------------------------------------------*/\r
1969 \r
1970 static int CreateAesKey(unsigned char *AesKey, const char* Key)\r
1971 {\r
1972         char* HashKey;\r
1973         uint32 HashKeyLen;\r
1974         uint32 results[10];\r
1975         int ByteOffset;\r
1976         int KeyIndex;\r
1977         int ResIndex;\r
1978 \r
1979         HashKeyLen = strlen(Key) + 16;\r
1980         if((HashKey = malloc(HashKeyLen + 1)) == NULL){\r
1981                 return (FFFTP_FAIL);\r
1982         }\r
1983 \r
1984         strcpy(HashKey, Key);\r
1985         strcat(HashKey, ">g^r=@N7=//z<[`:");\r
1986     sha_memory((uchar *)HashKey, HashKeyLen, results);\r
1987 \r
1988         strcpy(HashKey, Key);\r
1989         strcat(HashKey, "VG77dO1#EyC]$|C@");\r
1990     sha_memory((uchar *)HashKey, HashKeyLen, results + 5);\r
1991 \r
1992         KeyIndex = 0;\r
1993         ResIndex = 0;\r
1994         while(ResIndex < 8){\r
1995                 for(ByteOffset = 0; ByteOffset < 4; ByteOffset++){\r
1996                         AesKey[KeyIndex++] = (results[ResIndex] >> ByteOffset * 8) & 0xff;\r
1997                 }\r
1998                 ResIndex++;\r
1999         }\r
2000         free(HashKey);\r
2001 \r
2002         return (FFFTP_SUCCESS);\r
2003 }\r
2004 \r
2005 \r
2006 /*===== レジストリとINIファイルのアクセス処理 ============*/\r
2007 \r
2008 \r
2009 /*===== INIファイル用のレジストリデータ =====*/\r
2010 \r
2011 typedef struct regdatatbl {\r
2012         char KeyName[80+1];                     /* キー名 */\r
2013         char ValTbl[REG_SECT_MAX];      /* 値のテーブル */\r
2014         int ValLen;                                     /* 値データのバイト数 */\r
2015         int Mode;                                       /* キーのモード */\r
2016         struct regdatatbl *Next;\r
2017 } REGDATATBL;\r
2018 \r
2019 /*===== プロトタイプ =====*/\r
2020 \r
2021 static BOOL WriteOutRegToFile(REGDATATBL *Pos);\r
2022 static int ReadInReg(char *Name, REGDATATBL **Handle);\r
2023 // 暗号化通信対応\r
2024 //static int StrCatOut(char *Src, int Len, char *Dst);\r
2025 //static int StrReadIn(char *Src, int Max, char *Dst);\r
2026 static char *ScanValue(void *Handle, char *Name);\r
2027 \r
2028 \r
2029 /*===== ローカルなワーク =====*/\r
2030 \r
2031 static int TmpRegType;\r
2032 \r
2033 \r
2034 \r
2035 /*----- レジストリのタイプを設定する ------------------------------------------\r
2036 *\r
2037 *       Parameter\r
2038 *               int Type : タイプ (REGTYPE_xxx)\r
2039 *\r
2040 *       Return Value\r
2041 *               int ステータス\r
2042 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2043 *----------------------------------------------------------------------------*/\r
2044 \r
2045 static void SetRegType(int Type)\r
2046 {\r
2047         TmpRegType = Type;\r
2048         return;\r
2049 }\r
2050 \r
2051 \r
2052 /*----- レジストリ/INIファイルをオープンする(読み込み)-----------------------\r
2053 *\r
2054 *       Parameter\r
2055 *               char *Name : レジストリ名\r
2056 *               void **Handle : ハンドルを返すワーク\r
2057 *\r
2058 *       Return Value\r
2059 *               int ステータス\r
2060 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2061 *----------------------------------------------------------------------------*/\r
2062 \r
2063 static int OpenReg(char *Name, void **Handle)\r
2064 {\r
2065         int Sts;\r
2066         char Tmp[FMAX_PATH+1];\r
2067 \r
2068         Sts = FFFTP_FAIL;\r
2069         if(TmpRegType == REGTYPE_REG)\r
2070         {\r
2071                 strcpy(Tmp, "Software\\Sota\\");\r
2072                 strcat(Tmp, Name);\r
2073                 if(RegOpenKeyEx(HKEY_CURRENT_USER, Tmp, 0, KEY_READ, (HKEY *)Handle) == ERROR_SUCCESS)\r
2074                         Sts = FFFTP_SUCCESS;\r
2075         }\r
2076         else\r
2077         {\r
2078                 if((Sts = ReadInReg(Name, (REGDATATBL **)Handle)) == FFFTP_SUCCESS)\r
2079                         ((REGDATATBL *)(*Handle))->Mode = 0;\r
2080         }\r
2081         return(Sts);\r
2082 }\r
2083 \r
2084 \r
2085 /*----- レジストリ/INIファイルを作成する(書き込み)---------------------------\r
2086 *\r
2087 *       Parameter\r
2088 *               char *Name : レジストリ名\r
2089 *               void **Handle : ハンドルを返すワーク\r
2090 *\r
2091 *       Return Value\r
2092 *               int ステータス\r
2093 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2094 *----------------------------------------------------------------------------*/\r
2095 \r
2096 static int CreateReg(char *Name, void **Handle)\r
2097 {\r
2098         int Sts;\r
2099         char Tmp[FMAX_PATH+1];\r
2100         DWORD Dispos;\r
2101 \r
2102         Sts = FFFTP_FAIL;\r
2103         if(TmpRegType == REGTYPE_REG)\r
2104         {\r
2105                 strcpy(Tmp, "Software\\Sota\\");\r
2106                 strcat(Tmp, Name);\r
2107                 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
2108                         Sts = FFFTP_SUCCESS;\r
2109         }\r
2110         else\r
2111         {\r
2112                 if((*Handle = malloc(sizeof(REGDATATBL))) != NULL)\r
2113                 {\r
2114                         strcpy(((REGDATATBL *)(*Handle))->KeyName, Name);\r
2115                         ((REGDATATBL *)(*Handle))->ValLen = 0;\r
2116                         ((REGDATATBL *)(*Handle))->Next = NULL;\r
2117                         ((REGDATATBL *)(*Handle))->Mode = 1;\r
2118                         Sts = FFFTP_SUCCESS;\r
2119                 }\r
2120         }\r
2121         return(Sts);\r
2122 }\r
2123 \r
2124 \r
2125 /*----- レジストリ/INIファイルをクローズする ----------------------------------\r
2126 *\r
2127 *       Parameter\r
2128 *               void *Handle : ハンドル\r
2129 *\r
2130 *       Return Value\r
2131 *               int ステータス\r
2132 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2133 *----------------------------------------------------------------------------*/\r
2134 \r
2135 static int CloseReg(void *Handle)\r
2136 {\r
2137         REGDATATBL *Pos;\r
2138         REGDATATBL *Next;\r
2139         // ポータブル版判定\r
2140 //      FILE *Strm;\r
2141 \r
2142         if(TmpRegType == REGTYPE_REG)\r
2143         {\r
2144                 RegCloseKey(Handle);\r
2145 \r
2146                 /* INIファイルを削除 */\r
2147                 // ポータブル版判定\r
2148 //              if((Strm = fopen(AskIniFilePath(), "rt")) != NULL)\r
2149 //              {\r
2150 //                      fclose(Strm);\r
2151 //                      MoveFileToTrashCan(AskIniFilePath());\r
2152 //              }\r
2153         }\r
2154         else\r
2155         {\r
2156                 if(((REGDATATBL *)Handle)->Mode == 1)\r
2157                 {\r
2158                         if(WriteOutRegToFile(Handle) == TRUE)\r
2159                         {\r
2160 //                              /* レジストリをクリア */\r
2161 //                              ClearRegistry();\r
2162                         }\r
2163                 }\r
2164                 /* テーブルを削除 */\r
2165                 Pos = Handle;\r
2166                 while(Pos != NULL)\r
2167                 {\r
2168                         Next = Pos->Next;\r
2169                         free(Pos);\r
2170                         Pos = Next;\r
2171                 }\r
2172         }\r
2173         return(FFFTP_SUCCESS);\r
2174 }\r
2175 \r
2176 \r
2177 /*----- レジストリ情報をINIファイルに書き込む ---------------------------------\r
2178 *\r
2179 *       Parameter\r
2180 *               REGDATATBL *Pos : レジストリデータ\r
2181 *\r
2182 *       Return Value\r
2183 *               なし\r
2184 *----------------------------------------------------------------------------*/\r
2185 \r
2186 static BOOL WriteOutRegToFile(REGDATATBL *Pos)\r
2187 {\r
2188         FILE *Strm;\r
2189         char *Disp;\r
2190         BOOL Ret;\r
2191 \r
2192         Ret = FALSE;\r
2193         if((Strm = fopen(AskIniFilePath(), "wt")) != NULL)\r
2194         {\r
2195                 fprintf(Strm, MSGJPN239);\r
2196                 while(Pos != NULL)\r
2197                 {\r
2198                         fprintf(Strm, "\n[%s]\n", Pos->KeyName);\r
2199 \r
2200                         Disp = Pos->ValTbl;\r
2201                         while(Disp < (Pos->ValTbl + Pos->ValLen))\r
2202                         {\r
2203                                 fprintf(Strm, "%s\n", Disp);\r
2204                                 Disp = Disp + strlen(Disp) + 1;\r
2205                         }\r
2206                         Pos = Pos->Next;\r
2207                 }\r
2208                 fclose(Strm);\r
2209                 Ret = TRUE;\r
2210         }\r
2211         else\r
2212                 MessageBox(NULL, MSGJPN240, "FFFTP", MB_OK);\r
2213 \r
2214         return(Ret);\r
2215 }\r
2216 \r
2217 \r
2218 /*----- INIファイルからレジストリ情報を読み込む -------------------------------\r
2219 *\r
2220 *       Parameter\r
2221 *               char *Name : 名前\r
2222 *               void *Handle : ハンドル\r
2223 *\r
2224 *       Return Value\r
2225 *               int ステータス\r
2226 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2227 *----------------------------------------------------------------------------*/\r
2228 \r
2229 static int ReadInReg(char *Name, REGDATATBL **Handle)\r
2230 {\r
2231         FILE *Strm;\r
2232         char *Buf;\r
2233         char *Tmp;\r
2234         char *Data;\r
2235         REGDATATBL *New;\r
2236         REGDATATBL *Pos;\r
2237         int Sts;\r
2238 \r
2239         Sts = FFFTP_FAIL;\r
2240         *Handle = NULL;\r
2241         // バグ修正\r
2242         New = NULL;\r
2243 \r
2244         if((Strm = fopen(AskIniFilePath(), "rt")) != NULL)\r
2245         {\r
2246                 if((Buf = malloc(REG_SECT_MAX)) != NULL)\r
2247                 {\r
2248                         while(fgets(Buf, REG_SECT_MAX, Strm) != NULL)\r
2249                         {\r
2250                                 if(*Buf != '#')\r
2251                                 {\r
2252                                         if((Tmp = strchr(Buf, '\n')) != NULL)\r
2253                                                 *Tmp = NUL;\r
2254 \r
2255                                         if(*Buf == '[')\r
2256                                         {\r
2257                                                 if((New = malloc(sizeof(REGDATATBL))) != NULL)\r
2258                                                 {\r
2259                                                         if((Tmp = strchr(Buf, ']')) != NULL)\r
2260                                                                 *Tmp = NUL;\r
2261                                                         // バグ修正\r
2262 //                                                      strcpy(New->KeyName, Buf+1);\r
2263                                                         strncpy(New->KeyName, Buf+1, 80);\r
2264                                                         New->KeyName[80] = NUL;\r
2265                                                         New->ValLen = 0;\r
2266                                                         New->Next = NULL;\r
2267                                                         Data = New->ValTbl;\r
2268                                                 }\r
2269                                                 if(*Handle == NULL)\r
2270                                                         *Handle = New;\r
2271                                                 else\r
2272                                                 {\r
2273                                                         Pos = *Handle;\r
2274                                                         while(Pos->Next != NULL)\r
2275                                                                 Pos = Pos->Next;\r
2276                                                         Pos->Next = New;\r
2277                                                 }\r
2278                                         }\r
2279                                         else if(strlen(Buf) > 0)\r
2280                                         {\r
2281                                                 // バグ修正\r
2282 //                                              strcpy(Data, Buf);\r
2283 //                                              Data += strlen(Buf) + 1;\r
2284 //                                              New->ValLen += strlen(Buf) + 1;\r
2285                                                 if(Data != NULL && New != NULL)\r
2286                                                 {\r
2287                                                         if(New->ValLen + strlen(Buf) + 1 <= REG_SECT_MAX)\r
2288                                                         {\r
2289                                                                 strcpy(Data, Buf);\r
2290                                                                 Data += strlen(Buf) + 1;\r
2291                                                                 New->ValLen += strlen(Buf) + 1;\r
2292                                                         }\r
2293                                                 }\r
2294                                         }\r
2295                                 }\r
2296                         }\r
2297                         Sts = FFFTP_SUCCESS;\r
2298                         free(Buf);\r
2299                 }\r
2300                 fclose(Strm);\r
2301         }\r
2302         return(Sts);\r
2303 }\r
2304 \r
2305 \r
2306 /*----- サブキーをオープンする ------------------------------------------------\r
2307 *\r
2308 *       Parameter\r
2309 *               void *Parent : 親のハンドル\r
2310 *               char *Name : 名前\r
2311 *               void **Handle : ハンドルを返すワーク\r
2312 *\r
2313 *       Return Value\r
2314 *               int ステータス\r
2315 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2316 *----------------------------------------------------------------------------*/\r
2317 \r
2318 static int OpenSubKey(void *Parent, char *Name, void **Handle)\r
2319 {\r
2320         int Sts;\r
2321         char Key[80];\r
2322         REGDATATBL *Pos;\r
2323 \r
2324         Sts = FFFTP_FAIL;\r
2325         if(TmpRegType == REGTYPE_REG)\r
2326         {\r
2327                 if(RegOpenKeyEx(Parent, Name, 0, KEY_READ, (HKEY *)Handle) == ERROR_SUCCESS)\r
2328                         Sts = FFFTP_SUCCESS;\r
2329         }\r
2330         else\r
2331         {\r
2332                 strcpy(Key, ((REGDATATBL *)Parent)->KeyName);\r
2333                 strcat(Key, "\\");\r
2334                 strcat(Key, Name);\r
2335                 Pos = Parent;\r
2336                 while(Pos != NULL)\r
2337                 {\r
2338                         if(strcmp(Pos->KeyName, Key) == 0)\r
2339                         {\r
2340                                 *Handle = Pos;\r
2341                                 Sts = FFFTP_SUCCESS;\r
2342                                 break;\r
2343                         }\r
2344                         Pos = Pos->Next;\r
2345                 }\r
2346         }\r
2347         return(Sts);\r
2348 }\r
2349 \r
2350 \r
2351 /*----- サブキーを作成する ----------------------------------------------------\r
2352 *\r
2353 *       Parameter\r
2354 *               void *Parent : 親のハンドル\r
2355 *               char *Name : 名前\r
2356 *               void **Handle : ハンドルを返すワーク\r
2357 *\r
2358 *       Return Value\r
2359 *               int ステータス\r
2360 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2361 *----------------------------------------------------------------------------*/\r
2362 \r
2363 static int CreateSubKey(void *Parent, char *Name, void **Handle)\r
2364 {\r
2365         int Sts;\r
2366         DWORD Dispos;\r
2367         REGDATATBL *Pos;\r
2368 \r
2369         Sts = FFFTP_FAIL;\r
2370         if(TmpRegType == REGTYPE_REG)\r
2371         {\r
2372                 if(RegCreateKeyEx(Parent, Name, 0, "", REG_OPTION_NON_VOLATILE, KEY_SET_VALUE, NULL, (HKEY *)Handle, &Dispos) == ERROR_SUCCESS)\r
2373                         Sts = FFFTP_SUCCESS;\r
2374         }\r
2375         else\r
2376         {\r
2377                 if((*Handle = malloc(sizeof(REGDATATBL))) != NULL)\r
2378                 {\r
2379                         strcpy(((REGDATATBL *)(*Handle))->KeyName, ((REGDATATBL *)Parent)->KeyName);\r
2380                         strcat(((REGDATATBL *)(*Handle))->KeyName, "\\");\r
2381                         strcat(((REGDATATBL *)(*Handle))->KeyName, Name);\r
2382 \r
2383                         ((REGDATATBL *)(*Handle))->ValLen = 0;\r
2384                         ((REGDATATBL *)(*Handle))->Next = NULL;\r
2385 \r
2386                         Pos = (REGDATATBL *)Parent;\r
2387                         while(Pos->Next != NULL)\r
2388                                 Pos = Pos->Next;\r
2389                         Pos->Next = *Handle;\r
2390                         Sts = FFFTP_SUCCESS;\r
2391                 }\r
2392         }\r
2393         return(Sts);\r
2394 }\r
2395 \r
2396 \r
2397 /*----- サブキーをクローズする ------------------------------------------------\r
2398 *\r
2399 *       Parameter\r
2400 *               void *Handle : ハンドル\r
2401 *\r
2402 *       Return Value\r
2403 *               int ステータス\r
2404 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2405 *----------------------------------------------------------------------------*/\r
2406 \r
2407 static int CloseSubKey(void *Handle)\r
2408 {\r
2409         if(TmpRegType == REGTYPE_REG)\r
2410                 RegCloseKey(Handle);\r
2411         else\r
2412         {\r
2413                 /* Nothing */\r
2414         }\r
2415         return(FFFTP_SUCCESS);\r
2416 }\r
2417 \r
2418 \r
2419 /*----- サブキーを削除する ----------------------------------------------------\r
2420 *\r
2421 *       Parameter\r
2422 *               void *Handle : ハンドル\r
2423 *               char *Name : 名前\r
2424 *\r
2425 *       Return Value\r
2426 *               int ステータス\r
2427 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2428 *----------------------------------------------------------------------------*/\r
2429 \r
2430 static int DeleteSubKey(void *Handle, char *Name)\r
2431 {\r
2432         int Sts;\r
2433 \r
2434         Sts = FFFTP_FAIL;\r
2435         if(TmpRegType == REGTYPE_REG)\r
2436         {\r
2437                 if(RegDeleteKey(Handle, Name) == ERROR_SUCCESS)\r
2438                         Sts = FFFTP_SUCCESS;\r
2439         }\r
2440         else\r
2441         {\r
2442                 Sts = FFFTP_FAIL;\r
2443         }\r
2444         return(Sts);\r
2445 }\r
2446 \r
2447 \r
2448 /*----- 値を削除する ----------------------------------------------------------\r
2449 *\r
2450 *       Parameter\r
2451 *               void *Handle : ハンドル\r
2452 *               char *Name : 名前\r
2453 *\r
2454 *       Return Value\r
2455 *               int ステータス\r
2456 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2457 *----------------------------------------------------------------------------*/\r
2458 \r
2459 static int DeleteValue(void *Handle, char *Name)\r
2460 {\r
2461         int Sts;\r
2462 \r
2463         Sts = FFFTP_FAIL;\r
2464         if(TmpRegType == REGTYPE_REG)\r
2465         {\r
2466                 if(RegDeleteValue(Handle, Name) == ERROR_SUCCESS)\r
2467                         Sts = FFFTP_SUCCESS;\r
2468         }\r
2469         else\r
2470         {\r
2471                 Sts = FFFTP_FAIL;\r
2472         }\r
2473         return(Sts);\r
2474 }\r
2475 \r
2476 \r
2477 /*----- INT値を読み込む -------------------------------------------------------\r
2478 *\r
2479 *       Parameter\r
2480 *               void *Handle : ハンドル\r
2481 *               char *Name : 名前\r
2482 *               int *Value : INT値を返すワーク\r
2483 *\r
2484 *       Return Value\r
2485 *               int ステータス\r
2486 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2487 *----------------------------------------------------------------------------*/\r
2488 \r
2489 static int ReadIntValueFromReg(void *Handle, char *Name, int *Value)\r
2490 {\r
2491         int Sts;\r
2492         DWORD Size;\r
2493         char *Pos;\r
2494 \r
2495         Sts = FFFTP_FAIL;\r
2496         if(TmpRegType == REGTYPE_REG)\r
2497         {\r
2498                 Size = sizeof(int);\r
2499                 if(RegQueryValueEx(Handle, Name, NULL, NULL, (BYTE *)Value, &Size) == ERROR_SUCCESS)\r
2500                         Sts = FFFTP_SUCCESS;\r
2501         }\r
2502         else\r
2503         {\r
2504                 if((Pos = ScanValue(Handle, Name)) != NULL)\r
2505                 {\r
2506                         *Value = atoi(Pos);\r
2507                         Sts = FFFTP_SUCCESS;\r
2508                 }\r
2509         }\r
2510         // 全設定暗号化対応\r
2511         if(Sts == FFFTP_SUCCESS)\r
2512         {\r
2513                 if(EncryptSettings == YES)\r
2514                 {\r
2515                         UnmaskSettingsData(Name, strlen(Name), Value, sizeof(int), NO);\r
2516                         CalculateSettingsDataChecksum(Value, sizeof(int));\r
2517                 }\r
2518         }\r
2519         return(Sts);\r
2520 }\r
2521 \r
2522 \r
2523 /*----- INT値を書き込む -------------------------------------------------------\r
2524 *\r
2525 *       Parameter\r
2526 *               void *Handle : ハンドル\r
2527 *               char *Name : 名前\r
2528 *               int Value : INT値\r
2529 *\r
2530 *       Return Value\r
2531 *               int ステータス\r
2532 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2533 *----------------------------------------------------------------------------*/\r
2534 \r
2535 static int WriteIntValueToReg(void *Handle, char *Name, int Value)\r
2536 {\r
2537         REGDATATBL *Pos;\r
2538         char *Data;\r
2539         char Tmp[20];\r
2540 \r
2541         // 全設定暗号化対応\r
2542         if(EncryptSettings == YES)\r
2543                 MaskSettingsData(Name, strlen(Name), &Value, sizeof(int), NO);\r
2544         if(TmpRegType == REGTYPE_REG)\r
2545                 RegSetValueEx(Handle, Name, 0, REG_DWORD, (CONST BYTE *)&Value, sizeof(int));\r
2546         else\r
2547         {\r
2548                 Pos = (REGDATATBL *)Handle;\r
2549                 Data = Pos->ValTbl + Pos->ValLen;\r
2550                 strcpy(Data, Name);\r
2551                 strcat(Data, "=");\r
2552                 sprintf(Tmp, "%d", Value);\r
2553                 strcat(Data, Tmp);\r
2554                 Pos->ValLen += strlen(Data) + 1;\r
2555         }\r
2556         // 全設定暗号化対応\r
2557         if(EncryptSettings == YES)\r
2558         {\r
2559                 UnmaskSettingsData(Name, strlen(Name), &Value, sizeof(int), NO);\r
2560                 CalculateSettingsDataChecksum(&Value, sizeof(int));\r
2561         }\r
2562         return(FFFTP_SUCCESS);\r
2563 }\r
2564 \r
2565 \r
2566 /*----- 文字列を読み込む ------------------------------------------------------\r
2567 *\r
2568 *       Parameter\r
2569 *               void *Handle : ハンドル\r
2570 *               char *Name : 名前\r
2571 *               char *Str : 文字列を返すワーク\r
2572 *               DWORD Size : 最大サイズ\r
2573 *\r
2574 *       Return Value\r
2575 *               int ステータス\r
2576 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2577 *----------------------------------------------------------------------------*/\r
2578 \r
2579 static int ReadStringFromReg(void *Handle, char *Name, char *Str, DWORD Size)\r
2580 {\r
2581         int Sts;\r
2582         char *Pos;\r
2583         // UTF-8対応\r
2584         DWORD TempSize;\r
2585         char* pa0;\r
2586         wchar_t* pw0;\r
2587 \r
2588         Sts = FFFTP_FAIL;\r
2589         if(TmpRegType == REGTYPE_REG)\r
2590         {\r
2591                 if(RegQueryValueEx(Handle, Name, NULL, NULL, (BYTE *)Str, &Size) == ERROR_SUCCESS)\r
2592                 {\r
2593                         if(*(Str + Size - 1) != NUL)\r
2594                                 *(Str + Size) = NUL;\r
2595                         Sts = FFFTP_SUCCESS;\r
2596                 }\r
2597         }\r
2598         else\r
2599         {\r
2600                 if((Pos = ScanValue(Handle, Name)) != NULL)\r
2601                 {\r
2602                         // UTF-8対応\r
2603 //                      Size = min1(Size-1, strlen(Pos));\r
2604 //                      Size = StrReadIn(Pos, Size, Str);\r
2605 //                      *(Str + Size) = NUL;\r
2606 //                      Sts = FFFTP_SUCCESS;\r
2607                         switch(IniKanjiCode)\r
2608                         {\r
2609                         case KANJI_NOCNV:\r
2610                                 TempSize = min1(Size-1, strlen(Pos));\r
2611                                 TempSize = StrReadIn(Pos, TempSize, Str);\r
2612                                 *(Str + TempSize) = NUL;\r
2613                                 Sts = FFFTP_SUCCESS;\r
2614                                 break;\r
2615                         case KANJI_SJIS:\r
2616                                 if(pa0 = AllocateStringA(Size * 4))\r
2617                                 {\r
2618                                         if(pw0 = AllocateStringW(Size * 4 * 4))\r
2619                                         {\r
2620                                                 TempSize = min1((Size * 4) - 1, strlen(Pos));\r
2621                                                 TempSize = StrReadIn(Pos, TempSize, pa0);\r
2622                                                 *(pa0 + TempSize) = NUL;\r
2623                                                 AtoW(pw0, Size * 4 * 4, pa0, -1);\r
2624                                                 WtoM(Str, Size, pw0, -1);\r
2625                                                 TerminateStringM(Str, Size);\r
2626                                                 Sts = FFFTP_SUCCESS;\r
2627                                                 FreeDuplicatedString(pw0);\r
2628                                         }\r
2629                                         FreeDuplicatedString(pa0);\r
2630                                 }\r
2631                                 break;\r
2632                         }\r
2633                 }\r
2634         }\r
2635         // 全設定暗号化対応\r
2636         if(Sts == FFFTP_SUCCESS)\r
2637         {\r
2638                 if(EncryptSettings == YES)\r
2639                 {\r
2640                         UnmaskSettingsData(Name, strlen(Name), Str, strlen(Str) + 1, YES);\r
2641                         CalculateSettingsDataChecksum(Str, strlen(Str) + 1);\r
2642                 }\r
2643         }\r
2644         return(Sts);\r
2645 }\r
2646 \r
2647 \r
2648 /*----- 文字列を書き込む ------------------------------------------------------\r
2649 *\r
2650 *       Parameter\r
2651 *               void *Handle : ハンドル\r
2652 *               char *Name : 名前\r
2653 *               char *Str :文字列\r
2654 *\r
2655 *       Return Value\r
2656 *               int ステータス\r
2657 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2658 *----------------------------------------------------------------------------*/\r
2659 \r
2660 static int WriteStringToReg(void *Handle, char *Name, char *Str)\r
2661 {\r
2662         REGDATATBL *Pos;\r
2663         char *Data;\r
2664 \r
2665         // 全設定暗号化対応\r
2666         if(EncryptSettings == YES)\r
2667                 MaskSettingsData(Name, strlen(Name), Str, strlen(Str) + 1, YES);\r
2668         if(TmpRegType == REGTYPE_REG)\r
2669         // 全設定暗号化対応\r
2670 //              RegSetValueEx(Handle, Name, 0, REG_SZ, (CONST BYTE *)Str, strlen(Str)+1);\r
2671                 RegSetValueEx(Handle, Name, 0, EncryptSettings == YES ? REG_BINARY : REG_SZ, (CONST BYTE *)Str, strlen(Str)+1);\r
2672         else\r
2673         {\r
2674                 Pos = (REGDATATBL *)Handle;\r
2675                 Data = Pos->ValTbl + Pos->ValLen;\r
2676                 strcpy(Data, Name);\r
2677                 strcat(Data, "=");\r
2678                 Pos->ValLen += strlen(Data);\r
2679                 Data = Pos->ValTbl + Pos->ValLen;\r
2680                 Pos->ValLen += StrCatOut(Str, strlen(Str), Data) + 1;\r
2681         }\r
2682         // 全設定暗号化対応\r
2683         if(EncryptSettings == YES)\r
2684         {\r
2685                 UnmaskSettingsData(Name, strlen(Name), Str, strlen(Str) + 1, YES);\r
2686                 CalculateSettingsDataChecksum(Str, strlen(Str) + 1);\r
2687         }\r
2688         return(FFFTP_SUCCESS);\r
2689 }\r
2690 \r
2691 \r
2692 /*----- マルチ文字列を読み込む ------------------------------------------------\r
2693 *\r
2694 *       Parameter\r
2695 *               void *Handle : ハンドル\r
2696 *               char *Name : 名前\r
2697 *               char *Str : 文字列を返すワーク\r
2698 *               DWORD Size : 最大サイズ\r
2699 *\r
2700 *       Return Value\r
2701 *               int ステータス\r
2702 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2703 *----------------------------------------------------------------------------*/\r
2704 \r
2705 static int ReadMultiStringFromReg(void *Handle, char *Name, char *Str, DWORD Size)\r
2706 {\r
2707         int Sts;\r
2708         char *Pos;\r
2709         // UTF-8対応\r
2710         DWORD TempSize;\r
2711         char* pa0;\r
2712         wchar_t* pw0;\r
2713 \r
2714         Sts = FFFTP_FAIL;\r
2715         if(TmpRegType == REGTYPE_REG)\r
2716         {\r
2717                 if(RegQueryValueEx(Handle, Name, NULL, NULL, (BYTE *)Str, &Size) == ERROR_SUCCESS)\r
2718                 {\r
2719                         if(*(Str + Size - 1) != NUL)\r
2720                                 *(Str + Size) = NUL;\r
2721                         Sts = FFFTP_SUCCESS;\r
2722                 }\r
2723         }\r
2724         else\r
2725         {\r
2726                 if((Pos = ScanValue(Handle, Name)) != NULL)\r
2727                 {\r
2728                         // UTF-8対応\r
2729 //                      Size = min1(Size-1, strlen(Pos));\r
2730 //                      Size = StrReadIn(Pos, Size, Str);\r
2731 //                      *(Str + Size) = NUL;\r
2732 //                      Sts = FFFTP_SUCCESS;\r
2733                         switch(IniKanjiCode)\r
2734                         {\r
2735                         case KANJI_NOCNV:\r
2736                                 TempSize = min1(Size - 2, strlen(Pos));\r
2737                                 TempSize = StrReadIn(Pos, TempSize, Str);\r
2738                                 *(Str + TempSize) = NUL;\r
2739                                 *(Str + TempSize + 1) = NUL;\r
2740                                 Sts = FFFTP_SUCCESS;\r
2741                                 break;\r
2742                         case KANJI_SJIS:\r
2743                                 if(pa0 = AllocateStringA(Size * 4))\r
2744                                 {\r
2745                                         if(pw0 = AllocateStringW(Size * 4 * 4))\r
2746                                         {\r
2747                                                 TempSize = min1((Size * 4) - 2, strlen(Pos));\r
2748                                                 TempSize = StrReadIn(Pos, TempSize, pa0);\r
2749                                                 *(pa0 + TempSize) = NUL;\r
2750                                                 *(pa0 + TempSize + 1) = NUL;\r
2751                                                 AtoWMultiString(pw0, Size * 4 * 4, pa0);\r
2752                                                 WtoMMultiString(Str, Size, pw0);\r
2753                                                 TerminateStringM(Str, Size);\r
2754                                                 TerminateStringM(Str, Size - 1);\r
2755                                                 Sts = FFFTP_SUCCESS;\r
2756                                                 FreeDuplicatedString(pw0);\r
2757                                         }\r
2758                                         FreeDuplicatedString(pa0);\r
2759                                 }\r
2760                                 break;\r
2761                         }\r
2762                 }\r
2763         }\r
2764         // 全設定暗号化対応\r
2765         if(Sts == FFFTP_SUCCESS)\r
2766         {\r
2767                 if(EncryptSettings == YES)\r
2768                 {\r
2769                         UnmaskSettingsData(Name, strlen(Name), Str, StrMultiLen(Str) + 1, YES);\r
2770                         CalculateSettingsDataChecksum(Str, StrMultiLen(Str) + 1);\r
2771                 }\r
2772         }\r
2773         return(Sts);\r
2774 }\r
2775 \r
2776 \r
2777 /*----- マルチ文字列を書き込む ------------------------------------------------\r
2778 *\r
2779 *       Parameter\r
2780 *               void *Handle : ハンドル\r
2781 *               char *Name : 名前\r
2782 *               char *Str : 文字列\r
2783 *\r
2784 *       Return Value\r
2785 *               int ステータス\r
2786 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2787 *----------------------------------------------------------------------------*/\r
2788 \r
2789 static int WriteMultiStringToReg(void *Handle, char *Name, char *Str)\r
2790 {\r
2791         REGDATATBL *Pos;\r
2792         char *Data;\r
2793 \r
2794         // 全設定暗号化対応\r
2795         if(EncryptSettings == YES)\r
2796                 MaskSettingsData(Name, strlen(Name), Str, StrMultiLen(Str) + 1, YES);\r
2797         if(TmpRegType == REGTYPE_REG)\r
2798         // 全設定暗号化対応\r
2799 //              RegSetValueEx(Handle, Name, 0, REG_MULTI_SZ, (CONST BYTE *)Str, StrMultiLen(Str)+1);\r
2800                 RegSetValueEx(Handle, Name, 0, EncryptSettings == YES ? REG_BINARY : REG_MULTI_SZ, (CONST BYTE *)Str, StrMultiLen(Str)+1);\r
2801         else\r
2802         {\r
2803                 Pos = (REGDATATBL *)Handle;\r
2804                 Data = Pos->ValTbl + Pos->ValLen;\r
2805                 strcpy(Data, Name);\r
2806                 strcat(Data, "=");\r
2807                 Pos->ValLen += strlen(Data);\r
2808                 Data = Pos->ValTbl + Pos->ValLen;\r
2809                 Pos->ValLen += StrCatOut(Str, StrMultiLen(Str), Data) + 1;\r
2810         }\r
2811         // 全設定暗号化対応\r
2812         if(EncryptSettings == YES)\r
2813         {\r
2814                 UnmaskSettingsData(Name, strlen(Name), Str, StrMultiLen(Str) + 1, YES);\r
2815                 CalculateSettingsDataChecksum(Str, StrMultiLen(Str) + 1);\r
2816         }\r
2817         return(FFFTP_SUCCESS);\r
2818 }\r
2819 \r
2820 \r
2821 /*----- バイナリを読み込む-----------------------------------------------------\r
2822 *\r
2823 *       Parameter\r
2824 *               void *Handle : ハンドル\r
2825 *               char *Name : 名前\r
2826 *               void *Bin : バイナリを返すワーク\r
2827 *               DWORD Size : 最大サイズ\r
2828 *\r
2829 *       Return Value\r
2830 *               int ステータス\r
2831 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2832 *----------------------------------------------------------------------------*/\r
2833 \r
2834 static int ReadBinaryFromReg(void *Handle, char *Name, void *Bin, DWORD Size)\r
2835 {\r
2836         int Sts;\r
2837         char *Pos;\r
2838 \r
2839         Sts = FFFTP_FAIL;\r
2840         if(TmpRegType == REGTYPE_REG)\r
2841         {\r
2842                 if(RegQueryValueEx(Handle, Name, NULL, NULL, (BYTE *)Bin, &Size) == ERROR_SUCCESS)\r
2843                         Sts = FFFTP_SUCCESS;\r
2844         }\r
2845         else\r
2846         {\r
2847                 if((Pos = ScanValue(Handle, Name)) != NULL)\r
2848                 {\r
2849                         Size = min1(Size, strlen(Pos));\r
2850                         Size = StrReadIn(Pos, Size, Bin);\r
2851                         Sts = FFFTP_SUCCESS;\r
2852                 }\r
2853         }\r
2854         // 全設定暗号化対応\r
2855         if(Sts == FFFTP_SUCCESS)\r
2856         {\r
2857                 if(EncryptSettings == YES)\r
2858                 {\r
2859                         UnmaskSettingsData(Name, strlen(Name), Bin, Size, NO);\r
2860                         CalculateSettingsDataChecksum(Bin, Size);\r
2861                 }\r
2862         }\r
2863         return(Sts);\r
2864 }\r
2865 \r
2866 \r
2867 /*----- バイナリを書き込む ----------------------------------------------------\r
2868 *\r
2869 *       Parameter\r
2870 *               void *Handle : ハンドル\r
2871 *               char *Name : 名前\r
2872 *               void *Bin : バイナリ\r
2873 *               int Len : 長さ\r
2874 *\r
2875 *       Return Value\r
2876 *               int ステータス\r
2877 *                       FFFTP_SUCCESS/FFFTP_FAIL\r
2878 *----------------------------------------------------------------------------*/\r
2879 \r
2880 static int WriteBinaryToReg(void *Handle, char *Name, void *Bin, int Len)\r
2881 {\r
2882         REGDATATBL *Pos;\r
2883         char *Data;\r
2884 \r
2885         // 全設定暗号化対応\r
2886         if(EncryptSettings == YES)\r
2887                 MaskSettingsData(Name, strlen(Name), Bin, Len, NO);\r
2888         if(TmpRegType == REGTYPE_REG)\r
2889                 RegSetValueEx(Handle, Name, 0, REG_BINARY, (CONST BYTE *)Bin, Len);\r
2890         else\r
2891         {\r
2892                 Pos = (REGDATATBL *)Handle;\r
2893                 Data = Pos->ValTbl + Pos->ValLen;\r
2894                 strcpy(Data, Name);\r
2895                 strcat(Data, "=");\r
2896                 Pos->ValLen += strlen(Data);\r
2897                 Data = Pos->ValTbl + Pos->ValLen;\r
2898                 Pos->ValLen += StrCatOut(Bin, Len, Data) + 1;\r
2899         }\r
2900         // 全設定暗号化対応\r
2901         if(EncryptSettings == YES)\r
2902         {\r
2903                 UnmaskSettingsData(Name, strlen(Name), Bin, Len, NO);\r
2904                 CalculateSettingsDataChecksum(Bin, Len);\r
2905         }\r
2906         return(FFFTP_SUCCESS);\r
2907 }\r
2908 \r
2909 \r
2910 /*----- 文字列をバッファに追加書き込みする ------------------------------------\r
2911 *\r
2912 *       Parameter\r
2913 *               char *Src : 文字列\r
2914 *               int len : 文字列の長さ\r
2915 *               char *Dst : 書き込みするバッファ\r
2916 *\r
2917 *       Return Value\r
2918 *               int 追加したバイト数\r
2919 *----------------------------------------------------------------------------*/\r
2920 \r
2921 static int StrCatOut(char *Src, int Len, char *Dst)\r
2922 {\r
2923         int Count;\r
2924 \r
2925         Dst += strlen(Dst);\r
2926         Count = 0;\r
2927         for(; Len > 0; Len--)\r
2928         {\r
2929                 if(*Src == '\\')\r
2930                 {\r
2931                         *Dst++ = '\\';\r
2932                         *Dst++ = '\\';\r
2933                         Count += 2;\r
2934                 }\r
2935                 else if((*Src >= 0x20) && (*Src <= 0x7E))\r
2936                 {\r
2937                         *Dst++ = *Src;\r
2938                         Count++;\r
2939                 }\r
2940                 else\r
2941                 {\r
2942                         sprintf(Dst, "\\%02X", *(unsigned char *)Src);\r
2943                         Dst += 3;\r
2944                         Count += 3;\r
2945                 }\r
2946                 Src++;\r
2947         }\r
2948         *Dst = NUL;\r
2949         return(Count);\r
2950 }\r
2951 \r
2952 \r
2953 /*----- 文字列をバッファに読み込む --------------------------------------------\r
2954 *\r
2955 *       Parameter\r
2956 *               char *Src : 文字列\r
2957 *               int Max : 最大サイズ\r
2958 *               char *Dst : 書き込みするバッファ\r
2959 *\r
2960 *       Return Value\r
2961 *               int 読み込んだバイト数\r
2962 *----------------------------------------------------------------------------*/\r
2963 \r
2964 static int StrReadIn(char *Src, int Max, char *Dst)\r
2965 {\r
2966         int Count;\r
2967         int Tmp;\r
2968 \r
2969         Count = 0;\r
2970         while(*Src != NUL)\r
2971         {\r
2972                 if(Count >= Max)\r
2973                         break;\r
2974 \r
2975                 if(*Src == '\\')\r
2976                 {\r
2977                         Src++;\r
2978                         if(*Src == '\\')\r
2979                                 *Dst = '\\';\r
2980                         else\r
2981                         {\r
2982                                 sscanf(Src, "%02x", &Tmp);\r
2983                                 *Dst = Tmp;\r
2984                                 Src++;\r
2985                         }\r
2986                 }\r
2987                 else\r
2988                         *Dst = *Src;\r
2989 \r
2990                 Count++;\r
2991                 Dst++;\r
2992                 Src++;\r
2993         }\r
2994         return(Count);\r
2995 }\r
2996 \r
2997 \r
2998 /*----- 値を検索する ----------------------------------------------------------\r
2999 *\r
3000 *       Parameter\r
3001 *               char *Handle : ハンドル\r
3002 *               char *Name : 名前\r
3003 *\r
3004 *       Return Value\r
3005 *               char *値データの先頭\r
3006 *                       NULL=指定の名前の値が見つからない\r
3007 *----------------------------------------------------------------------------*/\r
3008 \r
3009 static char *ScanValue(void *Handle, char *Name)\r
3010 {\r
3011         REGDATATBL *Cur;\r
3012         char *Pos;\r
3013         char *Ret;\r
3014 \r
3015         Ret = NULL;\r
3016         Cur = Handle;\r
3017         Pos = Cur->ValTbl;\r
3018         while(Pos < (Cur->ValTbl + Cur->ValLen))\r
3019         {\r
3020                 if((strncmp(Name, Pos, strlen(Name)) == 0) &&\r
3021                    (*(Pos + strlen(Name)) == '='))\r
3022                 {\r
3023                         Ret = Pos + strlen(Name) + 1;\r
3024                         break;\r
3025                 }\r
3026                 Pos += strlen(Pos) + 1;\r
3027         }\r
3028         return(Ret);\r
3029 }\r
3030 \r
3031 \r
3032 /*----------- パスワードの妥当性を確認する ------------------------------------\r
3033 *\r
3034 *       Parameter\r
3035 *               char *Password: パスワード文字列\r
3036 *               char *HashStr: SHA-1ハッシュ文字列\r
3037 *\r
3038 *       Return Value\r
3039 *               int 0 不一致\r
3040 *                       1: 一致\r
3041 *                       2: 異常\r
3042 *----------------------------------------------------------------------------*/\r
3043 int CheckPasswordValidity( char* Password, int length, const char* HashStr )\r
3044 {\r
3045         ulong hash1[5];\r
3046         ulong hash2[5];\r
3047         \r
3048         int i, j;\r
3049         \r
3050         const char* p = HashStr;\r
3051         \r
3052         /* 空文字列は一致したことにする */\r
3053         if( HashStr[0] == NUL ) return 1;\r
3054 \r
3055         /* Hashをチェックするする*/\r
3056         if( strlen(HashStr) != 40 )     return 2;\r
3057 \r
3058         /* Hashをデコードする*/\r
3059         for( i = 0; i < 5; i++ ){\r
3060                 ulong decode = 0;\r
3061                 for( j = 0; j < 8; j++ ){\r
3062                         if( *p < 0x40 || 0x40 + 15 < *p ){\r
3063                                 return 2;\r
3064                         }\r
3065                         decode = (decode << 4 ) + (*p - 0x40);\r
3066                         ++p;\r
3067                 }\r
3068                 hash1[i] = decode;\r
3069         }\r
3070         \r
3071         /* Password をハッシュする */\r
3072         sha_memory( Password, length, hash2 );\r
3073         \r
3074         if( memcmp( (char*)hash1, (char*)hash2, sizeof( hash1 )) == 0 ){\r
3075                 return 1;\r
3076         }\r
3077         return 0;\r
3078 }\r
3079 \r
3080 /*----------- パスワードの妥当性チェックのための文字列を作成する ------------\r
3081 *\r
3082 *       Parameter\r
3083 *               char *Password: パスワード文字列\r
3084 *               char *Str: SHA-1ハッシュ文字列格納場所 (41bytes以上)\r
3085 *\r
3086 *       Return Value\r
3087 *               None\r
3088 *----------------------------------------------------------------------------*/\r
3089 void CreatePasswordHash( char* Password, int length, char* HashStr )\r
3090 {\r
3091         ulong hash[5];\r
3092         int i, j;\r
3093         unsigned char *p = (unsigned char *)HashStr;\r
3094 \r
3095         sha_memory( Password, length, hash );\r
3096 \r
3097         for( i = 0; i < 5; i++ ){\r
3098                 ulong rest = hash[i];\r
3099                 for( j = 0; j < 8; j++ ){\r
3100                         *p++ = (unsigned char)((rest & 0xf0000000) >> 28) + '@';\r
3101                         rest <<= 4;\r
3102                 }\r
3103         }\r
3104         *p = NUL;\r
3105 }\r
3106 \r
3107 void SetHashSalt( DWORD salt )\r
3108 {\r
3109         unsigned char* pos = &SecretKey[strlen(SecretKey) + 1];\r
3110         *pos++ = ( salt >> 24 ) & 0xff;\r
3111         *pos++ = ( salt >> 16 ) & 0xff;\r
3112         *pos++ = ( salt >>  8 ) & 0xff;\r
3113         *pos++ = ( salt       ) & 0xff;\r
3114         \r
3115         SecretKeyLength = strlen( SecretKey ) + 5;\r
3116 }\r
3117 \r
3118 /*----------- 乱数生成をする -------------------------------------------------\r
3119 *\r
3120 *       Parameter\r
3121 *\r
3122 *       Return Value\r
3123 *               ランダムな値:コンパイラVS2005/動作環境WinXP以上では rand_s から取得する\r
3124 *----------------------------------------------------------------------------*/\r
3125 DWORD GetRandamDWRODValue(void)\r
3126 {\r
3127         DWORD rndValue;\r
3128         int errorCode;\r
3129 #ifdef _CRT_RAND_S\r
3130         errno_t errnoRand_s;\r
3131         errnoRand_s = rand_s(&rndValue);\r
3132         errorCode = (0 != errnoRand_s ? 1 : 0);\r
3133 #else\r
3134         errorCode = 1;\r
3135 #endif\r
3136         if(0 != errorCode){\r
3137 #ifdef USE_RANDAM_C_RAND\r
3138                 rndValue = rand() | (rand() << 16);\r
3139 #else\r
3140                 /* rand() のかわりに、SHA-1とパフォーマンスカウンタを用いる */\r
3141                 ulong shaValue[5];\r
3142                 LARGE_INTEGER Counter;\r
3143                 \r
3144                 if(0 == IsRndSourceInit){\r
3145                         /* 初回取得時 */\r
3146                         HANDLE CurProcHandle;\r
3147                         HANDLE CurThreadHandle;\r
3148                         \r
3149                         if(DuplicateHandle(GetCurrentProcess(), GetCurrentProcess(), GetCurrentProcess(),\r
3150                                 &CurProcHandle, 0, FALSE, DUPLICATE_SAME_ACCESS))\r
3151                         {\r
3152                                 CloseHandle(CurProcHandle);\r
3153                         }\r
3154                         if(DuplicateHandle(GetCurrentProcess(), GetCurrentThread(), GetCurrentProcess(),\r
3155                                 &CurThreadHandle, 0, FALSE, DUPLICATE_SAME_ACCESS))\r
3156                         {\r
3157                                 CloseHandle(CurThreadHandle);\r
3158                         }\r
3159                         \r
3160                         /* _WIN64 では64bitだが、その場合はrand_sが大抵利用可能なのでここでは32bitのみ用いる */\r
3161                         RndSource[0] = (ulong)CurProcHandle;\r
3162                         RndSource[1] = (ulong)CurThreadHandle;\r
3163                         RndSource[2] = (ulong)GetTickCount();\r
3164                         RndSource[3] = (ulong)timeGetTime();\r
3165                         RndSource[4] = 0; /* カウントアップ */\r
3166                         RndSource[5] = RndSource[3] + 1;\r
3167                         IsRndSourceInit = 1;\r
3168                 }\r
3169                 RndSource[4]++;\r
3170                 RndSource[5] += 0x00010010;\r
3171                 if(QueryPerformanceCounter(&Counter)){\r
3172                         RndSource[6] = Counter.LowPart;\r
3173                         RndSource[7] = Counter.HighPart;\r
3174                         RndSource[8] = (ulong)rand();\r
3175                 }else{\r
3176                         RndSource[6] = (ulong)timeGetTime();\r
3177                         RndSource[7] = (ulong)rand();\r
3178                         RndSource[8] = (ulong)rand();\r
3179                 }\r
3180                 \r
3181                 sha_memory((char *)RndSource, sizeof(RndSource), shaValue);\r
3182                 rndValue = shaValue[0] ^ shaValue[1] ^ shaValue[2] ^ shaValue[3] ^ shaValue[4];\r
3183 #endif\r
3184         }\r
3185         return rndValue;\r
3186 }\r
3187 \r
3188 // 全設定暗号化対応\r
3189 void MaskSettingsData(const char* Salt, int SaltLength, void* Data, DWORD Size, int EscapeZero)\r
3190 {\r
3191         char Key[FMAX_PATH*2+1];\r
3192         ulong Hash[5];\r
3193         DWORD i;\r
3194         BYTE Mask[20];\r
3195         BYTE* p;\r
3196         memcpy(&Key[0], SecretKey, SecretKeyLength);\r
3197         memcpy(&Key[SecretKeyLength], Salt, SaltLength);\r
3198         sha_memory(Key, SecretKeyLength + SaltLength, Hash);\r
3199         // sha.cはビッグエンディアンのため\r
3200         for(i = 0; i < 5; i++)\r
3201                 Hash[i] = _byteswap_ulong(Hash[i]);\r
3202         memcpy(&Mask, &Hash, 20);\r
3203         p = (BYTE*)Data;\r
3204         for(i = 0; i < Size; i++)\r
3205         {\r
3206                 if(EscapeZero == YES)\r
3207                 {\r
3208                         if(p[i] != 0 && p[i] != Mask[i % sizeof(Mask)])\r
3209                                 p[i] ^= Mask[i % sizeof(Mask)];\r
3210                 }\r
3211                 else\r
3212                         p[i] ^= Mask[i % sizeof(Mask)];\r
3213         }\r
3214 }\r
3215 \r
3216 void UnmaskSettingsData(const char* Salt, int SaltLength, void* Data, DWORD Size, int EscapeZero)\r
3217 {\r
3218         char Key[FMAX_PATH*2+1];\r
3219         ulong Hash[5];\r
3220         DWORD i;\r
3221         BYTE Mask[20];\r
3222         BYTE* p;\r
3223         memcpy(&Key[0], SecretKey, SecretKeyLength);\r
3224         memcpy(&Key[SecretKeyLength], Salt, SaltLength);\r
3225         sha_memory(Key, SecretKeyLength + SaltLength, Hash);\r
3226         // sha.cはビッグエンディアンのため\r
3227         for(i = 0; i < 5; i++)\r
3228                 Hash[i] = _byteswap_ulong(Hash[i]);\r
3229         memcpy(&Mask, &Hash, 20);\r
3230         p = (BYTE*)Data;\r
3231         for(i = 0; i < Size; i++)\r
3232         {\r
3233                 if(EscapeZero == YES)\r
3234                 {\r
3235                         if(p[i] != 0 && p[i] != Mask[i % sizeof(Mask)])\r
3236                                 p[i] ^= Mask[i % sizeof(Mask)];\r
3237                 }\r
3238                 else\r
3239                         p[i] ^= Mask[i % sizeof(Mask)];\r
3240         }\r
3241 }\r
3242 \r
3243 void CalculateSettingsDataChecksum(void* Data, DWORD Size)\r
3244 {\r
3245         ulong Hash[5];\r
3246         DWORD i;\r
3247         BYTE Mask[20];\r
3248         sha_memory((char*)Data, Size, Hash);\r
3249         // sha.cはビッグエンディアンのため\r
3250         for(i = 0; i < 5; i++)\r
3251                 Hash[i] = _byteswap_ulong(Hash[i]);\r
3252         memcpy(&Mask, &Hash, 20);\r
3253         for(i = 0; i < 20; i++)\r
3254                 EncryptSettingsChecksum[i] ^= Mask[i];\r
3255 }\r
3256 \r
3257 // ポータブル版判定\r
3258 int IsRegAvailable()\r
3259 {\r
3260         int Sts;\r
3261         void* h;\r
3262         Sts = NO;\r
3263         SetRegType(REGTYPE_REG);\r
3264         if(OpenReg("FFFTP", &h) == FFFTP_SUCCESS)\r
3265         {\r
3266                 CloseReg(h);\r
3267                 Sts = YES;\r
3268         }\r
3269         return Sts;\r
3270 }\r
3271 \r
3272 int IsIniAvailable()\r
3273 {\r
3274         int Sts;\r
3275         void* h;\r
3276         Sts = NO;\r
3277         SetRegType(REGTYPE_INI);\r
3278         if(OpenReg("FFFTP", &h) == FFFTP_SUCCESS)\r
3279         {\r
3280                 CloseReg(h);\r
3281                 Sts = YES;\r
3282         }\r
3283         return Sts;\r
3284 }\r
3285 \r