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
87 static int DownLoadNonPassive(TRANSPACKET *Pkt, int *CancelCheckWork);
\r
88 static int DownLoadPassive(TRANSPACKET *Pkt, int *CancelCheckWork);
\r
89 static int DownLoadFile(TRANSPACKET *Pkt, SOCKET dSkt, int CreateMode, int *CancelCheckWork);
\r
90 static void DispDownloadFinishMsg(TRANSPACKET *Pkt, int iRetCode);
\r
92 //static int DispUpDownErrDialog(int ResID, HWND hWnd, char *Fname);
\r
93 static int DispUpDownErrDialog(int ResID, HWND hWnd, TRANSPACKET *Pkt);
\r
95 //static BOOL CALLBACK UpDownErrorDialogProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam);
\r
96 static INT_PTR CALLBACK UpDownErrorDialogProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam);
\r
97 static int SetDownloadResume(TRANSPACKET *Pkt, int ProcMode, LONGLONG Size, int *Mode, int *CancelCheckWork);
\r
99 //static BOOL CALLBACK NoResumeWndProc(HWND hDlg, UINT iMessage, WPARAM wParam, LPARAM lParam);
\r
100 static INT_PTR CALLBACK NoResumeWndProc(HWND hDlg, UINT iMessage, WPARAM wParam, LPARAM lParam);
\r
101 static int DoUpLoad(SOCKET cSkt, TRANSPACKET *Pkt);
\r
102 static int UpLoadNonPassive(TRANSPACKET *Pkt);
\r
103 static int UpLoadPassive(TRANSPACKET *Pkt);
\r
104 static int UpLoadFile(TRANSPACKET *Pkt, SOCKET dSkt);
\r
106 //static int TermCodeConvAndSend(TERMCODECONVINFO *tInfo, SOCKET Skt, char *Data, int Size, int Ascii);
\r
107 static int TermCodeConvAndSend(TERMCODECONVINFO *tInfo, SOCKET Skt, char *Data, int Size, int Ascii, int *CancelCheckWork);
\r
108 static void DispUploadFinishMsg(TRANSPACKET *Pkt, int iRetCode);
\r
109 static int SetUploadResume(TRANSPACKET *Pkt, int ProcMode, LONGLONG Size, int *Mode);
\r
110 static LRESULT CALLBACK TransDlgProc(HWND hDlg, UINT Msg, WPARAM wParam, LPARAM lParam);
\r
111 static void DispTransferStatus(HWND hWnd, int End, TRANSPACKET *Pkt);
\r
112 static void DispTransFileInfo(TRANSPACKET *Pkt, char *Title, int SkipButton, int Info);
\r
114 //static int GetAdrsAndPort(char *Str, char *Adrs, int *Port, int Max);
\r
115 static int GetAdrsAndPort(SOCKET Skt, char *Str, char *Adrs, int *Port, int Max);
\r
116 static int GetAdrsAndPortIPv4(SOCKET Skt, char *Str, char *Adrs, int *Port, int Max);
\r
117 static int GetAdrsAndPortIPv6(SOCKET Skt, char *Str, char *Adrs, int *Port, int Max);
\r
118 static int IsSpecialDevice(char *Fname);
\r
119 static int MirrorDelNotify(int Cur, int Notify, TRANSPACKET *Pkt);
\r
121 //static BOOL CALLBACK MirrorDeleteDialogCallBack(HWND hDlg, UINT iMessage, WPARAM wParam, LPARAM lParam);
\r
122 static INT_PTR CALLBACK MirrorDeleteDialogCallBack(HWND hDlg, UINT iMessage, WPARAM wParam, LPARAM lParam);
\r
123 static void SetErrorMsg(char *fmt, ...);
\r
125 static char* GetErrMsg();
\r
127 /*===== ローカルなワーク =====*/
\r
130 //static HANDLE hTransferThread;
\r
131 static HANDLE hTransferThread[MAX_DATA_CONNECTION];
\r
132 static int fTransferThreadExit = FALSE;
\r
134 static HANDLE hRunMutex; /* 転送スレッド実行ミューテックス */
\r
135 static HANDLE hListAccMutex; /* 転送ファイルアクセス用ミューテックス */
\r
137 static int TransFiles = 0; /* 転送待ちファイル数 */
\r
138 static TRANSPACKET *TransPacketBase = NULL; /* 転送ファイルリスト */
\r
140 static TRANSPACKET *NextTransPacketBase = NULL;
\r
143 //static int Canceled; /* 中止フラグ YES/NO */
\r
144 static int Canceled[MAX_DATA_CONNECTION]; /* 中止フラグ YES/NO */
\r
145 static int ClearAll; /* 全て中止フラグ YES/NO */
\r
147 static int ForceAbort; /* 転送中止フラグ */
\r
148 /* このフラグはスレッドを終了させるときに使う */
\r
151 //static LONGLONG AllTransSizeNow; /* 今回の転送で転送したサイズ */
\r
152 //static time_t TimeStart; /* 転送開始時間 */
\r
153 static LONGLONG AllTransSizeNow[MAX_DATA_CONNECTION]; /* 今回の転送で転送したサイズ */
\r
154 static time_t TimeStart[MAX_DATA_CONNECTION]; /* 転送開始時間 */
\r
156 static int KeepDlg = NO; /* 転送中ダイアログを消さないかどうか (YES/NO) */
\r
157 static int MoveToForeground = NO; /* ウインドウを前面に移動するかどうか (YES/NO) */
\r
160 //static char CurDir[FMAX_PATH+1] = { "" };
\r
161 static char CurDir[MAX_DATA_CONNECTION][FMAX_PATH+1];
\r
163 //static char ErrMsg[ERR_MSG_LEN+7];
\r
164 static char ErrMsg[MAX_DATA_CONNECTION+1][ERR_MSG_LEN+7];
\r
165 static DWORD ErrMsgThreadId[MAX_DATA_CONNECTION+1];
\r
166 static HANDLE hErrMsgMutex;
\r
169 static int WaitForMainThread = NO;
\r
171 static int TransferErrorMode = EXIST_OVW;
\r
172 static int TransferErrorNotify = NO;
\r
174 /*===== 外部参照 =====*/
\r
177 extern int SaveTimeStamp;
\r
179 // extern int TimeOut;
\r
180 extern int FwallType;
\r
181 extern int MirUpDelNotify;
\r
182 extern int MirDownDelNotify;
\r
183 extern int FolderAttr;
\r
184 extern int FolderAttrNum;
\r
187 /*----- ファイル転送スレッドを起動する ----------------------------------------
\r
194 *----------------------------------------------------------------------------*/
\r
196 int MakeTransferThread(void)
\r
201 hListAccMutex = CreateMutex( NULL, FALSE, NULL );
\r
202 hRunMutex = CreateMutex( NULL, TRUE, NULL );
\r
204 hErrMsgMutex = CreateMutex( NULL, FALSE, NULL );
\r
209 fTransferThreadExit = FALSE;
\r
211 // hTransferThread = (HANDLE)_beginthreadex(NULL, 0, TransferThread, 0, 0, &dwID);
\r
212 // if (hTransferThread == NULL)
\r
213 // return(FFFTP_FAIL); /* XXX */
\r
214 for(i = 0; i < MAX_DATA_CONNECTION; i++)
\r
216 hTransferThread[i] = (HANDLE)_beginthreadex(NULL, 0, TransferThread, (void*)i, 0, &dwID);
\r
217 if(hTransferThread[i] == NULL)
\r
221 return(FFFTP_SUCCESS);
\r
225 /*----- ファイル転送スレッドを終了する ----------------------------------------
\r
232 *----------------------------------------------------------------------------*/
\r
234 void CloseTransferThread(void)
\r
239 for(i = 0; i < MAX_DATA_CONNECTION; i++)
\r
244 fTransferThreadExit = TRUE;
\r
246 // while(WaitForSingleObject(hTransferThread, 10) == WAIT_TIMEOUT)
\r
248 // BackgrndMessageProc();
\r
251 // CloseHandle(hTransferThread);
\r
252 for(i = 0; i < MAX_DATA_CONNECTION; i++)
\r
254 while(WaitForSingleObject(hTransferThread[i], 10) == WAIT_TIMEOUT)
\r
256 BackgrndMessageProc();
\r
259 CloseHandle(hTransferThread[i]);
\r
262 ReleaseMutex( hRunMutex );
\r
264 CloseHandle( hListAccMutex );
\r
265 CloseHandle( hRunMutex );
\r
267 CloseHandle( hErrMsgMutex );
\r
272 /*----- 転送するファイル情報をリストに追加する --------------------------------
\r
275 * TRANSPACKET *Pkt : 転送ファイル情報
\r
276 * TRANSPACKET **Base : リストの先頭
\r
280 * FFFTP_SUCCESS/FFFTP_FAIL
\r
281 *----------------------------------------------------------------------------*/
\r
283 int AddTmpTransFileList(TRANSPACKET *Pkt, TRANSPACKET **Base)
\r
290 if((Pos = malloc(sizeof(TRANSPACKET))) != NULL)
\r
292 memcpy(Pos, Pkt, sizeof(TRANSPACKET));
\r
300 while(Prev->Next != NULL)
\r
304 Sts = FFFTP_SUCCESS;
\r
310 /*----- 転送するファイル情報リストをクリアする --------------------------------
\r
313 * TRANSPACKET **Base : リストの先頭
\r
317 *----------------------------------------------------------------------------*/
\r
319 void EraseTmpTransFileList(TRANSPACKET **Base)
\r
336 /*----- 転送するファイル情報リストから1つの情報を取り除く --------------------
\r
339 * TRANSPACKET *Pkt : 転送ファイル情報
\r
340 * TRANSPACKET **Base : リストの先頭
\r
344 * FFFTP_SUCCESS/FFFTP_FAIL
\r
345 *----------------------------------------------------------------------------*/
\r
347 int RemoveTmpTransFileListItem(TRANSPACKET **Base, int Num)
\r
359 Sts = FFFTP_SUCCESS;
\r
369 Prev->Next = Pos->Next;
\r
371 Sts = FFFTP_SUCCESS;
\r
380 /*----- 転送するファイル情報を転送ファイルリストに登録する --------------------
\r
383 * TRANSPACKET *Pkt : 転送ファイル情報
\r
387 *----------------------------------------------------------------------------*/
\r
389 void AddTransFileList(TRANSPACKET *Pkt)
\r
394 DispTransPacket(Pkt);
\r
397 // WaitForSingleObject(hListAccMutex, INFINITE);
\r
398 while(WaitForSingleObject(hListAccMutex, 0) == WAIT_TIMEOUT)
\r
400 WaitForMainThread = YES;
\r
401 BackgrndMessageProc();
\r
406 Pos = TransPacketBase;
\r
409 while(Pos->Next != NULL)
\r
412 if(AddTmpTransFileList(Pkt, &TransPacketBase) == FFFTP_SUCCESS)
\r
414 if((strncmp(Pkt->Cmd, "RETR", 4) == 0) ||
\r
415 (strncmp(Pkt->Cmd, "STOR", 4) == 0))
\r
418 PostMessage(GetMainHwnd(), WM_CHANGE_COND, 0, 0);
\r
422 if(NextTransPacketBase == NULL)
\r
425 NextTransPacketBase = Pos->Next;
\r
427 NextTransPacketBase = TransPacketBase;
\r
429 ReleaseMutex(hListAccMutex);
\r
431 WaitForMainThread = NO;
\r
437 /*----- 転送ファイル情報を転送ファイルリストに追加する ------------------------
\r
440 * TRANSPACKET *Pkt : 転送ファイル情報
\r
441 * TRANSPACKET **Base : リストの先頭
\r
448 *----------------------------------------------------------------------------*/
\r
450 void AppendTransFileList(TRANSPACKET *Pkt)
\r
455 // WaitForSingleObject(hListAccMutex, INFINITE);
\r
456 while(WaitForSingleObject(hListAccMutex, 0) == WAIT_TIMEOUT)
\r
458 WaitForMainThread = YES;
\r
459 BackgrndMessageProc();
\r
463 if(TransPacketBase == NULL)
\r
464 TransPacketBase = Pkt;
\r
467 Pos = TransPacketBase;
\r
468 while(Pos->Next != NULL)
\r
473 if(NextTransPacketBase == NULL)
\r
474 NextTransPacketBase = Pkt;
\r
478 DispTransPacket(Pkt);
\r
480 if((strncmp(Pkt->Cmd, "RETR", 4) == 0) ||
\r
481 (strncmp(Pkt->Cmd, "STOR", 4) == 0))
\r
484 PostMessage(GetMainHwnd(), WM_CHANGE_COND, 0, 0);
\r
489 ReleaseMutex(hListAccMutex);
\r
491 WaitForMainThread = NO;
\r
496 /*----- 転送ファイル情報を表示する --------------------------------------------
\r
499 * TRANSPACKET *Pkt : 転送ファイル情報
\r
503 *----------------------------------------------------------------------------*/
\r
505 static void DispTransPacket(TRANSPACKET *Pkt)
\r
507 if((strncmp(Pkt->Cmd, "RETR", 4) == 0) || (strncmp(Pkt->Cmd, "STOR", 4) == 0))
\r
508 DoPrintf("TransList Cmd=%s : %s : %s", Pkt->Cmd, Pkt->RemoteFile, Pkt->LocalFile);
\r
509 else if(strncmp(Pkt->Cmd, "R-", 2) == 0)
\r
510 DoPrintf("TransList Cmd=%s : %s", Pkt->Cmd, Pkt->RemoteFile);
\r
511 else if(strncmp(Pkt->Cmd, "L-", 2) == 0)
\r
512 DoPrintf("TransList Cmd=%s : %s", Pkt->Cmd, Pkt->LocalFile);
\r
513 else if(strncmp(Pkt->Cmd, "MKD", 3) == 0)
\r
515 if(strlen(Pkt->LocalFile) > 0)
\r
516 DoPrintf("TransList Cmd=%s : %s", Pkt->Cmd, Pkt->LocalFile);
\r
518 DoPrintf("TransList Cmd=%s : %s", Pkt->Cmd, Pkt->RemoteFile);
\r
521 DoPrintf("TransList Cmd=%s", Pkt->Cmd);
\r
526 /*----- 転送ファイルリストをクリアする ----------------------------------------
\r
533 *----------------------------------------------------------------------------*/
\r
535 static void EraseTransFileList(void)
\r
539 TRANSPACKET *NotDel;
\r
540 // TRANSPACKET Pkt;
\r
545 // WaitForSingleObject(hListAccMutex, INFINITE);
\r
546 while(WaitForSingleObject(hListAccMutex, 0) == WAIT_TIMEOUT)
\r
548 WaitForMainThread = YES;
\r
549 BackgrndMessageProc();
\r
552 New = TransPacketBase;
\r
555 /* 最後の"BACKCUR"は必要なので消さない */
\r
556 if(strcmp(New->Cmd, "BACKCUR") == 0)
\r
561 strcpy(NotDel->Cmd, "");
\r
565 // NotDel->Next = NULL;
\r
572 strcpy(New->Cmd, "");
\r
576 TransPacketBase = NotDel;
\r
578 NextTransPacketBase = NotDel;
\r
580 PostMessage(GetMainHwnd(), WM_CHANGE_COND, 0, 0);
\r
581 ReleaseMutex(hListAccMutex);
\r
583 WaitForMainThread = NO;
\r
586 // strcpy(Pkt.Cmd, "GOQUIT");
\r
587 // AddTransFileList(&Pkt);
\r
592 /*----- 転送中ダイアログを消さないようにするかどうかを設定 --------------------
\r
595 * int Sw : 転送中ダイアログを消さないかどうか (YES/NO)
\r
599 *----------------------------------------------------------------------------*/
\r
601 void KeepTransferDialog(int Sw)
\r
608 /*----- 現在転送中かどうかを返す ----------------------------------------------
\r
614 * int ステータス (YES/NO=転送中ではない)
\r
615 *----------------------------------------------------------------------------*/
\r
617 int AskTransferNow(void)
\r
619 return(TransPacketBase != NULL ? YES : NO);
\r
623 /*----- 転送するファイルの数を返す --------------------------------------------
\r
630 *----------------------------------------------------------------------------*/
\r
632 int AskTransferFileNum(void)
\r
634 return(TransFiles);
\r
638 /*----- 転送中ウインドウを前面に出す ------------------------------------------
\r
645 *----------------------------------------------------------------------------*/
\r
647 void GoForwardTransWindow(void)
\r
649 MoveToForeground = YES;
\r
654 /*----- 転送ソケットのカレントディレクトリ情報を初期化 ------------------------
\r
661 *----------------------------------------------------------------------------*/
\r
663 void InitTransCurDir(void)
\r
666 // strcpy(CurDir, "");
\r
668 for(i = 0; i < MAX_DATA_CONNECTION; i++)
\r
669 strcpy(CurDir[i], "");
\r
674 /*----- ファイル転送スレッドのメインループ ------------------------------------
\r
677 * void *Dummy : 使わない
\r
681 *----------------------------------------------------------------------------*/
\r
683 static ULONG WINAPI TransferThread(void *Dummy)
\r
687 char Tmp[FMAX_PATH+1];
\r
710 ThreadCount = (int)Dummy;
\r
711 CmdSkt = INVALID_SOCKET;
\r
712 NewCmdSkt = INVALID_SOCKET;
\r
713 TrnSkt = INVALID_SOCKET;
\r
714 SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_LOWEST);
\r
716 while((TransPacketBase != NULL) ||
\r
717 (WaitForSingleObject(hRunMutex, 200) == WAIT_TIMEOUT))
\r
719 if(fTransferThreadExit == TRUE)
\r
722 if(WaitForMainThread == YES)
\r
724 BackgrndMessageProc();
\r
729 // WaitForSingleObject(hListAccMutex, INFINITE);
\r
730 while(WaitForSingleObject(hListAccMutex, 0) == WAIT_TIMEOUT)
\r
732 BackgrndMessageProc();
\r
735 // memset(ErrMsg, NUL, ERR_MSG_LEN+7);
\r
736 memset(GetErrMsg(), NUL, ERR_MSG_LEN+7);
\r
739 Canceled[ThreadCount] = NO;
\r
741 while(TransPacketBase != NULL && strcmp(TransPacketBase->Cmd, "") == 0)
\r
743 Pos = TransPacketBase;
\r
744 TransPacketBase = TransPacketBase->Next;
\r
746 if(TransPacketBase == NULL)
\r
749 NewCmdSkt = AskCmdCtrlSkt();
\r
750 if(AskReuseCmdSkt() == YES && ThreadCount == 0)
\r
752 if(TransPacketBase && ThreadCount < AskMaxThreadCount())
\r
753 TrnSkt = AskTrnCtrlSkt();
\r
757 if(TransPacketBase && NewCmdSkt != INVALID_SOCKET && ThreadCount < AskMaxThreadCount())
\r
759 ReleaseMutex(hListAccMutex);
\r
760 if(TrnSkt == INVALID_SOCKET || NewCmdSkt != CmdSkt)
\r
761 ReConnectTrnSkt(&TrnSkt, &Canceled[ThreadCount]);
\r
763 CheckClosedAndReconnectTrnSkt(&TrnSkt, &Canceled[ThreadCount]);
\r
765 if(TrnSkt == INVALID_SOCKET)
\r
767 // 同時ログイン数制限に引っかかった可能性あり
\r
768 // 負荷を下げるために約10秒間待機
\r
770 while(NewCmdSkt != CmdSkt && i > 0)
\r
772 BackgrndMessageProc();
\r
777 LastUsed = timeGetTime();
\r
778 // WaitForSingleObject(hListAccMutex, INFINITE);
\r
779 while(WaitForSingleObject(hListAccMutex, 0) == WAIT_TIMEOUT)
\r
781 BackgrndMessageProc();
\r
787 if(TrnSkt != INVALID_SOCKET)
\r
790 // 60秒間使用されなければログアウト
\r
791 if(timeGetTime() - LastUsed > 60000 || NewCmdSkt == INVALID_SOCKET)
\r
793 ReleaseMutex(hListAccMutex);
\r
794 SendData(TrnSkt, "QUIT\r\n", 6, 0, &Canceled[ThreadCount]);
\r
796 TrnSkt = INVALID_SOCKET;
\r
797 // WaitForSingleObject(hListAccMutex, INFINITE);
\r
798 while(WaitForSingleObject(hListAccMutex, 0) == WAIT_TIMEOUT)
\r
800 BackgrndMessageProc();
\r
807 CmdSkt = NewCmdSkt;
\r
808 // if(TransPacketBase != NULL)
\r
809 if(TrnSkt != INVALID_SOCKET && NextTransPacketBase != NULL)
\r
811 Pos = NextTransPacketBase;
\r
812 NextTransPacketBase = NextTransPacketBase->Next;
\r
813 // ディレクトリ操作は非同期で行わない
\r
814 // ReleaseMutex(hListAccMutex);
\r
815 if(hWndTrans == NULL)
\r
817 // if((strncmp(TransPacketBase->Cmd, "RETR", 4) == 0) ||
\r
818 // (strncmp(TransPacketBase->Cmd, "STOR", 4) == 0) ||
\r
819 // (strncmp(TransPacketBase->Cmd, "MKD", 3) == 0) ||
\r
820 // (strncmp(TransPacketBase->Cmd, "L-", 2) == 0) ||
\r
821 // (strncmp(TransPacketBase->Cmd, "R-", 2) == 0))
\r
822 if((strncmp(Pos->Cmd, "RETR", 4) == 0) ||
\r
823 (strncmp(Pos->Cmd, "STOR", 4) == 0) ||
\r
824 (strncmp(Pos->Cmd, "MKD", 3) == 0) ||
\r
825 (strncmp(Pos->Cmd, "L-", 2) == 0) ||
\r
826 (strncmp(Pos->Cmd, "R-", 2) == 0))
\r
828 hWndTrans = CreateDialog(GetFtpInst(), MAKEINTRESOURCE(transfer_dlg), HWND_DESKTOP, (DLGPROC)TransDlgProc);
\r
829 if(MoveToForeground == YES)
\r
830 SetForegroundWindow(hWndTrans);
\r
831 ShowWindow(hWndTrans, SW_SHOWNOACTIVATE);
\r
832 GetWindowRect(hWndTrans, &WndRect);
\r
833 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
836 // TransPacketBase->hWndTrans = hWndTrans;
\r
837 Pos->hWndTrans = hWndTrans;
\r
838 Pos->ctrl_skt = TrnSkt;
\r
839 Pos->Abort = ABORT_NONE;
\r
840 Pos->ThreadCount = ThreadCount;
\r
842 if(hWndTrans != NULL)
\r
844 if(MoveToForeground == YES)
\r
846 SetForegroundWindow(hWndTrans);
\r
847 MoveToForeground = NO;
\r
851 if(hWndTrans != NULL)
\r
852 // SendMessage(hWndTrans, WM_SET_PACKET, 0, (LPARAM)TransPacketBase);
\r
853 SendMessage(hWndTrans, WM_SET_PACKET, 0, (LPARAM)Pos);
\r
855 // 中断後に受信バッファに応答が残っていると次のコマンドの応答が正しく処理できない
\r
856 RemoveReceivedData(TrnSkt);
\r
859 // if(strncmp(TransPacketBase->Cmd, "RETR", 4) == 0)
\r
860 if(strncmp(Pos->Cmd, "RETR", 4) == 0)
\r
862 // 一部TYPE、STOR(RETR)、PORT(PASV)を並列に処理できないホストがあるため
\r
863 // ReleaseMutex(hListAccMutex);
\r
865 // if(CheckPathViolation(TransPacketBase) == NO)
\r
866 if(CheckPathViolation(Pos) == NO)
\r
868 /* フルパスを使わないための処理 */
\r
869 // if(MakeNonFullPath(TransPacketBase, CurDir, Tmp) == FFFTP_SUCCESS)
\r
870 if(MakeNonFullPath(Pos, CurDir[Pos->ThreadCount], Tmp) == FFFTP_SUCCESS)
\r
872 // if(strncmp(TransPacketBase->Cmd, "RETR-S", 6) == 0)
\r
873 if(strncmp(Pos->Cmd, "RETR-S", 6) == 0)
\r
876 // DoSIZE(TransPacketBase->RemoteFile, &TransPacketBase->Size);
\r
877 // DoMDTM(TransPacketBase->RemoteFile, &TransPacketBase->Time);
\r
878 // strcpy(TransPacketBase->Cmd, "RETR ");
\r
879 DoSIZE(TrnSkt, Pos->RemoteFile, &Pos->Size, &Canceled[Pos->ThreadCount]);
\r
880 DoMDTM(TrnSkt, Pos->RemoteFile, &Pos->Time, &Canceled[Pos->ThreadCount]);
\r
881 strcpy(Pos->Cmd, "RETR ");
\r
885 // if(DoDownLoad(AskTrnCtrlSkt(), TransPacketBase, NO) == 429)
\r
887 // if(ReConnectTrnSkt() == FFFTP_SUCCESS)
\r
888 // DoDownLoad(AskTrnCtrlSkt(), TransPacketBase, NO, &Canceled);
\r
889 DoDownLoad(TrnSkt, Pos, NO, &Canceled[Pos->ThreadCount]);
\r
893 // 一部TYPE、STOR(RETR)、PORT(PASV)を並列に処理できないホストがあるため
\r
894 ReleaseMutex(hListAccMutex);
\r
897 // else if(strncmp(TransPacketBase->Cmd, "STOR", 4) == 0)
\r
898 else if(strncmp(Pos->Cmd, "STOR", 4) == 0)
\r
900 // 一部TYPE、STOR(RETR)、PORT(PASV)を並列に処理できないホストがあるため
\r
901 // ReleaseMutex(hListAccMutex);
\r
902 /* フルパスを使わないための処理 */
\r
903 // if(MakeNonFullPath(TransPacketBase, CurDir, Tmp) == FFFTP_SUCCESS)
\r
904 if(MakeNonFullPath(Pos, CurDir[Pos->ThreadCount], Tmp) == FFFTP_SUCCESS)
\r
907 // if(DoUpLoad(AskTrnCtrlSkt(), TransPacketBase) == 429)
\r
909 // if(ReConnectTrnSkt() == FFFTP_SUCCESS)
\r
910 // DoUpLoad(AskTrnCtrlSkt(), TransPacketBase);
\r
911 DoUpLoad(TrnSkt, Pos);
\r
914 // 一部TYPE、STOR(RETR)、PORT(PASV)を並列に処理できないホストがあるため
\r
915 ReleaseMutex(hListAccMutex);
\r
917 /* フォルダ作成(ローカルまたはホスト) */
\r
918 // else if(strncmp(TransPacketBase->Cmd, "MKD", 3) == 0)
\r
919 else if(strncmp(Pos->Cmd, "MKD", 3) == 0)
\r
921 // DispTransFileInfo(TransPacketBase, MSGJPN078, FALSE, YES);
\r
922 DispTransFileInfo(Pos, MSGJPN078, FALSE, YES);
\r
924 // if(strlen(TransPacketBase->RemoteFile) > 0)
\r
925 if(strlen(Pos->RemoteFile) > 0)
\r
927 /* フルパスを使わないための処理 */
\r
928 CwdSts = FTP_COMPLETE;
\r
930 // strcpy(Tmp, TransPacketBase->RemoteFile);
\r
931 strcpy(Tmp, Pos->RemoteFile);
\r
932 // if(ProcForNonFullpath(Tmp, CurDir, hWndTrans, 1) == FFFTP_FAIL)
\r
933 if(ProcForNonFullpath(TrnSkt, Tmp, CurDir[Pos->ThreadCount], hWndTrans, &Canceled[Pos->ThreadCount]) == FFFTP_FAIL)
\r
936 CwdSts = FTP_ERROR;
\r
939 if(CwdSts == FTP_COMPLETE)
\r
942 // CommandProcTrn(NULL, "MKD %s", Tmp);
\r
943 CommandProcTrn(TrnSkt, NULL, &Canceled[Pos->ThreadCount], "MKD %s", Tmp);
\r
944 /* すでにフォルダがある場合もあるので、 */
\r
945 /* ここではエラーチェックはしない */
\r
948 // CommandProcTrn(NULL, "%s %03d %s", AskHostChmodCmd(), FolderAttrNum, Tmp);
\r
949 CommandProcTrn(TrnSkt, NULL, &Canceled[Pos->ThreadCount], "%s %03d %s", AskHostChmodCmd(), FolderAttrNum, Tmp);
\r
952 // else if(strlen(TransPacketBase->LocalFile) > 0)
\r
953 else if(strlen(Pos->LocalFile) > 0)
\r
956 // DoLocalMKD(TransPacketBase->LocalFile);
\r
957 DoLocalMKD(Pos->LocalFile);
\r
959 ReleaseMutex(hListAccMutex);
\r
961 /* ディレクトリ作成(常にホスト側) */
\r
962 // else if(strncmp(TransPacketBase->Cmd, "R-MKD", 5) == 0)
\r
963 else if(strncmp(Pos->Cmd, "R-MKD", 5) == 0)
\r
965 // DispTransFileInfo(TransPacketBase, MSGJPN079, FALSE, YES);
\r
966 DispTransFileInfo(Pos, MSGJPN079, FALSE, YES);
\r
968 /* フルパスを使わないための処理 */
\r
969 // if(MakeNonFullPath(TransPacketBase, CurDir, Tmp) == FFFTP_SUCCESS)
\r
970 if(MakeNonFullPath(Pos, CurDir[Pos->ThreadCount], Tmp) == FFFTP_SUCCESS)
\r
973 // CommandProcTrn(NULL, "%s%s", TransPacketBase->Cmd+2, TransPacketBase->RemoteFile);
\r
974 CommandProcTrn(TrnSkt, NULL, &Canceled[Pos->ThreadCount], "%s%s", Pos->Cmd+2, Pos->RemoteFile);
\r
977 // CommandProcTrn(NULL, "%s %03d %s", AskHostChmodCmd(), FolderAttrNum, TransPacketBase->RemoteFile);
\r
978 CommandProcTrn(TrnSkt, NULL, &Canceled[Pos->ThreadCount], "%s %03d %s", AskHostChmodCmd(), FolderAttrNum, Pos->RemoteFile);
\r
980 ReleaseMutex(hListAccMutex);
\r
982 /* ディレクトリ削除(常にホスト側) */
\r
983 // else if(strncmp(TransPacketBase->Cmd, "R-RMD", 5) == 0)
\r
984 else if(strncmp(Pos->Cmd, "R-RMD", 5) == 0)
\r
986 // DispTransFileInfo(TransPacketBase, MSGJPN080, FALSE, YES);
\r
987 DispTransFileInfo(Pos, MSGJPN080, FALSE, YES);
\r
989 // DelNotify = MirrorDelNotify(WIN_REMOTE, DelNotify, TransPacketBase);
\r
990 DelNotify = MirrorDelNotify(WIN_REMOTE, DelNotify, Pos);
\r
991 if((DelNotify == YES) || (DelNotify == YES_ALL))
\r
993 /* フルパスを使わないための処理 */
\r
994 // if(MakeNonFullPath(TransPacketBase, CurDir, Tmp) == FFFTP_SUCCESS)
\r
995 if(MakeNonFullPath(Pos, CurDir[Pos->ThreadCount], Tmp) == FFFTP_SUCCESS)
\r
998 // CommandProcTrn(NULL, "%s%s", TransPacketBase->Cmd+2, TransPacketBase->RemoteFile);
\r
999 CommandProcTrn(TrnSkt, NULL, &Canceled[Pos->ThreadCount], "%s%s", Pos->Cmd+2, Pos->RemoteFile);
\r
1002 ReleaseMutex(hListAccMutex);
\r
1004 /* ファイル削除(常にホスト側) */
\r
1005 // else if(strncmp(TransPacketBase->Cmd, "R-DELE", 6) == 0)
\r
1006 else if(strncmp(Pos->Cmd, "R-DELE", 6) == 0)
\r
1008 // DispTransFileInfo(TransPacketBase, MSGJPN081, FALSE, YES);
\r
1009 DispTransFileInfo(Pos, MSGJPN081, FALSE, YES);
\r
1011 // DelNotify = MirrorDelNotify(WIN_REMOTE, DelNotify, TransPacketBase);
\r
1012 DelNotify = MirrorDelNotify(WIN_REMOTE, DelNotify, Pos);
\r
1013 if((DelNotify == YES) || (DelNotify == YES_ALL))
\r
1015 /* フルパスを使わないための処理 */
\r
1016 // if(MakeNonFullPath(TransPacketBase, CurDir, Tmp) == FFFTP_SUCCESS)
\r
1017 if(MakeNonFullPath(Pos, CurDir[Pos->ThreadCount], Tmp) == FFFTP_SUCCESS)
\r
1020 // CommandProcTrn(NULL, "%s%s", TransPacketBase->Cmd+2, TransPacketBase->RemoteFile);
\r
1021 CommandProcTrn(TrnSkt, NULL, &Canceled[Pos->ThreadCount], "%s%s", Pos->Cmd+2, Pos->RemoteFile);
\r
1024 ReleaseMutex(hListAccMutex);
\r
1026 /* ディレクトリ作成(常にローカル側) */
\r
1027 // else if(strncmp(TransPacketBase->Cmd, "L-MKD", 5) == 0)
\r
1028 else if(strncmp(Pos->Cmd, "L-MKD", 5) == 0)
\r
1030 // DispTransFileInfo(TransPacketBase, MSGJPN082, FALSE, YES);
\r
1031 DispTransFileInfo(Pos, MSGJPN082, FALSE, YES);
\r
1034 // DoLocalMKD(TransPacketBase->LocalFile);
\r
1035 DoLocalMKD(Pos->LocalFile);
\r
1036 ReleaseMutex(hListAccMutex);
\r
1038 /* ディレクトリ削除(常にローカル側) */
\r
1039 // else if(strncmp(TransPacketBase->Cmd, "L-RMD", 5) == 0)
\r
1040 else if(strncmp(Pos->Cmd, "L-RMD", 5) == 0)
\r
1042 // DispTransFileInfo(TransPacketBase, MSGJPN083, FALSE, YES);
\r
1043 DispTransFileInfo(Pos, MSGJPN083, FALSE, YES);
\r
1045 // DelNotify = MirrorDelNotify(WIN_LOCAL, DelNotify, TransPacketBase);
\r
1046 DelNotify = MirrorDelNotify(WIN_LOCAL, DelNotify, Pos);
\r
1047 if((DelNotify == YES) || (DelNotify == YES_ALL))
\r
1050 // DoLocalRMD(TransPacketBase->LocalFile);
\r
1051 DoLocalRMD(Pos->LocalFile);
\r
1053 ReleaseMutex(hListAccMutex);
\r
1055 /* ファイル削除(常にローカル側) */
\r
1056 // else if(strncmp(TransPacketBase->Cmd, "L-DELE", 6) == 0)
\r
1057 else if(strncmp(Pos->Cmd, "L-DELE", 6) == 0)
\r
1059 // DispTransFileInfo(TransPacketBase, MSGJPN084, FALSE, YES);
\r
1060 DispTransFileInfo(Pos, MSGJPN084, FALSE, YES);
\r
1062 // DelNotify = MirrorDelNotify(WIN_LOCAL, DelNotify, TransPacketBase);
\r
1063 DelNotify = MirrorDelNotify(WIN_LOCAL, DelNotify, Pos);
\r
1064 if((DelNotify == YES) || (DelNotify == YES_ALL))
\r
1067 // DoLocalDELE(TransPacketBase->LocalFile);
\r
1068 DoLocalDELE(Pos->LocalFile);
\r
1070 ReleaseMutex(hListAccMutex);
\r
1072 /* カレントディレクトリを設定 */
\r
1073 // else if(strcmp(TransPacketBase->Cmd, "SETCUR") == 0)
\r
1074 else if(strcmp(Pos->Cmd, "SETCUR") == 0)
\r
1076 // if(AskShareProh() == YES)
\r
1077 if(AskReuseCmdSkt() == NO || AskShareProh() == YES)
\r
1079 // if(strcmp(CurDir, TransPacketBase->RemoteFile) != 0)
\r
1080 if(strcmp(CurDir[Pos->ThreadCount], Pos->RemoteFile) != 0)
\r
1082 // if(CommandProcTrn(NULL, "CWD %s", TransPacketBase->RemoteFile)/100 != FTP_COMPLETE)
\r
1083 if(CommandProcTrn(TrnSkt, NULL, &Canceled[Pos->ThreadCount], "CWD %s", Pos->RemoteFile)/100 != FTP_COMPLETE)
\r
1085 DispCWDerror(hWndTrans);
\r
1090 // strcpy(CurDir, TransPacketBase->RemoteFile);
\r
1091 strcpy(CurDir[Pos->ThreadCount], Pos->RemoteFile);
\r
1092 ReleaseMutex(hListAccMutex);
\r
1094 /* カレントディレクトリを戻す */
\r
1095 // else if(strcmp(TransPacketBase->Cmd, "BACKCUR") == 0)
\r
1096 else if(strcmp(Pos->Cmd, "BACKCUR") == 0)
\r
1098 // if(AskShareProh() == NO)
\r
1099 if(AskReuseCmdSkt() == YES && AskShareProh() == NO)
\r
1101 // if(strcmp(CurDir, TransPacketBase->RemoteFile) != 0)
\r
1102 // CommandProcTrn(NULL, "CWD %s", TransPacketBase->RemoteFile);
\r
1103 // strcpy(CurDir, TransPacketBase->RemoteFile);
\r
1104 if(strcmp(CurDir[Pos->ThreadCount], Pos->RemoteFile) != 0)
\r
1105 CommandProcTrn(TrnSkt, NULL, &Canceled[Pos->ThreadCount], "CWD %s", Pos->RemoteFile);
\r
1106 strcpy(CurDir[Pos->ThreadCount], Pos->RemoteFile);
\r
1108 ReleaseMutex(hListAccMutex);
\r
1111 // else if(strcmp(TransPacketBase->Cmd, "GOQUIT") == 0)
\r
1112 // else if(strcmp(Pos->Cmd, "GOQUIT") == 0)
\r
1114 // ReleaseMutex(hListAccMutex);
\r
1118 ReleaseMutex(hListAccMutex);
\r
1120 /*===== 1つの処理終わり =====*/
\r
1122 if(ForceAbort == NO)
\r
1124 // WaitForSingleObject(hListAccMutex, INFINITE);
\r
1125 while(WaitForSingleObject(hListAccMutex, 0) == WAIT_TIMEOUT)
\r
1127 BackgrndMessageProc();
\r
1130 if(ClearAll == YES)
\r
1131 // EraseTransFileList();
\r
1133 for(i = 0; i < MAX_DATA_CONNECTION; i++)
\r
1134 Canceled[i] = YES;
\r
1135 EraseTransFileList();
\r
1141 // if((strncmp(TransPacketBase->Cmd, "RETR", 4) == 0) ||
\r
1142 // (strncmp(TransPacketBase->Cmd, "STOR", 4) == 0))
\r
1143 if((strncmp(Pos->Cmd, "RETR", 4) == 0) ||
\r
1144 (strncmp(Pos->Cmd, "STOR", 4) == 0))
\r
1147 if(TransFiles > 0)
\r
1149 PostMessage(GetMainHwnd(), WM_CHANGE_COND, 0, 0);
\r
1151 // Pos = TransPacketBase;
\r
1152 // TransPacketBase = TransPacketBase->Next;
\r
1156 ReleaseMutex(hListAccMutex);
\r
1158 if(BackgrndMessageProc() == YES)
\r
1160 WaitForSingleObject(hListAccMutex, INFINITE);
\r
1161 EraseTransFileList();
\r
1162 ReleaseMutex(hListAccMutex);
\r
1165 if(hWndTrans != NULL)
\r
1166 SendMessage(hWndTrans, WM_SET_PACKET, 0, 0);
\r
1168 strcpy(Pos->Cmd, "");
\r
1169 LastUsed = timeGetTime();
\r
1172 else if(TransPacketBase == NULL)
\r
1179 SoundPlay(SND_TRANS);
\r
1180 if(AskAutoExit() == NO)
\r
1183 PostMessage(GetMainHwnd(), WM_REFRESH_LOCAL_FLG, 0, 0);
\r
1185 PostMessage(GetMainHwnd(), WM_REFRESH_REMOTE_FLG, 0, 0);
\r
1189 PostMessage(GetMainHwnd(), WM_COMMAND, MAKEWPARAM(MENU_AUTO_EXIT, 0), 0);
\r
1193 ReleaseMutex(hListAccMutex);
\r
1196 if(hWndTrans != NULL)
\r
1198 DestroyWindow(hWndTrans);
\r
1201 // if(GoExit == YES)
\r
1203 // SoundPlay(SND_TRANS);
\r
1205 // if(AskAutoExit() == NO)
\r
1207 // if(Down == YES)
\r
1208 // PostMessage(GetMainHwnd(), WM_REFRESH_LOCAL_FLG, 0, 0);
\r
1210 // PostMessage(GetMainHwnd(), WM_REFRESH_REMOTE_FLG, 0, 0);
\r
1217 BackgrndMessageProc();
\r
1222 TransferErrorMode = AskTransferErrorMode();
\r
1223 TransferErrorNotify = AskTransferErrorNotify();
\r
1227 ReleaseMutex(hListAccMutex);
\r
1228 if(hWndTrans != NULL)
\r
1230 DestroyWindow(hWndTrans);
\r
1233 BackgrndMessageProc();
\r
1234 if(ThreadCount < AskMaxThreadCount())
\r
1240 if(AskReuseCmdSkt() == NO || ThreadCount > 0)
\r
1242 if(TrnSkt != INVALID_SOCKET)
\r
1244 SendData(TrnSkt, "QUIT\r\n", 6, 0, &Canceled[ThreadCount]);
\r
1252 /*----- フルパスを使わないファイルアクセスの準備 ------------------------------
\r
1255 * TRANSPACKET *Pkt : 転送パケット
\r
1256 * char *Cur : カレントディレクトリ
\r
1257 * char *Tmp : 作業用エリア
\r
1260 * int ステータス(FFFTP_SUCCESS/FFFTP_FAIL)
\r
1264 * このモジュール内で CWD を行ない、
\r
1265 * Pkt->RemoteFile にファイル名のみ残す。(パス名は消す)
\r
1266 *----------------------------------------------------------------------------*/
\r
1269 static int MakeNonFullPath(TRANSPACKET *Pkt, char *Cur, char *Tmp)
\r
1273 // Sts = ProcForNonFullpath(Pkt->RemoteFile, Cur, Pkt->hWndTrans, 1);
\r
1274 Sts = ProcForNonFullpath(Pkt->ctrl_skt, Pkt->RemoteFile, Cur, Pkt->hWndTrans, &Canceled[Pkt->ThreadCount]);
\r
1275 if(Sts == FFFTP_FAIL)
\r
1284 /*----- ダウンロードを行なう --------------------------------------------------
\r
1287 * SOCKET cSkt : コントロールソケット
\r
1288 * TRANSPACKET *Pkt : 転送ファイル情報
\r
1289 * int DirList : ディレクトリリストのダウンロード(YES/NO)
\r
1295 * このモジュールは、ファイル一覧の取得などを行なう際にメインのスレッド
\r
1296 * からも呼ばれる。メインのスレッドから呼ばれる時は Pkt->hWndTrans == NULL。
\r
1297 *----------------------------------------------------------------------------*/
\r
1299 int DoDownLoad(SOCKET cSkt, TRANSPACKET *Pkt, int DirList, int *CancelCheckWork)
\r
1302 char Reply[ERR_MSG_LEN+7];
\r
1304 Pkt->ctrl_skt = cSkt;
\r
1305 if(IsSpecialDevice(GetFileName(Pkt->LocalFile)) == YES)
\r
1308 SetTaskMsg(MSGJPN085, GetFileName(Pkt->LocalFile));
\r
1309 // エラーによってはダイアログが表示されない場合があるバグ対策
\r
1310 // DispDownloadFinishMsg(Pkt, iRetCode);
\r
1312 else if(Pkt->Mode != EXIST_IGNORE)
\r
1314 if(Pkt->Type == TYPE_I)
\r
1315 Pkt->KanjiCode = KANJI_NOCNV;
\r
1317 iRetCode = command(Pkt->ctrl_skt, Reply, CancelCheckWork, "TYPE %c", Pkt->Type);
\r
1318 if(iRetCode/100 < FTP_RETRY)
\r
1320 if(Pkt->hWndTrans != NULL)
\r
1323 // AllTransSizeNow = 0;
\r
1324 AllTransSizeNow[Pkt->ThreadCount] = 0;
\r
1327 DispTransFileInfo(Pkt, MSGJPN086, TRUE, YES);
\r
1329 DispTransFileInfo(Pkt, MSGJPN087, FALSE, NO);
\r
1333 // if(BackgrndMessageProc() == NO)
\r
1334 if(IsSFTPAttached(Pkt->ctrl_skt))
\r
1338 else if(BackgrndMessageProc() == NO)
\r
1340 if(AskPasvMode() != YES)
\r
1341 iRetCode = DownLoadNonPassive(Pkt, CancelCheckWork);
\r
1343 iRetCode = DownLoadPassive(Pkt, CancelCheckWork);
\r
1349 SetErrorMsg(Reply);
\r
1350 // エラーによってはダイアログが表示されない場合があるバグ対策
\r
1351 DispDownloadFinishMsg(Pkt, iRetCode);
\r
1355 DispTransFileInfo(Pkt, MSGJPN088, TRUE, YES);
\r
1356 SetTaskMsg(MSGJPN089, Pkt->RemoteFile);
\r
1363 /*----- 通常モードでファイルをダウンロード ------------------------------------
\r
1366 * TRANSPACKET *Pkt : 転送ファイル情報
\r
1370 *----------------------------------------------------------------------------*/
\r
1372 static int DownLoadNonPassive(TRANSPACKET *Pkt, int *CancelCheckWork)
\r
1376 SOCKET data_socket = INVALID_SOCKET; // data channel socket
\r
1377 SOCKET listen_socket = INVALID_SOCKET; // data listen socket
\r
1379 // char Buf[1024];
\r
1380 char Buf[FMAX_PATH+1024];
\r
1383 // struct sockaddr_in saSockAddr1;
\r
1384 struct sockaddr_in saSockAddrIPv4;
\r
1385 struct sockaddr_in6 saSockAddrIPv6;
\r
1386 char Reply[ERR_MSG_LEN+7];
\r
1388 if((listen_socket = GetFTPListenSocket(Pkt->ctrl_skt, CancelCheckWork)) != INVALID_SOCKET)
\r
1390 if(SetDownloadResume(Pkt, Pkt->Mode, Pkt->ExistSize, &CreateMode, CancelCheckWork) == YES)
\r
1392 sprintf(Buf, "%s%s", Pkt->Cmd, Pkt->RemoteFile);
\r
1393 iRetCode = command(Pkt->ctrl_skt, Reply, CancelCheckWork, "%s", Buf);
\r
1394 if(iRetCode/100 == FTP_PRELIM)
\r
1397 // if(SocksGet2ndBindReply(listen_socket, &data_socket) == FFFTP_FAIL)
\r
1398 if(SocksGet2ndBindReply(listen_socket, &data_socket, CancelCheckWork) == FFFTP_FAIL)
\r
1401 // iLength = sizeof(saSockAddr1);
\r
1402 // data_socket = do_accept(listen_socket, (struct sockaddr *)&saSockAddr1, (int *)&iLength);
\r
1403 switch(AskCurNetType())
\r
1406 iLength=sizeof(saSockAddrIPv4);
\r
1407 data_socket = do_accept(listen_socket,(struct sockaddr *)&saSockAddrIPv4, (int *)&iLength);
\r
1410 iLength=sizeof(saSockAddrIPv6);
\r
1411 data_socket = do_accept(listen_socket,(struct sockaddr *)&saSockAddrIPv6, (int *)&iLength);
\r
1415 if(shutdown(listen_socket, 1) != 0)
\r
1416 ReportWSError("shutdown listen", WSAGetLastError());
\r
1417 listen_socket = DoClose(listen_socket);
\r
1419 if(data_socket == INVALID_SOCKET)
\r
1421 SetErrorMsg(MSGJPN280);
\r
1422 ReportWSError("accept", WSAGetLastError());
\r
1427 // DoPrintf("Skt=%u : accept from %s port %u", data_socket, inet_ntoa(saSockAddr1.sin_addr), ntohs(saSockAddr1.sin_port));
\r
1429 switch(AskCurNetType())
\r
1432 DoPrintf("Skt=%u : accept from %s port %u", data_socket, inet_ntoa(saSockAddrIPv4.sin_addr), ntohs(saSockAddrIPv4.sin_port));
\r
1435 DoPrintf("Skt=%u : accept from %s port %u", data_socket, inet6_ntoa(saSockAddrIPv6.sin6_addr), ntohs(saSockAddrIPv6.sin6_port));
\r
1441 if(data_socket != INVALID_SOCKET)
\r
1443 // 一部TYPE、STOR(RETR)、PORT(PASV)を並列に処理できないホストがあるため
\r
1444 ReleaseMutex(hListAccMutex);
\r
1446 // iRetCode = DownLoadFile(Pkt, data_socket, CreateMode, CancelCheckWork);
\r
1447 if(IsSSLAttached(Pkt->ctrl_skt))
\r
1449 if(AttachSSL(data_socket, Pkt->ctrl_skt, CancelCheckWork))
\r
1450 iRetCode = DownLoadFile(Pkt, data_socket, CreateMode, CancelCheckWork);
\r
1455 iRetCode = DownLoadFile(Pkt, data_socket, CreateMode, CancelCheckWork);
\r
1456 // data_socket = DoClose(data_socket);
\r
1461 SetErrorMsg(Reply);
\r
1462 SetTaskMsg(MSGJPN090);
\r
1463 listen_socket = DoClose(listen_socket);
\r
1473 SetErrorMsg(MSGJPN279);
\r
1475 // エラーによってはダイアログが表示されない場合があるバグ対策
\r
1476 // DispDownloadFinishMsg(Pkt, iRetCode);
\r
1482 /*----- Passiveモードでファイルをダウンロード ---------------------------------
\r
1485 * TRANSPACKET *Pkt : 転送ファイル情報
\r
1489 *----------------------------------------------------------------------------*/
\r
1491 static int DownLoadPassive(TRANSPACKET *Pkt, int *CancelCheckWork)
\r
1494 SOCKET data_socket = INVALID_SOCKET; // data channel socket
\r
1496 // char Buf[1024];
\r
1497 char Buf[FMAX_PATH+1024];
\r
1504 char Reply[ERR_MSG_LEN+7];
\r
1507 // iRetCode = command(Pkt->ctrl_skt, Buf, CancelCheckWork, "PASV");
\r
1508 switch(AskCurNetType())
\r
1511 iRetCode = command(Pkt->ctrl_skt, Buf, CancelCheckWork, "PASV");
\r
1514 iRetCode = command(Pkt->ctrl_skt, Buf, CancelCheckWork, "EPSV");
\r
1517 if(iRetCode/100 == FTP_COMPLETE)
\r
1520 // if(GetAdrsAndPort(Buf, Adrs, &Port, 19) == FFFTP_SUCCESS)
\r
1521 if(GetAdrsAndPort(Pkt->ctrl_skt, Buf, Adrs, &Port, 39) == FFFTP_SUCCESS)
\r
1523 if((data_socket = connectsock(Adrs, Port, MSGJPN091, CancelCheckWork)) != INVALID_SOCKET)
\r
1527 if(setsockopt(data_socket, IPPROTO_TCP, TCP_NODELAY, (LPSTR)&Flg, sizeof(Flg)) == SOCKET_ERROR)
\r
1528 ReportWSError("setsockopt", WSAGetLastError());
\r
1530 if(SetDownloadResume(Pkt, Pkt->Mode, Pkt->ExistSize, &CreateMode, CancelCheckWork) == YES)
\r
1532 sprintf(Buf, "%s%s", Pkt->Cmd, Pkt->RemoteFile);
\r
1533 iRetCode = command(Pkt->ctrl_skt, Reply, CancelCheckWork, "%s", Buf);
\r
1534 if(iRetCode/100 == FTP_PRELIM)
\r
1536 // 一部TYPE、STOR(RETR)、PORT(PASV)を並列に処理できないホストがあるため
\r
1537 ReleaseMutex(hListAccMutex);
\r
1539 // iRetCode = DownLoadFile(Pkt, data_socket, CreateMode, CancelCheckWork);
\r
1540 if(IsSSLAttached(Pkt->ctrl_skt))
\r
1542 if(AttachSSL(data_socket, Pkt->ctrl_skt, CancelCheckWork))
\r
1543 iRetCode = DownLoadFile(Pkt, data_socket, CreateMode, CancelCheckWork);
\r
1548 iRetCode = DownLoadFile(Pkt, data_socket, CreateMode, CancelCheckWork);
\r
1549 // data_socket = DoClose(data_socket);
\r
1553 SetErrorMsg(Reply);
\r
1554 SetTaskMsg(MSGJPN092);
\r
1555 data_socket = DoClose(data_socket);
\r
1567 SetErrorMsg(MSGJPN093);
\r
1568 SetTaskMsg(MSGJPN093);
\r
1575 // エラーによってはダイアログが表示されない場合があるバグ対策
\r
1576 // DispDownloadFinishMsg(Pkt, iRetCode);
\r
1582 /*----- ダウンロードの実行 ----------------------------------------------------
\r
1585 * TRANSPACKET *Pkt : 転送ファイル情報
\r
1586 * SOCKET dSkt : データソケット
\r
1587 * int CreateMode : ファイル作成モード (CREATE_ALWAYS/OPEN_ALWAYS)
\r
1594 * ダイアログを出す(Pkt->hWndTrans!=NULL)場合、インターバルタイマで経過を表示する
\r
1595 * ダイアログを出さない場合、このルーチンからDispDownloadSize()を呼ぶ
\r
1596 *----------------------------------------------------------------------------*/
\r
1598 static int DownLoadFile(TRANSPACKET *Pkt, SOCKET dSkt, int CreateMode, int *CancelCheckWork)
\r
1601 char Buf[BUFSIZE];
\r
1602 char Buf2[BUFSIZE+3];
\r
1603 HANDLE iFileHandle;
\r
1604 SECURITY_ATTRIBUTES Sec;
\r
1606 CODECONVINFO cInfo;
\r
1608 // fd_set ReadFds;
\r
1609 // struct timeval Tout;
\r
1610 // struct timeval *ToutPtr;
\r
1613 char TmpBuf[ONELINE_BUF_SIZE];
\r
1614 DWORD dwFileAttributes;
\r
1616 #ifdef SET_BUFFER_SIZE
\r
1617 /* Add by H.Shirouzu at 2002/10/02 */
\r
1618 int buf_size = SOCKBUF_SIZE;
\r
1619 for ( ; buf_size > 0; buf_size /= 2)
\r
1620 if (setsockopt(dSkt, SOL_SOCKET, SO_RCVBUF, (char *)&buf_size, sizeof(buf_size)) == 0)
\r
1625 // 念のため受信バッファを無効にする
\r
1626 #ifdef DISABLE_TRANSFER_NETWORK_BUFFERS
\r
1628 setsockopt(dSkt, SOL_SOCKET, SO_RCVBUF, (char *)&buf_size, sizeof(buf_size));
\r
1631 Pkt->Abort = ABORT_NONE;
\r
1633 Sec.nLength = sizeof(SECURITY_ATTRIBUTES);
\r
1634 Sec.lpSecurityDescriptor = NULL;
\r
1635 Sec.bInheritHandle = FALSE;
\r
1637 dwFileAttributes = GetFileAttributes(Pkt->LocalFile);
\r
1638 if (dwFileAttributes != INVALID_FILE_ATTRIBUTES && (dwFileAttributes & FILE_ATTRIBUTE_READONLY)) {
\r
1640 if (MessageBox(GetMainHwnd(), MSGJPN296, MSGJPN086, MB_YESNO) == IDYES) {
\r
1642 SetFileAttributes(Pkt->LocalFile, dwFileAttributes ^ FILE_ATTRIBUTE_READONLY);
\r
1646 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
1649 char Buf3[(BUFSIZE + 3) * 4];
\r
1650 CODECONVINFO cInfo2;
\r
1651 int ProcessedBOM = NO;
\r
1652 // 4GB超対応(kaokunさん提供)
\r
1654 if(CreateMode == OPEN_ALWAYS)
\r
1655 // 4GB超対応(kaokunさん提供)
\r
1656 // SetFilePointer(iFileHandle, 0, 0, FILE_END);
\r
1657 SetFilePointer(iFileHandle, 0, &High, FILE_END);
\r
1659 if(Pkt->hWndTrans != NULL)
\r
1662 // TimeStart = time(NULL);
\r
1663 TimeStart[Pkt->ThreadCount] = time(NULL);
\r
1664 SetTimer(Pkt->hWndTrans, TIMER_DISPLAY, DISPLAY_TIMING, NULL);
\r
1667 InitCodeConvInfo(&cInfo);
\r
1668 cInfo.KanaCnv = Pkt->KanaCnv;
\r
1670 InitCodeConvInfo(&cInfo2);
\r
1671 cInfo2.KanaCnv = Pkt->KanaCnv;
\r
1673 /*===== ファイルを受信するループ =====*/
\r
1674 while((Pkt->Abort == ABORT_NONE) && (ForceAbort == NO))
\r
1676 // FD_ZERO(&ReadFds);
\r
1677 // FD_SET(dSkt, &ReadFds);
\r
1678 // ToutPtr = NULL;
\r
1679 // if(TimeOut != 0)
\r
1681 // Tout.tv_sec = TimeOut;
\r
1682 // Tout.tv_usec = 0;
\r
1683 // ToutPtr = &Tout;
\r
1685 // iNumBytes = select(0, &ReadFds, NULL, NULL, ToutPtr);
\r
1686 // if(iNumBytes == SOCKET_ERROR)
\r
1688 // ReportWSError("select", WSAGetLastError());
\r
1689 // if(Pkt->Abort == ABORT_NONE)
\r
1690 // Pkt->Abort = ABORT_ERROR;
\r
1693 // else if(iNumBytes == 0)
\r
1695 // SetErrorMsg(MSGJPN094);
\r
1696 // SetTaskMsg(MSGJPN094);
\r
1697 // Pkt->Abort = ABORT_ERROR;
\r
1701 if((iNumBytes = do_recv(dSkt, Buf, BUFSIZE, 0, &TimeOutErr, CancelCheckWork)) <= 0)
\r
1703 if(TimeOutErr == YES)
\r
1705 SetErrorMsg(MSGJPN094);
\r
1706 SetTaskMsg(MSGJPN094);
\r
1707 if(Pkt->hWndTrans != NULL)
\r
1709 if(Pkt->Abort == ABORT_NONE)
\r
1710 Pkt->Abort = ABORT_ERROR;
\r
1712 else if(iNumBytes == SOCKET_ERROR)
\r
1714 if(Pkt->Abort == ABORT_NONE)
\r
1715 Pkt->Abort = ABORT_ERROR;
\r
1721 if(Pkt->KanjiCode != KANJI_NOCNV)
\r
1724 cInfo.StrLen = iNumBytes;
\r
1726 cInfo.BufSize = BUFSIZE+3;
\r
1729 // ここで全てUTF-8へ変換する
\r
1730 // TODO: SJIS以外も直接UTF-8へ変換
\r
1731 // if(Pkt->KanjiCode == KANJI_JIS)
\r
1732 // Continue = ConvJIStoSJIS(&cInfo);
\r
1734 // Continue = ConvEUCtoSJIS(&cInfo);
\r
1735 switch(Pkt->KanjiCode)
\r
1738 switch(Pkt->KanjiCodeDesired)
\r
1741 // memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
1742 // cInfo2.OutLen = cInfo.StrLen;
\r
1745 Continue = ConvSJIStoJIS(&cInfo);
\r
1746 cInfo2.Str = cInfo.Buf;
\r
1747 cInfo2.StrLen = cInfo.OutLen;
\r
1748 cInfo2.Buf = Buf3;
\r
1749 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1750 ConvJIStoSJIS(&cInfo2);
\r
1753 Continue = ConvSJIStoJIS(&cInfo);
\r
1754 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
1755 cInfo2.OutLen = cInfo.OutLen;
\r
1758 Continue = ConvSJIStoEUC(&cInfo);
\r
1759 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
1760 cInfo2.OutLen = cInfo.OutLen;
\r
1763 Continue = ConvSJIStoUTF8N(&cInfo);
\r
1764 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
1765 cInfo2.OutLen = cInfo.OutLen;
\r
1767 case KANJI_UTF8BOM:
\r
1768 if(ProcessedBOM == NO)
\r
1770 memcpy(Buf3, "\xEF\xBB\xBF", 3);
\r
1771 cInfo2.OutLen = 3;
\r
1773 ProcessedBOM = YES;
\r
1776 Continue = ConvSJIStoUTF8N(&cInfo);
\r
1777 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
1778 cInfo2.OutLen = cInfo.OutLen;
\r
1783 switch(Pkt->KanjiCodeDesired)
\r
1786 Continue = ConvJIStoSJIS(&cInfo);
\r
1787 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
1788 cInfo2.OutLen = cInfo.OutLen;
\r
1791 // memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
1792 // cInfo2.OutLen = cInfo.StrLen;
\r
1795 Continue = ConvJIStoSJIS(&cInfo);
\r
1796 cInfo2.Str = cInfo.Buf;
\r
1797 cInfo2.StrLen = cInfo.OutLen;
\r
1798 cInfo2.Buf = Buf3;
\r
1799 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1800 ConvSJIStoJIS(&cInfo2);
\r
1803 Continue = ConvJIStoSJIS(&cInfo);
\r
1804 cInfo2.Str = cInfo.Buf;
\r
1805 cInfo2.StrLen = cInfo.OutLen;
\r
1806 cInfo2.Buf = Buf3;
\r
1807 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1808 ConvSJIStoEUC(&cInfo2);
\r
1811 Continue = ConvJIStoSJIS(&cInfo);
\r
1812 cInfo2.Str = cInfo.Buf;
\r
1813 cInfo2.StrLen = cInfo.OutLen;
\r
1814 cInfo2.Buf = Buf3;
\r
1815 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1816 ConvSJIStoUTF8N(&cInfo2);
\r
1818 case KANJI_UTF8BOM:
\r
1819 if(ProcessedBOM == NO)
\r
1821 memcpy(Buf3, "\xEF\xBB\xBF", 3);
\r
1822 cInfo2.OutLen = 3;
\r
1824 ProcessedBOM = YES;
\r
1827 Continue = ConvJIStoSJIS(&cInfo);
\r
1828 cInfo2.Str = cInfo.Buf;
\r
1829 cInfo2.StrLen = cInfo.OutLen;
\r
1830 cInfo2.Buf = Buf3;
\r
1831 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1832 ConvSJIStoUTF8N(&cInfo2);
\r
1837 switch(Pkt->KanjiCodeDesired)
\r
1840 Continue = ConvEUCtoSJIS(&cInfo);
\r
1841 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
1842 cInfo2.OutLen = cInfo.OutLen;
\r
1845 Continue = ConvEUCtoSJIS(&cInfo);
\r
1846 cInfo2.Str = cInfo.Buf;
\r
1847 cInfo2.StrLen = cInfo.OutLen;
\r
1848 cInfo2.Buf = Buf3;
\r
1849 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1850 ConvSJIStoJIS(&cInfo2);
\r
1853 // memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
1854 // cInfo2.OutLen = cInfo.StrLen;
\r
1857 Continue = ConvEUCtoSJIS(&cInfo);
\r
1858 cInfo2.Str = cInfo.Buf;
\r
1859 cInfo2.StrLen = cInfo.OutLen;
\r
1860 cInfo2.Buf = Buf3;
\r
1861 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1862 ConvSJIStoEUC(&cInfo2);
\r
1865 Continue = ConvEUCtoSJIS(&cInfo);
\r
1866 cInfo2.Str = cInfo.Buf;
\r
1867 cInfo2.StrLen = cInfo.OutLen;
\r
1868 cInfo2.Buf = Buf3;
\r
1869 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1870 ConvSJIStoUTF8N(&cInfo2);
\r
1872 case KANJI_UTF8BOM:
\r
1873 if(ProcessedBOM == NO)
\r
1875 memcpy(Buf3, "\xEF\xBB\xBF", 3);
\r
1876 cInfo2.OutLen = 3;
\r
1878 ProcessedBOM = YES;
\r
1881 Continue = ConvEUCtoSJIS(&cInfo);
\r
1882 cInfo2.Str = cInfo.Buf;
\r
1883 cInfo2.StrLen = cInfo.OutLen;
\r
1884 cInfo2.Buf = Buf3;
\r
1885 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1886 ConvSJIStoUTF8N(&cInfo2);
\r
1891 switch(Pkt->KanjiCodeDesired)
\r
1894 Continue = ConvUTF8NtoSJIS(&cInfo);
\r
1895 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
1896 cInfo2.OutLen = cInfo.OutLen;
\r
1899 Continue = ConvUTF8NtoSJIS(&cInfo);
\r
1900 cInfo2.Str = cInfo.Buf;
\r
1901 cInfo2.StrLen = cInfo.OutLen;
\r
1902 cInfo2.Buf = Buf3;
\r
1903 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1904 ConvSJIStoJIS(&cInfo2);
\r
1907 Continue = ConvUTF8NtoSJIS(&cInfo);
\r
1908 cInfo2.Str = cInfo.Buf;
\r
1909 cInfo2.StrLen = cInfo.OutLen;
\r
1910 cInfo2.Buf = Buf3;
\r
1911 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1912 ConvSJIStoEUC(&cInfo2);
\r
1915 memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
1916 cInfo2.OutLen = cInfo.StrLen;
\r
1919 case KANJI_UTF8BOM:
\r
1920 if(ProcessedBOM == NO)
\r
1922 memcpy(Buf3, "\xEF\xBB\xBF", 3);
\r
1923 cInfo2.OutLen = 3;
\r
1925 ProcessedBOM = YES;
\r
1928 memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
1929 cInfo2.OutLen = cInfo.StrLen;
\r
1934 case KANJI_UTF8BOM:
\r
1935 if(ProcessedBOM == NO)
\r
1937 if(memcmp(Buf, "\xEF\xBB\xBF", 3) == 0)
\r
1940 cInfo.StrLen -= 3;
\r
1942 cInfo2.OutLen = 0;
\r
1943 switch(Pkt->KanjiCodeDesired)
\r
1945 case KANJI_UTF8BOM:
\r
1946 memcpy(Buf3, "\xEF\xBB\xBF", 3);
\r
1947 cInfo2.OutLen = 3;
\r
1951 ProcessedBOM = YES;
\r
1954 switch(Pkt->KanjiCodeDesired)
\r
1957 Continue = ConvUTF8NtoSJIS(&cInfo);
\r
1958 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
1959 cInfo2.OutLen = cInfo.OutLen;
\r
1962 Continue = ConvUTF8NtoSJIS(&cInfo);
\r
1963 cInfo2.Str = cInfo.Buf;
\r
1964 cInfo2.StrLen = cInfo.OutLen;
\r
1965 cInfo2.Buf = Buf3;
\r
1966 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1967 ConvSJIStoJIS(&cInfo2);
\r
1970 Continue = ConvUTF8NtoSJIS(&cInfo);
\r
1971 cInfo2.Str = cInfo.Buf;
\r
1972 cInfo2.StrLen = cInfo.OutLen;
\r
1973 cInfo2.Buf = Buf3;
\r
1974 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
1975 ConvSJIStoEUC(&cInfo2);
\r
1978 memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
1979 cInfo2.OutLen = cInfo.StrLen;
\r
1982 case KANJI_UTF8BOM:
\r
1983 memcpy(Buf3, cInfo.Str, cInfo.StrLen);
\r
1984 cInfo2.OutLen = cInfo.StrLen;
\r
1990 // if(WriteFile(iFileHandle, Buf2, cInfo.OutLen, &Writed, NULL) == FALSE)
\r
1991 if(WriteFile(iFileHandle, Buf3, cInfo2.OutLen, &Writed, NULL) == FALSE)
\r
1992 Pkt->Abort = ABORT_DISKFULL;
\r
1994 while((Continue == YES) && (Pkt->Abort == ABORT_NONE));
\r
1998 if(WriteFile(iFileHandle, Buf, iNumBytes, &Writed, NULL) == FALSE)
\r
1999 Pkt->Abort = ABORT_DISKFULL;
\r
2002 Pkt->ExistSize += iNumBytes;
\r
2003 if(Pkt->hWndTrans != NULL)
\r
2005 // AllTransSizeNow += iNumBytes;
\r
2006 AllTransSizeNow[Pkt->ThreadCount] += iNumBytes;
\r
2009 /* 転送ダイアログを出さない時の経過表示 */
\r
2010 DispDownloadSize(Pkt->ExistSize);
\r
2013 if(BackgrndMessageProc() == YES)
\r
2017 /* 書き残したデータを書き込む */
\r
2018 if(Pkt->KanjiCode != KANJI_NOCNV)
\r
2021 cInfo.BufSize = BUFSIZE+3;
\r
2022 FlushRestData(&cInfo);
\r
2023 switch(Pkt->KanjiCode)
\r
2026 switch(Pkt->KanjiCodeDesired)
\r
2030 cInfo2.Str = cInfo.Buf;
\r
2031 cInfo2.StrLen = cInfo.OutLen;
\r
2032 cInfo2.Buf = Buf3;
\r
2033 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2034 ConvJIStoSJIS(&cInfo2);
\r
2037 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2038 cInfo2.OutLen = cInfo.OutLen;
\r
2041 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2042 cInfo2.OutLen = cInfo.OutLen;
\r
2045 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2046 cInfo2.OutLen = cInfo.OutLen;
\r
2048 case KANJI_UTF8BOM:
\r
2049 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2050 cInfo2.OutLen = cInfo.OutLen;
\r
2055 switch(Pkt->KanjiCodeDesired)
\r
2058 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2059 cInfo2.OutLen = cInfo.OutLen;
\r
2063 cInfo2.Str = cInfo.Buf;
\r
2064 cInfo2.StrLen = cInfo.OutLen;
\r
2065 cInfo2.Buf = Buf3;
\r
2066 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2067 ConvSJIStoJIS(&cInfo2);
\r
2070 cInfo2.Str = cInfo.Buf;
\r
2071 cInfo2.StrLen = cInfo.OutLen;
\r
2072 cInfo2.Buf = Buf3;
\r
2073 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2074 ConvSJIStoEUC(&cInfo2);
\r
2077 cInfo2.Str = cInfo.Buf;
\r
2078 cInfo2.StrLen = cInfo.OutLen;
\r
2079 cInfo2.Buf = Buf3;
\r
2080 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2081 ConvSJIStoUTF8N(&cInfo2);
\r
2083 case KANJI_UTF8BOM:
\r
2084 cInfo2.Str = cInfo.Buf;
\r
2085 cInfo2.StrLen = cInfo.OutLen;
\r
2086 cInfo2.Buf = Buf3;
\r
2087 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2088 ConvSJIStoUTF8N(&cInfo2);
\r
2093 switch(Pkt->KanjiCodeDesired)
\r
2096 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2097 cInfo2.OutLen = cInfo.OutLen;
\r
2100 cInfo2.Str = cInfo.Buf;
\r
2101 cInfo2.StrLen = cInfo.OutLen;
\r
2102 cInfo2.Buf = Buf3;
\r
2103 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2104 ConvSJIStoJIS(&cInfo2);
\r
2108 cInfo2.Str = cInfo.Buf;
\r
2109 cInfo2.StrLen = cInfo.OutLen;
\r
2110 cInfo2.Buf = Buf3;
\r
2111 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2112 ConvSJIStoEUC(&cInfo2);
\r
2115 cInfo2.Str = cInfo.Buf;
\r
2116 cInfo2.StrLen = cInfo.OutLen;
\r
2117 cInfo2.Buf = Buf3;
\r
2118 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2119 ConvSJIStoUTF8N(&cInfo2);
\r
2121 case KANJI_UTF8BOM:
\r
2122 cInfo2.Str = cInfo.Buf;
\r
2123 cInfo2.StrLen = cInfo.OutLen;
\r
2124 cInfo2.Buf = Buf3;
\r
2125 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2126 ConvSJIStoUTF8N(&cInfo2);
\r
2131 switch(Pkt->KanjiCodeDesired)
\r
2134 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2135 cInfo2.OutLen = cInfo.OutLen;
\r
2138 cInfo2.Str = cInfo.Buf;
\r
2139 cInfo2.StrLen = cInfo.OutLen;
\r
2140 cInfo2.Buf = Buf3;
\r
2141 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2142 ConvSJIStoJIS(&cInfo2);
\r
2145 cInfo2.Str = cInfo.Buf;
\r
2146 cInfo2.StrLen = cInfo.OutLen;
\r
2147 cInfo2.Buf = Buf3;
\r
2148 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2149 ConvSJIStoEUC(&cInfo2);
\r
2152 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2153 cInfo2.OutLen = cInfo.OutLen;
\r
2155 case KANJI_UTF8BOM:
\r
2156 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2157 cInfo2.OutLen = cInfo.OutLen;
\r
2161 case KANJI_UTF8BOM:
\r
2162 switch(Pkt->KanjiCodeDesired)
\r
2165 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2166 cInfo2.OutLen = cInfo.OutLen;
\r
2169 cInfo2.Str = cInfo.Buf;
\r
2170 cInfo2.StrLen = cInfo.OutLen;
\r
2171 cInfo2.Buf = Buf3;
\r
2172 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2173 ConvSJIStoJIS(&cInfo2);
\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 ConvSJIStoEUC(&cInfo2);
\r
2183 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2184 cInfo2.OutLen = cInfo.OutLen;
\r
2186 case KANJI_UTF8BOM:
\r
2187 memcpy(Buf3, cInfo.Buf, cInfo.OutLen);
\r
2188 cInfo2.OutLen = cInfo.OutLen;
\r
2193 // if(WriteFile(iFileHandle, Buf2, cInfo.OutLen, &Writed, NULL) == FALSE)
\r
2194 if(WriteFile(iFileHandle, Buf3, cInfo2.OutLen, &Writed, NULL) == FALSE)
\r
2195 Pkt->Abort = ABORT_DISKFULL;
\r
2196 cInfo2.Buf = Buf3;
\r
2197 cInfo2.BufSize = (BUFSIZE + 3) * 4;
\r
2198 FlushRestData(&cInfo2);
\r
2199 if(WriteFile(iFileHandle, Buf3, cInfo2.OutLen, &Writed, NULL) == FALSE)
\r
2200 Pkt->Abort = ABORT_DISKFULL;
\r
2204 if(Pkt->hWndTrans != NULL)
\r
2206 KillTimer(Pkt->hWndTrans, TIMER_DISPLAY);
\r
2207 DispTransferStatus(Pkt->hWndTrans, YES, Pkt);
\r
2209 // TimeStart = time(NULL) - TimeStart + 1;
\r
2210 TimeStart[Pkt->ThreadCount] = time(NULL) - TimeStart[Pkt->ThreadCount] + 1;
\r
2214 /* 転送ダイアログを出さない時の経過表示を消す */
\r
2215 DispDownloadSize(-1);
\r
2218 /* ファイルのタイムスタンプを合わせる */
\r
2219 if((SaveTimeStamp == YES) &&
\r
2220 ((Pkt->Time.dwLowDateTime != 0) || (Pkt->Time.dwHighDateTime != 0)))
\r
2222 SetFileTime(iFileHandle, &Pkt->Time, &Pkt->Time, &Pkt->Time);
\r
2225 CloseHandle(iFileHandle);
\r
2227 if(iNumBytes == SOCKET_ERROR)
\r
2228 ReportWSError("recv",WSAGetLastError());
\r
2232 SetErrorMsg(MSGJPN095, Pkt->LocalFile);
\r
2233 SetTaskMsg(MSGJPN095, Pkt->LocalFile);
\r
2234 Pkt->Abort = ABORT_ERROR;
\r
2238 if(shutdown(dSkt, 1) != 0)
\r
2239 ReportWSError("shutdown", WSAGetLastError());
\r
2243 if(ForceAbort == NO)
\r
2245 /* Abortをホストに伝える */
\r
2246 if(Pkt->Abort != ABORT_NONE && iFileHandle != INVALID_HANDLE_VALUE)
\r
2248 SendData(Pkt->ctrl_skt, "\xFF\xF4\xFF", 3, MSG_OOB, CancelCheckWork); /* MSG_OOBに注意 */
\r
2249 SendData(Pkt->ctrl_skt, "\xF2", 1, 0, CancelCheckWork);
\r
2250 command(Pkt->ctrl_skt, NULL, CancelCheckWork, "ABOR");
\r
2254 iRetCode = ReadReplyMessage(Pkt->ctrl_skt, Buf, 1024, CancelCheckWork, TmpBuf);
\r
2257 //DoPrintf("##DOWN REPLY : %s", Buf);
\r
2259 if(Pkt->Abort == ABORT_DISKFULL)
\r
2261 SetErrorMsg(MSGJPN096);
\r
2262 SetTaskMsg(MSGJPN096);
\r
2265 // if(iRetCode >= FTP_RETRY)
\r
2266 if((iRetCode/100) >= FTP_RETRY)
\r
2268 if(Pkt->Abort != ABORT_NONE)
\r
2275 /*----- ダウンロード終了/中止時のメッセージを表示 ----------------------------
\r
2278 * TRANSPACKET *Pkt : 転送ファイル情報
\r
2279 * int iRetCode : 応答コード
\r
2283 *----------------------------------------------------------------------------*/
\r
2285 static void DispDownloadFinishMsg(TRANSPACKET *Pkt, int iRetCode)
\r
2287 char Fname[FMAX_PATH+1];
\r
2290 ReleaseMutex(hListAccMutex);
\r
2291 if(ForceAbort == NO)
\r
2293 if((iRetCode/100) >= FTP_CONTINUE)
\r
2295 strcpy(Fname, Pkt->RemoteFile);
\r
2297 #if defined(HAVE_OPENVMS)
\r
2298 /* OpenVMSの場合、空ディレクトリへ移動すると550 File not foundになって
\r
2299 * エラーダイアログやエラーメッセージが出るので何もしない */
\r
2300 if (AskHostType() == HTYPE_VMS)
\r
2303 #if defined(HAVE_TANDEM)
\r
2304 /* HP Nonstop Server の場合、ファイルのない subvol へ移動すると550 File not found
\r
2305 * になるが問題ないのでエラーダイアログやエラーメッセージを出さないため */
\r
2306 if (AskHostType() == HTYPE_TANDEM)
\r
2311 // if((strncmp(Pkt->Cmd, "NLST", 4) == 0) || (strncmp(Pkt->Cmd, "LIST", 4) == 0))
\r
2312 if((strncmp(Pkt->Cmd, "NLST", 4) == 0) || (strncmp(Pkt->Cmd, "LIST", 4) == 0) || (strncmp(Pkt->Cmd, "MLSD", 4) == 0))
\r
2314 SetTaskMsg(MSGJPN097);
\r
2315 strcpy(Fname, MSGJPN098);
\r
2318 // else if((Pkt->hWndTrans != NULL) && (TimeStart != 0))
\r
2319 // SetTaskMsg(MSGJPN099, TimeStart, Pkt->ExistSize/TimeStart);
\r
2320 else if((Pkt->hWndTrans != NULL) && (TimeStart[Pkt->ThreadCount] != 0))
\r
2321 SetTaskMsg(MSGJPN099, TimeStart[Pkt->ThreadCount], Pkt->ExistSize/TimeStart[Pkt->ThreadCount]);
\r
2323 SetTaskMsg(MSGJPN100);
\r
2325 if(Pkt->Abort != ABORT_USER)
\r
2327 // 全て中止を選択後にダイアログが表示されるバグ対策
\r
2328 // if(DispUpDownErrDialog(downerr_dlg, Pkt->hWndTrans, Fname) == NO)
\r
2330 // if(Canceled[Pkt->ThreadCount] == NO && ClearAll == NO && DispUpDownErrDialog(downerr_dlg, Pkt->hWndTrans, Fname) == NO)
\r
2331 // ClearAll = YES;
\r
2332 if(Canceled[Pkt->ThreadCount] == NO && ClearAll == NO)
\r
2334 if(strncmp(Pkt->Cmd, "RETR", 4) == 0 || strncmp(Pkt->Cmd, "STOR", 4) == 0)
\r
2336 if(TransferErrorNotify == YES && DispUpDownErrDialog(downerr_dlg, Pkt->hWndTrans, Pkt) == NO)
\r
2340 Pkt->Mode = TransferErrorMode;
\r
2341 AddTransFileList(Pkt);
\r
2350 // if((strncmp(Pkt->Cmd, "NLST", 4) == 0) || (strncmp(Pkt->Cmd, "LIST", 4) == 0))
\r
2351 if((strncmp(Pkt->Cmd, "NLST", 4) == 0) || (strncmp(Pkt->Cmd, "LIST", 4) == 0) || (strncmp(Pkt->Cmd, "MLSD", 4) == 0))
\r
2352 SetTaskMsg(MSGJPN101, Pkt->ExistSize);
\r
2354 // else if((Pkt->hWndTrans != NULL) && (TimeStart != 0))
\r
2355 // SetTaskMsg(MSGJPN102, TimeStart, Pkt->ExistSize/TimeStart);
\r
2356 else if((Pkt->hWndTrans != NULL) && (TimeStart[Pkt->ThreadCount] != 0))
\r
2357 // "0 B/S"と表示されるバグを修正
\r
2358 // 原因は%dにあたる部分に64ビット値が渡されているため
\r
2359 // SetTaskMsg(MSGJPN102, TimeStart[Pkt->ThreadCount], Pkt->ExistSize/TimeStart[Pkt->ThreadCount]);
\r
2360 SetTaskMsg(MSGJPN102, (LONG)TimeStart[Pkt->ThreadCount], (LONG)(Pkt->ExistSize/TimeStart[Pkt->ThreadCount]));
\r
2362 SetTaskMsg(MSGJPN103, Pkt->ExistSize);
\r
2369 /*----- ダウンロード/アップロードエラーのダイアログを表示 --------------------
\r
2372 * int RedID : ダイアログボックスのリソースID
\r
2373 * HWND hWnd : 書き込み中ダイアログのウインドウ
\r
2374 * char *Fname : ファイル名
\r
2377 * int ステータス (YES=中止/NO=全て中止)
\r
2378 *----------------------------------------------------------------------------*/
\r
2381 //static int DispUpDownErrDialog(int ResID, HWND hWnd, char *Fname)
\r
2382 static int DispUpDownErrDialog(int ResID, HWND hWnd, TRANSPACKET *Pkt)
\r
2385 hWnd = GetMainHwnd();
\r
2387 SoundPlay(SND_ERROR);
\r
2389 // return(DialogBoxParam(GetFtpInst(), MAKEINTRESOURCE(ResID), hWnd, UpDownErrorDialogProc, (LPARAM)Fname));
\r
2390 return(DialogBoxParam(GetFtpInst(), MAKEINTRESOURCE(ResID), hWnd, UpDownErrorDialogProc, (LPARAM)Pkt));
\r
2394 /*----- ダウンロードエラー/アップロードエラーダイアログのコールバック --------
\r
2397 * HWND hDlg : ウインドウハンドル
\r
2398 * UINT message : メッセージ番号
\r
2399 * WPARAM wParam : メッセージの WPARAM 引数
\r
2400 * LPARAM lParam : メッセージの LPARAM 引数
\r
2404 *----------------------------------------------------------------------------*/
\r
2407 //static BOOL CALLBACK UpDownErrorDialogProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
\r
2408 static INT_PTR CALLBACK UpDownErrorDialogProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
\r
2410 static TRANSPACKET *Pkt;
\r
2411 static const RADIOBUTTON DownExistButton[] = {
\r
2412 { DOWN_EXIST_OVW, EXIST_OVW },
\r
2413 { DOWN_EXIST_RESUME, EXIST_RESUME },
\r
2414 { DOWN_EXIST_IGNORE, EXIST_IGNORE }
\r
2416 #define DOWNEXISTBUTTONS (sizeof(DownExistButton)/sizeof(RADIOBUTTON))
\r
2420 case WM_INITDIALOG :
\r
2421 Pkt = (TRANSPACKET *)lParam;
\r
2422 // SendDlgItemMessage(hDlg, UPDOWN_ERR_FNAME, WM_SETTEXT, 0, (LPARAM)lParam);
\r
2423 SendDlgItemMessage(hDlg, UPDOWN_ERR_FNAME, WM_SETTEXT, 0, (LPARAM)Pkt->RemoteFile);
\r
2425 // SendDlgItemMessage(hDlg, UPDOWN_ERR_MSG, WM_SETTEXT, 0, (LPARAM)ErrMsg);
\r
2426 SendDlgItemMessage(hDlg, UPDOWN_ERR_MSG, WM_SETTEXT, 0, (LPARAM)GetErrMsg());
\r
2428 if((Pkt->Type == TYPE_A) || (Pkt->ExistSize <= 0))
\r
2429 EnableWindow(GetDlgItem(hDlg, DOWN_EXIST_RESUME), FALSE);
\r
2431 SetRadioButtonByValue(hDlg, TransferErrorMode, DownExistButton, DOWNEXISTBUTTONS);
\r
2435 switch(GET_WM_COMMAND_ID(wParam, lParam))
\r
2438 TransferErrorNotify = NO;
\r
2439 /* ここに break はない */
\r
2442 TransferErrorMode = AskRadioButtonValue(hDlg, DownExistButton, DOWNEXISTBUTTONS);
\r
2443 EndDialog(hDlg, YES);
\r
2447 EndDialog(hDlg, NO);
\r
2451 // hHelpWin = HtmlHelp(NULL, AskHelpFilePath(), HH_HELP_CONTEXT, IDH_HELP_TOPIC_0000009);
\r
2460 /*----- ダウンロードのリジュームの準備を行う ----------------------------------
\r
2463 * TRANSPACKET *Pkt : 転送ファイル情報
\r
2464 * iont ProcMode : 処理モード(EXIST_xxx)
\r
2465 * LONGLONG Size : ロード済みのファイルのサイズ
\r
2466 * int *Mode : ファイル作成モード (CREATE_xxxx)
\r
2469 * int 転送を行うかどうか(YES/NO=このファイルを中止/NO_ALL=全て中止)
\r
2472 * Pkt->ExistSizeのセットを行なう
\r
2473 *----------------------------------------------------------------------------*/
\r
2475 static int SetDownloadResume(TRANSPACKET *Pkt, int ProcMode, LONGLONG Size, int *Mode, int *CancelCheckWork)
\r
2479 char Reply[ERR_MSG_LEN+7];
\r
2484 Pkt->ExistSize = 0;
\r
2485 *Mode = CREATE_ALWAYS;
\r
2487 if(ProcMode == EXIST_RESUME)
\r
2489 iRetCode = command(Pkt->ctrl_skt, Reply, CancelCheckWork, "REST %s", MakeNumString(Size, Tmp, FALSE));
\r
2490 if(iRetCode/100 < FTP_RETRY)
\r
2493 if(Pkt->hWndTrans != NULL)
\r
2494 Pkt->ExistSize = Size;
\r
2495 *Mode = OPEN_ALWAYS;
\r
2499 Com = DialogBox(GetFtpInst(), MAKEINTRESOURCE(noresume_dlg), Pkt->hWndTrans, NoResumeWndProc);
\r
2502 if(Com == NO_ALL) /* 全て中止 */
\r
2504 Pkt->Abort = ABORT_USER;
\r
2512 /*----- resumeエラーダイアログのコールバック ----------------------------------
\r
2515 * HWND hDlg : ウインドウハンドル
\r
2516 * UINT message : メッセージ番号
\r
2517 * WPARAM wParam : メッセージの WPARAM 引数
\r
2518 * LPARAM lParam : メッセージの LPARAM 引数
\r
2522 *----------------------------------------------------------------------------*/
\r
2525 //static BOOL CALLBACK NoResumeWndProc(HWND hDlg, UINT iMessage, WPARAM wParam, LPARAM lParam)
\r
2526 static INT_PTR CALLBACK NoResumeWndProc(HWND hDlg, UINT iMessage, WPARAM wParam, LPARAM lParam)
\r
2530 case WM_INITDIALOG :
\r
2534 switch(GET_WM_COMMAND_ID(wParam, lParam))
\r
2537 EndDialog(hDlg, YES);
\r
2541 EndDialog(hDlg, NO);
\r
2544 case RESUME_CANCEL_ALL :
\r
2545 EndDialog(hDlg, NO_ALL);
\r
2555 /*----- アップロードを行なう --------------------------------------------------
\r
2558 * SOCKET cSkt : コントロールソケット
\r
2559 * TRANSPACKET *Pkt : 転送ファイル情報
\r
2563 *----------------------------------------------------------------------------*/
\r
2565 static int DoUpLoad(SOCKET cSkt, TRANSPACKET *Pkt)
\r
2568 char Reply[ERR_MSG_LEN+7];
\r
2570 Pkt->ctrl_skt = cSkt;
\r
2572 if(Pkt->Mode != EXIST_IGNORE)
\r
2574 if(CheckFileReadable(Pkt->LocalFile) == FFFTP_SUCCESS)
\r
2576 if(Pkt->Type == TYPE_I)
\r
2577 Pkt->KanjiCode = KANJI_NOCNV;
\r
2580 // iRetCode = command(Pkt->ctrl_skt, Reply, &Canceled, "TYPE %c", Pkt->Type);
\r
2581 iRetCode = command(Pkt->ctrl_skt, Reply, &Canceled[Pkt->ThreadCount], "TYPE %c", Pkt->Type);
\r
2582 if(iRetCode/100 < FTP_RETRY)
\r
2584 if(Pkt->Mode == EXIST_UNIQUE)
\r
2585 strcpy(Pkt->Cmd, "STOU ");
\r
2587 if(Pkt->hWndTrans != NULL)
\r
2588 DispTransFileInfo(Pkt, MSGJPN104, TRUE, YES);
\r
2591 // if(BackgrndMessageProc() == NO)
\r
2592 if(IsSFTPAttached(Pkt->ctrl_skt))
\r
2596 else if(BackgrndMessageProc() == NO)
\r
2598 if(AskPasvMode() != YES)
\r
2599 iRetCode = UpLoadNonPassive(Pkt);
\r
2601 iRetCode = UpLoadPassive(Pkt);
\r
2607 SetErrorMsg(Reply);
\r
2610 if((Pkt->Attr != -1) && ((iRetCode/100) == FTP_COMPLETE))
\r