llxfer.cpp
上传用户:king477883
上传日期:2021-03-01
资源大小:9553k
文件大小:8k
源码类别:

游戏引擎

开发平台:

C++ Builder

  1. /** 
  2.  * @file llxfer.cpp
  3.  * @brief implementation of LLXfer class for a single xfer.
  4.  *
  5.  * $LicenseInfo:firstyear=2001&license=viewergpl$
  6.  * 
  7.  * Copyright (c) 2001-2010, Linden Research, Inc.
  8.  * 
  9.  * Second Life Viewer Source Code
  10.  * The source code in this file ("Source Code") is provided by Linden Lab
  11.  * to you under the terms of the GNU General Public License, version 2.0
  12.  * ("GPL"), unless you have obtained a separate licensing agreement
  13.  * ("Other License"), formally executed by you and Linden Lab.  Terms of
  14.  * the GPL can be found in doc/GPL-license.txt in this distribution, or
  15.  * online at http://secondlifegrid.net/programs/open_source/licensing/gplv2
  16.  * 
  17.  * There are special exceptions to the terms and conditions of the GPL as
  18.  * it is applied to this Source Code. View the full text of the exception
  19.  * in the file doc/FLOSS-exception.txt in this software distribution, or
  20.  * online at
  21.  * http://secondlifegrid.net/programs/open_source/licensing/flossexception
  22.  * 
  23.  * By copying, modifying or distributing this software, you acknowledge
  24.  * that you have read and understood your obligations described above,
  25.  * and agree to abide by those obligations.
  26.  * 
  27.  * ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO
  28.  * WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY,
  29.  * COMPLETENESS OR PERFORMANCE.
  30.  * $/LicenseInfo$
  31.  */
  32. #include "linden_common.h"
  33. #include "llxfer.h"
  34. #include "lluuid.h"
  35. #include "llerror.h"
  36. #include "llmath.h"
  37. #include "u64.h"
  38. //number of bytes sent in each message
  39. const U32 LL_XFER_CHUNK_SIZE = 1000;
  40. const U32 LLXfer::XFER_FILE = 1;
  41. const U32 LLXfer::XFER_VFILE = 2;
  42. const U32 LLXfer::XFER_MEM = 3;
  43. ///////////////////////////////////////////////////////////
  44. LLXfer::LLXfer (S32 chunk_size)
  45. {
  46. init(chunk_size);
  47. }
  48. ///////////////////////////////////////////////////////////
  49. LLXfer::~LLXfer ()
  50. {
  51. cleanup();
  52. }
  53. ///////////////////////////////////////////////////////////
  54. void LLXfer::init (S32 chunk_size)
  55. {
  56. mID = 0;
  57. mPacketNum = -1; // there's a preincrement before sending the zeroth packet
  58. mXferSize = 0;
  59. mStatus = e_LL_XFER_UNINITIALIZED;
  60. mNext = NULL;
  61. mWaitingForACK = FALSE;
  62. mCallback = NULL;
  63. mCallbackDataHandle = NULL;
  64. mCallbackResult = 0;
  65. mBufferContainsEOF = FALSE;
  66. mBuffer = NULL;
  67. mBufferLength = 0;
  68. mBufferStartOffset = 0;
  69. mRetries = 0;
  70. if (chunk_size < 1)
  71. {
  72. chunk_size = LL_XFER_CHUNK_SIZE;
  73. }
  74. mChunkSize = chunk_size;
  75. }
  76. ///////////////////////////////////////////////////////////
  77. void LLXfer::cleanup ()
  78. {
  79. if (mBuffer)
  80. {
  81. delete[] mBuffer;
  82. mBuffer = NULL;
  83. }
  84. }
  85. ///////////////////////////////////////////////////////////
  86. S32 LLXfer::startSend (U64 xfer_id, const LLHost &remote_host)
  87. {
  88. llwarns << "undifferentiated LLXfer::startSend for " << getFileName() << llendl;
  89. return (-1);
  90. }
  91. ///////////////////////////////////////////////////////////
  92. void LLXfer::setXferSize (S32 xfer_size)
  93. {
  94. mXferSize = xfer_size;
  95. // cout << "starting transfer of size: " << xfer_size << endl;
  96. }
  97. ///////////////////////////////////////////////////////////
  98. S32 LLXfer::startDownload()
  99. {
  100. llwarns << "undifferentiated LLXfer::startDownload for " << getFileName()
  101. << llendl;
  102. return (-1);
  103. }
  104. ///////////////////////////////////////////////////////////
  105. S32 LLXfer::receiveData (char *datap, S32 data_size)
  106. {
  107. S32 retval = 0;
  108. if (((S32) mBufferLength + data_size) > getMaxBufferSize())
  109. {
  110. retval = flush();
  111. }
  112. if (!retval)
  113. {
  114. if (datap != NULL)
  115. {
  116. memcpy(&mBuffer[mBufferLength],datap,data_size); /*Flawfinder: ignore*/
  117. mBufferLength += data_size;
  118. }
  119. else
  120. {
  121. llerrs << "NULL data passed in receiveData" << llendl;
  122. }
  123. }
  124. return (retval);
  125. }
  126. ///////////////////////////////////////////////////////////
  127. S32 LLXfer::flush()
  128. {
  129. // only files have somewhere to flush to
  130. // if we get called with a flush it means we've blown past our
  131. // allocated buffer size
  132. return (-1);
  133. }
  134. ///////////////////////////////////////////////////////////
  135. S32 LLXfer::suck(S32 start_position)
  136. {
  137. llwarns << "Attempted to send a packet outside the buffer bounds in LLXfer::suck()" << llendl;
  138. return (-1);
  139. }
  140. ///////////////////////////////////////////////////////////
  141. void LLXfer::sendPacket(S32 packet_num)
  142. {
  143. char fdata_buf[LL_XFER_LARGE_PAYLOAD+4]; /* Flawfinder: ignore */
  144. S32 fdata_size = mChunkSize;
  145. BOOL last_packet = FALSE;
  146. S32 num_copy = 0;
  147. // if the desired packet is not in our current buffered excerpt from the file. . . 
  148. if (((U32)packet_num*fdata_size < mBufferStartOffset) 
  149. || ((U32)llmin((U32)mXferSize,(U32)((U32)(packet_num+1)*fdata_size)) > mBufferStartOffset + mBufferLength))
  150. {
  151. if (suck(packet_num*fdata_size))  // returns non-zero on failure
  152. {
  153. abort(LL_ERR_EOF);
  154. return;
  155. }
  156. }
  157. S32 desired_read_position = 0;
  158. desired_read_position = packet_num * fdata_size - mBufferStartOffset;
  159. fdata_size = llmin((S32)mBufferLength-desired_read_position, mChunkSize);
  160. if (fdata_size < 0)
  161. {
  162. llwarns << "negative data size in xfer send, aborting" << llendl;
  163. abort(LL_ERR_EOF);
  164. return;
  165. }
  166. if (((U32)(desired_read_position + fdata_size) >= (U32)mBufferLength) && (mBufferContainsEOF))
  167. {
  168. last_packet = TRUE;
  169. }
  170. if (packet_num)
  171. num_copy = llmin(fdata_size, (S32)sizeof(fdata_buf));
  172. num_copy = llmin(num_copy, (S32)(mBufferLength - desired_read_position));
  173. if (num_copy > 0)
  174. {
  175. memcpy(fdata_buf,&mBuffer[desired_read_position],num_copy); /*Flawfinder: ignore*/
  176. }
  177. }
  178. else  
  179. {
  180. // if we're the first packet, encode size as an additional S32
  181. // at start of data.
  182. num_copy = llmin(fdata_size, (S32)(sizeof(fdata_buf)-sizeof(S32)));
  183. num_copy = llmin(
  184. num_copy,
  185. (S32)(mBufferLength - desired_read_position));
  186. if (num_copy > 0)
  187. {
  188. memcpy( /*Flawfinder: ignore*/
  189. fdata_buf + sizeof(S32),
  190. &mBuffer[desired_read_position],
  191. num_copy);
  192. }
  193. fdata_size += sizeof(S32);
  194. htonmemcpy(fdata_buf,&mXferSize, MVT_S32, sizeof(S32));
  195. }
  196. S32 encoded_packetnum = encodePacketNum(packet_num,last_packet);
  197. if (fdata_size)
  198. {
  199. // send the packet 
  200. gMessageSystem->newMessageFast(_PREHASH_SendXferPacket);
  201. gMessageSystem->nextBlockFast(_PREHASH_XferID);
  202. gMessageSystem->addU64Fast(_PREHASH_ID, mID);
  203. gMessageSystem->addU32Fast(_PREHASH_Packet, encoded_packetnum);
  204. gMessageSystem->nextBlockFast(_PREHASH_DataPacket);
  205. gMessageSystem->addBinaryDataFast(_PREHASH_Data, &fdata_buf,fdata_size);
  206. gMessageSystem->sendMessage(mRemoteHost);
  207. ACKTimer.reset();
  208. mWaitingForACK = TRUE;
  209. }
  210. if (last_packet)
  211. {
  212. mStatus = e_LL_XFER_COMPLETE;
  213. }
  214. else
  215. {
  216. mStatus = e_LL_XFER_IN_PROGRESS;
  217. }
  218. }
  219. ///////////////////////////////////////////////////////////
  220. void LLXfer::sendNextPacket()
  221. {
  222. mRetries = 0;
  223. sendPacket(++mPacketNum);
  224. }
  225. ///////////////////////////////////////////////////////////
  226. void LLXfer::resendLastPacket()
  227. {
  228. mRetries++;
  229. sendPacket(mPacketNum);
  230. }
  231. ///////////////////////////////////////////////////////////
  232. S32 LLXfer::processEOF()
  233. {
  234. S32 retval = 0;
  235. mStatus = e_LL_XFER_COMPLETE;
  236. if (LL_ERR_NOERR == mCallbackResult)
  237. {
  238. llinfos << "xfer from " << mRemoteHost << " complete: " << getFileName()
  239. << llendl;
  240. }
  241. else
  242. {
  243. llinfos << "xfer from " << mRemoteHost << " failed, code "
  244. << mCallbackResult << ": " << getFileName() << llendl;
  245. }
  246. if (mCallback)
  247. {
  248. mCallback(mCallbackDataHandle,mCallbackResult,LL_EXSTAT_NONE);
  249. }
  250. return(retval);
  251. }
  252. ///////////////////////////////////////////////////////////
  253. S32 LLXfer::encodePacketNum(S32 packet_num, BOOL is_EOF)
  254. {
  255. if (is_EOF)
  256. {
  257. packet_num |= 0x80000000;
  258. }
  259. return packet_num;
  260. }
  261. ///////////////////////////////////////////////////////////
  262. void LLXfer::abort (S32 result_code)
  263. {
  264. mCallbackResult = result_code;
  265. llinfos << "Aborting xfer from " << mRemoteHost << " named " << getFileName()
  266. << " - error: " << result_code << llendl;
  267. gMessageSystem->newMessageFast(_PREHASH_AbortXfer);
  268. gMessageSystem->nextBlockFast(_PREHASH_XferID);
  269. gMessageSystem->addU64Fast(_PREHASH_ID, mID);
  270. gMessageSystem->addS32Fast(_PREHASH_Result, result_code);
  271. gMessageSystem->sendMessage(mRemoteHost);
  272. mStatus = e_LL_XFER_ABORTED;
  273. }
  274. ///////////////////////////////////////////////////////////
  275. std::string LLXfer::getFileName() 
  276. {
  277. return U64_to_str(mID);
  278. }
  279. ///////////////////////////////////////////////////////////
  280. U32 LLXfer::getXferTypeTag()
  281. {
  282. return 0;
  283. }
  284. ///////////////////////////////////////////////////////////
  285. S32 LLXfer::getMaxBufferSize ()
  286. {
  287. return(mXferSize);
  288. }
  289. std::ostream& operator<< (std::ostream& os, LLXfer &hh)
  290. {
  291. os << hh.getFileName() ;
  292. return os;
  293. }