OSDN Git Service

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