OSDN Git Service

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