1 /*=============================================================================
\r
5 ===============================================================================
\r
6 / Copyright (C) 1997-2007 Sota. All rights reserved.
\r
8 / Redistribution and use in source and binary forms, with or without
\r
9 / modification, are permitted provided that the following conditions
\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
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
30 /* このソースは一部、WS_FTP Version 93.12.05 のソースを参考にしました。 */
\r
31 /* スレッドの作成/終了に関して、樋口殿作成のパッチを組み込みました。 */
\r
34 一部、高速化のためのコード追加 by H.Shirouzu at 2002/10/02
\r
42 #include <mbstring.h>
\r
45 //#include <winsock.h>
\r
46 #include <winsock2.h>
\r
47 #include <windowsx.h>
\r
48 #include <commctrl.h>
\r
49 #include <process.h>
\r
52 #include "resource.h"
\r
54 #define SET_BUFFER_SIZE
\r
56 /* Add by H.Shirouzu at 2002/10/02 */
\r
58 #define BUFSIZE (32 * 1024)
\r
59 #define SOCKBUF_SIZE (256 * 1024)
\r
62 #ifdef DISABLE_TRANSFER_NETWORK_BUFFERS
\r
64 #define BUFSIZE (64 * 1024) // RWIN値以下で充分な大きさが望ましいと思われる。
\r
65 #undef SET_BUFFER_SIZE
\r
68 #define TIMER_DISPLAY 1 /* 表示更新用タイマのID */
\r
69 #define DISPLAY_TIMING 500 /* 表示更新時間 0.5秒 */
\r
71 #define ERR_MSG_LEN 1024
\r
81 /*===== プロトタイプ =====*/
\r
83 static void DispTransPacket(TRANSPACKET *Pkt);
\r
84 static void EraseTransFileList(void);
\r
85 static ULONG WINAPI TransferThread(void *Dummy);
\r
86 static int MakeNonFullPath(TRANSPACKET *Pkt, char *CurDir, char *Tmp);
\r
88 static int SetDownloadedFileTime(TRANSPACKET *Pkt);
\r
89 static int DownloadNonPassive(TRANSPACKET *Pkt, int *CancelCheckWork);
\r
90 static int DownloadPassive(TRANSPACKET *Pkt, int *CancelCheckWork);
\r
91 static int DownloadFile(TRANSPACKET *Pkt, SOCKET dSkt, int CreateMode, int *CancelCheckWork);
\r
92 static void DispDownloadFinishMsg(TRANSPACKET *Pkt, int iRetCode);
\r
94 //static int DispUpDownErrDialog(int ResID, HWND hWnd, char *Fname);
\r
95 static int DispUpDownErrDialog(int ResID, HWND hWnd, TRANSPACKET *Pkt);
\r
97 //static BOOL CALLBACK UpDownErrorDialogProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam);
\r
98 static INT_PTR CALLBACK UpDownErrorDialogProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam);
\r
99 static int SetDownloadResume(TRANSPACKET *Pkt, int ProcMode, LONGLONG Size, int *Mode, int *CancelCheckWork);
\r
101 //static BOOL CALLBACK NoResumeWndProc(HWND hDlg, UINT iMessage, WPARAM wParam, LPARAM lParam);
\r
102 static INT_PTR CALLBACK NoResumeWndProc(HWND hDlg, UINT iMessage, WPARAM wParam, LPARAM lParam);
\r
103 static int DoUpload(SOCKET cSkt, TRANSPACKET *Pkt);
\r
104 static int UploadNonPassive(TRANSPACKET *Pkt);
\r
105 static int UploadPassive(TRANSPACKET *Pkt);
\r
106 static int UploadFile(TRANSPACKET *Pkt, SOCKET dSkt);
\r
108 //static int TermCodeConvAndSend(TERMCODECONVINFO *tInfo, SOCKET Skt, char *Data, int Size, int Ascii);
\r
109 static int TermCodeConvAndSend(TERMCODECONVINFO *tInfo, SOCKET Skt, char *Data, int Size, int Ascii, int *CancelCheckWork);
\r
110 static void DispUploadFinishMsg(TRANSPACKET *Pkt, int iRetCode);
\r
111 static int SetUploadResume(TRANSPACKET *Pkt, int ProcMode, LONGLONG Size, int *Mode);
\r
112 static LRESULT CALLBACK TransDlgProc(HWND hDlg, UINT Msg, WPARAM wParam, LPARAM lParam);
\r
113 static void DispTransferStatus(HWND hWnd, int End, TRANSPACKET *Pkt);
\r
114 static void DispTransFileInfo(TRANSPACKET *Pkt, char *Title, int SkipButton, int Info);
\r
116 //static int GetAdrsAndPort(char *Str, char *Adrs, int *Port, int Max);
\r
117 static int GetAdrsAndPort(SOCKET Skt, char *Str, char *Adrs, int *Port, int Max);
\r
118 static int GetAdrsAndPortIPv4(SOCKET Skt, char *Str, char *Adrs, int *Port, int Max);
\r
119 static int GetAdrsAndPortIPv6(SOCKET Skt, char *Str, char *Adrs, int *Port, int Max);
\r
120 static int IsSpecialDevice(char *Fname);
\r
121 static int MirrorDelNotify(int Cur, int Notify, TRANSPACKET *Pkt);
\r
123 //static BOOL CALLBACK MirrorDeleteDialogCallBack(HWND hDlg, UINT iMessage, WPARAM wParam, LPARAM lParam);
\r
124 static INT_PTR CALLBACK MirrorDeleteDialogCallBack(HWND hDlg, UINT iMessage, WPARAM wParam, LPARAM lParam);
\r
125 static void SetErrorMsg(char *fmt, ...);
\r
127 static char* GetErrMsg();
\r
129 /*===== ローカルなワーク =====*/
\r
132 //static HANDLE hTransferThread;
\r
133 static HANDLE hTransferThread[MAX_DATA_CONNECTION];
\r
134 static int fTransferThreadExit = FALSE;
\r
136 static HANDLE hRunMutex; /* 転送スレッド実行ミューテックス */
\r
137 static HANDLE hListAccMutex; /* 転送ファイルアクセス用ミューテックス */
\r
139 static int TransFiles = 0; /* 転送待ちファイル数 */
\r
140 static TRANSPACKET *TransPacketBase = NULL; /* 転送ファイルリスト */
\r
142 static TRANSPACKET *NextTransPacketBase = NULL;
\r
145 //static int Canceled; /* 中止フラグ YES/NO */
\r
146 static int Canceled[MAX_DATA_CONNECTION]; /* 中止フラグ YES/NO */
\r
147 static int ClearAll; /* 全て中止フラグ YES/NO */
\r
149 static int ForceAbort; /* 転送中止フラグ */
\r
150 /* このフラグはスレッドを終了させるときに使う */
\r
153 //static LONGLONG AllTransSizeNow; /* 今回の転送で転送したサイズ */
\r
154 //static time_t TimeStart; /* 転送開始時間 */
\r
155 static LONGLONG AllTransSizeNow[MAX_DATA_CONNECTION]; /* 今回の転送で転送したサイズ */
\r
156 static time_t TimeStart[MAX_DATA_CONNECTION]; /* 転送開始時間 */
\r
158 static int KeepDlg = NO; /* 転送中ダイアログを消さないかどうか (YES/NO) */
\r
159 static int MoveToForeground = NO; /* ウインドウを前面に移動するかどうか (YES/NO) */
\r
162 //static char CurDir[FMAX_PATH+1] = { "" };
\r
163 static char CurDir[MAX_DATA_CONNECTION][FMAX_PATH+1];
\r
165 //static char ErrMsg[ERR_MSG_LEN+7];
\r
166 static char ErrMsg[MAX_DATA_CONNECTION+1][ERR_MSG_LEN+7];
\r
167 static DWORD ErrMsgThreadId[MAX_DATA_CONNECTION+1];
\r
168 static HANDLE hErrMsgMutex;
\r
171 static int WaitForMainThread = NO;
\r
173 static int TransferErrorMode = EXIST_OVW;
\r
174 static int TransferErrorNotify = NO;
\r
176 static LONGLONG TransferSizeLeft = 0;
\r
177 static LONGLONG TransferSizeTotal = 0;
\r
178 static int TransferErrorDisplay = 0;
\r
180 /*===== 外部参照 =====*/
\r
183 extern int SaveTimeStamp;
\r
185 // extern int TimeOut;
\r
186 extern int FwallType;
\r
187 extern int MirUpDelNotify;
\r
188 extern int MirDownDelNotify;
\r
189 extern int FolderAttr;
\r
190 extern int FolderAttrNum;
\r
192 extern int SendQuit;
\r
194 extern time_t LastDataConnectionTime;
\r
197 /*----- ファイル転送スレッドを起動する ----------------------------------------
\r
204 *----------------------------------------------------------------------------*/
\r
206 int MakeTransferThread(void)
\r
211 hListAccMutex = CreateMutex( NULL, FALSE, NULL );
\r
212 hRunMutex = CreateMutex( NULL, TRUE, NULL );
\r
214 hErrMsgMutex = CreateMutex( NULL, FALSE, NULL );
\r
219 fTransferThreadExit = FALSE;
\r
221 // hTransferThread = (HANDLE)_beginthreadex(NULL, 0, TransferThread, 0, 0, &dwID);
\r
222 // if (hTransferThread == NULL)
\r
223 // return(FFFTP_FAIL); /* XXX */
\r
224 for(i = 0; i < MAX_DATA_CONNECTION; i++)
\r
226 hTransferThread[i] = (HANDLE)_beginthreadex(NULL, 0, TransferThread, (void*)i, 0, &dwID);
\r
227 if(hTransferThread[i] == NULL)
\r
231 return(FFFTP_SUCCESS);
\r
235 /*----- ファイル転送スレッドを終了する ----------------------------------------
\r
242 *----------------------------------------------------------------------------*/
\r
244 void CloseTransferThread(void)
\r
249 for(i = 0; i < MAX_DATA_CONNECTION; i++)
\r
253 // ForceAbort = YES;
\r
255 fTransferThreadExit = TRUE;
\r
257 // while(WaitForSingleObject(hTransferThread, 10) == WAIT_TIMEOUT)
\r
259 // BackgrndMessageProc();
\r
262 // CloseHandle(hTransferThread);
\r
263 for(i = 0; i < MAX_DATA_CONNECTION; i++)
\r
265 while(WaitForSingleObject(hTransferThread[i], 10) == WAIT_TIMEOUT)
\r
267 BackgrndMessageProc();
\r
270 CloseHandle(hTransferThread[i]);
\r
273 ReleaseMutex( hRunMutex );
\r
275 CloseHandle( hListAccMutex );
\r
276 CloseHandle( hRunMutex );
\r
278 CloseHandle( hErrMsgMutex );
\r
284 void AbortAllTransfer()
\r
287 while(TransPacketBase != NULL)
\r
289 for(i = 0; i < MAX_DATA_CONNECTION; i++)
\r
292 if(BackgrndMessageProc() == YES)
\r
299 /*----- 転送するファイル情報をリストに追加する --------------------------------
\r
302 * TRANSPACKET *Pkt : 転送ファイル情報
\r
303 * TRANSPACKET **Base : リストの先頭
\r
307 * FFFTP_SUCCESS/FFFTP_FAIL
\r
308 *----------------------------------------------------------------------------*/
\r
310 int AddTmpTransFileList(TRANSPACKET *Pkt, TRANSPACKET **Base)
\r
317 if((Pos = malloc(sizeof(TRANSPACKET))) != NULL)
\r
319 memcpy(Pos, Pkt, sizeof(TRANSPACKET));
\r
327 while(Prev->Next != NULL)
\r
331 Sts = FFFTP_SUCCESS;
\r
337 /*----- 転送するファイル情報リストをクリアする --------------------------------
\r
340 * TRANSPACKET **Base : リストの先頭
\r
344 *----------------------------------------------------------------------------*/
\r
346 void EraseTmpTransFileList(TRANSPACKET **Base)
\r
363 /*----- 転送するファイル情報リストから1つの情報を取り除く --------------------
\r
366 * TRANSPACKET *Pkt : 転送ファイル情報
\r
367 * TRANSPACKET **Base : リストの先頭
\r
371 * FFFTP_SUCCESS/FFFTP_FAIL
\r
372 *----------------------------------------------------------------------------*/
\r
374 int RemoveTmpTransFileListItem(TRANSPACKET **Base, int Num)
\r
386 Sts = FFFTP_SUCCESS;
\r
396 Prev->Next = Pos->Next;
\r
398 Sts = FFFTP_SUCCESS;
\r
407 /*----- 転送するファイル情報を転送ファイルリストに登録する --------------------
\r
410 * TRANSPACKET *Pkt : 転送ファイル情報
\r
414 *----------------------------------------------------------------------------*/
\r
416 void AddTransFileList(TRANSPACKET *Pkt)
\r
421 DispTransPacket(Pkt);
\r
424 // WaitForSingleObject(hListAccMutex, INFINITE);
\r
425 while(WaitForSingleObject(hListAccMutex, 0) == WAIT_TIMEOUT)
\r
427 WaitForMainThread = YES;
\r
428 BackgrndMessageProc();
\r
433 Pos = TransPacketBase;
\r
436 while(Pos->Next != NULL)
\r
439 if(AddTmpTransFileList(Pkt, &TransPacketBase) == FFFTP_SUCCESS)
\r
441 if((strncmp(Pkt->Cmd, "RETR", 4) == 0) ||
\r
442 (strncmp(Pkt->Cmd, "STOR", 4) == 0))
\r
446 TransferSizeLeft += Pkt->Size;
\r
447 TransferSizeTotal += Pkt->Size;
\r
448 PostMessage(GetMainHwnd(), WM_CHANGE_COND, 0, 0);
\r
452 if(NextTransPacketBase == NULL)
\r
455 NextTransPacketBase = Pos->Next;
\r
457 NextTransPacketBase = TransPacketBase;
\r
459 ReleaseMutex(hListAccMutex);
\r
461 WaitForMainThread = NO;
\r
468 void AddNullTransFileList()
\r
471 memset(&Pkt, 0, sizeof(TRANSPACKET));
\r
472 strcpy(Pkt.Cmd, "NULL");
\r
473 AddTransFileList(&Pkt);
\r
476 /*----- 転送ファイル情報を転送ファイルリストに追加する ------------------------
\r
479 * TRANSPACKET *Pkt : 転送ファイル情報
\r
480 * TRANSPACKET **Base : リストの先頭
\r
487 *----------------------------------------------------------------------------*/
\r
489 void AppendTransFileList(TRANSPACKET *Pkt)
\r
494 // WaitForSingleObject(hListAccMutex, INFINITE);
\r
495 while(WaitForSingleObject(hListAccMutex, 0) == WAIT_TIMEOUT)
\r
497 WaitForMainThread = YES;
\r
498 BackgrndMessageProc();
\r
502 if(TransPacketBase == NULL)
\r
503 TransPacketBase = Pkt;
\r
506 Pos = TransPacketBase;
\r
507 while(Pos->Next != NULL)
\r
512 if(NextTransPacketBase == NULL)
\r
513 NextTransPacketBase = Pkt;
\r
517 DispTransPacket(Pkt);
\r
519 if((strncmp(Pkt->Cmd, "RETR", 4) == 0) ||
\r
520 (strncmp(Pkt->Cmd, "STOR", 4) == 0))
\r
524 TransferSizeLeft += Pkt->Size;
\r
525 TransferSizeTotal += Pkt->Size;
\r
526 PostMessage(GetMainHwnd(), WM_CHANGE_COND, 0, 0);
\r
531 ReleaseMutex(hListAccMutex);
\r
533 WaitForMainThread = NO;
\r
538 /*----- 転送ファイル情報を表示する --------------------------------------------
\r
541 * TRANSPACKET *Pkt : 転送ファイル情報
\r
545 *----------------------------------------------------------------------------*/
\r
547 static void DispTransPacket(TRANSPACKET *Pkt)
\r
549 if((strncmp(Pkt->Cmd, "RETR", 4) == 0) || (strncmp(Pkt->Cmd, "STOR", 4) == 0))
\r
550 DoPrintf("TransList Cmd=%s : %s : %s", Pkt->Cmd, Pkt->RemoteFile, Pkt->LocalFile);
\r
551 else if(strncmp(Pkt->Cmd, "R-", 2) == 0)
\r
552 DoPrintf("TransList Cmd=%s : %s", Pkt->Cmd, Pkt->RemoteFile);
\r
553 else if(strncmp(Pkt->Cmd, "L-", 2) == 0)
\r
554 DoPrintf("TransList Cmd=%s : %s", Pkt->Cmd, Pkt->LocalFile);
\r
555 else if(strncmp(Pkt->Cmd, "MKD", 3) == 0)
\r
557 if(strlen(Pkt->LocalFile) > 0)
\r
558 DoPrintf("TransList Cmd=%s : %s", Pkt->Cmd, Pkt->LocalFile);
\r
560 DoPrintf("TransList Cmd=%s : %s", Pkt->Cmd, Pkt->RemoteFile);
\r
563 DoPrintf("TransList Cmd=%s", Pkt->Cmd);
\r
568 /*----- 転送ファイルリストをクリアする ----------------------------------------
\r
575 *----------------------------------------------------------------------------*/
\r
577 static void EraseTransFileList(void)
\r
581 TRANSPACKET *NotDel;
\r
582 // TRANSPACKET Pkt;
\r
587 // WaitForSingleObject(hListAccMutex, INFINITE);
\r
588 while(WaitForSingleObject(hListAccMutex, 0) == WAIT_TIMEOUT)
\r
590 WaitForMainThread = YES;
\r
591 BackgrndMessageProc();
\r
594 New = TransPacketBase;
\r
597 /* 最後の"BACKCUR"は必要なので消さない */
\r
598 if(strcmp(New->Cmd, "BACKCUR") == 0)
\r
603 strcpy(NotDel->Cmd, "");
\r
607 // NotDel->Next = NULL;
\r
614 strcpy(New->Cmd, "");
\r
618 TransPacketBase = NotDel;
\r
620 NextTransPacketBase = NotDel;
\r
623 TransferSizeLeft = 0;
\r
624 TransferSizeTotal = 0;
\r
625 PostMessage(GetMainHwnd(), WM_CHANGE_COND, 0, 0);
\r
626 ReleaseMutex(hListAccMutex);
\r
628 WaitForMainThread = NO;
\r
631 // strcpy(Pkt.Cmd, "GOQUIT");
\r
632 // AddTransFileList(&Pkt);
\r
637 /*----- 転送中ダイアログを消さないようにするかどうかを設定 --------------------
\r
640 * int Sw : 転送中ダイアログを消さないかどうか (YES/NO)
\r
644 *----------------------------------------------------------------------------*/
\r
646 void KeepTransferDialog(int Sw)
\r
653 /*----- 現在転送中かどうかを返す ----------------------------------------------
\r
659 * int ステータス (YES/NO=転送中ではない)
\r
660 *----------------------------------------------------------------------------*/
\r
662 int AskTransferNow(void)
\r
664 return(TransPacketBase != NULL ? YES : NO);
\r
668 /*----- 転送するファイルの数を返す --------------------------------------------
\r
675 *----------------------------------------------------------------------------*/
\r
677 int AskTransferFileNum(void)
\r
679 return(TransFiles);
\r
683 /*----- 転送中ウインドウを前面に出す ------------------------------------------
\r
690 *----------------------------------------------------------------------------*/
\r
692 void GoForwardTransWindow(void)
\r
694 MoveToForeground = YES;
\r
699 /*----- 転送ソケットのカレントディレクトリ情報を初期化 ------------------------
\r
706 *----------------------------------------------------------------------------*/
\r
708 void InitTransCurDir(void)
\r
711 // strcpy(CurDir, "");
\r
713 for(i = 0; i < MAX_DATA_CONNECTION; i++)
\r
714 strcpy(CurDir[i], "");
\r
719 /*----- ファイル転送スレッドのメインループ ------------------------------------
\r
722 * void *Dummy : 使わない
\r
726 *----------------------------------------------------------------------------*/
\r
728 static ULONG WINAPI TransferThread(void *Dummy)
\r
732 char Tmp[FMAX_PATH+1];
\r
755 ThreadCount = (int)Dummy;
\r
756 TrnSkt = INVALID_SOCKET;
\r
758 SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_LOWEST);
\r
760 while((TransPacketBase != NULL) ||
\r
761 (WaitForSingleObject(hRunMutex, 200) == WAIT_TIMEOUT))
\r
763 if(fTransferThreadExit == TRUE)
\r
766 if(WaitForMainThread == YES)
\r
768 BackgrndMessageProc();
\r
773 // WaitForSingleObject(hListAccMutex, INFINITE);
\r
774 while(WaitForSingleObject(hListAccMutex, 0) == WAIT_TIMEOUT)
\r
776 BackgrndMessageProc();
\r
779 // memset(ErrMsg, NUL, ERR_MSG_LEN+7);
\r
780 memset(GetErrMsg(), NUL, ERR_MSG_LEN+7);
\r
783 Canceled[ThreadCount] = NO;
\r
785 while(TransPacketBase != NULL && strcmp(TransPacketBase->Cmd, "") == 0)
\r
787 Pos = TransPacketBase;
\r
788 TransPacketBase = TransPacketBase->Next;
\r
790 if(TransPacketBase == NULL)
\r
793 if(AskReuseCmdSkt() == YES && ThreadCount == 0)
\r
795 TrnSkt = AskTrnCtrlSkt();
\r
796 // セッションあたりの転送量制限対策
\r
797 if(TrnSkt != INVALID_SOCKET && AskErrorReconnect() == YES && LastError == YES)
\r
799 ReleaseMutex(hListAccMutex);
\r
800 PostMessage(GetMainHwnd(), WM_RECONNECTSOCKET, 0, 0);
\r
802 TrnSkt = INVALID_SOCKET;
\r
803 // WaitForSingleObject(hListAccMutex, INFINITE);
\r
804 while(WaitForSingleObject(hListAccMutex, 0) == WAIT_TIMEOUT)
\r
806 BackgrndMessageProc();
\r
813 // セッションあたりの転送量制限対策
\r
814 if(TrnSkt != INVALID_SOCKET && AskErrorReconnect() == YES && LastError == YES)
\r
816 ReleaseMutex(hListAccMutex);
\r
817 DoQUIT(TrnSkt, &Canceled[ThreadCount]);
\r
819 TrnSkt = INVALID_SOCKET;
\r
820 // WaitForSingleObject(hListAccMutex, INFINITE);
\r
821 while(WaitForSingleObject(hListAccMutex, 0) == WAIT_TIMEOUT)
\r
823 BackgrndMessageProc();
\r
827 if(TransPacketBase && AskConnecting() == YES && ThreadCount < AskMaxThreadCount())
\r
829 ReleaseMutex(hListAccMutex);
\r
830 if(TrnSkt == INVALID_SOCKET)
\r
831 ReConnectTrnSkt(&TrnSkt, &Canceled[ThreadCount]);
\r
833 CheckClosedAndReconnectTrnSkt(&TrnSkt, &Canceled[ThreadCount]);
\r
835 if(TrnSkt == INVALID_SOCKET)
\r
837 // 同時ログイン数制限に引っかかった可能性あり
\r
838 // 負荷を下げるために約10秒間待機
\r
842 BackgrndMessageProc();
\r
847 LastUsed = timeGetTime();
\r
848 // WaitForSingleObject(hListAccMutex, INFINITE);
\r
849 while(WaitForSingleObject(hListAccMutex, 0) == WAIT_TIMEOUT)
\r
851 BackgrndMessageProc();
\r
857 if(TrnSkt != INVALID_SOCKET)
\r
860 // 60秒間使用されなければログアウト
\r
861 if(timeGetTime() - LastUsed > 60000 || AskConnecting() == NO || ThreadCount >= AskMaxThreadCount())
\r
863 ReleaseMutex(hListAccMutex);
\r
864 DoQUIT(TrnSkt, &Canceled[ThreadCount]);
\r
866 TrnSkt = INVALID_SOCKET;
\r
867 // WaitForSingleObject(hListAccMutex, INFINITE);
\r
868 while(WaitForSingleObject(hListAccMutex, 0) == WAIT_TIMEOUT)
\r
870 BackgrndMessageProc();
\r
878 // if(TransPacketBase != NULL)
\r
879 if(TrnSkt != INVALID_SOCKET && NextTransPacketBase != NULL)
\r
881 Pos = NextTransPacketBase;
\r
882 NextTransPacketBase = NextTransPacketBase->Next;
\r
883 // ディレクトリ操作は非同期で行わない
\r
884 // ReleaseMutex(hListAccMutex);
\r
885 if(hWndTrans == NULL)
\r
887 // if((strncmp(TransPacketBase->Cmd, "RETR", 4) == 0) ||
\r
888 // (strncmp(TransPacketBase->Cmd, "STOR", 4) == 0) ||
\r
889 // (strncmp(TransPacketBase->Cmd, "MKD", 3) == 0) ||
\r
890 // (strncmp(TransPacketBase->Cmd, "L-", 2) == 0) ||
\r
891 // (strncmp(TransPacketBase->Cmd, "R-", 2) == 0))
\r
892 if((strncmp(Pos->Cmd, "RETR", 4) == 0) ||
\r
893 (strncmp(Pos->Cmd, "STOR", 4) == 0) ||
\r
894 (strncmp(Pos->Cmd, "MKD", 3) == 0) ||
\r
895 (strncmp(Pos->Cmd, "L-", 2) == 0) ||
\r
896 (strncmp(Pos->Cmd, "R-", 2) == 0))
\r
898 hWndTrans = CreateDialog(GetFtpInst(), MAKEINTRESOURCE(transfer_dlg), HWND_DESKTOP, (DLGPROC)TransDlgProc);
\r
899 if(MoveToForeground == YES)
\r
900 SetForegroundWindow(hWndTrans);
\r
901 ShowWindow(hWndTrans, SW_SHOWNOACTIVATE);
\r
902 GetWindowRect(hWndTrans, &WndRect);
\r
903 SetWindowPos(hWndTrans, NULL, WndRect.left, WndRect.top + (WndRect.bottom - WndRect.top) * ThreadCount - (WndRect.bottom - WndRect.top) * (AskMaxThreadCount() - 1) / 2, 0, 0, SWP_NOSIZE | SWP_NOZORDER);
\r
906 // TransPacketBase->hWndTrans = hWndTrans;
\r
907 Pos->hWndTrans = hWndTrans;
\r
908 Pos->ctrl_skt = TrnSkt;
\r
909 Pos->Abort = ABORT_NONE;
\r
910 Pos->ThreadCount = ThreadCount;
\r
912 if(hWndTrans != NULL)
\r
914 if(MoveToForeground == YES)
\r
916 SetForegroundWindow(hWndTrans);
\r
917 MoveToForeground = NO;
\r
921 if(hWndTrans != NULL)
\r
922 // SendMessage(hWndTrans, WM_SET_PACKET, 0, (LPARAM)TransPacketBase);
\r
923 SendMessage(hWndTrans, WM_SET_PACKET, 0, (LPARAM)Pos);
\r
925 // 中断後に受信バッファに応答が残っていると次のコマンドの応答が正しく処理できない
\r
926 RemoveReceivedData(TrnSkt);
\r
929 // if(strncmp(TransPacketBase->Cmd, "RETR", 4) == 0)
\r
930 if(strncmp(Pos->Cmd, "RETR", 4) == 0)
\r
932 // 一部TYPE、STOR(RETR)、PORT(PASV)を並列に処理できないホストがあるため
\r
933 // ReleaseMutex(hListAccMutex);
\r
935 // if(CheckPathViolation(TransPacketBase) == NO)
\r
936 if(CheckPathViolation(Pos) == NO)
\r
938 /* フルパスを使わないための処理 */
\r
939 // if(MakeNonFullPath(TransPacketBase, CurDir, Tmp) == FFFTP_SUCCESS)
\r
940 if(MakeNonFullPath(Pos, CurDir[Pos->ThreadCount], Tmp) == FFFTP_SUCCESS)
\r
942 // if(strncmp(TransPacketBase->Cmd, "RETR-S", 6) == 0)
\r
943 if(strncmp(Pos->Cmd, "RETR-S", 6) == 0)
\r
946 // DoSIZE(TransPacketBase->RemoteFile, &TransPacketBase->Size);
\r
947 // DoMDTM(TransPacketBase->RemoteFile, &TransPacketBase->Time);
\r
948 // strcpy(TransPacketBase->Cmd, "RETR ");
\r
949 DoSIZE(TrnSkt, Pos->RemoteFile, &Pos->Size, &Canceled[Pos->ThreadCount]);
\r
950 DoMDTM(TrnSkt, Pos->RemoteFile, &Pos->Time, &Canceled[Pos->ThreadCount]);
\r
951 strcpy(Pos->Cmd, "RETR ");
\r
955 // if(DoDownload(AskTrnCtrlSkt(), TransPacketBase, NO) == 429)
\r
957 // if(ReConnectTrnSkt() == FFFTP_SUCCESS)
\r
958 // DoDownload(AskTrnCtrlSkt(), TransPacketBase, NO, &Canceled);
\r
960 if(Pos->NoTransfer == NO)
\r
962 Sts = DoDownload(TrnSkt, Pos, NO, &Canceled[Pos->ThreadCount]) / 100;
\r
963 if(Sts != FTP_COMPLETE)
\r
968 if((SaveTimeStamp == YES) &&
\r
969 ((Pos->Time.dwLowDateTime != 0) || (Pos->Time.dwHighDateTime != 0)))
\r
971 SetDownloadedFileTime(Pos);
\r
975 // 一部TYPE、STOR(RETR)、PORT(PASV)を並列に処理できないホストがあるため
\r
976 ReleaseMutex(hListAccMutex);
\r
979 // else if(strncmp(TransPacketBase->Cmd, "STOR", 4) == 0)
\r
980 else if(strncmp(Pos->Cmd, "STOR", 4) == 0)
\r
982 // 一部TYPE、STOR(RETR)、PORT(PASV)を並列に処理できないホストがあるため
\r
983 // ReleaseMutex(hListAccMutex);
\r
984 /* フルパスを使わないための処理 */
\r
985 // if(MakeNonFullPath(TransPacketBase, CurDir, Tmp) == FFFTP_SUCCESS)
\r
986 if(MakeNonFullPath(Pos, CurDir[Pos->ThreadCount], Tmp) == FFFTP_SUCCESS)
\r
989 // if(DoUpload(AskTrnCtrlSkt(), TransPacketBase) == 429)
\r
991 // if(ReConnectTrnSkt() == FFFTP_SUCCESS)
\r
992 // DoUpload(AskTrnCtrlSkt(), TransPacketBase);
\r
994 if(Pos->NoTransfer == NO)
\r
996 Sts = DoUpload(TrnSkt, Pos) / 100;
\r
997 if(Sts != FTP_COMPLETE)
\r
1002 /* ファイルのタイムスタンプを合わせる */
\r
1003 if((SaveTimeStamp == YES) &&
\r
1004 ((Pos->Time.dwLowDateTime != 0) || (Pos->Time.dwHighDateTime != 0)))
\r
1006 DoMFMT(TrnSkt, Pos->RemoteFile, &Pos->Time, &Canceled[Pos->ThreadCount]);
\r
1009 // 一部TYPE、STOR(RETR)、PORT(PASV)を並列に処理できないホストがあるため
\r
1010 ReleaseMutex(hListAccMutex);
\r
1012 /* フォルダ作成(ローカルまたはホスト) */
\r
1013 // else if(strncmp(TransPacketBase->Cmd, "MKD", 3) == 0)
\r
1014 else if(strncmp(Pos->Cmd, "MKD", 3) == 0)
\r
1016 // DispTransFileInfo(TransPacketBase, MSGJPN078, FALSE, YES);
\r
1017 DispTransFileInfo(Pos, MSGJPN078, FALSE, YES);
\r
1019 // if(strlen(TransPacketBase->RemoteFile) > 0)
\r
1020 if(strlen(Pos->RemoteFile) > 0)
\r
1022 /* フルパスを使わないための処理 */
\r
1023 CwdSts = FTP_COMPLETE;
\r
1025 // strcpy(Tmp, TransPacketBase->RemoteFile);
\r
1026 strcpy(Tmp, Pos->RemoteFile);
\r
1027 // if(ProcForNonFullpath(Tmp, CurDir, hWndTrans, 1) == FFFTP_FAIL)
\r
1028 if(ProcForNonFullpath(TrnSkt, Tmp, CurDir[Pos->ThreadCount], hWndTrans, &Canceled[Pos->ThreadCount]) == FFFTP_FAIL)
\r
1031 CwdSts = FTP_ERROR;
\r
1034 if(CwdSts == FTP_COMPLETE)
\r
1037 // CommandProcTrn(NULL, "MKD %s", Tmp);
\r
1038 CommandProcTrn(TrnSkt, NULL, &Canceled[Pos->ThreadCount], "MKD %s", Tmp);
\r
1039 /* すでにフォルダがある場合もあるので、 */
\r
1040 /* ここではエラーチェックはしない */
\r
1043 // CommandProcTrn(NULL, "%s %03d %s", AskHostChmodCmd(), FolderAttrNum, Tmp);
\r
1044 CommandProcTrn(TrnSkt, NULL, &Canceled[Pos->ThreadCount], "%s %03d %s", AskHostChmodCmd(), FolderAttrNum, Tmp);
\r
1047 // else if(strlen(TransPacketBase->LocalFile) > 0)
\r
1048 else if(strlen(Pos->LocalFile) > 0)
\r
1051 // DoLocalMKD(TransPacketBase->LocalFile);
\r
1052 DoLocalMKD(Pos->LocalFile);
\r
1054 ReleaseMutex(hListAccMutex);
\r
1056 /* ディレクトリ作成(常にホスト側) */
\r
1057 // else if(strncmp(TransPacketBase->Cmd, "R-MKD", 5) == 0)
\r
1058 else if(strncmp(Pos->Cmd, "R-MKD", 5) == 0)
\r
1060 // DispTransFileInfo(TransPacketBase, MSGJPN079, FALSE, YES);
\r
1061 DispTransFileInfo(Pos, MSGJPN079, FALSE, YES);
\r
1063 /* フルパスを使わないための処理 */
\r
1064 // if(MakeNonFullPath(TransPacketBase, CurDir, Tmp) == FFFTP_SUCCESS)
\r
1065 if(MakeNonFullPath(Pos, CurDir[Pos->ThreadCount], Tmp) == FFFTP_SUCCESS)
\r
1068 // CommandProcTrn(NULL, "%s%s", TransPacketBase->Cmd+2, TransPacketBase->RemoteFile);
\r
1069 CommandProcTrn(TrnSkt, NULL, &Canceled[Pos->ThreadCount], "%s%s", Pos->Cmd+2, Pos->RemoteFile);
\r
1072 // CommandProcTrn(NULL, "%s %03d %s", AskHostChmodCmd(), FolderAttrNum, TransPacketBase->RemoteFile);
\r
1073 CommandProcTrn(TrnSkt, NULL, &Canceled[Pos->ThreadCount], "%s %03d %s", AskHostChmodCmd(), FolderAttrNum, Pos->RemoteFile);
\r
1075 ReleaseMutex(hListAccMutex);
\r
1077 /* ディレクトリ削除(常にホスト側) */
\r
1078 // else if(strncmp(TransPacketBase->Cmd, "R-RMD", 5) == 0)
\r
1079 else if(strncmp(Pos->Cmd, "R-RMD", 5) == 0)
\r
1081 // DispTransFileInfo(TransPacketBase, MSGJPN080, FALSE, YES);
\r
1082 DispTransFileInfo(Pos, MSGJPN080, FALSE, YES);
\r
1084 // DelNotify = MirrorDelNotify(WIN_REMOTE, DelNotify, TransPacketBase);
\r
1085 DelNotify = MirrorDelNotify(WIN_REMOTE, DelNotify, Pos);
\r
1086 if((DelNotify == YES) || (DelNotify == YES_ALL))
\r
1088 /* フルパスを使わないための処理 */
\r
1089 // if(MakeNonFullPath(TransPacketBase, CurDir, Tmp) == FFFTP_SUCCESS)
\r
1090 if(MakeNonFullPath(Pos, CurDir[Pos->ThreadCount], Tmp) == FFFTP_SUCCESS)
\r
1093 // CommandProcTrn(NULL, "%s%s", TransPacketBase->Cmd+2, TransPacketBase->RemoteFile);
\r
1094 CommandProcTrn(TrnSkt, NULL, &Canceled[Pos->ThreadCount], "%s%s", Pos->Cmd+2, Pos->RemoteFile);
\r
1097 ReleaseMutex(hListAccMutex);
\r
1099 /* ファイル削除(常にホスト側) */
\r
1100 // else if(strncmp(TransPacketBase->Cmd, "R-DELE", 6) == 0)
\r
1101 else if(strncmp(Pos->Cmd, "R-DELE", 6) == 0)
\r
1103 // DispTransFileInfo(TransPacketBase, MSGJPN081, FALSE, YES);
\r
1104 DispTransFileInfo(Pos, MSGJPN081, FALSE, YES);
\r
1106 // DelNotify = MirrorDelNotify(WIN_REMOTE, DelNotify, TransPacketBase);
\r
1107 DelNotify = MirrorDelNotify(WIN_REMOTE, DelNotify, Pos);
\r
1108 if((DelNotify == YES) || (DelNotify == YES_ALL))
\r
1110 /* フルパスを使わないための処理 */
\r
1111 // if(MakeNonFullPath(TransPacketBase, CurDir, Tmp) == FFFTP_SUCCESS)
\r
1112 if(MakeNonFullPath(Pos, CurDir[Pos->ThreadCount], Tmp) == FFFTP_SUCCESS)
\r
1115 // CommandProcTrn(NULL, "%s%s", TransPacketBase->Cmd+2, TransPacketBase->RemoteFile);
\r
1116 CommandProcTrn(TrnSkt, NULL, &Canceled[Pos->ThreadCount], "%s%s", Pos->Cmd+2, Pos->RemoteFile);
\r
1119 ReleaseMutex(hListAccMutex);
\r
1121 /* ディレクトリ作成(常にローカル側) */
\r
1122 // else if(strncmp(TransPacketBase->Cmd, "L-MKD", 5) == 0)
\r
1123 else if(strncmp(Pos->Cmd, "L-MKD", 5) == 0)
\r
1125 // DispTransFileInfo(TransPacketBase, MSGJPN082, FALSE, YES);
\r
1126 DispTransFileInfo(Pos, MSGJPN082, FALSE, YES);
\r
1129 // DoLocalMKD(TransPacketBase->LocalFile);
\r
1130 DoLocalMKD(Pos->LocalFile);
\r
1131 ReleaseMutex(hListAccMutex);
\r
1133 /* ディレクトリ削除(常にローカル側) */
\r
1134 // else if(strncmp(TransPacketBase->Cmd, "L-RMD", 5) == 0)
\r
1135 else if(strncmp(Pos->Cmd, "L-RMD", 5) == 0)
\r
1137 // DispTransFileInfo(TransPacketBase, MSGJPN083, FALSE, YES);
\r
1138 DispTransFileInfo(Pos, MSGJPN083, FALSE, YES);
\r
1140 // DelNotify = MirrorDelNotify(WIN_LOCAL, DelNotify, TransPacketBase);
\r
1141 DelNotify = MirrorDelNotify(WIN_LOCAL, DelNotify, Pos);
\r
1142 if((DelNotify == YES) || (DelNotify == YES_ALL))
\r
1145 // DoLocalRMD(TransPacketBase->LocalFile);
\r
1146 DoLocalRMD(Pos->LocalFile);
\r
1148 ReleaseMutex(hListAccMutex);
\r
1150 /* ファイル削除(常にローカル側) */
\r
1151 // else if(strncmp(TransPacketBase->Cmd, "L-DELE", 6) == 0)
\r
1152 else if(strncmp(Pos->Cmd, "L-DELE", 6) == 0)
\r
1154 // DispTransFileInfo(TransPacketBase, MSGJPN084, FALSE, YES);
\r
1155 DispTransFileInfo(Pos, MSGJPN084, FALSE, YES);
\r
1157 // DelNotify = MirrorDelNotify(WIN_LOCAL, DelNotify, TransPacketBase);
\r
1158 DelNotify = MirrorDelNotify(WIN_LOCAL, DelNotify, Pos);
\r
1159 if((DelNotify == YES) || (DelNotify == YES_ALL))
\r
1162 // DoLocalDELE(TransPacketBase->LocalFile);
\r
1163 DoLocalDELE(Pos->LocalFile);
\r
1165 ReleaseMutex(hListAccMutex);
\r
1167 /* カレントディレクトリを設定 */
\r
1168 // else if(strcmp(TransPacketBase->Cmd, "SETCUR") == 0)
\r
1169 else if(strcmp(Pos->Cmd, "SETCUR") == 0)
\r
1171 // if(AskShareProh() == YES)
\r
1172 if(AskReuseCmdSkt() == NO || AskShareProh() == YES)
\r
1174 // if(strcmp(CurDir, TransPacketBase->RemoteFile) != 0)
\r
1175 if(strcmp(CurDir[Pos->ThreadCount], Pos->RemoteFile) != 0)
\r
1177 // if(CommandProcTrn(NULL, "CWD %s", TransPacketBase->RemoteFile)/100 != FTP_COMPLETE)
\r
1178 if(CommandProcTrn(TrnSkt, NULL, &Canceled[Pos->ThreadCount], "CWD %s", Pos->RemoteFile)/100 != FTP_COMPLETE)
\r
1180 DispCWDerror(hWndTrans);
\r
1185 // strcpy(CurDir, TransPacketBase->RemoteFile);
\r
1186 strcpy(CurDir[Pos->ThreadCount], Pos->RemoteFile);
\r
1187 ReleaseMutex(hListAccMutex);
\r
1189 /* カレントディレクトリを戻す */
\r
1190 // else if(strcmp(TransPacketBase->Cmd, "BACKCUR") == 0)
\r
1191 else if(strcmp(Pos->Cmd, "BACKCUR") == 0)
\r
1193 // if(AskShareProh() == NO)
\r
1194 if(AskReuseCmdSkt() == YES && AskShareProh() == NO)
\r
1196 // if(strcmp(CurDir, TransPacketBase->RemoteFile) != 0)
\r
1197 // CommandProcTrn(NULL, "CWD %s", TransPacketBase->RemoteFile);
\r
1198 // strcpy(CurDir, TransPacketBase->RemoteFile);
\r
1199 if(strcmp(CurDir[Pos->ThreadCount], Pos->RemoteFile) != 0)
\r
1200 CommandProcTrn(TrnSkt, NULL, &Canceled[Pos->ThreadCount], "CWD %s", Pos->RemoteFile);
\r
1201 strcpy(CurDir[Pos->ThreadCount], Pos->RemoteFile);
\r
1203 ReleaseMutex(hListAccMutex);
\r
1206 // else if(strcmp(TransPacketBase->Cmd, "GOQUIT") == 0)
\r
1207 // else if(strcmp(Pos->Cmd, "GOQUIT") == 0)
\r
1209 // ReleaseMutex(hListAccMutex);
\r
1213 else if(strcmp(Pos->Cmd, "NULL") == 0)
\r
1217 ReleaseMutex(hListAccMutex);
\r
1220 ReleaseMutex(hListAccMutex);
\r
1222 /*===== 1つの処理終わり =====*/
\r
1224 if(ForceAbort == NO)
\r
1226 // WaitForSingleObject(hListAccMutex, INFINITE);
\r
1227 while(WaitForSingleObject(hListAccMutex, 0) == WAIT_TIMEOUT)
\r
1229 BackgrndMessageProc();
\r
1232 if(ClearAll == YES)
\r
1233 // EraseTransFileList();
\r
1235 for(i = 0; i < MAX_DATA_CONNECTION; i++)
\r
1236 Canceled[i] = YES;
\r
1238 strcpy(Pos->Cmd, "");
\r
1240 EraseTransFileList();
\r
1245 // if((strncmp(TransPacketBase->Cmd, "RETR", 4) == 0) ||
\r
1246 // (strncmp(TransPacketBase->Cmd, "STOR", 4) == 0))
\r
1247 if((strncmp(Pos->Cmd, "RETR", 4) == 0) ||
\r
1248 (strncmp(Pos->Cmd, "STOR", 4) == 0) ||
\r
1249 (strncmp(Pos->Cmd, "STOU", 4) == 0))
\r
1252 if(TransFiles > 0)
\r
1255 if(TransferSizeLeft > 0)
\r
1256 TransferSizeLeft -= Pos->Size;
\r
1257 if(TransferSizeLeft < 0)
\r
1258 TransferSizeLeft = 0;
\r
1259 if(TransFiles == 0)
\r
1260 TransferSizeTotal = 0;
\r
1261 PostMessage(GetMainHwnd(), WM_CHANGE_COND, 0, 0);
\r
1263 // Pos = TransPacketBase;
\r
1264 // TransPacketBase = TransPacketBase->Next;
\r
1268 ReleaseMutex(hListAccMutex);
\r
1270 if(BackgrndMessageProc() == YES)
\r
1272 WaitForSingleObject(hListAccMutex, INFINITE);
\r
1273 EraseTransFileList();
\r
1274 ReleaseMutex(hListAccMutex);
\r
1277 if(hWndTrans != NULL)
\r
1278 SendMessage(hWndTrans, WM_SET_PACKET, 0, 0);
\r
1280 strcpy(Pos->Cmd, "");
\r
1281 LastUsed = timeGetTime();
\r
1284 else if(TransPacketBase == NULL)
\r
1291 SoundPlay(SND_TRANS);
\r
1292 if(AskAutoExit() == NO)
\r
1295 PostMessage(GetMainHwnd(), WM_REFRESH_LOCAL_FLG, 0, 0);
\r
1297 PostMessage(GetMainHwnd(), WM_REFRESH_REMOTE_FLG, 0, 0);
\r
1301 PostMessage(GetMainHwnd(), WM_COMMAND, MAKEWPARAM(MENU_AUTO_EXIT, 0), 0);
\r
1305 ReleaseMutex(hListAccMutex);
\r
1308 if(hWndTrans != NULL)
\r
1310 DestroyWindow(hWndTrans);
\r
1313 // if(GoExit == YES)
\r
1315 // SoundPlay(SND_TRANS);
\r
1317 // if(AskAutoExit() == NO)
\r
1319 // if(Down == YES)
\r
1320 // PostMessage(GetMainHwnd(), WM_REFRESH_LOCAL_FLG, 0, 0);
\r
1322 // PostMessage(GetMainHwnd(), WM_REFRESH_REMOTE_FLG, 0, 0);
\r
1329 BackgrndMessageProc();
\r
1334 TransferErrorMode = AskTransferErrorMode();
\r
1335 TransferErrorNotify = AskTransferErrorNotify();
\r
1339 ReleaseMutex(hListAccMutex);
\r
1340 if(hWndTrans != NULL)
\r
1342 DestroyWindow(hWndTrans);
\r
1345 BackgrndMessageProc();
\r
1346 if(ThreadCount < AskMaxThreadCount())
\r
1352 if(AskReuseCmdSkt() == NO || ThreadCount > 0)
\r
1354 if(TrnSkt != INVALID_SOCKET)
\r
1356 SendData(TrnSkt, "QUIT\r\n", 6, 0, &Canceled[ThreadCount]);
\r
1364 /*----- フルパスを使わないファイルアクセスの準備 ------------------------------
\r
1367 * TRANSPACKET *Pkt : 転送パケット
\r
1368 * char *Cur : カレントディレクトリ
\r
1369 * char *Tmp : 作業用エリア
\r
1372 * int ステータス(FFFTP_SUCCESS/FFFTP_FAIL)
\r
1376 * このモジュール内で CWD を行ない、
\r
1377 * Pkt->RemoteFile にファイル名のみ残す。(パス名は消す)
\r
1378 *----------------------------------------------------------------------------*/
\r
1381 static int MakeNonFullPath(TRANSPACKET *Pkt, char *Cur, char *Tmp)
\r
1385 // Sts = ProcForNonFullpath(Pkt->RemoteFile, Cur, Pkt->hWndTrans, 1);
\r
1386 Sts = ProcForNonFullpath(Pkt->ctrl_skt, Pkt->RemoteFile, Cur, Pkt->hWndTrans, &Canceled[Pkt->ThreadCount]);
\r
1387 if(Sts == FFFTP_FAIL)
\r
1397 static int SetDownloadedFileTime(TRANSPACKET *Pkt)
\r
1402 if((hFile = CreateFile(Pkt->LocalFile, GENERIC_WRITE | GENERIC_READ, FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL)) != INVALID_HANDLE_VALUE)
\r
1404 if(SetFileTime(hFile, &Pkt->Time, &Pkt->Time, &Pkt->Time))
\r
1405 Sts = FFFTP_SUCCESS;
\r
1406 CloseHandle(hFile);
\r
1411 /*----- ダウンロードを行なう --------------------------------------------------
\r
1414 * SOCKET cSkt : コントロールソケット
\r
1415 * TRANSPACKET *Pkt : 転送ファイル情報
\r
1416 * int DirList : ディレクトリリストのダウンロード(YES/NO)
\r
1422 * このモジュールは、ファイル一覧の取得などを行なう際にメインのスレッド
\r
1423 * からも呼ばれる。メインのスレッドから呼ばれる時は Pkt->hWndTrans == NULL。
\r
1424 *----------------------------------------------------------------------------*/
\r
1426 int DoDownload(SOCKET cSkt, TRANSPACKET *Pkt, int DirList, int *CancelCheckWork)
\r
1429 char Reply[ERR_MSG_LEN+7];
\r
1431 Pkt->ctrl_skt = cSkt;
\r
1432 if(IsSpecialDevice(GetFileName(Pkt->LocalFile)) == YES)
\r
1435 SetTaskMsg(MSGJPN085, GetFileName(Pkt->LocalFile));
\r
1436 // エラーによってはダイアログが表示されない場合があるバグ対策
\r
1437 // DispDownloadFinishMsg(Pkt, iRetCode);
\r
1439 else if(Pkt->Mode != EXIST_IGNORE)
\r
1441 if(Pkt->Type == TYPE_I)
\r
1442 Pkt->KanjiCode = KANJI_NOCNV;
\r
1444 iRetCode = command(Pkt->ctrl_skt, Reply, CancelCheckWork, "TYPE %c", Pkt->Type);
\r
1445 if(iRetCode/100 < FTP_RETRY)
\r
1447 if(Pkt->hWndTrans != NULL)
\r
1450 // AllTransSizeNow = 0;
\r
1451 AllTransSizeNow[Pkt->ThreadCount] = 0;
\r
1454 DispTransFileInfo(Pkt, MSGJPN086, TRUE, YES);
\r
1456 DispTransFileInfo(Pkt, MSGJPN087, FALSE, NO);
\r
1459 if(BackgrndMessageProc() == NO)
\r
1461 if(AskPasvMode() != YES)
\r
1462 iRetCode = DownloadNonPassive(Pkt, CancelCheckWork);
\r
1464 iRetCode = DownloadPassive(Pkt, CancelCheckWork);
\r
1470 SetErrorMsg(Reply);
\r
1471 // エラーによってはダイアログが表示されない場合があるバグ対策
\r
1472 DispDownloadFinishMsg(Pkt, iRetCode);
\r
1476 DispTransFileInfo(Pkt, MSGJPN088, TRUE, YES);
\r
1477 SetTaskMsg(MSGJPN089, Pkt->RemoteFile);
\r
1484 /*----- 通常モードでファイルをダウンロード ------------------------------------
\r
1487 * TRANSPACKET *Pkt : 転送ファイル情報
\r
1491 *----------------------------------------------------------------------------*/
\r
1493 static int DownloadNonPassive(TRANSPACKET *Pkt, int *CancelCheckWork)
\r
1497 SOCKET data_socket = INVALID_SOCKET; // data channel socket
\r
1498 SOCKET listen_socket = INVALID_SOCKET; // data listen socket
\r
1500 // char Buf[1024];
\r
1501 char Buf[FMAX_PATH+1024];
\r
1504 // struct sockaddr_in saSockAddr1;
\r
1505 struct sockaddr_in saSockAddrIPv4;
\r
1506 struct sockaddr_in6 saSockAddrIPv6;
\r
1509 char Reply[ERR_MSG_LEN+7];
\r
1511 if((listen_socket = GetFTPListenSocket(Pkt->ctrl_skt, CancelCheckWork)) != INVALID_SOCKET)
\r
1513 if(SetDownloadResume(Pkt, Pkt->Mode, Pkt->ExistSize, &CreateMode, CancelCheckWork) == YES)
\r
1515 sprintf(Buf, "%s%s", Pkt->Cmd, Pkt->RemoteFile);
\r
1516 iRetCode = command(Pkt->ctrl_skt, Reply, CancelCheckWork, "%s", Buf);
\r
1517 if(iRetCode/100 == FTP_PRELIM)
\r
1520 // if(SocksGet2ndBindReply(listen_socket, &data_socket) == FFFTP_FAIL)
\r
1521 if(SocksGet2ndBindReply(listen_socket, &data_socket, CancelCheckWork) == FFFTP_FAIL)
\r
1524 // iLength = sizeof(saSockAddr1);
\r
1525 // data_socket = do_accept(listen_socket, (struct sockaddr *)&saSockAddr1, (int *)&iLength);
\r
1526 switch(AskCurNetType())
\r
1529 iLength=sizeof(saSockAddrIPv4);
\r
1530 data_socket = do_accept(listen_socket,(struct sockaddr *)&saSockAddrIPv4, (int *)&iLength);
\r
1533 iLength=sizeof(saSockAddrIPv6);
\r
1534 data_socket = do_accept(listen_socket,(struct sockaddr *)&saSockAddrIPv6, (int *)&iLength);
\r
1538 if(shutdown(listen_socket, 1) != 0)
\r
1539 ReportWSError("shutdown listen", WSAGetLastError());
\r
1541 if(IsUPnPLoaded() == YES)
\r
1543 if(GetAsyncTableDataMapPort(listen_socket, &Port) == YES)
\r
1544 RemovePortMapping(Port);
\r
1546 listen_socket = DoClose(listen_socket);
\r
1548 if(data_socket == INVALID_SOCKET)
\r
1550 SetErrorMsg(MSGJPN280);
\r
1551 ReportWSError("accept", WSAGetLastError());
\r
1556 // DoPrintf("Skt=%u : accept from %s port %u", data_socket, inet_ntoa(saSockAddr1.sin_addr), ntohs(saSockAddr1.sin_port));
\r
1558 switch(AskCurNetType())
\r
1561 DoPrintf("Skt=%u : accept from %s port %u", data_socket, inet_ntoa(saSockAddrIPv4.sin_addr), ntohs(saSockAddrIPv4.sin_port));
\r
1564 DoPrintf("Skt=%u : accept from %s port %u", data_socket, inet6_ntoa(saSockAddrIPv6.sin6_addr), ntohs(saSockAddrIPv6.sin6_port));
\r
1570 if(data_socket != INVALID_SOCKET)
\r
1572 // 一部TYPE、STOR(RETR)、PORT(PASV)を並列に処理できないホストがあるため
\r
1573 ReleaseMutex(hListAccMutex);
\r
1575 // iRetCode = DownloadFile(Pkt, data_socket, CreateMode, CancelCheckWork);
\r
1576 if(IsSSLAttached(Pkt->ctrl_skt))
\r
1578 if(AttachSSL(data_socket, Pkt->ctrl_skt, CancelCheckWork, FALSE, NULL))
\r
1579 iRetCode = DownloadFile(Pkt, data_socket, CreateMode, CancelCheckWork);
\r
1584 iRetCode = DownloadFile(Pkt, data_socket, CreateMode, CancelCheckWork);
\r
1585 // data_socket = DoClose(data_socket);
\r
1590 SetErrorMsg(Reply);
\r
1591 SetTaskMsg(MSGJPN090);
\r
1593 if(IsUPnPLoaded() == YES)
\r
1595 if(GetAsyncTableDataMapPort(listen_socket, &Port) == YES)
\r
1596 RemovePortMapping(Port);
\r
1598 listen_socket = DoClose(listen_socket);
\r
1604 // iRetCode = 500;
\r
1607 if(IsUPnPLoaded() == YES)
\r
1609 if(GetAsyncTableDataMapPort(listen_socket, &Port) == YES)
\r
1610 RemovePortMapping(Port);
\r
1612 listen_socket = DoClose(listen_socket);
\r
1619 SetErrorMsg(MSGJPN279);
\r
1621 // エラーによってはダイアログが表示されない場合があるバグ対策
\r
1622 // DispDownloadFinishMsg(Pkt, iRetCode);
\r
1628 /*----- Passiveモードでファイルをダウンロード ---------------------------------
\r
1631 * TRANSPACKET *Pkt : 転送ファイル情報
\r
1635 *----------------------------------------------------------------------------*/
\r
1637 static int DownloadPassive(TRANSPACKET *Pkt, int *CancelCheckWork)
\r
1640 SOCKET data_socket = INVALID_SOCKET; // data channel socket
\r
1642 // char Buf[1024];
\r
1643 char Buf[FMAX_PATH+1024];
\r
1650 char Reply[ERR_MSG_LEN+7];
\r
1653 // iRetCode = command(Pkt->ctrl_skt, Buf, CancelCheckWork, "PASV");
\r
1654 switch(AskCurNetType())
\r
1657 iRetCode = command(Pkt->ctrl_skt, Buf, CancelCheckWork, "PASV");
\r
1660 iRetCode = command(Pkt->ctrl_skt, Buf, CancelCheckWork, "EPSV");
\r
1663 if(iRetCode/100 == FTP_COMPLETE)
\r
1666 // if(GetAdrsAndPort(Buf, Adrs, &Port, 19) == FFFTP_SUCCESS)
\r
1667 if(GetAdrsAndPort(Pkt->ctrl_skt, Buf, Adrs, &Port, 39) == FFFTP_SUCCESS)
\r
1669 if((data_socket = connectsock(Adrs, Port, MSGJPN091, CancelCheckWork)) != INVALID_SOCKET)
\r
1673 if(setsockopt(data_socket, IPPROTO_TCP, TCP_NODELAY, (LPSTR)&Flg, sizeof(Flg)) == SOCKET_ERROR)
\r
1674 ReportWSError("setsockopt", WSAGetLastError());
\r
1676 if(SetDownloadResume(Pkt, Pkt->Mode, Pkt->ExistSize, &CreateMode, CancelCheckWork) == YES)
\r
1678 sprintf(Buf, "%s%s", Pkt->Cmd, Pkt->RemoteFile);
\r
1679 iRetCode = command(Pkt->ctrl_skt, Reply, CancelCheckWork, "%s", Buf);
\r
1680 if(iRetCode/100 == FTP_PRELIM)
\r
1682 // 一部TYPE、STOR(RETR)、PORT(PASV)を並列に処理できないホストがあるため
\r
1683 ReleaseMutex(hListAccMutex);
\r
1685 // iRetCode = DownloadFile(Pkt, data_socket, CreateMode, CancelCheckWork);
\r
1686 if(IsSSLAttached(Pkt->ctrl_skt))
\r
1688 if(AttachSSL(data_socket, Pkt->ctrl_skt, CancelCheckWork, FALSE, NULL))
\r
1689 iRetCode = DownloadFile(Pkt, data_socket, CreateMode, CancelCheckWork);
\r
1694 iRetCode = DownloadFile(Pkt, data_socket, CreateMode, CancelCheckWork);
\r
1695 // data_socket = DoClose(data_socket);
\r
1699 SetErrorMsg(Reply);
\r
1700 SetTaskMsg(MSGJPN092);
\r
1701 data_socket = DoClose(data_socket);
\r
1713 SetErrorMsg(MSGJPN093);
\r
1714 SetTaskMsg(MSGJPN093);
\r
1721 // エラーによってはダイアログが表示されない場合があるバグ対策
\r
1722 // DispDownloadFinishMsg(Pkt, iRetCode);
\r
1728 /*----- ダウンロードの実行 ----------------------------------------------------
\r
1731 * TRANSPACKET *Pkt : 転送ファイル情報
\r
1732 * SOCKET dSkt : データソケット
\r
1733 * int CreateMode : ファイル作成モード (CREATE_ALWAYS/OPEN_ALWAYS)
\r
1740 * ダイアログを出す(Pkt->hWndTrans!=NULL)場合、インターバルタイマで経過を表示する
\r
1741 * ダイアログを出さない場合、このルーチンからDispDownloadSize()を呼ぶ
\r
1742 *----------------------------------------------------------------------------*/
\r
1744 static int DownloadFile(TRANSPACKET *Pkt, SOCKET dSkt, int CreateMode, int *CancelCheckWork)
\r
1747 char Buf[BUFSIZE];
\r
1748 char Buf2[BUFSIZE+3];
\r
1749 HANDLE iFileHandle;
\r
1750 SECURITY_ATTRIBUTES Sec;
\r
1752 CODECONVINFO cInfo;
\r
1754 // fd_set ReadFds;
\r
1755 // struct timeval Tout;
\r
1756 // struct timeval *ToutPtr;
\r
1759 char TmpBuf[ONELINE_BUF_SIZE];
\r
1760 DWORD dwFileAttributes;
\r
1762 #ifdef SET_BUFFER_SIZE
\r
1763 /* Add by H.Shirouzu at 2002/10/02 */
\r
1764 int buf_size = SOCKBUF_SIZE;
\r
1765 for ( ; buf_size > 0; buf_size /= 2)
\r
1766 if (setsockopt(dSkt, SOL_SOCKET, SO_RCVBUF, (char *)&buf_size, sizeof(buf_size)) == 0)
\r
1771 // 念のため受信バッファを無効にする
\r
1772 #ifdef DISABLE_TRANSFER_NETWORK_BUFFERS
\r
1774 setsockopt(dSkt, SOL_SOCKET, SO_RCVBUF, (char *)&buf_size, sizeof(buf_size));
\r
1777 Pkt->Abort = ABORT_NONE;
\r
1779 Sec.nLength = sizeof(SECURITY_ATTRIBUTES);
\r
1780 Sec.lpSecurityDescriptor = NULL;
\r
1781 Sec.bInheritHandle = FALSE;
\r
1783 dwFileAttributes = GetFileAttributes(Pkt->LocalFile);
\r
1784 if (dwFileAttributes != INVALID_FILE_ATTRIBUTES && (dwFileAttributes & FILE_ATTRIBUTE_READONLY)) {
\r
1786 if (MessageBox(GetMainHwnd(), MSGJPN296, MSGJPN086, MB_YESNO) == IDYES) {
\r
1788 SetFileAttributes(Pkt->LocalFile, dwFileAttributes ^ FILE_ATTRIBUTE_READONLY);
\r
1792 if((iFileHandle = CreateFile(Pkt->LocalFile, GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, &Sec, CreateMode, FILE_ATTRIBUTE_NORMAL, NULL)) != INVALID_HANDLE_VALUE)
\r
1795 char Buf3[(BUFSIZE + 3) * 4];
\r
1796 CODECONVINFO cInfo2;
\r
1797 int ProcessedBOM = NO;
\r
1798 // 4GB超対応(kaokunさん提供)
\r
1800 if(CreateMode == OPEN_ALWAYS)
\r
1801 // 4GB超対応(kaokunさん提供)
\r
1802 // SetFilePointer(iFileHandle, 0, 0, FILE_END);
\r
1803 SetFilePointer(iFileHandle, 0, &High, FILE_END);
\r
1805 if(Pkt->hWndTrans != NULL)
\r
1808 // TimeStart = time(NULL);
\r
1809 TimeStart[Pkt->ThreadCount] = time(NULL);
\r
1810 SetTimer(Pkt->hWndTrans, TIMER_DISPLAY, DISPLAY_TIMING, NULL);
\r
1813 InitCodeConvInfo(&cInfo);
\r
1814 cInfo.KanaCnv = Pkt->KanaCnv;
\r
1816 InitCodeConvInfo(&cInfo2);
\r
1817 cInfo2.KanaCnv = Pkt->KanaCnv;
\r
1819 /*===== ファイルを受信するループ =====*/
\r
1820 while((Pkt->Abort == ABORT_NONE) && (ForceAbort == NO))
\r
1822 // FD_ZERO(&ReadFds);
\r
1823 // FD_SET(dSkt, &ReadFds);
\r
1824 // ToutPtr = NULL;
\r
1825 // if(TimeOut != 0)
\r
1827 // Tout.tv_sec = TimeOut;
\r
1828 // Tout.tv_usec = 0;
\r
1829 // ToutPtr = &Tout;
\r
1831 // iNumBytes = select(0, &ReadFds, NULL, NULL, ToutPtr);
\r
1832 // if(iNumBytes == SOCKET_ERROR)
\r
1834 // ReportWSError("select", WSAGetLastError());
\r
1835 // if(Pkt->Abort == ABORT_NONE)
\r
1836 // Pkt->Abort = ABORT_ERROR;
\r
1839 // else if(iNumBytes == 0)
\r
1841 // SetErrorMsg(MSGJPN094);
\r
1842 // SetTaskMsg(MSGJPN094);
\r
1843 // Pkt->Abort = ABORT_ERROR;
\r
1847 if((iNumBytes = do_recv(dSkt, Buf, BUFSIZE, 0, &TimeOutErr, CancelCheckWork)) <= 0)
\r
1849 if(TimeOutErr == YES)
\r
1851 SetErrorMsg(MSGJPN094);
\r
1852 SetTaskMsg(MSGJPN094);
\r
1853 if(Pkt->hWndTrans != NULL)
\r
1855 if(Pkt->Abort == ABORT_NONE)
\r
1856 Pkt->Abort = ABORT_ERROR;
\r
1858 else if(iNumBytes == SOCKET_ERROR)
\r
1860 if(Pkt->Abort == ABORT_NONE)
\r
1861 Pkt->Abort = ABORT_ERROR;
\r
1867 if(Pkt->KanjiCode != KANJI_NOCNV)
\r
1870 cInfo.StrLen = iNumBytes;
\r
1872 cInfo.BufSize = BUFSIZE+3;
\r
1875 // ここで全てUTF-8へ変換する
\r
1876 // TODO: SJIS以外も直接UTF-8へ変換
\r
1877 // if(Pkt->KanjiCode == KANJI_JIS)
\r
1878 // Continue = ConvJIStoSJIS(&cInfo);
\r
1880 // Continue = ConvEUCtoSJIS(&cInfo);
\r
1881 switch(Pkt->KanjiCode)
\r
1884 switch(Pkt->KanjiCodeDesired)
\r
1887 // memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
1888 // cInfo2.OutLen = cInfo.StrLen;
\r
1891 Continue = ConvSJIStoJIS(&cInfo);
\r
1892 cInfo2.Str = cInfo.Buf;
\r
1893 cInfo2.StrLen = cInfo.OutLen;
\r
1894 cInfo2.Buf = Buf3;
\r
1895 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1896 ConvJIStoSJIS(&cInfo2);
\r
1899 Continue = ConvSJIStoJIS(&cInfo);
\r
1900 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
1901 cInfo2.OutLen = cInfo.OutLen;
\r
1904 Continue = ConvSJIStoEUC(&cInfo);
\r
1905 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
1906 cInfo2.OutLen = cInfo.OutLen;
\r
1909 Continue = ConvSJIStoUTF8N(&cInfo);
\r
1910 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
1911 cInfo2.OutLen = cInfo.OutLen;
\r
1913 case KANJI_UTF8BOM:
\r
1914 if(ProcessedBOM == NO)
\r
1916 memcpy(Buf3, "\xEF\xBB\xBF", 3);
\r
1917 cInfo2.OutLen = 3;
\r
1919 ProcessedBOM = YES;
\r
1922 Continue = ConvSJIStoUTF8N(&cInfo);
\r
1923 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
1924 cInfo2.OutLen = cInfo.OutLen;
\r
1929 switch(Pkt->KanjiCodeDesired)
\r
1932 Continue = ConvJIStoSJIS(&cInfo);
\r
1933 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
1934 cInfo2.OutLen = cInfo.OutLen;
\r
1937 // memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
1938 // cInfo2.OutLen = cInfo.StrLen;
\r
1941 Continue = ConvJIStoSJIS(&cInfo);
\r
1942 cInfo2.Str = cInfo.Buf;
\r
1943 cInfo2.StrLen = cInfo.OutLen;
\r
1944 cInfo2.Buf = Buf3;
\r
1945 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1946 ConvSJIStoJIS(&cInfo2);
\r
1949 Continue = ConvJIStoSJIS(&cInfo);
\r
1950 cInfo2.Str = cInfo.Buf;
\r
1951 cInfo2.StrLen = cInfo.OutLen;
\r
1952 cInfo2.Buf = Buf3;
\r
1953 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1954 ConvSJIStoEUC(&cInfo2);
\r
1957 Continue = ConvJIStoSJIS(&cInfo);
\r
1958 cInfo2.Str = cInfo.Buf;
\r
1959 cInfo2.StrLen = cInfo.OutLen;
\r
1960 cInfo2.Buf = Buf3;
\r
1961 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1962 ConvSJIStoUTF8N(&cInfo2);
\r
1964 case KANJI_UTF8BOM:
\r
1965 if(ProcessedBOM == NO)
\r
1967 memcpy(Buf3, "\xEF\xBB\xBF", 3);
\r
1968 cInfo2.OutLen = 3;
\r
1970 ProcessedBOM = YES;
\r
1973 Continue = ConvJIStoSJIS(&cInfo);
\r
1974 cInfo2.Str = cInfo.Buf;
\r
1975 cInfo2.StrLen = cInfo.OutLen;
\r
1976 cInfo2.Buf = Buf3;
\r
1977 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1978 ConvSJIStoUTF8N(&cInfo2);
\r
1983 switch(Pkt->KanjiCodeDesired)
\r
1986 Continue = ConvEUCtoSJIS(&cInfo);
\r
1987 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
1988 cInfo2.OutLen = cInfo.OutLen;
\r
1991 Continue = ConvEUCtoSJIS(&cInfo);
\r
1992 cInfo2.Str = cInfo.Buf;
\r
1993 cInfo2.StrLen = cInfo.OutLen;
\r
1994 cInfo2.Buf = Buf3;
\r
1995 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1996 ConvSJIStoJIS(&cInfo2);
\r
1999 // memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
2000 // cInfo2.OutLen = cInfo.StrLen;
\r
2003 Continue = ConvEUCtoSJIS(&cInfo);
\r
2004 cInfo2.Str = cInfo.Buf;
\r
2005 cInfo2.StrLen = cInfo.OutLen;
\r
2006 cInfo2.Buf = Buf3;
\r
2007 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2008 ConvSJIStoEUC(&cInfo2);
\r
2011 Continue = ConvEUCtoSJIS(&cInfo);
\r
2012 cInfo2.Str = cInfo.Buf;
\r
2013 cInfo2.StrLen = cInfo.OutLen;
\r
2014 cInfo2.Buf = Buf3;
\r
2015 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2016 ConvSJIStoUTF8N(&cInfo2);
\r
2018 case KANJI_UTF8BOM:
\r
2019 if(ProcessedBOM == NO)
\r
2021 memcpy(Buf3, "\xEF\xBB\xBF", 3);
\r
2022 cInfo2.OutLen = 3;
\r
2024 ProcessedBOM = YES;
\r
2027 Continue = ConvEUCtoSJIS(&cInfo);
\r
2028 cInfo2.Str = cInfo.Buf;
\r
2029 cInfo2.StrLen = cInfo.OutLen;
\r
2030 cInfo2.Buf = Buf3;
\r
2031 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2032 ConvSJIStoUTF8N(&cInfo2);
\r
2037 switch(Pkt->KanjiCodeDesired)
\r
2040 Continue = ConvUTF8NtoSJIS(&cInfo);
\r
2041 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2042 cInfo2.OutLen = cInfo.OutLen;
\r
2045 Continue = ConvUTF8NtoSJIS(&cInfo);
\r
2046 cInfo2.Str = cInfo.Buf;
\r
2047 cInfo2.StrLen = cInfo.OutLen;
\r
2048 cInfo2.Buf = Buf3;
\r
2049 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2050 ConvSJIStoJIS(&cInfo2);
\r
2053 Continue = ConvUTF8NtoSJIS(&cInfo);
\r
2054 cInfo2.Str = cInfo.Buf;
\r
2055 cInfo2.StrLen = cInfo.OutLen;
\r
2056 cInfo2.Buf = Buf3;
\r
2057 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2058 ConvSJIStoEUC(&cInfo2);
\r
2061 memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
2062 cInfo2.OutLen = cInfo.StrLen;
\r
2065 case KANJI_UTF8BOM:
\r
2066 if(ProcessedBOM == NO)
\r
2068 memcpy(Buf3, "\xEF\xBB\xBF", 3);
\r
2069 cInfo2.OutLen = 3;
\r
2071 ProcessedBOM = YES;
\r
2074 memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
2075 cInfo2.OutLen = cInfo.StrLen;
\r
2080 case KANJI_UTF8BOM:
\r
2081 if(ProcessedBOM == NO)
\r
2083 if(memcmp(Buf, "\xEF\xBB\xBF", 3) == 0)
\r
2086 cInfo.StrLen -= 3;
\r
2088 cInfo2.OutLen = 0;
\r
2089 switch(Pkt->KanjiCodeDesired)
\r
2091 case KANJI_UTF8BOM:
\r
2092 memcpy(Buf3, "\xEF\xBB\xBF", 3);
\r
2093 cInfo2.OutLen = 3;
\r
2097 ProcessedBOM = YES;
\r
2100 switch(Pkt->KanjiCodeDesired)
\r
2103 Continue = ConvUTF8NtoSJIS(&cInfo);
\r
2104 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2105 cInfo2.OutLen = cInfo.OutLen;
\r
2108 Continue = ConvUTF8NtoSJIS(&cInfo);
\r
2109 cInfo2.Str = cInfo.Buf;
\r
2110 cInfo2.StrLen = cInfo.OutLen;
\r
2111 cInfo2.Buf = Buf3;
\r
2112 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2113 ConvSJIStoJIS(&cInfo2);
\r
2116 Continue = ConvUTF8NtoSJIS(&cInfo);
\r
2117 cInfo2.Str = cInfo.Buf;
\r
2118 cInfo2.StrLen = cInfo.OutLen;
\r
2119 cInfo2.Buf = Buf3;
\r
2120 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2121 ConvSJIStoEUC(&cInfo2);
\r
2124 memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
2125 cInfo2.OutLen = cInfo.StrLen;
\r
2128 case KANJI_UTF8BOM:
\r
2129 memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
2130 cInfo2.OutLen = cInfo.StrLen;
\r
2136 // if(WriteFile(iFileHandle, Buf2, cInfo.OutLen, &Writed, NULL) == FALSE)
\r
2137 if(WriteFile(iFileHandle, Buf3, cInfo2.OutLen, &Writed, NULL) == FALSE)
\r
2138 Pkt->Abort = ABORT_DISKFULL;
\r
2140 while((Continue == YES) && (Pkt->Abort == ABORT_NONE));
\r
2144 if(WriteFile(iFileHandle, Buf, iNumBytes, &Writed, NULL) == FALSE)
\r
2145 Pkt->Abort = ABORT_DISKFULL;
\r
2148 Pkt->ExistSize += iNumBytes;
\r
2149 if(Pkt->hWndTrans != NULL)
\r
2151 // AllTransSizeNow += iNumBytes;
\r
2152 AllTransSizeNow[Pkt->ThreadCount] += iNumBytes;
\r
2155 /* 転送ダイアログを出さない時の経過表示 */
\r
2156 DispDownloadSize(Pkt->ExistSize);
\r
2159 if(BackgrndMessageProc() == YES)
\r
2163 /* 書き残したデータを書き込む */
\r
2164 if(Pkt->KanjiCode != KANJI_NOCNV)
\r
2167 cInfo.BufSize = BUFSIZE+3;
\r
2168 FlushRestData(&cInfo);
\r
2169 switch(Pkt->KanjiCode)
\r
2172 switch(Pkt->KanjiCodeDesired)
\r
2176 cInfo2.Str = cInfo.Buf;
\r
2177 cInfo2.StrLen = cInfo.OutLen;
\r
2178 cInfo2.Buf = Buf3;
\r
2179 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2180 ConvJIStoSJIS(&cInfo2);
\r
2183 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2184 cInfo2.OutLen = cInfo.OutLen;
\r
2187 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2188 cInfo2.OutLen = cInfo.OutLen;
\r
2191 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2192 cInfo2.OutLen = cInfo.OutLen;
\r
2194 case KANJI_UTF8BOM:
\r
2195 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2196 cInfo2.OutLen = cInfo.OutLen;
\r
2201 switch(Pkt->KanjiCodeDesired)
\r
2204 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2205 cInfo2.OutLen = cInfo.OutLen;
\r
2209 cInfo2.Str = cInfo.Buf;
\r
2210 cInfo2.StrLen = cInfo.OutLen;
\r
2211 cInfo2.Buf = Buf3;
\r
2212 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2213 ConvSJIStoJIS(&cInfo2);
\r
2216 cInfo2.Str = cInfo.Buf;
\r
2217 cInfo2.StrLen = cInfo.OutLen;
\r
2218 cInfo2.Buf = Buf3;
\r
2219 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2220 ConvSJIStoEUC(&cInfo2);
\r
2223 cInfo2.Str = cInfo.Buf;
\r
2224 cInfo2.StrLen = cInfo.OutLen;
\r
2225 cInfo2.Buf = Buf3;
\r
2226 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2227 ConvSJIStoUTF8N(&cInfo2);
\r
2229 case KANJI_UTF8BOM:
\r
2230 cInfo2.Str = cInfo.Buf;
\r
2231 cInfo2.StrLen = cInfo.OutLen;
\r
2232 cInfo2.Buf = Buf3;
\r
2233 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2234 ConvSJIStoUTF8N(&cInfo2);
\r
2239 switch(Pkt->KanjiCodeDesired)
\r
2242 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2243 cInfo2.OutLen = cInfo.OutLen;
\r
2246 cInfo2.Str = cInfo.Buf;
\r
2247 cInfo2.StrLen = cInfo.OutLen;
\r
2248 cInfo2.Buf = Buf3;
\r
2249 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2250 ConvSJIStoJIS(&cInfo2);
\r
2254 cInfo2.Str = cInfo.Buf;
\r
2255 cInfo2.StrLen = cInfo.OutLen;
\r
2256 cInfo2.Buf = Buf3;
\r
2257 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2258 ConvSJIStoEUC(&cInfo2);
\r
2261 cInfo2.Str = cInfo.Buf;
\r
2262 cInfo2.StrLen = cInfo.OutLen;
\r
2263 cInfo2.Buf = Buf3;
\r
2264 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2265 ConvSJIStoUTF8N(&cInfo2);
\r
2267 case KANJI_UTF8BOM:
\r
2268 cInfo2.Str = cInfo.Buf;
\r
2269 cInfo2.StrLen = cInfo.OutLen;
\r
2270 cInfo2.Buf = Buf3;
\r
2271 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2272 ConvSJIStoUTF8N(&cInfo2);
\r
2277 switch(Pkt->KanjiCodeDesired)
\r
2280 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2281 cInfo2.OutLen = cInfo.OutLen;
\r
2284 cInfo2.Str = cInfo.Buf;
\r
2285 cInfo2.StrLen = cInfo.OutLen;
\r
2286 cInfo2.Buf = Buf3;
\r
2287 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2288 ConvSJIStoJIS(&cInfo2);
\r
2291 cInfo2.Str = cInfo.Buf;
\r
2292 cInfo2.StrLen = cInfo.OutLen;
\r
2293 cInfo2.Buf = Buf3;
\r
2294 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2295 ConvSJIStoEUC(&cInfo2);
\r
2298 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2299 cInfo2.OutLen = cInfo.OutLen;
\r
2301 case KANJI_UTF8BOM:
\r
2302 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2303 cInfo2.OutLen = cInfo.OutLen;
\r
2307 case KANJI_UTF8BOM:
\r
2308 switch(Pkt->KanjiCodeDesired)
\r
2311 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2312 cInfo2.OutLen = cInfo.OutLen;
\r
2315 cInfo2.Str = cInfo.Buf;
\r
2316 cInfo2.StrLen = cInfo.OutLen;
\r
2317 cInfo2.Buf = Buf3;
\r
2318 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2319 ConvSJIStoJIS(&cInfo2);
\r
2322 cInfo2.Str = cInfo.Buf;
\r
2323 cInfo2.StrLen = cInfo.OutLen;
\r
2324 cInfo2.Buf = Buf3;
\r
2325 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2326 ConvSJIStoEUC(&cInfo2);
\r
2329 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2330 cInfo2.OutLen = cInfo.OutLen;
\r
2332 case KANJI_UTF8BOM:
\r
2333 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2334 cInfo2.OutLen = cInfo.OutLen;
\r
2339 // if(WriteFile(iFileHandle, Buf2, cInfo.OutLen, &Writed, NULL) == FALSE)
\r
2340 if(WriteFile(iFileHandle, Buf3, cInfo2.OutLen, &Writed, NULL) == FALSE)
\r
2341 Pkt->Abort = ABORT_DISKFULL;
\r
2342 cInfo2.Buf = Buf3;
\r
2343 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2344 FlushRestData(&cInfo2);
\r
2345 if(WriteFile(iFileHandle, Buf3, cInfo2.OutLen, &Writed, NULL) == FALSE)
\r
2346 Pkt->Abort = ABORT_DISKFULL;
\r
2350 if(Pkt->hWndTrans != NULL)
\r
2352 KillTimer(Pkt->hWndTrans, TIMER_DISPLAY);
\r
2353 DispTransferStatus(Pkt->hWndTrans, YES, Pkt);
\r
2355 // TimeStart = time(NULL) - TimeStart + 1;
\r
2356 TimeStart[Pkt->ThreadCount] = time(NULL) - TimeStart[Pkt->ThreadCount] + 1;
\r
2360 /* 転送ダイアログを出さない時の経過表示を消す */
\r
2361 DispDownloadSize(-1);
\r
2364 /* ファイルのタイムスタンプを合わせる */
\r
2366 // if((SaveTimeStamp == YES) &&
\r
2367 // ((Pkt->Time.dwLowDateTime != 0) || (Pkt->Time.dwHighDateTime != 0)))
\r
2369 // SetFileTime(iFileHandle, &Pkt->Time, &Pkt->Time, &Pkt->Time);
\r
2372 CloseHandle(iFileHandle);
\r
2374 if(iNumBytes == SOCKET_ERROR)
\r
2375 ReportWSError("recv",WSAGetLastError());
\r
2379 SetErrorMsg(MSGJPN095, Pkt->LocalFile);
\r
2380 SetTaskMsg(MSGJPN095, Pkt->LocalFile);
\r
2381 Pkt->Abort = ABORT_ERROR;
\r
2385 if(shutdown(dSkt, 1) != 0)
\r
2386 ReportWSError("shutdown", WSAGetLastError());
\r
2389 LastDataConnectionTime = time(NULL);
\r
2392 if(ForceAbort == NO)
\r
2394 /* Abortをホストに伝える */
\r
2395 if(Pkt->Abort != ABORT_NONE && iFileHandle != INVALID_HANDLE_VALUE)
\r
2397 SendData(Pkt->ctrl_skt, "\xFF\xF4\xFF", 3, MSG_OOB, CancelCheckWork); /* MSG_OOBに注意 */
\r
2398 SendData(Pkt->ctrl_skt, "\xF2", 1, 0, CancelCheckWork);
\r
2399 command(Pkt->ctrl_skt, NULL, CancelCheckWork, "ABOR");
\r
2403 iRetCode = ReadReplyMessage(Pkt->ctrl_skt, Buf, 1024, CancelCheckWork, TmpBuf);
\r
2406 //DoPrintf("##DOWN REPLY : %s", Buf);
\r
2408 if(Pkt->Abort == ABORT_DISKFULL)
\r
2410 SetErrorMsg(MSGJPN096);
\r
2411 SetTaskMsg(MSGJPN096);
\r
2414 // if(iRetCode >= FTP_RETRY)
\r
2415 if((iRetCode/100) >= FTP_RETRY)
\r
2417 if(Pkt->Abort != ABORT_NONE)
\r
2424 /*----- ダウンロード終了/中止時のメッセージを表示 ----------------------------
\r
2427 * TRANSPACKET *Pkt : 転送ファイル情報
\r
2428 * int iRetCode : 応答コード
\r
2432 *----------------------------------------------------------------------------*/
\r
2434 static void DispDownloadFinishMsg(TRANSPACKET *Pkt, int iRetCode)
\r
2436 char Fname[FMAX_PATH+1];
\r
2439 ReleaseMutex(hListAccMutex);
\r
2440 if(ForceAbort == NO)
\r
2442 if((iRetCode/100) >= FTP_CONTINUE)
\r
2444 strcpy(Fname, Pkt->RemoteFile);
\r
2446 #if defined(HAVE_OPENVMS)
\r
2447 /* OpenVMSの場合、空ディレクトリへ移動すると550 File not foundになって
\r
2448 * エラーダイアログやエラーメッセージが出るので何もしない */
\r
2449 if (AskHostType() == HTYPE_VMS)
\r
2452 #if defined(HAVE_TANDEM)
\r
2453 /* HP Nonstop Server の場合、ファイルのない subvol へ移動すると550 File not found
\r
2454 * になるが問題ないのでエラーダイアログやエラーメッセージを出さないため */
\r
2455 if (AskHostType() == HTYPE_TANDEM)
\r
2460 // if((strncmp(Pkt->Cmd, "NLST", 4) == 0) || (strncmp(Pkt->Cmd, "LIST", 4) == 0))
\r
2461 if((strncmp(Pkt->Cmd, "NLST", 4) == 0) || (strncmp(Pkt->Cmd, "LIST", 4) == 0) || (strncmp(Pkt->Cmd, "MLSD", 4) == 0))
\r
2463 SetTaskMsg(MSGJPN097);
\r
2464 strcpy(Fname, MSGJPN098);
\r
2467 // else if((Pkt->hWndTrans != NULL) && (TimeStart != 0))
\r
2468 // SetTaskMsg(MSGJPN099, TimeStart, Pkt->ExistSize/TimeStart);
\r
2469 else if((Pkt->hWndTrans != NULL) && (TimeStart[Pkt->ThreadCount] != 0))
\r
2470 SetTaskMsg(MSGJPN099, TimeStart[Pkt->ThreadCount], Pkt->ExistSize/TimeStart[Pkt->ThreadCount]);
\r
2472 SetTaskMsg(MSGJPN100);
\r
2474 if(Pkt->Abort != ABORT_USER)
\r
2476 // 全て中止を選択後にダイアログが表示されるバグ対策
\r
2477 // if(DispUpDownErrDialog(downerr_dlg, Pkt->hWndTrans, Fname) == NO)
\r
2479 // if(Canceled[Pkt->ThreadCount] == NO && ClearAll == NO && DispUpDownErrDialog(downerr_dlg, Pkt->hWndTrans, Fname) == NO)
\r
2480 // ClearAll = YES;
\r
2481 if(Canceled[Pkt->ThreadCount] == NO && ClearAll == NO)
\r
2483 if(strncmp(Pkt->Cmd, "RETR", 4) == 0 || strncmp(Pkt->Cmd, "STOR", 4) == 0)
\r
2486 TransferErrorDisplay++;
\r
2487 if(TransferErrorNotify == YES && DispUpDownErrDialog(downerr_dlg, Pkt->hWndTrans, Pkt) == NO)
\r
2491 Pkt->Mode = TransferErrorMode;
\r
2492 AddTransFileList(Pkt);
\r
2495 TransferErrorDisplay--;
\r
2503 // if((strncmp(Pkt->Cmd, "NLST", 4) == 0) || (strncmp(Pkt->Cmd, "LIST", 4) == 0))
\r
2504 if((strncmp(Pkt->Cmd, "NLST", 4) == 0) || (strncmp(Pkt->Cmd, "LIST", 4) == 0) || (strncmp(Pkt->Cmd, "MLSD", 4) == 0))
\r
2505 SetTaskMsg(MSGJPN101, Pkt->ExistSize);
\r
2507 // else if((Pkt->hWndTrans != NULL) && (TimeStart != 0))
\r
2508 // SetTaskMsg(MSGJPN102, TimeStart, Pkt->ExistSize/TimeStart);
\r
2509 else if((Pkt->hWndTrans != NULL) && (TimeStart[Pkt->ThreadCount] != 0))
\r
2510 // "0 B/S"と表示されるバグを修正
\r
2511 // 原因は%dにあたる部分に64ビット値が渡されているため
\r
2512 // SetTaskMsg(MSGJPN102, TimeStart[Pkt->ThreadCount], Pkt->ExistSize/TimeStart[Pkt->ThreadCount]);
\r
2513 SetTaskMsg(MSGJPN102, (LONG)TimeStart[Pkt->ThreadCount], (LONG)(Pkt->ExistSize/TimeStart[Pkt->ThreadCount]));
\r
2515 SetTaskMsg(MSGJPN103, Pkt->ExistSize);
\r
2522 /*----- ダウンロード/アップロードエラーのダイアログを表示 --------------------
\r
2525 * int RedID : ダイアログボックスのリソースID
\r
2526 * HWND hWnd : 書き込み中ダイアログのウインドウ
\r
2527 * char *Fname : ファイル名
\r
2530 * int ステータス (YES=中止/NO=全て中止)
\r
2531 *----------------------------------------------------------------------------*/
\r
2534 //static int DispUpDownErrDialog(int ResID, HWND hWnd, char *Fname)
\r
2535 static int DispUpDownErrDialog(int ResID, HWND hWnd, TRANSPACKET *Pkt)
\r
2538 hWnd = GetMainHwnd();
\r
2540 SoundPlay(SND_ERROR);
\r
2542 // return(DialogBoxParam(GetFtpInst(), MAKEINTRESOURCE(ResID), hWnd, UpDownErrorDialogProc, (LPARAM)Fname));
\r
2543 return(DialogBoxParam(GetFtpInst(), MAKEINTRESOURCE(ResID), hWnd, UpDownErrorDialogProc, (LPARAM)Pkt));
\r
2547 /*----- ダウンロードエラー/アップロードエラーダイアログのコールバック --------
\r
2550 * HWND hDlg : ウインドウハンドル
\r
2551 * UINT message : メッセージ番号
\r
2552 * WPARAM wParam : メッセージの WPARAM 引数
\r
2553 * LPARAM lParam : メッセージの LPARAM 引数
\r
2557 *----------------------------------------------------------------------------*/
\r
2560 //static BOOL CALLBACK UpDownErrorDialogProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
\r
2561 static INT_PTR CALLBACK UpDownErrorDialogProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
\r
2563 static TRANSPACKET *Pkt;
\r
2564 static const RADIOBUTTON DownExistButton[] = {
\r
2565 { DOWN_EXIST_OVW, EXIST_OVW },
\r
2566 { DOWN_EXIST_RESUME, EXIST_RESUME },
\r
2567 { DOWN_EXIST_IGNORE, EXIST_IGNORE }
\r
2569 #define DOWNEXISTBUTTONS (sizeof(DownExistButton)/sizeof(RADIOBUTTON))
\r
2573 case WM_INITDIALOG :
\r
2574 Pkt = (TRANSPACKET *)lParam;
\r
2575 // SendDlgItemMessage(hDlg, UPDOWN_ERR_FNAME, WM_SETTEXT, 0, (LPARAM)lParam);
\r
2576 SendDlgItemMessage(hDlg, UPDOWN_ERR_FNAME, WM_SETTEXT, 0, (LPARAM)Pkt->RemoteFile);
\r
2578 // SendDlgItemMessage(hDlg, UPDOWN_ERR_MSG, WM_SETTEXT, 0, (LPARAM)ErrMsg);
\r
2579 SendDlgItemMessage(hDlg, UPDOWN_ERR_MSG, WM_SETTEXT, 0, (LPARAM)GetErrMsg());
\r
2581 if((Pkt->Type == TYPE_A) || (Pkt->ExistSize <= 0))
\r
2582 EnableWindow(GetDlgItem(hDlg, DOWN_EXIST_RESUME), FALSE);
\r
2584 SetRadioButtonByValue(hDlg, TransferErrorMode, DownExistButton, DOWNEXISTBUTTONS);
\r
2588 switch(GET_WM_COMMAND_ID(wParam, lParam))
\r
2591 TransferErrorNotify = NO;
\r
2592 /* ここに break はない */
\r
2595 TransferErrorMode = AskRadioButtonValue(hDlg, DownExistButton, DOWNEXISTBUTTONS);
\r
2596 EndDialog(hDlg, YES);
\r
2600 EndDialog(hDlg, NO);
\r
2604 // hHelpWin = HtmlHelp(NULL, AskHelpFilePath(), HH_HELP_CONTEXT, IDH_HELP_TOPIC_0000009);
\r
2613 /*----- ダウンロードのリジュームの準備を行う ----------------------------------
\r
2616 * TRANSPACKET *Pkt : 転送ファイル情報
\r
2617 * iont ProcMode : 処理モード(EXIST_xxx)
\r
2618 * LONGLONG Size : ロード済みのファイルのサイズ
\r
2619 * int *Mode : ファイル作成モード (CREATE_xxxx)
\r
2622 * int 転送を行うかどうか(YES/NO=このファイルを中止/NO_ALL=全て中止)
\r
2625 * Pkt->ExistSizeのセットを行なう
\r
2626 *----------------------------------------------------------------------------*/
\r
2628 static int SetDownloadResume(TRANSPACKET *Pkt, int ProcMode, LONGLONG Size, int *Mode, int *CancelCheckWork)
\r
2632 char Reply[ERR_MSG_LEN+7];
\r
2637 Pkt->ExistSize = 0;
\r
2638 *Mode = CREATE_ALWAYS;
\r
2640 if(ProcMode == EXIST_RESUME)
\r
2642 iRetCode = command(Pkt->ctrl_skt, Reply, CancelCheckWork, "REST %s", MakeNumString(Size, Tmp, FALSE));
\r
2643 if(iRetCode/100 < FTP_RETRY)
\r
2646 if(Pkt->hWndTrans != NULL)
\r
2647 Pkt->ExistSize = Size;
\r
2648 *Mode = OPEN_ALWAYS;
\r
2652 Com = DialogBox(GetFtpInst(), MAKEINTRESOURCE(noresume_dlg), Pkt->hWndTrans, NoResumeWndProc);
\r
2655 if(Com == NO_ALL) /* 全て中止 */
\r
2657 Pkt->Abort = ABORT_USER;
\r
2665 /*----- resumeエラーダイアログのコールバック ----------------------------------
\r
2668 * HWND hDlg : ウインドウハンドル
\r
2669 * UINT message : メッセージ番号
\r
2670 * WPARAM wParam : メッセージの WPARAM 引数
\r
2671 * LPARAM lParam : メッセージの LPARAM 引数
\r
2675 *----------------------------------------------------------------------------*/
\r
2678 //static BOOL CALLBACK NoResumeWndProc(HWND hDlg, UINT iMessage, WPARAM wParam, LPARAM lParam)
\r
2679 static INT_PTR CALLBACK NoResumeWndProc(HWND hDlg, UINT iMessage, WPARAM wParam, LPARAM lParam)
\r
2683 case WM_INITDIALOG :
\r
2687 switch(GET_WM_COMMAND_ID(wParam, lParam))
\r
2690 EndDialog(hDlg, YES);
\r
2694 EndDialog(hDlg, NO);
\r
2697 case RESUME_CANCEL_ALL :
\r
2698 EndDialog(hDlg, NO_ALL);
\r
2708 /*----- アップロードを行なう --------------------------------------------------
\r
2711 * SOCKET cSkt : コントロールソケット
\r
2712 * TRANSPACKET *Pkt : 転送ファイル情報
\r
2716 *----------------------------------------------------------------------------*/
\r
2718 static int DoUpload(SOCKET cSkt, TRANSPACKET *Pkt)
\r
2721 char Reply[ERR_MSG_LEN+7];
\r
2723 Pkt->ctrl_skt = cSkt;
\r
2725 if(Pkt->Mode != EXIST_IGNORE)
\r
2727 if(CheckFileReadable(Pkt->LocalFile) == FFFTP_SUCCESS)
\r
2729 if(Pkt->Type == TYPE_I)
\r
2730 Pkt->KanjiCode = KANJI_NOCNV;
\r
2733 // iRetCode = command(Pkt->ctrl_skt, Reply, &Canceled, "TYPE %c", Pkt->Type);
\r
2734 iRetCode = command(Pkt->ctrl_skt, Reply, &Canceled[Pkt->ThreadCount], "TYPE %c", Pkt->Type);
\r
2735 if(iRetCode/100 < FTP_RETRY)
\r
2737 if(Pkt->Mode == EXIST_UNIQUE)
\r
2738 strcpy(Pkt->Cmd, "STOU ");
\r
2740 if(Pkt->hWndTrans != NULL)
\r
2741 DispTransFileInfo(Pkt, MSGJPN104, TRUE, YES);
\r
2743 if(BackgrndMessageProc() == NO)
\r
2745 if(AskPasvMode() != YES)
\r
2746 iRetCode = UploadNonPassive(Pkt);
\r
2748 iRetCode = UploadPassive(Pkt);
\r
2754 SetErrorMsg(Reply);
\r
2757 if((Pkt->Attr != -1) && ((iRetCode/100) == FTP_COMPLETE))
\r
2759 // command(Pkt->ctrl_skt, Reply, &Canceled, "%s %03X %s", AskHostChmodCmd(), Pkt->Attr, Pkt->RemoteFile);
\r
2760 command(Pkt->ctrl_skt, Reply, &Canceled[Pkt->ThreadCount], "%s %03X %s", AskHostChmodCmd(), Pkt->Attr, Pkt->RemoteFile);
\r
2764 SetErrorMsg(MSGJPN105, Pkt->LocalFile);
\r
2765 SetTaskMsg(MSGJPN105, Pkt->LocalFile);
\r
2767 Pkt->Abort = ABORT_ERROR;
\r
2768 // エラーによってはダイアログが表示されない場合があるバグ対策
\r
2769 // DispUploadFinishMsg(Pkt, iRetCode);
\r
2771 // エラーによってはダイアログが表示されない場合があるバグ対策
\r
2772 DispUploadFinishMsg(Pkt, iRetCode);
\r
2776 DispTransFileInfo(Pkt, MSGJPN106, TRUE, YES);
\r
2777 SetTaskMsg(MSGJPN107, Pkt->LocalFile);
\r
2784 /*----- 通常モードでファイルをアップロード ------------------------------------
\r
2787 * TRANSPACKET *Pkt : 転送ファイル情報
\r
2791 *----------------------------------------------------------------------------*/
\r
2793 static int UploadNonPassive(TRANSPACKET *Pkt)
\r
2797 SOCKET data_socket = INVALID_SOCKET; // data channel socket
\r
2798 SOCKET listen_socket = INVALID_SOCKET; // data listen socket
\r
2800 // char Buf[1024];
\r
2801 char Buf[FMAX_PATH+1024];
\r
2803 // struct sockaddr_in saSockAddr1;
\r
2804 struct sockaddr_in saSockAddrIPv4;
\r
2805 struct sockaddr_in6 saSockAddrIPv6;
\r
2809 char Reply[ERR_MSG_LEN+7];
\r
2812 // if((listen_socket = GetFTPListenSocket(Pkt->ctrl_skt, &Canceled)) != INVALID_SOCKET)
\r
2813 if((listen_socket = GetFTPListenSocket(Pkt->ctrl_skt, &Canceled[Pkt->ThreadCount])) != INVALID_SOCKET)
\r
2815 SetUploadResume(Pkt, Pkt->Mode, Pkt->ExistSize, &Resume);
\r
2817 #if defined(HAVE_TANDEM)
\r
2818 if(AskHostType() == HTYPE_TANDEM && AskOSS() == NO && Pkt->Type != TYPE_A) {
\r
2819 if( Pkt->PriExt == DEF_PRIEXT && Pkt->SecExt == DEF_SECEXT && Pkt->MaxExt == DEF_MAXEXT) {
\r
2820 // EXTENTがデフォルトのときはコードのみ
\r
2821 sprintf(Buf, "%s%s,%d", Pkt->Cmd, Pkt->RemoteFile, Pkt->FileCode);
\r
2823 sprintf(Buf, "%s%s,%d,%d,%d,%d", Pkt->Cmd, Pkt->RemoteFile, Pkt->FileCode, Pkt->PriExt, Pkt->SecExt, Pkt->MaxExt);
\r
2827 sprintf(Buf, "%s%s", Pkt->Cmd, Pkt->RemoteFile);
\r
2829 sprintf(Buf, "%s%s", "APPE ", Pkt->RemoteFile);
\r
2832 // iRetCode = command(Pkt->ctrl_skt, Reply, &Canceled, "%s", Buf);
\r
2833 iRetCode = command(Pkt->ctrl_skt, Reply, &Canceled[Pkt->ThreadCount], "%s", Buf);
\r
2834 if((iRetCode/100) == FTP_PRELIM)
\r
2837 // 応答の形式に規格が無くファイル名を取得できないため属性変更を無効化
\r
2838 if(Pkt->Mode == EXIST_UNIQUE)
\r
2841 // if(SocksGet2ndBindReply(listen_socket, &data_socket) == FFFTP_FAIL)
\r
2842 if(SocksGet2ndBindReply(listen_socket, &data_socket, &Canceled[Pkt->ThreadCount]) == FFFTP_FAIL)
\r
2845 // iLength=sizeof(saSockAddr1);
\r
2846 // data_socket = do_accept(listen_socket,(struct sockaddr *)&saSockAddr1, (int *)&iLength);
\r
2847 switch(AskCurNetType())
\r
2850 iLength=sizeof(saSockAddrIPv4);
\r
2851 data_socket = do_accept(listen_socket,(struct sockaddr *)&saSockAddrIPv4, (int *)&iLength);
\r
2854 iLength=sizeof(saSockAddrIPv6);
\r
2855 data_socket = do_accept(listen_socket,(struct sockaddr *)&saSockAddrIPv6, (int *)&iLength);
\r
2859 if(shutdown(listen_socket, 1) != 0)
\r
2860 ReportWSError("shutdown listen", WSAGetLastError());
\r
2862 if(IsUPnPLoaded() == YES)
\r
2864 if(GetAsyncTableDataMapPort(listen_socket, &Port) == YES)
\r
2865 RemovePortMapping(Port);
\r
2867 listen_socket = DoClose(listen_socket);
\r
2869 if(data_socket == INVALID_SOCKET)
\r
2871 SetErrorMsg(MSGJPN280);
\r
2872 ReportWSError("accept", WSAGetLastError());
\r
2877 // DoPrintf("Skt=%u : accept from %s port %u", data_socket, inet_ntoa(saSockAddr1.sin_addr), ntohs(saSockAddr1.sin_port));
\r
2879 switch(AskCurNetType())
\r
2882 DoPrintf("Skt=%u : accept from %s port %u", data_socket, inet_ntoa(saSockAddrIPv4.sin_addr), ntohs(saSockAddrIPv4.sin_port));
\r
2885 DoPrintf("Skt=%u : accept from %s port %u", data_socket, inet6_ntoa(saSockAddrIPv6.sin6_addr), ntohs(saSockAddrIPv6.sin6_port));
\r
2891 if(data_socket != INVALID_SOCKET)
\r
2893 // 一部TYPE、STOR(RETR)、PORT(PASV)を並列に処理できないホストがあるため
\r
2894 ReleaseMutex(hListAccMutex);
\r
2896 // iRetCode = UploadFile(Pkt, data_socket);
\r
2897 if(IsSSLAttached(Pkt->ctrl_skt))
\r
2899 if(AttachSSL(data_socket, Pkt->ctrl_skt, &Canceled[Pkt->ThreadCount], FALSE, NULL))
\r
2900 iRetCode = UploadFile(Pkt, data_socket);
\r
2905 iRetCode = UploadFile(Pkt, data_socket);
\r
2906 data_socket = DoClose(data_socket);
\r
2911 SetErrorMsg(Reply);
\r
2912 SetTaskMsg(MSGJPN108);
\r
2914 if(IsUPnPLoaded() == YES)
\r
2916 if(GetAsyncTableDataMapPort(listen_socket, &Port) == YES)
\r
2917 RemovePortMapping(Port);
\r
2919 listen_socket = DoClose(listen_socket);
\r
2925 SetErrorMsg(MSGJPN279);
\r
2928 // エラーによってはダイアログが表示されない場合があるバグ対策
\r
2929 // DispUploadFinishMsg(Pkt, iRetCode);
\r
2935 /*----- Passiveモードでファイルをアップロード ---------------------------------
\r
2938 * TRANSPACKET *Pkt : 転送ファイル情報
\r
2942 *----------------------------------------------------------------------------*/
\r
2944 static int UploadPassive(TRANSPACKET *Pkt)
\r
2947 SOCKET data_socket = INVALID_SOCKET; // data channel socket
\r
2949 // char Buf[1024];
\r
2950 char Buf[FMAX_PATH+1024];
\r
2957 char Reply[ERR_MSG_LEN+7];
\r
2960 // iRetCode = command(Pkt->ctrl_skt, Buf, &Canceled, "PASV");
\r
2962 // iRetCode = command(Pkt->ctrl_skt, Buf, &Canceled[Pkt->ThreadCount], "PASV");
\r
2963 switch(AskCurNetType())
\r
2966 iRetCode = command(Pkt->ctrl_skt, Buf, &Canceled[Pkt->ThreadCount], "PASV");
\r
2969 iRetCode = command(Pkt->ctrl_skt, Buf, &Canceled[Pkt->ThreadCount], "EPSV");
\r
2972 if(iRetCode/100 == FTP_COMPLETE)
\r
2975 // if(GetAdrsAndPort(Buf, Adrs, &Port, 19) == FFFTP_SUCCESS)
\r
2976 if(GetAdrsAndPort(Pkt->ctrl_skt, Buf, Adrs, &Port, 39) == FFFTP_SUCCESS)
\r
2979 // if((data_socket = connectsock(Adrs, Port, MSGJPN109, &Canceled)) != INVALID_SOCKET)
\r
2980 if((data_socket = connectsock(Adrs, Port, MSGJPN109, &Canceled[Pkt->ThreadCount])) != INVALID_SOCKET)
\r
2984 if(setsockopt(data_socket, IPPROTO_TCP, TCP_NODELAY, (LPSTR)&Flg, sizeof(Flg)) == SOCKET_ERROR)
\r
2985 ReportWSError("setsockopt", WSAGetLastError());
\r
2987 SetUploadResume(Pkt, Pkt->Mode, Pkt->ExistSize, &Resume);
\r
2989 #if defined(HAVE_TANDEM)
\r
2990 if(AskHostType() == HTYPE_TANDEM && AskOSS() == NO && Pkt->Type != TYPE_A) {
\r
2991 if( Pkt->PriExt == DEF_PRIEXT && Pkt->SecExt == DEF_SECEXT && Pkt->MaxExt == DEF_MAXEXT) {
\r
2992 // EXTENTがデフォルトのときはコードのみ
\r
2993 sprintf(Buf, "%s%s,%d", Pkt->Cmd, Pkt->RemoteFile, Pkt->FileCode);
\r
2995 sprintf(Buf, "%s%s,%d,%d,%d,%d", Pkt->Cmd, Pkt->RemoteFile, Pkt->FileCode, Pkt->PriExt, Pkt->SecExt, Pkt->MaxExt);
\r
2999 sprintf(Buf, "%s%s", Pkt->Cmd, Pkt->RemoteFile);
\r
3001 sprintf(Buf, "%s%s", "APPE ", Pkt->RemoteFile);
\r
3004 // iRetCode = command(Pkt->ctrl_skt, Reply, &Canceled, "%s", Buf);
\r
3005 iRetCode = command(Pkt->ctrl_skt, Reply, &Canceled[Pkt->ThreadCount], "%s", Buf);
\r
3006 if(iRetCode/100 == FTP_PRELIM)
\r
3009 // 応答の形式に規格が無くファイル名を取得できないため属性変更を無効化
\r
3010 if(Pkt->Mode == EXIST_UNIQUE)
\r
3012 // 一部TYPE、STOR(RETR)、PORT(PASV)を並列に処理できないホストがあるため
\r
3013 ReleaseMutex(hListAccMutex);
\r
3015 // iRetCode = UploadFile(Pkt, data_socket);
\r
3016 if(IsSSLAttached(Pkt->ctrl_skt))
\r
3018 if(AttachSSL(data_socket, Pkt->ctrl_skt, &Canceled[Pkt->ThreadCount], FALSE, NULL))
\r
3019 iRetCode = UploadFile(Pkt, data_socket);
\r
3024 iRetCode = UploadFile(Pkt, data_socket);
\r
3026 data_socket = DoClose(data_socket);
\r
3030 SetErrorMsg(Reply);
\r
3031 SetTaskMsg(MSGJPN110);
\r
3032 data_socket = DoClose(data_socket);
\r
3038 SetErrorMsg(MSGJPN281);
\r
3045 SetTaskMsg(MSGJPN111);
\r
3052 // エラーによってはダイアログが表示されない場合があるバグ対策
\r
3053 // DispUploadFinishMsg(Pkt, iRetCode);
\r
3059 /*----- アップロードの実行 ----------------------------------------------------
\r
3062 * TRANSPACKET *Pkt : 転送ファイル情報
\r
3063 * SOCKET dSkt : データソケット
\r
3069 * 転送の経過表示は、インターバルタイマで経過を表示する
\r
3070 * 転送ダイアログを出さないでアップロードすることはない
\r
3071 *----------------------------------------------------------------------------*/
\r
3073 static int UploadFile(TRANSPACKET *Pkt, SOCKET dSkt)
\r
3076 HANDLE iFileHandle;
\r
3077 SECURITY_ATTRIBUTES Sec;
\r
3078 char Buf[BUFSIZE];
\r
3079 char Buf2[BUFSIZE+3];
\r
3080 CODECONVINFO cInfo;
\r
3081 TERMCODECONVINFO tInfo;
\r
3088 char TmpBuf[ONELINE_BUF_SIZE];
\r
3092 #ifdef SET_BUFFER_SIZE
\r
3093 /* Add by H.Shirouzu at 2002/10/02 */
\r
3094 int buf_size = SOCKBUF_SIZE;
\r
3095 for ( ; buf_size > 0; buf_size /= 2)
\r
3096 if (setsockopt(dSkt, SOL_SOCKET, SO_SNDBUF, (char *)&buf_size, sizeof(buf_size)) == 0)
\r
3101 // 念のため送信バッファを無効にする
\r
3102 #ifdef DISABLE_TRANSFER_NETWORK_BUFFERS
\r
3104 setsockopt(dSkt, SOL_SOCKET, SO_SNDBUF, (char *)&buf_size, sizeof(buf_size));
\r
3107 Pkt->Abort = ABORT_NONE;
\r
3109 Sec.nLength = sizeof(SECURITY_ATTRIBUTES);
\r
3110 Sec.lpSecurityDescriptor = NULL;
\r
3111 Sec.bInheritHandle = FALSE;
\r
3113 if((iFileHandle = CreateFile(Pkt->LocalFile, GENERIC_READ,
\r
3114 FILE_SHARE_READ|FILE_SHARE_WRITE, &Sec, OPEN_EXISTING, 0, NULL)) != INVALID_HANDLE_VALUE)
\r
3117 char Buf3[(BUFSIZE + 3) * 4];
\r
3118 CODECONVINFO cInfo2;
\r
3119 int ProcessedBOM = NO;
\r
3120 if(Pkt->hWndTrans != NULL)
\r
3122 Low = GetFileSize(iFileHandle, &High);
\r
3123 Pkt->Size = MakeLongLong(High, Low);
\r
3125 High = (DWORD)HIGH32(Pkt->ExistSize);
\r
3126 Low = (DWORD)LOW32(Pkt->ExistSize);
\r
3127 SetFilePointer(iFileHandle, Low, &High, FILE_BEGIN);
\r
3130 // AllTransSizeNow = 0;
\r
3131 // TimeStart = time(NULL);
\r
3132 AllTransSizeNow[Pkt->ThreadCount] = 0;
\r
3133 TimeStart[Pkt->ThreadCount] = time(NULL);
\r
3134 SetTimer(Pkt->hWndTrans, TIMER_DISPLAY, DISPLAY_TIMING, NULL);
\r
3137 InitCodeConvInfo(&cInfo);
\r
3138 cInfo.KanaCnv = Pkt->KanaCnv;
\r
3139 InitTermCodeConvInfo(&tInfo);
\r
3141 InitCodeConvInfo(&cInfo2);
\r
3142 cInfo2.KanaCnv = Pkt->KanaCnv;
\r
3144 /*===== ファイルを送信するループ =====*/
\r
3145 while((Pkt->Abort == ABORT_NONE) &&
\r
3146 (ForceAbort == NO) &&
\r
3147 (ReadFile(iFileHandle, Buf, BUFSIZE, &iNumBytes, NULL) == TRUE))
\r
3149 if(iNumBytes == 0)
\r
3154 if((RmEOF == YES) && (Pkt->Type == TYPE_A))
\r
3156 if((EofPos = memchr(Buf, 0x1A, iNumBytes)) != NULL)
\r
3157 iNumBytes = EofPos - Buf;
\r
3161 if(Pkt->KanjiCode != KANJI_NOCNV)
\r
3164 cInfo.StrLen = iNumBytes;
\r
3166 cInfo.BufSize = BUFSIZE+3;
\r
3169 // ここで全てUTF-8へ変換する
\r
3170 // TODO: SJIS以外も直接UTF-8へ変換
\r
3171 // if(Pkt->KanjiCode == KANJI_JIS)
\r
3172 // Continue = ConvSJIStoJIS(&cInfo);
\r
3174 // Continue = ConvSJIStoEUC(&cInfo);
\r
3175 switch(Pkt->KanjiCodeDesired)
\r
3178 switch(Pkt->KanjiCode)
\r
3181 // memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
3182 // cInfo2.OutLen = cInfo.StrLen;
\r
3185 Continue = ConvSJIStoJIS(&cInfo);
\r
3186 cInfo2.Str = cInfo.Buf;
\r
3187 cInfo2.StrLen = cInfo.OutLen;
\r
3188 cInfo2.Buf = Buf3;
\r
3189 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3190 ConvJIStoSJIS(&cInfo2);
\r
3193 Continue = ConvSJIStoJIS(&cInfo);
\r
3194 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3195 cInfo2.OutLen = cInfo.OutLen;
\r
3198 Continue = ConvSJIStoEUC(&cInfo);
\r
3199 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3200 cInfo2.OutLen = cInfo.OutLen;
\r
3203 Continue = ConvSJIStoUTF8N(&cInfo);
\r
3204 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3205 cInfo2.OutLen = cInfo.OutLen;
\r
3207 case KANJI_UTF8BOM:
\r
3208 if(ProcessedBOM == NO)
\r
3210 memcpy(Buf3, "\xEF\xBB\xBF", 3);
\r
3211 cInfo2.OutLen = 3;
\r
3213 ProcessedBOM = YES;
\r
3216 Continue = ConvSJIStoUTF8N(&cInfo);
\r
3217 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3218 cInfo2.OutLen = cInfo.OutLen;
\r
3223 switch(Pkt->KanjiCode)
\r
3226 Continue = ConvJIStoSJIS(&cInfo);
\r
3227 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3228 cInfo2.OutLen = cInfo.OutLen;
\r
3231 // memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
3232 // cInfo2.OutLen = cInfo.StrLen;
\r
3235 Continue = ConvJIStoSJIS(&cInfo);
\r
3236 cInfo2.Str = cInfo.Buf;
\r
3237 cInfo2.StrLen = cInfo.OutLen;
\r
3238 cInfo2.Buf = Buf3;
\r
3239 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3240 ConvSJIStoJIS(&cInfo2);
\r
3243 Continue = ConvJIStoSJIS(&cInfo);
\r
3244 cInfo2.Str = cInfo.Buf;
\r
3245 cInfo2.StrLen = cInfo.OutLen;
\r
3246 cInfo2.Buf = Buf3;
\r
3247 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3248 ConvSJIStoEUC(&cInfo2);
\r
3251 Continue = ConvJIStoSJIS(&cInfo);
\r
3252 cInfo2.Str = cInfo.Buf;
\r
3253 cInfo2.StrLen = cInfo.OutLen;
\r
3254 cInfo2.Buf = Buf3;
\r
3255 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3256 ConvSJIStoUTF8N(&cInfo2);
\r
3258 case KANJI_UTF8BOM:
\r
3259 if(ProcessedBOM == NO)
\r
3261 memcpy(Buf3, "\xEF\xBB\xBF", 3);
\r
3262 cInfo2.OutLen = 3;
\r
3264 ProcessedBOM = YES;
\r
3267 Continue = ConvJIStoSJIS(&cInfo);
\r
3268 cInfo2.Str = cInfo.Buf;
\r
3269 cInfo2.StrLen = cInfo.OutLen;
\r
3270 cInfo2.Buf = Buf3;
\r
3271 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3272 ConvSJIStoUTF8N(&cInfo2);
\r
3277 switch(Pkt->KanjiCode)
\r
3280 Continue = ConvEUCtoSJIS(&cInfo);
\r
3281 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3282 cInfo2.OutLen = cInfo.OutLen;
\r
3285 Continue = ConvEUCtoSJIS(&cInfo);
\r
3286 cInfo2.Str = cInfo.Buf;
\r
3287 cInfo2.StrLen = cInfo.OutLen;
\r
3288 cInfo2.Buf = Buf3;
\r
3289 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3290 ConvSJIStoJIS(&cInfo2);
\r
3293 // memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
3294 // cInfo2.OutLen = cInfo.StrLen;
\r
3297 Continue = ConvEUCtoSJIS(&cInfo);
\r
3298 cInfo2.Str = cInfo.Buf;
\r
3299 cInfo2.StrLen = cInfo.OutLen;
\r
3300 cInfo2.Buf = Buf3;
\r
3301 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3302 ConvSJIStoEUC(&cInfo2);
\r
3305 Continue = ConvEUCtoSJIS(&cInfo);
\r
3306 cInfo2.Str = cInfo.Buf;
\r
3307 cInfo2.StrLen = cInfo.OutLen;
\r
3308 cInfo2.Buf = Buf3;
\r
3309 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3310 ConvSJIStoUTF8N(&cInfo2);
\r
3312 case KANJI_UTF8BOM:
\r
3313 if(ProcessedBOM == NO)
\r
3315 memcpy(Buf3, "\xEF\xBB\xBF", 3);
\r
3316 cInfo2.OutLen = 3;
\r
3318 ProcessedBOM = YES;
\r
3321 Continue = ConvEUCtoSJIS(&cInfo);
\r
3322 cInfo2.Str = cInfo.Buf;
\r
3323 cInfo2.StrLen = cInfo.OutLen;
\r
3324 cInfo2.Buf = Buf3;
\r
3325 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3326 ConvSJIStoUTF8N(&cInfo2);
\r
3331 switch(Pkt->KanjiCode)
\r
3334 Continue = ConvUTF8NtoSJIS(&cInfo);
\r
3335 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3336 cInfo2.OutLen = cInfo.OutLen;
\r
3339 Continue = ConvUTF8NtoSJIS(&cInfo);
\r
3340 cInfo2.Str = cInfo.Buf;
\r
3341 cInfo2.StrLen = cInfo.OutLen;
\r
3342 cInfo2.Buf = Buf3;
\r
3343 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3344 ConvSJIStoJIS(&cInfo2);
\r
3347 Continue = ConvUTF8NtoSJIS(&cInfo);
\r
3348 cInfo2.Str = cInfo.Buf;
\r
3349 cInfo2.StrLen = cInfo.OutLen;
\r
3350 cInfo2.Buf = Buf3;
\r
3351 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3352 ConvSJIStoEUC(&cInfo2);
\r
3355 memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
3356 cInfo2.OutLen = cInfo.StrLen;
\r
3359 case KANJI_UTF8BOM:
\r
3360 if(ProcessedBOM == NO)
\r
3362 memcpy(Buf3, "\xEF\xBB\xBF", 3);
\r
3363 cInfo2.OutLen = 3;
\r
3365 ProcessedBOM = YES;
\r
3368 memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
3369 cInfo2.OutLen = cInfo.StrLen;
\r
3374 case KANJI_UTF8BOM:
\r
3375 if(ProcessedBOM == NO)
\r
3377 if(memcmp(Buf, "\xEF\xBB\xBF", 3) == 0)
\r
3380 cInfo.StrLen -= 3;
\r
3382 cInfo2.OutLen = 0;
\r
3383 switch(Pkt->KanjiCode)
\r
3385 case KANJI_UTF8BOM:
\r
3386 memcpy(Buf3, "\xEF\xBB\xBF", 3);
\r
3387 cInfo2.OutLen = 3;
\r
3391 ProcessedBOM = YES;
\r
3394 switch(Pkt->KanjiCode)
\r
3397 Continue = ConvUTF8NtoSJIS(&cInfo);
\r
3398 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3399 cInfo2.OutLen = cInfo.OutLen;
\r
3402 Continue = ConvUTF8NtoSJIS(&cInfo);
\r
3403 cInfo2.Str = cInfo.Buf;
\r
3404 cInfo2.StrLen = cInfo.OutLen;
\r
3405 cInfo2.Buf = Buf3;
\r
3406 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3407 ConvSJIStoJIS(&cInfo2);
\r
3410 Continue = ConvUTF8NtoSJIS(&cInfo);
\r
3411 cInfo2.Str = cInfo.Buf;
\r
3412 cInfo2.StrLen = cInfo.OutLen;
\r
3413 cInfo2.Buf = Buf3;
\r
3414 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3415 ConvSJIStoEUC(&cInfo2);
\r
3418 memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
3419 cInfo2.OutLen = cInfo.StrLen;
\r
3422 case KANJI_UTF8BOM:
\r
3423 memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
3424 cInfo2.OutLen = cInfo.StrLen;
\r
3431 // if(TermCodeConvAndSend(&tInfo, dSkt, Buf2, cInfo.OutLen, Pkt->Type) == FFFTP_FAIL)
\r
3432 if(TermCodeConvAndSend(&tInfo, dSkt, Buf3, cInfo2.OutLen, Pkt->Type, &Canceled[Pkt->ThreadCount]) == FFFTP_FAIL)
\r
3434 Pkt->Abort = ABORT_ERROR;
\r
3438 while(Continue == YES);
\r
3443 // if(TermCodeConvAndSend(&tInfo, dSkt, Buf, iNumBytes, Pkt->Type) == FFFTP_FAIL)
\r
3444 if(TermCodeConvAndSend(&tInfo, dSkt, Buf, iNumBytes, Pkt->Type, &Canceled[Pkt->ThreadCount]) == FFFTP_FAIL)
\r
3445 Pkt->Abort = ABORT_ERROR;
\r
3448 Pkt->ExistSize += iNumBytes;
\r
3449 if(Pkt->hWndTrans != NULL)
\r
3451 // AllTransSizeNow += iNumBytes;
\r
3452 AllTransSizeNow[Pkt->ThreadCount] += iNumBytes;
\r
3454 if(BackgrndMessageProc() == YES)
\r
3457 if(EofPos != NULL)
\r
3461 if((ForceAbort == NO) && (Pkt->Abort == ABORT_NONE))
\r
3464 if(Pkt->KanjiCode != KANJI_NOCNV)
\r
3467 cInfo.BufSize = BUFSIZE+3;
\r
3468 FlushRestData(&cInfo);
\r
3469 switch(Pkt->KanjiCodeDesired)
\r
3472 switch(Pkt->KanjiCode)
\r
3476 cInfo2.Str = cInfo.Buf;
\r
3477 cInfo2.StrLen = cInfo.OutLen;
\r
3478 cInfo2.Buf = Buf3;
\r
3479 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3480 ConvJIStoSJIS(&cInfo2);
\r
3483 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3484 cInfo2.OutLen = cInfo.OutLen;
\r
3487 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3488 cInfo2.OutLen = cInfo.OutLen;
\r
3491 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3492 cInfo2.OutLen = cInfo.OutLen;
\r
3494 case KANJI_UTF8BOM:
\r
3495 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3496 cInfo2.OutLen = cInfo.OutLen;
\r
3501 switch(Pkt->KanjiCode)
\r
3504 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3505 cInfo2.OutLen = cInfo.OutLen;
\r
3509 cInfo2.Str = cInfo.Buf;
\r
3510 cInfo2.StrLen = cInfo.OutLen;
\r
3511 cInfo2.Buf = Buf3;
\r
3512 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3513 ConvSJIStoJIS(&cInfo2);
\r
3516 cInfo2.Str = cInfo.Buf;
\r
3517 cInfo2.StrLen = cInfo.OutLen;
\r
3518 cInfo2.Buf = Buf3;
\r
3519 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3520 ConvSJIStoEUC(&cInfo2);
\r
3523 cInfo2.Str = cInfo.Buf;
\r
3524 cInfo2.StrLen = cInfo.OutLen;
\r
3525 cInfo2.Buf = Buf3;
\r
3526 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3527 ConvSJIStoUTF8N(&cInfo2);
\r
3529 case KANJI_UTF8BOM:
\r
3530 cInfo2.Str = cInfo.Buf;
\r
3531 cInfo2.StrLen = cInfo.OutLen;
\r
3532 cInfo2.Buf = Buf3;
\r
3533 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3534 ConvSJIStoUTF8N(&cInfo2);
\r
3539 switch(Pkt->KanjiCode)
\r
3542 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3543 cInfo2.OutLen = cInfo.OutLen;
\r
3546 cInfo2.Str = cInfo.Buf;
\r
3547 cInfo2.StrLen = cInfo.OutLen;
\r
3548 cInfo2.Buf = Buf3;
\r
3549 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3550 ConvSJIStoJIS(&cInfo2);
\r
3554 cInfo2.Str = cInfo.Buf;
\r
3555 cInfo2.StrLen = cInfo.OutLen;
\r
3556 cInfo2.Buf = Buf3;
\r
3557 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3558 ConvSJIStoEUC(&cInfo2);
\r
3561 cInfo2.Str = cInfo.Buf;
\r
3562 cInfo2.StrLen = cInfo.OutLen;
\r
3563 cInfo2.Buf = Buf3;
\r
3564 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3565 ConvSJIStoUTF8N(&cInfo2);
\r
3567 case KANJI_UTF8BOM:
\r
3568 cInfo2.Str = cInfo.Buf;
\r
3569 cInfo2.StrLen = cInfo.OutLen;
\r
3570 cInfo2.Buf = Buf3;
\r
3571 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3572 ConvSJIStoUTF8N(&cInfo2);
\r
3577 switch(Pkt->KanjiCode)
\r
3580 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3581 cInfo2.OutLen = cInfo.OutLen;
\r
3584 cInfo2.Str = cInfo.Buf;
\r
3585 cInfo2.StrLen = cInfo.OutLen;
\r
3586 cInfo2.Buf = Buf3;
\r
3587 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3588 ConvSJIStoJIS(&cInfo2);
\r
3591 cInfo2.Str = cInfo.Buf;
\r
3592 cInfo2.StrLen = cInfo.OutLen;
\r
3593 cInfo2.Buf = Buf3;
\r
3594 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3595 ConvSJIStoEUC(&cInfo2);
\r
3598 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3599 cInfo2.OutLen = cInfo.OutLen;
\r
3601 case KANJI_UTF8BOM:
\r
3602 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3603 cInfo2.OutLen = cInfo.OutLen;
\r
3607 case KANJI_UTF8BOM:
\r
3608 switch(Pkt->KanjiCode)
\r
3611 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3612 cInfo2.OutLen = cInfo.OutLen;
\r
3615 cInfo2.Str = cInfo.Buf;
\r
3616 cInfo2.StrLen = cInfo.OutLen;
\r
3617 cInfo2.Buf = Buf3;
\r
3618 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3619 ConvSJIStoJIS(&cInfo2);
\r
3622 cInfo2.Str = cInfo.Buf;
\r
3623 cInfo2.StrLen = cInfo.OutLen;
\r
3624 cInfo2.Buf = Buf3;
\r
3625 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3626 ConvSJIStoEUC(&cInfo2);
\r
3629 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3630 cInfo2.OutLen = cInfo.OutLen;
\r
3632 case KANJI_UTF8BOM:
\r
3633 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
3634 cInfo2.OutLen = cInfo.OutLen;
\r
3640 // if(TermCodeConvAndSend(&tInfo, dSkt, Buf2, cInfo.OutLen, Pkt->Type) == FFFTP_FAIL)
\r
3641 if(TermCodeConvAndSend(&tInfo, dSkt, Buf3, cInfo2.OutLen, Pkt->Type, &Canceled[Pkt->ThreadCount]) == FFFTP_FAIL)
\r
3642 Pkt->Abort = ABORT_ERROR;
\r
3643 cInfo2.Buf = Buf3;
\r
3644 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
3645 FlushRestData(&cInfo2);
\r
3646 if(TermCodeConvAndSend(&tInfo, dSkt, Buf3, cInfo2.OutLen, Pkt->Type, &Canceled[Pkt->ThreadCount]) == FFFTP_FAIL)
\r
3647 Pkt->Abort = ABORT_ERROR;
\r
3651 tInfo.BufSize = BUFSIZE+3;
\r
3652 FlushRestTermCodeConvData(&tInfo);
\r
3654 // if(SendData(dSkt, Buf2, tInfo.OutLen, 0, &Canceled) == FFFTP_FAIL)
\r
3655 if(SendData(dSkt, Buf2, tInfo.OutLen, 0, &Canceled[Pkt->ThreadCount]) == FFFTP_FAIL)
\r
3656 Pkt->Abort = ABORT_ERROR;
\r
3660 if(Pkt->hWndTrans != NULL)
\r
3662 KillTimer(Pkt->hWndTrans, TIMER_DISPLAY);
\r
3663 DispTransferStatus(Pkt->hWndTrans, YES, Pkt);
\r
3665 // TimeStart = time(NULL) - TimeStart + 1;
\r
3666 TimeStart[Pkt->ThreadCount] = time(NULL) - TimeStart[Pkt->ThreadCount] + 1;
\r
3668 CloseHandle(iFileHandle);
\r
3672 SetErrorMsg(MSGJPN112, Pkt->LocalFile);
\r
3673 SetTaskMsg(MSGJPN112, Pkt->LocalFile);
\r
3674 Pkt->Abort = ABORT_ERROR;
\r
3678 LastDataConnectionTime = time(NULL);
\r
3679 if(shutdown(dSkt, 1) != 0)
\r
3680 ReportWSError("shutdown", WSAGetLastError());
\r
3684 while(do_recv(dSkt, Buf, BUFSIZE, 0, &TimeOutErr, &Canceled) > 0)
\r
3689 // iRetCode = ReadReplyMessage(Pkt->ctrl_skt, Buf, 1024, &Canceled, TmpBuf);
\r
3690 iRetCode = ReadReplyMessage(Pkt->ctrl_skt, Buf, 1024, &Canceled[Pkt->ThreadCount], TmpBuf);
\r
3693 //DoPrintf("##UP REPLY : %s", Buf);
\r
3696 // if(iRetCode >= FTP_RETRY)
\r
3697 if((iRetCode/100) >= FTP_RETRY)
\r
3700 if(Pkt->Abort != ABORT_NONE)
\r
3707 /*----- バッファの内容を改行コード変換して送信 --------------------------------
\r
3710 * TERMCODECONVINFO *tInfo : 改行コード変換パケット
\r
3711 * SOCKET Skt : ソケット
\r
3712 * char *Data : データ
\r
3713 * int Size : データのサイズ
\r
3714 * int Ascii : モード (TYPE_xx)
\r
3718 *----------------------------------------------------------------------------*/
\r
3721 //static int TermCodeConvAndSend(TERMCODECONVINFO *tInfo, SOCKET Skt, char *Data, int Size, int Ascii)
\r
3722 static int TermCodeConvAndSend(TERMCODECONVINFO *tInfo, SOCKET Skt, char *Data, int Size, int Ascii, int *CancelCheckWork)
\r
3724 char Buf3[BUFSIZE*2];
\r
3728 Ret = FFFTP_SUCCESS;
\r
3730 // CR-LF以外の改行コードを変換しないモードはここへ追加
\r
3731 if(Ascii == TYPE_A)
\r
3733 tInfo->Str = Data;
\r
3734 tInfo->StrLen = Size;
\r
3735 tInfo->Buf = Buf3;
\r
3736 tInfo->BufSize = BUFSIZE*2;
\r
3739 Continue = ConvTermCodeToCRLF(tInfo);
\r
3741 // if((Ret = SendData(Skt, Buf3, tInfo->OutLen, 0, &Canceled)) == FFFTP_FAIL)
\r
3742 if((Ret = SendData(Skt, Buf3, tInfo->OutLen, 0, CancelCheckWork)) == FFFTP_FAIL)
\r
3745 while(Continue == YES);
\r
3749 // Ret = SendData(Skt, Data, Size, 0, &Canceled);
\r
3750 Ret = SendData(Skt, Data, Size, 0, CancelCheckWork);
\r
3756 /*----- アップロード終了/中止時のメッセージを表示 ----------------------------
\r
3759 * TRANSPACKET *Pkt : 転送ファイル情報
\r
3760 * int iRetCode : 応答コード
\r
3764 *----------------------------------------------------------------------------*/
\r
3766 static void DispUploadFinishMsg(TRANSPACKET *Pkt, int iRetCode)
\r
3769 ReleaseMutex(hListAccMutex);
\r
3770 if(ForceAbort == NO)
\r
3772 if((iRetCode/100) >= FTP_CONTINUE)
\r
3775 // if((Pkt->hWndTrans != NULL) && (TimeStart != 0))
\r
3776 // SetTaskMsg(MSGJPN113, TimeStart, Pkt->ExistSize/TimeStart);
\r
3777 if((Pkt->hWndTrans != NULL) && (TimeStart[Pkt->ThreadCount] != 0))
\r
3778 SetTaskMsg(MSGJPN113, TimeStart[Pkt->ThreadCount], Pkt->ExistSize/TimeStart[Pkt->ThreadCount]);
\r
3780 SetTaskMsg(MSGJPN114);
\r
3782 if(Pkt->Abort != ABORT_USER)
\r
3784 // 全て中止を選択後にダイアログが表示されるバグ対策
\r
3785 // if(DispUpDownErrDialog(uperr_dlg, Pkt->hWndTrans, Pkt->LocalFile) == NO)
\r
3787 // if(Canceled[Pkt->ThreadCount] == NO && ClearAll == NO && DispUpDownErrDialog(uperr_dlg, Pkt->hWndTrans, Pkt->LocalFile) == NO)
\r
3788 // ClearAll = YES;
\r
3789 if(Canceled[Pkt->ThreadCount] == NO && ClearAll == NO)
\r
3791 if(strncmp(Pkt->Cmd, "RETR", 4) == 0 || strncmp(Pkt->Cmd, "STOR", 4) == 0)
\r
3794 TransferErrorDisplay++;
\r
3795 if(TransferErrorNotify == YES && DispUpDownErrDialog(uperr_dlg, Pkt->hWndTrans, Pkt) == NO)
\r
3799 Pkt->Mode = TransferErrorMode;
\r
3800 AddTransFileList(Pkt);
\r
3803 TransferErrorDisplay--;
\r
3811 // if((Pkt->hWndTrans != NULL) && (TimeStart != 0))
\r
3812 // SetTaskMsg(MSGJPN115, TimeStart, Pkt->ExistSize/TimeStart);
\r
3813 if((Pkt->hWndTrans != NULL) && (TimeStart[Pkt->ThreadCount] != 0))
\r
3814 // "0 B/S"と表示されるバグを修正
\r
3815 // 原因は%dにあたる部分に64ビット値が渡されているため
\r
3816 // SetTaskMsg(MSGJPN115, TimeStart[Pkt->ThreadCount], Pkt->ExistSize/TimeStart[Pkt->ThreadCount]);
\r
3817 SetTaskMsg(MSGJPN115, (LONG)TimeStart[Pkt->ThreadCount], (LONG)(Pkt->ExistSize/TimeStart[Pkt->ThreadCount]));
\r
3819 SetTaskMsg(MSGJPN116);
\r
3826 /*----- アップロードのリジュームの準備を行う ----------------------------------
\r
3829 * TRANSPACKET *Pkt : 転送ファイル情報
\r
3830 * iont ProcMode : 処理モード(EXIST_xxx)
\r
3831 * LONGLONG Size : ホストにあるファイルのサイズ
\r
3832 * int *Mode : リジュームを行うかどうか (YES/NO)
\r
3838 * Pkt->ExistSizeのセットを行なう
\r
3839 *----------------------------------------------------------------------------*/
\r
3841 static int SetUploadResume(TRANSPACKET *Pkt, int ProcMode, LONGLONG Size, int *Mode)
\r
3843 Pkt->ExistSize = 0;
\r
3845 if(ProcMode == EXIST_RESUME)
\r
3847 if(Pkt->hWndTrans != NULL)
\r
3849 Pkt->ExistSize = Size;
\r
3857 /*----- 転送中ダイアログボックスのコールバック --------------------------------
\r
3860 * HWND hDlg : ウインドウハンドル
\r
3861 * UINT message : メッセージ番号
\r
3862 * WPARAM wParam : メッセージの WPARAM 引数
\r
3863 * LPARAM lParam : メッセージの LPARAM 引数
\r
3867 *----------------------------------------------------------------------------*/
\r
3869 static LRESULT CALLBACK TransDlgProc(HWND hDlg, UINT Msg, WPARAM wParam, LPARAM lParam)
\r
3875 // static TRANSPACKET *Pkt;
\r
3881 case WM_INITDIALOG :
\r
3882 GetWindowRect(hDlg, &RectDlg);
\r
3883 RectDlg.right -= RectDlg.left;
\r
3884 RectDlg.bottom -= RectDlg.top;
\r
3885 GetWindowRect(GetMainHwnd(), &RectPar);
\r
3887 ((RectPar.right + RectPar.left) / 2) - (RectDlg.right / 2),
\r
3888 ((RectPar.bottom + RectPar.top) / 2) - (RectDlg.bottom / 2),
\r
3893 hMenu = GetSystemMenu(hDlg, FALSE);
\r
3894 EnableMenuItem(hMenu, SC_CLOSE, MF_GRAYED);
\r
3898 switch(LOWORD(wParam))
\r
3900 case TRANS_STOP_NEXT :
\r
3904 case TRANS_STOP_ALL :
\r
3906 for(i = 0; i < MAX_DATA_CONNECTION; i++)
\r
3907 Canceled[i] = YES;
\r
3908 /* ここに break はない */
\r
3912 // if(!(Pkt = (TRANSPACKET*)GetWindowLong(hDlg, GWL_USERDATA)))
\r
3913 if(!(Pkt = (TRANSPACKET*)GetWindowLongPtr(hDlg, GWLP_USERDATA)))
\r
3915 Pkt->Abort = ABORT_USER;
\r
3916 // Canceled = YES;
\r
3917 Canceled[Pkt->ThreadCount] = YES;
\r
3923 if(wParam == TIMER_DISPLAY)
\r
3925 if(MoveToForeground == YES)
\r
3926 SetForegroundWindow(hDlg);
\r
3927 MoveToForeground = NO;
\r
3928 KillTimer(hDlg, TIMER_DISPLAY);
\r
3930 // if(!(Pkt = (TRANSPACKET*)GetWindowLong(hDlg, GWL_USERDATA)))
\r
3931 if(!(Pkt = (TRANSPACKET*)GetWindowLongPtr(hDlg, GWLP_USERDATA)))
\r
3933 DispTransferStatus(hDlg, NO, Pkt);
\r
3934 SetTimer(hDlg, TIMER_DISPLAY, DISPLAY_TIMING, NULL);
\r
3938 case WM_SET_PACKET :
\r
3939 // Pkt = (TRANSPACKET *)lParam;
\r
3941 // SetWindowLong(hDlg, GWL_USERDATA, (LONG)lParam);
\r
3942 SetWindowLongPtr(hDlg, GWLP_USERDATA, (LONG_PTR)lParam);
\r
3949 /*----- 転送ステータスを表示 --------------------------------------------------
\r
3952 * HWND hWnd : ウインドウハンドル
\r
3953 * int End : 転送が完了したかどうか (YES/NO)
\r
3957 *----------------------------------------------------------------------------*/
\r
3959 static void DispTransferStatus(HWND hWnd, int End, TRANSPACKET *Pkt)
\r
3973 SendMessage(hWnd, WM_GETTEXT, 79, (LPARAM)Str);
\r
3974 if((Pos = strchr(Str, ')')) != NULL)
\r
3978 sprintf(Tmp, "(%d)%s", AskTransferFileNum(), Pos);
\r
3979 SendMessage(hWnd, WM_SETTEXT, 0, (LPARAM)Tmp);
\r
3981 if(Pkt->Abort == ABORT_NONE)
\r
3986 // TotalLap = time(NULL) - TimeStart + 1;
\r
3987 TotalLap = time(NULL) - TimeStart[Pkt->ThreadCount] + 1;
\r
3992 // Bps = AllTransSizeNow / TotalLap;
\r
3993 Bps = AllTransSizeNow[Pkt->ThreadCount] / TotalLap;
\r
3994 Transed = Pkt->Size - Pkt->ExistSize;
\r
3996 if(Pkt->Size <= 0)
\r
3997 sprintf(Tmp, "%d ", Pkt->ExistSize);
\r
3998 else if(Pkt->Size < 1024)
\r
3999 sprintf(Tmp, "%s / %s ", MakeNumString(Pkt->ExistSize, Num1, TRUE), MakeNumString(Pkt->Size, Num2, TRUE));
\r
4001 sprintf(Tmp, "%sk / %sk ", MakeNumString(Pkt->ExistSize/1024, Num1, TRUE), MakeNumString(Pkt->Size/1024, Num2, TRUE));
\r
4005 sprintf(Tmp, "( 0 B/S )");
\r
4006 else if(Bps < 1000)
\r
4007 sprintf(Tmp, "( %s B/S )", MakeNumString(Bps, Num1, TRUE));
\r
4009 sprintf(Tmp, "( %s.%02d KB/S )", MakeNumString(Bps/1000, Num1, TRUE), (int)((Bps%1000)/10));
\r
4012 if((Bps > 0) && (Pkt->Size > 0) && (Transed >= 0))
\r
4014 sprintf(Tmp, " %d:%02d", (int)((Transed/Bps)/60), (int)((Transed/Bps)%60));
\r
4018 strcat(Str, " ??:??");
\r
4021 strcpy(Str, MSGJPN117);
\r
4024 strcpy(Str, MSGJPN118);
\r
4026 SendDlgItemMessage(hWnd, TRANS_STATUS, WM_SETTEXT, 0, (LPARAM)Str);
\r
4028 if(Pkt->Size <= 0)
\r
4030 else if(Pkt->Size < 1024*1024)
\r
4031 Per = (int)(Pkt->ExistSize * 100 / Pkt->Size);
\r
4033 Per = (int)((Pkt->ExistSize/1024) * 100 / (Pkt->Size/1024));
\r
4034 SendDlgItemMessage(hWnd, TRANS_TIME_BAR, PBM_SETPOS, Per, 0);
\r
4040 /*----- 転送するファイルの情報を表示 ------------------------------------------
\r
4043 * TRANSPACKET *Pkt : 転送ファイル情報
\r
4044 * char *Title : ウインドウのタイトル
\r
4045 * int SkipButton : 「このファイルを中止」ボタンの有無 (TRUE/FALSE)
\r
4046 * int Info : ファイル情報を表示するかどうか (YES/NO)
\r
4050 *----------------------------------------------------------------------------*/
\r
4052 static void DispTransFileInfo(TRANSPACKET *Pkt, char *Title, int SkipButton, int Info)
\r
4056 if(Pkt->hWndTrans != NULL)
\r
4058 EnableWindow(GetDlgItem(Pkt->hWndTrans, IDCANCEL), SkipButton);
\r
4060 sprintf(Tmp, "(%d)%s", AskTransferFileNum(), Title);
\r
4061 SendMessage(Pkt->hWndTrans, WM_SETTEXT, 0, (LPARAM)Tmp);
\r
4062 SendDlgItemMessage(Pkt->hWndTrans, TRANS_STATUS, WM_SETTEXT, 0, (LPARAM)"");
\r
4064 SendDlgItemMessage(Pkt->hWndTrans, TRANS_TIME_BAR, PBM_SETRANGE, 0, MAKELPARAM(0, 100));
\r
4065 SendDlgItemMessage(Pkt->hWndTrans, TRANS_TIME_BAR, PBM_SETSTEP, 1, 0);
\r
4066 SendDlgItemMessage(Pkt->hWndTrans, TRANS_TIME_BAR, PBM_SETPOS, 0, 0);
\r
4070 DispStaticText(GetDlgItem(Pkt->hWndTrans, TRANS_REMOTE), Pkt->RemoteFile);
\r
4071 DispStaticText(GetDlgItem(Pkt->hWndTrans, TRANS_LOCAL), Pkt->LocalFile);
\r
4073 if(Pkt->Type == TYPE_I)
\r
4074 SendDlgItemMessage(Pkt->hWndTrans, TRANS_MODE, WM_SETTEXT, 0, (LPARAM)MSGJPN119);
\r
4075 else if(Pkt->Type == TYPE_A)
\r
4076 SendDlgItemMessage(Pkt->hWndTrans, TRANS_MODE, WM_SETTEXT, 0, (LPARAM)MSGJPN120);
\r
4079 if(Pkt->KanjiCode == KANJI_NOCNV)
\r
4080 SendDlgItemMessage(Pkt->hWndTrans, TRANS_KANJI, WM_SETTEXT, 0, (LPARAM)MSGJPN121);
\r
4081 else if(Pkt->KanjiCode == KANJI_SJIS)
\r
4082 SendDlgItemMessage(Pkt->hWndTrans, TRANS_KANJI, WM_SETTEXT, 0, (LPARAM)MSGJPN305);
\r
4083 else if(Pkt->KanjiCode == KANJI_JIS)
\r
4084 SendDlgItemMessage(Pkt->hWndTrans, TRANS_KANJI, WM_SETTEXT, 0, (LPARAM)MSGJPN122);
\r
4085 else if(Pkt->KanjiCode == KANJI_EUC)
\r
4086 SendDlgItemMessage(Pkt->hWndTrans, TRANS_KANJI, WM_SETTEXT, 0, (LPARAM)MSGJPN123);
\r
4087 else if(Pkt->KanjiCode == KANJI_UTF8N)
\r
4088 SendDlgItemMessage(Pkt->hWndTrans, TRANS_KANJI, WM_SETTEXT, 0, (LPARAM)MSGJPN306);
\r
4089 else if(Pkt->KanjiCode == KANJI_UTF8BOM)
\r
4090 SendDlgItemMessage(Pkt->hWndTrans, TRANS_KANJI, WM_SETTEXT, 0, (LPARAM)MSGJPN329);
\r
4094 SendDlgItemMessage(Pkt->hWndTrans, TRANS_REMOTE, WM_SETTEXT, 0, (LPARAM)"");
\r
4095 SendDlgItemMessage(Pkt->hWndTrans, TRANS_LOCAL, WM_SETTEXT, 0, (LPARAM)"");
\r
4096 SendDlgItemMessage(Pkt->hWndTrans, TRANS_MODE, WM_SETTEXT, 0, (LPARAM)"");
\r
4097 SendDlgItemMessage(Pkt->hWndTrans, TRANS_KANJI, WM_SETTEXT, 0, (LPARAM)"");
\r
4104 /*----- PASVコマンドの戻り値からアドレスとポート番号を抽出 --------------------
\r
4107 * char *Str : PASVコマンドのリプライ
\r
4108 * char *Adrs : アドレスのコピー先 ("www.xxx.yyy.zzz")
\r
4109 * int *Port : ポート番号をセットするワーク
\r
4110 * int Max : アドレス文字列の最大長
\r
4114 * FFFTP_SUCCESS/FFFTP_FAIL
\r
4115 *----------------------------------------------------------------------------*/
\r
4117 static int GetAdrsAndPort(SOCKET Skt, char *Str, char *Adrs, int *Port, int Max)
\r
4120 Result = FFFTP_FAIL;
\r
4121 switch(AskCurNetType())
\r
4124 Result = GetAdrsAndPortIPv4(Skt, Str, Adrs, Port, Max);
\r
4127 Result = GetAdrsAndPortIPv6(Skt, Str, Adrs, Port, Max);
\r
4135 //static int GetAdrsAndPort(char *Str, char *Adrs, int *Port, int Max)
\r
4136 static int GetAdrsAndPortIPv4(SOCKET Skt, char *Str, char *Adrs, int *Port, int Max)
\r
4140 // コンマではなくドットを返すホストがあるため
\r
4144 struct sockaddr_in SockAddr;
\r
4148 Pos = strchr(Str, '(');
\r
4152 Btm = strchr(Pos, ',');
\r
4153 // コンマではなくドットを返すホストがあるため
\r
4155 Btm = strchr(Pos, '.');
\r
4159 // コンマではなくドットを返すホストがあるため
\r
4160 // Btm = strchr(Btm, ',');
\r
4162 Btm = strchr(OldBtm, ',');
\r
4164 Btm = strchr(OldBtm, '.');
\r
4168 // コンマではなくドットを返すホストがあるため
\r
4169 // Btm = strchr(Btm, ',');
\r
4171 Btm = strchr(OldBtm, ',');
\r
4173 Btm = strchr(OldBtm, '.');
\r
4177 // コンマではなくドットを返すホストがあるため
\r
4178 // Btm = strchr(Btm, ',');
\r
4180 Btm = strchr(OldBtm, ',');
\r
4182 Btm = strchr(OldBtm, '.');
\r
4185 if((Btm - Pos) <= Max)
\r
4188 // strncpy(Adrs, Pos, Btm - Pos);
\r
4189 // *(Adrs + (Btm - Pos)) = NUL;
\r
4190 // ReplaceAll(Adrs, ',', '.');
\r
4191 if(AskNoPasvAdrs() == NO)
\r
4193 strncpy(Adrs, Pos, Btm - Pos);
\r
4194 *(Adrs + (Btm - Pos)) = NUL;
\r
4195 ReplaceAll(Adrs, ',', '.');
\r
4199 if(GetAsyncTableDataIPv4(Skt, &SockAddr, NULL) == YES)
\r
4200 AddressToStringIPv4(Adrs, &SockAddr.sin_addr);
\r
4204 Btm = strchr(Pos, ',');
\r
4205 // コンマではなくドットを返すホストがあるため
\r
4207 Btm = strchr(Pos, '.');
\r
4211 *Port = (atoi(Pos) * 0x100) + atoi(Btm);
\r
4212 Sts = FFFTP_SUCCESS;
\r
4225 static int GetAdrsAndPortIPv6(SOCKET Skt, char *Str, char *Adrs, int *Port, int Max)
\r
4231 struct sockaddr_in6 SockAddr;
\r
4235 Btm = strchr(Str, '(');
\r
4239 Btm = strchr(Btm, '|');
\r
4243 Btm = strchr(Pos, '|');
\r
4247 Btm = strchr(Pos, '|');
\r
4250 if((Btm - Pos) <= Max)
\r
4253 if(AskNoPasvAdrs() == NO && (Btm - Pos) > 0)
\r
4255 strncpy(Adrs, Pos, Btm - Pos);
\r
4256 *(Adrs + (Btm - Pos)) = NUL;
\r
4260 // i = sizeof(SockAddr);
\r
4261 // if(getpeername(Skt, (struct sockaddr*)&SockAddr, &i) != SOCKET_ERROR)
\r
4262 if(GetAsyncTableDataIPv6(Skt, &SockAddr, NULL) == YES)
\r
4263 AddressToStringIPv6(Adrs, &SockAddr.sin6_addr);
\r
4267 Btm = strchr(Pos, '|');
\r
4271 *Port = atoi(Pos);
\r
4272 Btm = strchr(Btm, ')');
\r
4274 Sts = FFFTP_SUCCESS;
\r
4285 /*----- Windowsのスペシャルデバイスかどうかを返す -----------------------------
\r
4288 * char *Fname : ファイル名
\r
4291 * int ステータス (YES/NO)
\r
4292 *----------------------------------------------------------------------------*/
\r
4294 static int IsSpecialDevice(char *Fname)
\r
4300 // if((_stricmp(Fname, "CON") == 0) ||
\r
4301 // (_stricmp(Fname, "PRN") == 0) ||
\r
4302 // (_stricmp(Fname, "AUX") == 0) ||
\r
4303 // (_strnicmp(Fname, "CON.", 4) == 0) ||
\r
4304 // (_strnicmp(Fname, "PRN.", 4) == 0) ||
\r
4305 // (_strnicmp(Fname, "AUX.", 4) == 0))
\r
4309 if(_strnicmp(Fname, "AUX", 3) == 0|| _strnicmp(Fname, "CON", 3) == 0 || _strnicmp(Fname, "NUL", 3) == 0 || _strnicmp(Fname, "PRN", 3) == 0)
\r
4311 if(*(Fname + 3) == '\0' || *(Fname + 3) == '.')
\r
4314 else if(_strnicmp(Fname, "COM", 3) == 0 || _strnicmp(Fname, "LPT", 3) == 0)
\r
4316 if(isdigit(*(Fname + 3)) != 0)
\r
4318 if(*(Fname + 4) == '\0' || *(Fname + 4) == '.')
\r
4326 /*----- ミラーリングでのファイル削除確認 --------------------------------------
\r
4331 * TRANSPACKET *Pkt
\r
4335 *----------------------------------------------------------------------------*/
\r
4337 static int MirrorDelNotify(int Cur, int Notify, TRANSPACKET *Pkt)
\r
4339 MIRRORDELETEINFO DelInfo;
\r
4342 if(((Cur == WIN_LOCAL) && (MirDownDelNotify == NO)) ||
\r
4343 ((Cur == WIN_REMOTE) && (MirUpDelNotify == NO)))
\r
4348 if(Notify != YES_ALL)
\r
4350 DelInfo.Cur = Cur;
\r
4351 DelInfo.Pkt = Pkt;
\r
4352 hWnd = Pkt->hWndTrans;
\r
4354 hWnd = GetMainHwnd();
\r
4355 Notify = DialogBoxParam(GetFtpInst(), MAKEINTRESOURCE(delete_dlg), hWnd, MirrorDeleteDialogCallBack, (LPARAM)&DelInfo);
\r
4361 /*----- ミラーリングでのファイル削除ダイアログのコールバック ------------------
\r
4364 * HWND hDlg : ウインドウハンドル
\r
4365 * UINT message : メッセージ番号
\r
4366 * WPARAM wParam : メッセージの WPARAM 引数
\r
4367 * LPARAM lParam : メッセージの LPARAM 引数
\r
4371 *----------------------------------------------------------------------------*/
\r
4374 //static BOOL CALLBACK MirrorDeleteDialogCallBack(HWND hDlg, UINT iMessage, WPARAM wParam, LPARAM lParam)
\r
4375 static INT_PTR CALLBACK MirrorDeleteDialogCallBack(HWND hDlg, UINT iMessage, WPARAM wParam, LPARAM lParam)
\r
4377 static MIRRORDELETEINFO *DelInfo;
\r
4380 case WM_INITDIALOG :
\r
4381 DelInfo = (MIRRORDELETEINFO *)lParam;
\r
4383 if(DelInfo->Cur == WIN_LOCAL)
\r
4385 SendMessage(hDlg, WM_SETTEXT, 0, (LPARAM)MSGJPN124);
\r
4386 SendDlgItemMessage(hDlg, DELETE_TEXT, WM_SETTEXT, 0, (LPARAM)DelInfo->Pkt->LocalFile);
\r
4390 SendMessage(hDlg, WM_SETTEXT, 0, (LPARAM)MSGJPN125);
\r
4391 SendDlgItemMessage(hDlg, DELETE_TEXT, WM_SETTEXT, 0, (LPARAM)DelInfo->Pkt->RemoteFile);
\r
4396 switch(GET_WM_COMMAND_ID(wParam, lParam))
\r
4399 EndDialog(hDlg, YES);
\r
4403 EndDialog(hDlg, NO);
\r
4407 EndDialog(hDlg, YES_ALL);
\r
4412 EndDialog(hDlg, NO_ALL);
\r
4424 static void SetErrorMsg(char *fmt, ...)
\r
4429 // if(strlen(ErrMsg) == 0)
\r
4430 if(strlen(GetErrMsg()) == 0)
\r
4432 va_start(Args, fmt);
\r
4434 // wvsprintf(ErrMsg, fmt, Args);
\r
4435 wvsprintf(GetErrMsg(), fmt, Args);
\r
4446 /*----- ダウンロード時の不正なパスをチェック ----------------------------------
\r
4449 * TRANSPACKET *packet : ダウンロード情報
\r
4452 * int YES=不正なパス/NO=問題ないパス
\r
4453 *----------------------------------------------------------------------------*/
\r
4454 int CheckPathViolation(TRANSPACKET *packet)
\r
4459 if((strncmp(packet->RemoteFile, "..\\", 3) == 0) ||
\r
4460 (strncmp(packet->RemoteFile, "../", 3) == 0) ||
\r
4461 (strstr(packet->RemoteFile, "\\..\\") != NULL) ||
\r
4462 (strstr(packet->RemoteFile, "/../") != NULL))
\r
4464 msg = malloc(strlen(MSGJPN297) + strlen(packet->RemoteFile) + 1);
\r
4467 sprintf(msg, MSGJPN297, packet->RemoteFile);
\r
4468 MessageBox(GetMainHwnd(), msg, MSGJPN086, MB_OK);
\r
4478 static char* GetErrMsg()
\r
4484 WaitForSingleObject(hErrMsgMutex, INFINITE);
\r
4485 ThreadId = GetCurrentThreadId();
\r
4487 while(i < MAX_DATA_CONNECTION + 1)
\r
4489 if(ErrMsgThreadId[i] == ThreadId)
\r
4499 while(i < MAX_DATA_CONNECTION + 1)
\r
4501 if(ErrMsgThreadId[i] == 0)
\r
4503 ErrMsgThreadId[i] = ThreadId;
\r
4510 ReleaseMutex(hErrMsgMutex);
\r
4515 LONGLONG AskTransferSizeLeft(void)
\r
4517 return(TransferSizeLeft);
\r
4520 LONGLONG AskTransferSizeTotal(void)
\r
4522 return(TransferSizeTotal);
\r
4525 int AskTransferErrorDisplay(void)
\r
4527 return(TransferErrorDisplay);
\r