123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327 |
- /**
- * @file llxfer_vfile.cpp
- * @brief implementation of LLXfer_VFile class for a single xfer (vfile).
- *
- * $LicenseInfo:firstyear=2002&license=viewergpl$
- *
- * Copyright (c) 2002-2009, Linden Research, Inc.
- *
- * Second Life Viewer Source Code
- * The source code in this file ("Source Code") is provided by Linden Lab
- * to you under the terms of the GNU General Public License, version 2.0
- * ("GPL"), unless you have obtained a separate licensing agreement
- * ("Other License"), formally executed by you and Linden Lab. Terms of
- * the GPL can be found in doc/GPL-license.txt in this distribution, or
- * online at http://secondlifegrid.net/programs/open_source/licensing/gplv2
- *
- * There are special exceptions to the terms and conditions of the GPL as
- * it is applied to this Source Code. View the full text of the exception
- * in the file doc/FLOSS-exception.txt in this software distribution, or
- * online at
- * http://secondlifegrid.net/programs/open_source/licensing/flossexception
- *
- * By copying, modifying or distributing this software, you acknowledge
- * that you have read and understood your obligations described above,
- * and agree to abide by those obligations.
- *
- * ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO
- * WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY,
- * COMPLETENESS OR PERFORMANCE.
- * $/LicenseInfo$
- */
- #include "linden_common.h"
- #include "llxfer_vfile.h"
- #include "lldir.h"
- #include "llfilesystem.h"
- #include "llmath.h"
- LLXfer_VFile::LLXfer_VFile()
- : LLXfer(-1),
- mDeleteTempFile(false)
- {
- init(LLUUID::null, LLAssetType::AT_NONE);
- }
- LLXfer_VFile::LLXfer_VFile(const LLUUID& local_id, LLAssetType::EType type)
- : LLXfer(-1),
- mDeleteTempFile(false)
- {
- init(local_id, type);
- }
- LLXfer_VFile::~LLXfer_VFile()
- {
- cleanup();
- }
- void LLXfer_VFile::init(const LLUUID& local_id, LLAssetType::EType type)
- {
- mLocalID = local_id;
- mType = type;
- mVFile = NULL;
- std::string id_string;
- mLocalID.toString(id_string);
- mName = llformat("VFile %s:%s", id_string.c_str(),
- LLAssetType::lookup(mType));
- }
- void LLXfer_VFile::cleanup()
- {
- if (mTempID.notNull() && mDeleteTempFile)
- {
- LLFileSystem file(mTempID);
- if (file.exists())
- {
- file.remove();
- }
- else
- {
- llwarns << "No matching cache file " << file.getName()
- << ". Nothing deleted." << llendl;
- }
- }
- if (mVFile)
- {
- delete mVFile;
- mVFile = NULL;
- }
- LLXfer::cleanup();
- }
- S32 LLXfer_VFile::initializeRequest(U64 xfer_id, const LLUUID& local_id,
- const LLUUID& remote_id,
- LLAssetType::EType type,
- const LLHost& remote_host,
- void (*callback)(void**, S32, LLExtStat),
- void** user_data)
- {
- mRemoteHost = remote_host;
- mLocalID = local_id;
- mRemoteID = remote_id;
- mType = type;
- mID = xfer_id;
- mCallback = callback;
- mCallbackDataHandle = user_data;
- mCallbackResult = LL_ERR_NOERR;
- std::string id_string;
- mLocalID.toString(id_string);
- mName = llformat("VFile %s:%s", id_string.c_str(),
- LLAssetType::lookup(mType));
- llinfos << "Requesting " << mName << llendl;
- if (mBuffer)
- {
- delete[] mBuffer;
- mBuffer = NULL;
- }
- mBuffer = new char[LL_MAX_XFER_FILE_BUFFER];
- mBufferLength = 0;
- mPacketNum = 0;
- mTempID.generate();
- mDeleteTempFile = true;
- mStatus = e_LL_XFER_PENDING;
- return LL_ERR_NOERR;
- }
- S32 LLXfer_VFile::startDownload()
- {
- LLMessageSystem* msg = gMessageSystemp;
- msg->newMessageFast(_PREHASH_RequestXfer);
- msg->nextBlockFast(_PREHASH_XferID);
- msg->addU64Fast(_PREHASH_ID, mID);
- msg->addStringFast(_PREHASH_Filename, "");
- msg->addU8("FilePath", (U8) LL_PATH_NONE);
- msg->addBool("DeleteOnCompletion", false);
- msg->addBool("UseBigPackets", mChunkSize == LL_XFER_LARGE_PAYLOAD);
- msg->addUUIDFast(_PREHASH_VFileID, mRemoteID);
- msg->addS16Fast(_PREHASH_VFileType, (S16)mType);
- msg->sendReliable(mRemoteHost);
- mStatus = e_LL_XFER_IN_PROGRESS;
- return LL_ERR_NOERR;
- }
- S32 LLXfer_VFile::startSend(U64 xfer_id, const LLHost& remote_host)
- {
- mRemoteHost = remote_host;
- mID = xfer_id;
- mPacketNum = -1;
- if (mBuffer)
- {
- delete[] mBuffer;
- }
- mBuffer = new char[LL_MAX_XFER_FILE_BUFFER];
- mBufferLength = 0;
- mBufferStartOffset = 0;
- if (mVFile)
- {
- delete mVFile;
- }
- mVFile = new LLFileSystem(mLocalID);
- if (!mVFile->exists())
- {
- llwarns << "Cannot read cache file " << mVFile->getName()
- << ". Aborted." << llendl;
- delete mVFile;
- mVFile = NULL;
- return LL_ERR_FILE_NOT_FOUND;
- }
- S32 size = mVFile->getSize();
- if (size <= 0)
- {
- llwarns << "Empty cache file " << mVFile->getName() << ". Aborted."
- << llendl;
- delete mVFile;
- mVFile = NULL;
- return LL_ERR_FILE_EMPTY;
- }
- setXferSize(size);
- mStatus = e_LL_XFER_PENDING;
- return LL_ERR_NOERR;
- }
- void LLXfer_VFile::closeFileHandle()
- {
- if (mVFile)
- {
- delete mVFile;
- mVFile = NULL;
- }
- }
- S32 LLXfer_VFile::reopenFileHandle()
- {
- if (!mVFile)
- {
- mVFile = new LLFileSystem(mLocalID);
- if (!mVFile->exists())
- {
- llwarns << "Cannot read cache file; " << mVFile->getName()
- << llendl;
- delete mVFile;
- mVFile = NULL;
- return LL_ERR_FILE_NOT_FOUND;
- }
- }
- return LL_ERR_NOERR;
- }
- void LLXfer_VFile::setXferSize(S32 xfer_size)
- {
- LLXfer::setXferSize(xfer_size);
- // Do not do this on the server side, where we have a persistent mVFile
- // It would be nice if LLXFers could tell which end of the pipe they were
- if (!mVFile)
- {
- LLFileSystem file(mTempID, LLFileSystem::APPEND);
- }
- }
- S32 LLXfer_VFile::suck(S32 start_position)
- {
- S32 retval = 0;
- if (mVFile)
- {
- // Grab a buffer from the right place in the file
- if (!mVFile->seek(start_position, 0))
- {
- llwarns << "VFile Xfer Can't seek to position: " << start_position
- << " - File length: " << mVFile->getSize()
- << " - While sending file " << mLocalID << llendl;
- return -1;
- }
- if (mVFile->read((U8*)mBuffer, LL_MAX_XFER_FILE_BUFFER))
- {
- mBufferLength = mVFile->getLastBytesRead();
- mBufferStartOffset = start_position;
- mBufferContainsEOF = mVFile->eof();
- }
- else
- {
- retval = -1;
- }
- }
- else
- {
- retval = -1;
- }
- return retval;
- }
- S32 LLXfer_VFile::flush()
- {
- if (mBufferLength)
- {
- LLFileSystem file(mTempID, LLFileSystem::APPEND);
- file.write((U8*)mBuffer, mBufferLength);
- mBufferLength = 0;
- }
- return LL_ERR_NOERR;
- }
- S32 LLXfer_VFile::processEOF()
- {
- mStatus = e_LL_XFER_COMPLETE;
- flush();
- if (!mCallbackResult)
- {
- LLFileSystem file(mTempID);
- if (file.exists())
- {
- if (file.rename(mLocalID))
- {
- // Rename worked and the original file is gone. Clear flag so
- // that we do not attempt to delete the gone file in cleanup().
- mDeleteTempFile = false;
- }
- else
- {
- llwarns << "Unable to rename cache file: " << file.getName()
- << llendl;
- }
- }
- else
- {
- llwarns << "Cannot open cache file: " << file.getName() << llendl;
- }
- }
- if (mVFile)
- {
- delete mVFile;
- mVFile = NULL;
- }
- return LLXfer::processEOF();
- }
|