// Copyright 2016 The SwiftShader Authors. All Rights Reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #include "Direct3DVertexBuffer9.hpp" #include "Direct3DDevice9.hpp" #include "Resource.hpp" #include "Debug.hpp" #include <assert.h> namespace D3D9 { Direct3DVertexBuffer9::Direct3DVertexBuffer9(Direct3DDevice9 *device, unsigned int length, unsigned long usage, long FVF, D3DPOOL pool) : Direct3DResource9(device, D3DRTYPE_VERTEXBUFFER, pool, length), length(length), usage(usage), FVF(FVF) { if(FVF) { unsigned int stride = 0; switch(FVF & D3DFVF_POSITION_MASK) { case D3DFVF_XYZ: stride += 12; break; case D3DFVF_XYZRHW: stride += 16; break; case D3DFVF_XYZB1: stride += 16; break; case D3DFVF_XYZB2: stride += 20; break; case D3DFVF_XYZB3: stride += 24; break; case D3DFVF_XYZB4: stride += 28; break; case D3DFVF_XYZB5: stride += 32; break; case D3DFVF_XYZW: stride += 16; break; } if(FVF & D3DFVF_NORMAL) stride += 12; if(FVF & D3DFVF_PSIZE) stride += 4; if(FVF & D3DFVF_DIFFUSE) stride += 4; if(FVF & D3DFVF_SPECULAR) stride += 4; switch((FVF & D3DFVF_TEXCOUNT_MASK) >> D3DFVF_TEXCOUNT_SHIFT) { case 8: stride += 4 + 4 * ((1 + (FVF >> 30)) % 4); case 7: stride += 4 + 4 * ((1 + (FVF >> 28)) % 4); case 6: stride += 4 + 4 * ((1 + (FVF >> 26)) % 4); case 5: stride += 4 + 4 * ((1 + (FVF >> 24)) % 4); case 4: stride += 4 + 4 * ((1 + (FVF >> 22)) % 4); case 3: stride += 4 + 4 * ((1 + (FVF >> 20)) % 4); case 2: stride += 4 + 4 * ((1 + (FVF >> 18)) % 4); case 1: stride += 4 + 4 * ((1 + (FVF >> 16)) % 4); case 0: break; default: ASSERT(false); } ASSERT(length >= stride); // FIXME } vertexBuffer = new sw::Resource(length + 192 + 1024); // NOTE: Applications can 'overshoot' while writing vertices lockCount = 0; } Direct3DVertexBuffer9::~Direct3DVertexBuffer9() { vertexBuffer->destruct(); } long Direct3DVertexBuffer9::QueryInterface(const IID &iid, void **object) { CriticalSection cs(device); TRACE(""); if(iid == IID_IDirect3DVertexBuffer9 || iid == IID_IDirect3DResource9 || iid == IID_IUnknown) { AddRef(); *object = this; return S_OK; } *object = 0; return NOINTERFACE(iid); } unsigned long Direct3DVertexBuffer9::AddRef() { TRACE(""); return Direct3DResource9::AddRef(); } unsigned long Direct3DVertexBuffer9::Release() { TRACE(""); return Direct3DResource9::Release(); } long Direct3DVertexBuffer9::FreePrivateData(const GUID &guid) { CriticalSection cs(device); TRACE(""); return Direct3DResource9::FreePrivateData(guid); } long Direct3DVertexBuffer9::GetPrivateData(const GUID &guid, void *data, unsigned long *size) { CriticalSection cs(device); TRACE(""); return Direct3DResource9::GetPrivateData(guid, data, size); } void Direct3DVertexBuffer9::PreLoad() { CriticalSection cs(device); TRACE(""); Direct3DResource9::PreLoad(); } long Direct3DVertexBuffer9::SetPrivateData(const GUID &guid, const void *data, unsigned long size, unsigned long flags) { CriticalSection cs(device); TRACE(""); return Direct3DResource9::SetPrivateData(guid, data, size, flags); } long Direct3DVertexBuffer9::GetDevice(IDirect3DDevice9 **device) { CriticalSection cs(this->device); TRACE(""); return Direct3DResource9::GetDevice(device); } unsigned long Direct3DVertexBuffer9::SetPriority(unsigned long newPriority) { CriticalSection cs(device); TRACE(""); return Direct3DResource9::SetPriority(newPriority); } unsigned long Direct3DVertexBuffer9::GetPriority() { CriticalSection cs(device); TRACE(""); return Direct3DResource9::GetPriority(); } D3DRESOURCETYPE Direct3DVertexBuffer9::GetType() { CriticalSection cs(device); TRACE(""); return Direct3DResource9::GetType(); } long Direct3DVertexBuffer9::Lock(unsigned int offset, unsigned int size, void **data, unsigned long flags) { CriticalSection cs(device); TRACE(""); if(offset == 0 && size == 0) // Lock whole buffer { size = length; } if(!data || offset + size > length) { return INVALIDCALL(); } void *buffer; if(flags & D3DLOCK_DISCARD/* && usage & D3DUSAGE_DYNAMIC*/) { vertexBuffer->destruct(); vertexBuffer = new sw::Resource(length + 192 + 1024); // NOTE: Applications can 'overshoot' while writing vertices buffer = (void*)vertexBuffer->data(); } else if(flags & D3DLOCK_NOOVERWRITE/* && usage & D3DUSAGE_DYNAMIC*/) { buffer = (void*)vertexBuffer->data(); } else { buffer = vertexBuffer->lock(sw::PUBLIC); lockCount++; } *data = (unsigned char*)buffer + offset; return D3D_OK; } long Direct3DVertexBuffer9::Unlock() { CriticalSection cs(device); TRACE(""); if(lockCount > 0) { vertexBuffer->unlock(); lockCount--; } return D3D_OK; } long Direct3DVertexBuffer9::GetDesc(D3DVERTEXBUFFER_DESC *description) { CriticalSection cs(device); TRACE(""); if(!description) { return INVALIDCALL(); } description->FVF = FVF; description->Format = D3DFMT_VERTEXDATA; description->Pool = pool; description->Size = length; description->Type = D3DRTYPE_VERTEXBUFFER; description->Usage = usage; return D3D_OK; } int Direct3DVertexBuffer9::getLength() const { return length; } sw::Resource *Direct3DVertexBuffer9::getResource() const { return vertexBuffer; } }