//+------------------------------------------------------------------------- // // Microsoft Windows // // Copyright (C) Microsoft Corporation, 1997 - 1999 // // File: xfer.h // //-------------------------------------------------------------------------- #ifndef _XFER_H_ #define _XFER_H_ #include #define SERVICE_NAME_1 "OBEX:IrXfer" #define SERVICE_NAME_2 "OBEX" #define GUARD_MAGIC 0x45454545 #define TIMEOUT_INFINITE (~0UL) #define ERROR_DESCRIPTION_LENGTH 1000 #define IRDA_DEVICE_NAME_LENGTH 22 // // OBEX parameter codes // #define OBEX_PARAM_UNICODE 0x00 #define OBEX_PARAM_STREAM 0x40 #define OBEX_PARAM_1BYTE 0x80 #define OBEX_PARAM_4BYTE 0xC0 #define OBEX_PARAM_TYPE_MASK 0xC0 #define OBEX_PARAM_COUNT ( 0x00 | OBEX_PARAM_4BYTE ) #define OBEX_PARAM_NAME ( 0x01 | OBEX_PARAM_UNICODE ) #define OBEX_PARAM_LENGTH ( 0x03 | OBEX_PARAM_4BYTE ) #define OBEX_PARAM_UNIX_TIME ( 0x04 | OBEX_PARAM_4BYTE ) #define OBEX_PARAM_ISO_TIME ( 0x04 | OBEX_PARAM_STREAM ) #define OBEX_PARAM_BODY ( 0x08 | OBEX_PARAM_STREAM ) #define OBEX_PARAM_BODY_END ( 0x09 | OBEX_PARAM_STREAM ) #define OBEX_PARAM_WHO ( 0x0A | OBEX_PARAM_STREAM ) // #define OBEX_PARAM_LEN 1 #define PRIVATE_PARAM_WIN32_ERROR ( 0x30 | OBEX_PARAM_4BYTE ) // for xfer.c typedef struct { BOOL fXferInProgress; // transfer in progress __int64 dwTotalSize; // total size of transfer __int64 dwTotalSent; // number of bytes sent in this transfer __int64 dwFileSize; // total size of current file __int64 dwFileSent; // number of bytes sent of the current file } DATA_XFER, *LPDATA_XFER; typedef struct { FILETIME filetime; // file time WCHAR szFileName[MAX_PATH]; // name of file WCHAR szFileSave[MAX_PATH]; // path+name of final file WCHAR szFileTemp[MAX_PATH]; // path+name of temp file used HANDLE hFile; // file handle (of szFileTemp) } DATA_FILE, *LPDATA_FILE; // for obex.c typedef struct { BYTE1 b1Flags; // setpath flags BYTE1 b1Constants; // setpath constants } DATA_PATH, *LPDATA_PATH; typedef struct { BOOL fWaiting; // indicates if waiting for a reply BYTE1 b1Status; // response status (error/success) } DATA_REPLY, *LPDATA_REPLY; typedef enum { osIDLE = 0, osCONN = 1, osFILE = 2 } OBEXSTATE; typedef struct { LPSTORE lpStore; OBEXSTATE state; BYTE1 b1Version; // peer's version of obex BYTE1 b1Flags; // connection flags BYTE2 b2MaxPacket; // peer's maximum packet size } DATA_CONN, *LPDATA_CONN; // for status.c typedef struct { DWORD dwDeviceID; BOOL fClear; } TARGET_ITEM, *LPTARGET_ITEM; //------------------------------------- enum TRANSFER_STATE { BLANK, CONNECTING, ACCEPTING, READING, WRITING, CLOSING }; enum OBEX_DIALECT { dialUnknown = 0, dialWin95, dialNt5 }; typedef class FILE_TRANSFER * PFILE_TRANSFER; class FILE_TRANSFER { public: FILE_TRANSFER( ); ~FILE_TRANSFER(); BOOL Xfer_Init( wchar_t * files, unsigned length, OBEX_DIALECT dialect , OBEX_DEVICE_TYPE DeviceType, BOOL CreateSocket, SOCKET ListenSocket ); BOOL SendReplyObex( BYTE1 ObexCode ); BOOL SendReplyWin32( BYTE1 b1Opcode, DWORD status ); void BeginSend( DWORD DeviceId, OBEX_DEVICE_TYPE DeviceType, error_status_t * pStatus, FAILURE_LOCATION * pLocation ); void Deactivate(); DWORD SyncAccept( VOID ); handle_t _rpcHandle; COOKIE _cookie; inline long DecrementRefCount() { EnterCriticalSection(&m_Lock); long count = --_refs; DbgLog3(SEV_INFO, "[%x] %p: refs = %d\n", (DWORD) _cookie, this, count); if (0 == count) { #if DBG DbgPrint("irmon: freeing transfer\n"); #endif LeaveCriticalSection(&m_Lock); delete this; return count; } else { LeaveCriticalSection(&m_Lock); } return count; } inline BOOL IsActive( VOID ) { return (_state != ACCEPTING); } inline long IncrementRefCount() { EnterCriticalSection(&m_Lock); long count = ++_refs; DbgLog3(SEV_INFO, "[%x] %p: refs = %d\n", (DWORD) _cookie, this, count); LeaveCriticalSection(&m_Lock); return count; } void Cancel() { _fCancelled = TRUE; } COOKIE GetCookie( VOID ) { return _cookie; } void Send(); void RecordDeviceName( SOCKADDR_IRDA * s ); VOID StopListening( VOID ) { IncrementRefCount(); EnterCriticalSection(&m_Lock); m_StopListening=TRUE; if (m_ListenSocket != INVALID_SOCKET) { closesocket(m_ListenSocket); m_ListenSocket=INVALID_SOCKET; } LeaveCriticalSection(&m_Lock); DecrementRefCount(); return; } private: CRITICAL_SECTION m_Lock; BOOL m_StopListening; SOCKET m_ListenSocket; OBEX_DEVICE_TYPE m_DeviceType; BOOL _fCancelled; BOOL _fInUiReceiveList; XFER_TYPE _xferType; TRANSFER_STATE _state; SOCKET _socket; wchar_t * _files; long _refs; HANDLE _event; BOOL _fWriteable; OBEX_DIALECT _dialect; wchar_t _DeviceName[MAX_PATH]; HANDLE _waitHandle; BYTE _buffer[ cbSOCK_BUFFER_SIZE + 16 + sizeof(SOCKADDR_IRDA) + 16 + sizeof(SOCKADDR_IRDA) ]; DWORD _guard; void HandleClosure( DWORD error ); // for sock.c error_status_t Sock_Request( LPVOID lpvData, DWORD dwDataSize ); error_status_t Sock_Respond( LPVOID lpvData, DWORD dwDataSize ); error_status_t Sock_EstablishConnection( DWORD dwDeviceID ,OBEX_DEVICE_TYPE DeviceType); VOID Sock_BreakConnection( SOCKET * pSock ); error_status_t _SendDataOnSocket( SOCKET sock, LPVOID lpvData, DWORD dwDataSize ); VOID _BreakConnection( SOCKET sock ); VOID _ReadConnection( SOCKET sock ); error_status_t Sock_CheckForReply( long Timeout ); // for xfer.c UINT _uObjsReceived; WCHAR _szRecvFolder[MAX_PATH]; DATA_FILE _dataFileRecv; DATA_XFER _dataXferRecv; BOOL Xfer_FileSetSize( BYTE4 b4Size ); error_status_t Xfer_FileWriteBody( LPVOID lpvData, BYTE2 b2Size, BOOL fFinal ); VOID Xfer_ConnEnd( VOID ); error_status_t Xfer_ConnStart( VOID ); VOID Xfer_FileAbort( VOID ); VOID Xfer_FileInit( VOID ); error_status_t Xfer_FileSetName( LPWSTR szName ); VOID Xfer_FileSetTime( FILETIME * FileTime ); error_status_t Xfer_SetPath( LPWSTR szPath ); VOID Xfer_SetSize( BYTE4 b4Size ); error_status_t _PutFileBody( HANDLE hFile, wchar_t FileName[] ); error_status_t _SendFile( LPWSTR szFile ); error_status_t _SendFolder( LPWSTR szFolder ); error_status_t _FileStart( VOID ); error_status_t _FileEnd( BOOL fSave ); VOID _Send_StartXfer( __int64 dwTotalSize, LPWSTR szDst ); VOID _Send_EndXfer( VOID ); error_status_t _SetReceiveFolder( LPWSTR szFolder ); // for progress.c DWORD _dwTimeStart; DWORD _dwSecondsLeft; int _CurrentPercentage; VOID _FormatTime( LPWSTR sz, DWORD dwSeconds ); // for obex.c DATA_CONN _dataRecv; DATA_PATH _dataPath; DATA_REPLY _dataReply; ULONG _blocksSent; ULONG _blocksAcked; ULONG _blockSize; __int64 _completedFilesSize; __int64 _currentFileSize; __int64 _currentFileAcked; DWORD _lastAckTime; error_status_t Obex_Abort( error_status_t status ); error_status_t Obex_Connect( __int64 dwTotalSize ); error_status_t Obex_Disconnect( error_status_t status ); error_status_t Obex_PutBegin( LPWSTR wszObj, __int64 dwObjSize, FILETIME * pFileTime ); error_status_t Obex_PutBody( wchar_t FileName[], LPBYTE1 pb1Data, BYTE2 b2DataSize, BOOL fFinal ); error_status_t Obex_SetPath( LPWSTR wszPath ); BOOL Obex_ConsumePackets( XFER_TYPE xferType, error_status_t * pStatus ); BOOL Obex_Init( VOID ); BOOL Obex_ReceiveData( XFER_TYPE xferType, LPVOID lpvData, DWORD dwDataSize ); VOID Obex_Reset( VOID ); error_status_t _WaitForReply( DWORD dwTimeout, BYTE1 b1NeededReply ); error_status_t _Put( LPWSTR wszObj, __int64 dwObjSize, FILETIME * Time, LPBYTE1 pb1Data, BYTE2 b2DataSize, BOOL fFinal ); error_status_t _Request( LPSTORE lpStore, BYTE1 b1NeededReply ); error_status_t _Respond( LPSTORE lpStore ); INT _ValidOpcode( OBEXSTATE state, BYTE1 b1Opcode ); BOOL _HandleAbort( BYTE1 b1Opcode, BYTE2 b2Length ); BOOL _HandleBadRequest( BYTE1 b1Opcode ); BOOL _HandleConnect( BYTE1 b1Opcode, BYTE2 b2Length ); BOOL _HandleDisconnect( BYTE1 b1Opcode, BYTE2 b2Length ); BOOL _HandleNotImplemented( BYTE1 b1Opcode ); error_status_t _HandlePut( BYTE1 b1Opcode, BYTE2 b2Length, BOOL fFinal ); error_status_t _HandleResponse( BYTE1 b1Status, BYTE2 b2Length ); error_status_t _HandleRequest( BYTE1 b1Opcode, BYTE2 b2Length ); error_status_t _HandleSetPath( BYTE1 b1Opcode, BYTE2 b2Length ); error_status_t _ParseParams( BYTE1 b1Opcode, BYTE2 b2Length ); BOOL _PokePacketSizeIntoStore( LPSTORE lpStore ); VOID _SetState( LPDATA_CONN lpDataConn, OBEXSTATE os ); VOID _WaitInit( VOID ); VOID _WriteBody( LPVOID lpvData, BYTE2 b2Size, BOOL fFinal ); BYTE2 _SkipHeader( BYTE1 b1Param, LPSTORE lpStore ); BYTE1 StatusToReplyCode( BYTE1 b1Opcode, DWORD status ); error_status_t ObexStatusToWin32( BYTE1 ObexStatus ); BOOL Activate(); }; error_status_t _GetObjListStats( LPWSTR lpszObjList, LPDWORD lpdwFiles, LPDWORD lpdwFolders, __int64 * pTotalSize ); DWORD ReportFileError( DWORD mc, WCHAR * file, DWORD error ); BYTE1 WinErrorToObexError( DWORD Win32Error ); extern RPC_BINDING_HANDLE rpcBinding; extern BOOL g_fAllowReceives; extern wchar_t g_DuplicateFileTemplate[]; extern wchar_t g_UnknownDeviceName[]; extern MUTEX * g_Mutex; #include extern "C" { FILE_TRANSFER* InitializeSocket( char ServiceName[] ); FILE_TRANSFER * ListenForTransfer( SOCKET ListenSocket, OBEX_DEVICE_TYPE DeviceType ); } #endif // _XFER_H_