| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351 |
- /*
- -----------------------------------------------------------------------------
- This source file is part of OGRE
- (Object-oriented Graphics Rendering Engine)
- For the latest info, see http://www.ogre3d.org/
- Copyright (c) 2000-2011 Torus Knot Software Ltd
- Permission is hereby granted, free of charge, to any person obtaining a copy
- of this software and associated documentation files (the "Software"), to deal
- in the Software without restriction, including without limitation the rights
- to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- copies of the Software, and to permit persons to whom the Software is
- furnished to do so, subject to the following conditions:
- The above copyright notice and this permission notice shall be included in
- all copies or substantial portions of the Software.
- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
- THE SOFTWARE.
- -----------------------------------------------------------------------------
- */
- #include "CmD3D9HardwareVertexBuffer.h"
- #include "CmD3D9Mappings.h"
- #include "CmException.h"
- #include "CmD3D9HardwareBufferManager.h"
- #include "CmD3D9RenderSystem.h"
- #include "CmD3D9Device.h"
- #include "CmD3D9ResourceManager.h"
- namespace CamelotEngine {
- //---------------------------------------------------------------------
- D3D9HardwareVertexBuffer::D3D9HardwareVertexBuffer(HardwareBufferManagerBase* mgr, size_t vertexSize,
- size_t numVertices, HardwareBuffer::Usage usage,
- bool useSystemMemory, bool useShadowBuffer)
- : HardwareVertexBuffer(mgr, vertexSize, numVertices, usage, useSystemMemory, useShadowBuffer)
- {
- D3D9_DEVICE_ACCESS_CRITICAL_SECTION
- D3DPOOL eResourcePool;
-
- #if OGRE_D3D_MANAGE_BUFFERS
- eResourcePool = useSystemMemory? D3DPOOL_SYSTEMMEM :
- // If not system mem, use managed pool UNLESS buffer is discardable
- // if discardable, keeping the software backing is expensive
- (usage & HardwareBuffer::HBU_DISCARDABLE)? D3DPOOL_DEFAULT : D3DPOOL_MANAGED;
- #else
- eResourcePool = useSystemMemory? D3DPOOL_SYSTEMMEM : D3DPOOL_DEFAULT;
- #endif
- // Set the desired memory pool.
- mBufferDesc.Pool = eResourcePool;
- // Allocate the system memory buffer.
- mSystemMemoryBuffer = new char [getSizeInBytes()];
- memset(mSystemMemoryBuffer, 0, getSizeInBytes());
- // Case we have to create this buffer resource on loading.
- if (D3D9RenderSystem::getResourceManager()->getCreationPolicy() == RCP_CREATE_ON_ALL_DEVICES)
- {
- for (UINT32 i = 0; i < D3D9RenderSystem::getResourceCreationDeviceCount(); ++i)
- {
- IDirect3DDevice9* d3d9Device = D3D9RenderSystem::getResourceCreationDevice(i);
- createBuffer(d3d9Device, mBufferDesc.Pool);
- }
- }
- }
- //---------------------------------------------------------------------
- D3D9HardwareVertexBuffer::~D3D9HardwareVertexBuffer()
- {
- D3D9_DEVICE_ACCESS_CRITICAL_SECTION
- DeviceToBufferResourcesIterator it = mMapDeviceToBufferResources.begin();
- while (it != mMapDeviceToBufferResources.end())
- {
- SAFE_RELEASE(it->second->mBuffer);
- SAFE_DELETE(it->second);
- ++it;
- }
- mMapDeviceToBufferResources.clear();
- SAFE_DELETE_ARRAY(mSystemMemoryBuffer);
- }
- //---------------------------------------------------------------------
- void* D3D9HardwareVertexBuffer::lockImpl(size_t offset,
- size_t length, LockOptions options)
- {
- D3D9_DEVICE_ACCESS_CRITICAL_SECTION
- if (options != HBL_READ_ONLY)
- {
- DeviceToBufferResourcesIterator it = mMapDeviceToBufferResources.begin();
- while (it != mMapDeviceToBufferResources.end())
- {
- BufferResources* bufferResources = it->second;
- bufferResources->mOutOfDate = true;
- if(bufferResources->mLockLength > 0)
- {
- size_t highPoint = std::max( offset + length,
- bufferResources->mLockOffset + bufferResources->mLockLength );
- bufferResources->mLockOffset = std::min( bufferResources->mLockOffset, offset );
- bufferResources->mLockLength = highPoint - bufferResources->mLockOffset;
- }
- else
- {
- if (offset < bufferResources->mLockOffset)
- bufferResources->mLockOffset = offset;
- if (length > bufferResources->mLockLength)
- bufferResources->mLockLength = length;
- }
-
- if (bufferResources->mLockOptions != HBL_DISCARD)
- bufferResources->mLockOptions = options;
- ++it;
- }
- }
- return mSystemMemoryBuffer + offset;
- }
- //---------------------------------------------------------------------
- void D3D9HardwareVertexBuffer::unlockImpl(void)
- {
- D3D9_DEVICE_ACCESS_CRITICAL_SECTION
- DeviceToBufferResourcesIterator it = mMapDeviceToBufferResources.begin();
- // TODO PORT - Don't know what the next frame number is. Add a method for that
- //UINT32 nextFrameNumber = Root::getSingleton().getNextFrameNumber();
- UINT32 nextFrameNumber = 0;
- while (it != mMapDeviceToBufferResources.end())
- {
- BufferResources* bufferResources = it->second;
- if (bufferResources->mOutOfDate &&
- bufferResources->mBuffer != NULL&&
- nextFrameNumber - bufferResources->mLastUsedFrame <= 1)
- updateBufferResources(mSystemMemoryBuffer, bufferResources);
- ++it;
- }
- }
- //---------------------------------------------------------------------
- void D3D9HardwareVertexBuffer::readData(size_t offset, size_t length,
- void* pDest)
- {
- // There is no functional interface in D3D, just do via manual
- // lock, copy & unlock
- void* pSrc = this->lock(offset, length, HardwareBuffer::HBL_READ_ONLY);
- memcpy(pDest, pSrc, length);
- this->unlock();
- }
- //---------------------------------------------------------------------
- void D3D9HardwareVertexBuffer::writeData(size_t offset, size_t length,
- const void* pSource,
- bool discardWholeBuffer)
- {
- // There is no functional interface in D3D, just do via manual
- // lock, copy & unlock
- void* pDst = this->lock(offset, length,
- discardWholeBuffer ? HardwareBuffer::HBL_DISCARD : HardwareBuffer::HBL_NORMAL);
- memcpy(pDst, pSource, length);
- this->unlock();
- }
- //---------------------------------------------------------------------
- void D3D9HardwareVertexBuffer::notifyOnDeviceCreate(IDirect3DDevice9* d3d9Device)
- {
- D3D9_DEVICE_ACCESS_CRITICAL_SECTION
- if (D3D9RenderSystem::getResourceManager()->getCreationPolicy() == RCP_CREATE_ON_ALL_DEVICES)
- createBuffer(d3d9Device, mBufferDesc.Pool);
- }
- //---------------------------------------------------------------------
- void D3D9HardwareVertexBuffer::notifyOnDeviceDestroy(IDirect3DDevice9* d3d9Device)
- {
- D3D9_DEVICE_ACCESS_CRITICAL_SECTION
- DeviceToBufferResourcesIterator it = mMapDeviceToBufferResources.find(d3d9Device);
- if (it != mMapDeviceToBufferResources.end())
- {
- SAFE_RELEASE(it->second->mBuffer);
- SAFE_DELETE(it->second);
- mMapDeviceToBufferResources.erase(it);
- }
- }
- //---------------------------------------------------------------------
- void D3D9HardwareVertexBuffer::notifyOnDeviceLost(IDirect3DDevice9* d3d9Device)
- {
- D3D9_DEVICE_ACCESS_CRITICAL_SECTION
- if (mBufferDesc.Pool == D3DPOOL_DEFAULT)
- {
- DeviceToBufferResourcesIterator it = mMapDeviceToBufferResources.find(d3d9Device);
- if (it != mMapDeviceToBufferResources.end())
- {
- SAFE_RELEASE(it->second->mBuffer);
- }
- }
- }
- //---------------------------------------------------------------------
- void D3D9HardwareVertexBuffer::notifyOnDeviceReset(IDirect3DDevice9* d3d9Device)
- {
- D3D9_DEVICE_ACCESS_CRITICAL_SECTION
- if (D3D9RenderSystem::getResourceManager()->getCreationPolicy() == RCP_CREATE_ON_ALL_DEVICES)
- createBuffer(d3d9Device, mBufferDesc.Pool);
- }
- //---------------------------------------------------------------------
- void D3D9HardwareVertexBuffer::createBuffer(IDirect3DDevice9* d3d9Device, D3DPOOL ePool)
- {
- D3D9_DEVICE_ACCESS_CRITICAL_SECTION
- BufferResources* bufferResources;
- HRESULT hr;
- DeviceToBufferResourcesIterator it;
- // Find the vertex buffer of this device.
- it = mMapDeviceToBufferResources.find(d3d9Device);
- if (it != mMapDeviceToBufferResources.end())
- {
- bufferResources = it->second;
- SAFE_RELEASE(bufferResources->mBuffer);
- }
- else
- {
- bufferResources = new BufferResources;
- mMapDeviceToBufferResources[d3d9Device] = bufferResources;
- }
- bufferResources->mBuffer = NULL;
- bufferResources->mOutOfDate = true;
- bufferResources->mLockOffset = 0;
- bufferResources->mLockLength = getSizeInBytes();
- bufferResources->mLockOptions = HBL_NORMAL;
- // TODO PORT - Don't know what the next frame number is. Add a method for that
- //bufferResources->mLastUsedFrame = Root::getSingleton().getNextFrameNumber();
- bufferResources->mLastUsedFrame = 0;
-
- // Create the vertex buffer
- hr = d3d9Device->CreateVertexBuffer(
- static_cast<UINT>(mSizeInBytes),
- D3D9Mappings::get(mUsage),
- 0, // No FVF here, thank you.
- ePool,
- &bufferResources->mBuffer,
- NULL);
- if (FAILED(hr))
- {
- String msg = DXGetErrorDescription(hr);
- OGRE_EXCEPT(Exception::ERR_RENDERINGAPI_ERROR,
- "Cannot restore D3D9 vertex buffer: " + msg,
- "D3D9HardwareVertexBuffer::createBuffer");
- }
- hr = bufferResources->mBuffer->GetDesc(&mBufferDesc);
- if (FAILED(hr))
- {
- String msg = DXGetErrorDescription(hr);
- OGRE_EXCEPT(Exception::ERR_RENDERINGAPI_ERROR,
- "Cannot get D3D9 Vertex buffer desc: " + msg,
- "D3D9HardwareVertexBuffer::createBuffer");
- }
- }
- //---------------------------------------------------------------------
- IDirect3DVertexBuffer9* D3D9HardwareVertexBuffer::getD3D9VertexBuffer(void)
- {
- IDirect3DDevice9* d3d9Device = D3D9RenderSystem::getActiveD3D9Device();
- DeviceToBufferResourcesIterator it;
- // Find the vertex buffer of this device.
- it = mMapDeviceToBufferResources.find(d3d9Device);
- // Case vertex buffer was not found for the current device -> create it.
- if (it == mMapDeviceToBufferResources.end() || it->second->mBuffer == NULL)
- {
- createBuffer(d3d9Device, mBufferDesc.Pool);
- it = mMapDeviceToBufferResources.find(d3d9Device);
- }
- if (it->second->mOutOfDate)
- updateBufferResources(mSystemMemoryBuffer, it->second);
-
- // TODO PORT - Don't know what the next frame number is. Add a method for that
- //it->second->mLastUsedFrame = Root::getSingleton().getNextFrameNumber();
- it->second->mLastUsedFrame = 0;
- return it->second->mBuffer;
- }
- //---------------------------------------------------------------------
- bool D3D9HardwareVertexBuffer::updateBufferResources(const char* systemMemoryBuffer,
- BufferResources* bufferResources)
- {
- assert(bufferResources != NULL);
- assert(bufferResources->mBuffer != NULL);
- assert(bufferResources->mOutOfDate);
-
- void* dstBytes;
- HRESULT hr;
-
- // Lock the buffer.
- hr = bufferResources->mBuffer->Lock(
- static_cast<UINT>(bufferResources->mLockOffset),
- static_cast<UINT>(bufferResources->mLockLength),
- &dstBytes,
- D3D9Mappings::get(bufferResources->mLockOptions, mUsage));
- if (FAILED(hr))
- {
- String msg = DXGetErrorDescription(hr);
- OGRE_EXCEPT(Exception::ERR_RENDERINGAPI_ERROR,
- "Cannot lock D3D9 vertex buffer: " + msg,
- "D3D9HardwareVertexBuffer::updateBuffer");
- }
- memcpy(dstBytes, systemMemoryBuffer + bufferResources->mLockOffset, bufferResources->mLockLength);
- // Unlock the buffer.
- hr = bufferResources->mBuffer->Unlock();
- if (FAILED(hr))
- {
- String msg = DXGetErrorDescription(hr);
- OGRE_EXCEPT(Exception::ERR_RENDERINGAPI_ERROR,
- "Cannot unlock D3D9 vertex buffer: " + msg,
- "D3D9HardwareVertexBuffer::updateBuffer");
- }
- bufferResources->mOutOfDate = false;
- bufferResources->mLockOffset = mSizeInBytes;
- bufferResources->mLockLength = 0;
- bufferResources->mLockOptions = HBL_NORMAL;
- return true;
- }
- }
|