summaryrefslogtreecommitdiffstats
path: root/o3d/command_buffer/service
diff options
context:
space:
mode:
Diffstat (limited to 'o3d/command_buffer/service')
-rw-r--r--o3d/command_buffer/service/build.scons114
-rw-r--r--o3d/command_buffer/service/cross/big_test.cc106
-rw-r--r--o3d/command_buffer/service/cross/big_test_helpers.h70
-rw-r--r--o3d/command_buffer/service/cross/buffer_rpc.cc136
-rw-r--r--o3d/command_buffer/service/cross/buffer_rpc.h101
-rw-r--r--o3d/command_buffer/service/cross/buffer_rpc_test.cc171
-rw-r--r--o3d/command_buffer/service/cross/cmd_buffer_engine.cc290
-rw-r--r--o3d/command_buffer/service/cross/cmd_buffer_engine.h214
-rw-r--r--o3d/command_buffer/service/cross/cmd_buffer_engine_test.cc626
-rw-r--r--o3d/command_buffer/service/cross/cmd_parser.cc92
-rw-r--r--o3d/command_buffer/service/cross/cmd_parser.h113
-rw-r--r--o3d/command_buffer/service/cross/cmd_parser_test.cc315
-rw-r--r--o3d/command_buffer/service/cross/effect_utils.cc70
-rw-r--r--o3d/command_buffer/service/cross/effect_utils.h56
-rw-r--r--o3d/command_buffer/service/cross/effect_utils_test.cc86
-rw-r--r--o3d/command_buffer/service/cross/gapi_decoder.cc881
-rw-r--r--o3d/command_buffer/service/cross/gapi_decoder.h131
-rw-r--r--o3d/command_buffer/service/cross/gl/effect_gl.cc633
-rw-r--r--o3d/command_buffer/service/cross/gl/effect_gl.h151
-rw-r--r--o3d/command_buffer/service/cross/gl/gapi_gl.cc141
-rw-r--r--o3d/command_buffer/service/cross/gl/gapi_gl.h395
-rw-r--r--o3d/command_buffer/service/cross/gl/geometry_gl.cc548
-rw-r--r--o3d/command_buffer/service/cross/gl/geometry_gl.h144
-rw-r--r--o3d/command_buffer/service/cross/gl/gl_utils.h63
-rw-r--r--o3d/command_buffer/service/cross/gl/sampler_gl.cc235
-rw-r--r--o3d/command_buffer/service/cross/gl/sampler_gl.h87
-rw-r--r--o3d/command_buffer/service/cross/gl/states_gl.cc342
-rw-r--r--o3d/command_buffer/service/cross/gl/texture_gl.cc678
-rw-r--r--o3d/command_buffer/service/cross/gl/texture_gl.h223
-rw-r--r--o3d/command_buffer/service/cross/mocks.h149
-rw-r--r--o3d/command_buffer/service/cross/plugin.cc439
-rw-r--r--o3d/command_buffer/service/cross/resource.cc102
-rw-r--r--o3d/command_buffer/service/cross/resource.h249
-rw-r--r--o3d/command_buffer/service/cross/resource_test.cc128
-rw-r--r--o3d/command_buffer/service/cross/texture_utils.cc104
-rw-r--r--o3d/command_buffer/service/cross/texture_utils.h158
-rw-r--r--o3d/command_buffer/service/linux/big_test_main.cc125
-rw-r--r--o3d/command_buffer/service/linux/x_utils.cc93
-rw-r--r--o3d/command_buffer/service/linux/x_utils.h78
-rw-r--r--o3d/command_buffer/service/win/big_test_main.cc163
-rw-r--r--o3d/command_buffer/service/win/d3d9/d3d9_utils.h102
-rw-r--r--o3d/command_buffer/service/win/d3d9/effect_d3d9.cc569
-rw-r--r--o3d/command_buffer/service/win/d3d9/effect_d3d9.h127
-rw-r--r--o3d/command_buffer/service/win/d3d9/gapi_d3d9.cc305
-rw-r--r--o3d/command_buffer/service/win/d3d9/gapi_d3d9.h384
-rw-r--r--o3d/command_buffer/service/win/d3d9/geometry_d3d9.cc428
-rw-r--r--o3d/command_buffer/service/win/d3d9/geometry_d3d9.h126
-rw-r--r--o3d/command_buffer/service/win/d3d9/sampler_d3d9.cc199
-rw-r--r--o3d/command_buffer/service/win/d3d9/sampler_d3d9.h85
-rw-r--r--o3d/command_buffer/service/win/d3d9/states_d3d9.cc349
-rw-r--r--o3d/command_buffer/service/win/d3d9/texture_d3d9.cc636
-rw-r--r--o3d/command_buffer/service/win/d3d9/texture_d3d9.h245
-rw-r--r--o3d/command_buffer/service/win/plugin.def6
-rw-r--r--o3d/command_buffer/service/win/plugin.rc100
54 files changed, 12661 insertions, 0 deletions
diff --git a/o3d/command_buffer/service/build.scons b/o3d/command_buffer/service/build.scons
new file mode 100644
index 0000000..17ec60f
--- /dev/null
+++ b/o3d/command_buffer/service/build.scons
@@ -0,0 +1,114 @@
+# Copyright 2009, Google Inc.
+# All rights reserved.
+#
+# Redistribution and use in source and binary forms, with or without
+# modification, are permitted provided that the following conditions are
+# met:
+#
+# * Redistributions of source code must retain the above copyright
+# notice, this list of conditions and the following disclaimer.
+# * Redistributions in binary form must reproduce the above
+# copyright notice, this list of conditions and the following disclaimer
+# in the documentation and/or other materials provided with the
+# distribution.
+# * Neither the name of Google Inc. nor the names of its
+# contributors may be used to endorse or promote products derived from
+# this software without specific prior written permission.
+#
+# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+
+Import('env')
+
+INPUTS = [
+ 'cross/buffer_rpc.cc',
+ 'cross/cmd_parser.cc',
+ 'cross/cmd_buffer_engine.cc',
+ 'cross/effect_utils.cc',
+ 'cross/gapi_decoder.cc',
+ 'cross/resource.cc',
+ 'cross/texture_utils.cc',
+]
+
+# Build the big tests
+BIG_TEST_SERVER = [
+ 'cross/big_test.cc',
+]
+
+# Add some flags and libraries to the build environment for the unit tests
+env.Append(
+ LIBPATH = [
+ '$NACL_LIB_DIR',
+ ],
+ LIBS = [
+ 'o3dCmdBuf_service',
+ 'o3dCmdBuf_common',
+ 'o3d_base',
+ ] + env['NACL_HTP_LIBS'] + env['ICU_LIBS'],
+)
+
+if env['TARGET_PLATFORM'] == 'WINDOWS':
+ env.Append(CCFLAGS = ['/Wp64'],
+ LIBS = [# System libs.
+ 'd3d9',
+ # TODO: remove link-time dependency on d3dx9, using
+ # dynamic loading instead.
+ 'd3dx9',
+ 'dxerr',
+ 'shell32'],
+ LINKFLAGS=['/SUBSYSTEM:CONSOLE'])
+ INPUTS += ['win/d3d9/effect_d3d9.cc',
+ 'win/d3d9/gapi_d3d9.cc',
+ 'win/d3d9/geometry_d3d9.cc',
+ 'win/d3d9/sampler_d3d9.cc',
+ 'win/d3d9/states_d3d9.cc',
+ 'win/d3d9/texture_d3d9.cc']
+ BIG_TEST_SERVER += ['win/big_test_main.cc']
+elif env['TARGET_PLATFORM'] == 'LINUX':
+ env.Append(LIBS = ['GL', 'GLEW', 'Cg', 'CgGL'],
+ CPPPATH = ['$CG_DIR/include'])
+ INPUTS += ['cross/gl/effect_gl.cc',
+ 'cross/gl/gapi_gl.cc',
+ 'cross/gl/geometry_gl.cc',
+ 'cross/gl/sampler_gl.cc',
+ 'cross/gl/states_gl.cc',
+ 'cross/gl/texture_gl.cc',
+ 'linux/x_utils.cc']
+ BIG_TEST_SERVER += ['linux/big_test_main.cc']
+
+# Create a target library from the sources called 'o3dCmdBuf'
+o3dcmdbuf_lib = env.ComponentLibrary('o3dCmdBuf_service', INPUTS)
+
+# TODO: why can't these be ComponentTestProgram?
+
+# Create a target executable program called 'o3dCmdBuf_bigtest_server' from the list
+# of big test server files.
+big_test_server = env.Program('o3dCmdBuf_bigtest_server', BIG_TEST_SERVER)
+
+# Copy the resulting executable to the Artifacts directory.
+env.Replicate('$ARTIFACTS_DIR', big_test_server)
+
+plugin_env = env.Clone();
+plugin_env.Append(CPPPATH = ['$NPAPI_DIR/include'])
+
+PLUGIN_INPUTS = [
+ 'cross/plugin.cc',
+ plugin_env.SharedObject(
+ 'npn_api',
+ '$NIXYSA_DIR/static_glue/npapi/npn_api.cc'),
+]
+if env['TARGET_PLATFORM'] == 'WINDOWS':
+ PLUGIN_INPUTS += ['win/plugin.def',
+ env.RES('win/plugin.rc')]
+plugin = plugin_env.SharedLibrary('npo3d_cb_plugin', PLUGIN_INPUTS)
+plugin_env.Replicate('$ARTIFACTS_DIR', plugin)
diff --git a/o3d/command_buffer/service/cross/big_test.cc b/o3d/command_buffer/service/cross/big_test.cc
new file mode 100644
index 0000000..b68a9da
--- /dev/null
+++ b/o3d/command_buffer/service/cross/big_test.cc
@@ -0,0 +1,106 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains a "big" test of the whole command buffer service, making
+// sure all the pieces fit together.
+//
+// Currently this checks that the RPC mechanism properly forwards call to the
+// service thread.
+
+#include <build/build_config.h>
+#include "command_buffer/common/cross/rpc_imc.h"
+#include "command_buffer/service/cross/big_test_helpers.h"
+#include "command_buffer/service/cross/buffer_rpc.h"
+#include "command_buffer/service/cross/cmd_buffer_engine.h"
+#include "command_buffer/service/cross/gapi_decoder.h"
+#include "native_client/service_runtime/nrd_xfer_lib/nrd_all_modules.h"
+
+namespace o3d {
+namespace command_buffer {
+
+nacl::SocketAddress g_address = { "command-buffer" };
+
+// Main function. Creates a socket, and waits for an incoming connection on it.
+// Then run the engine main loop.
+void BigTest() {
+ NaClNrdAllModulesInit();
+ GAPIDecoder decoder(g_gapi);
+ CommandBufferEngine engine(&decoder);
+ decoder.set_engine(&engine);
+ nacl::Handle server_socket = nacl::BoundSocket(&g_address);
+ nacl::Handle handles[1];
+ nacl::MessageHeader msg;
+ msg.iov = NULL;
+ msg.iov_length = 0;
+ msg.handles = handles;
+ msg.handle_count = 1;
+ int r = nacl::ReceiveDatagram(server_socket, &msg, 0);
+ DCHECK_NE(r, -1);
+ nacl::Close(server_socket);
+
+ nacl::HtpHandle htp_handle = nacl::CreateImcDesc(handles[0]);
+ IMCMessageProcessor processor(htp_handle, engine.rpc_impl());
+ engine.set_process_interface(&processor);
+ IMCSender sender(htp_handle);
+ engine.set_client_rpc(&sender);
+
+ bool result = g_gapi->Initialize();
+ DCHECK(result);
+
+ bool running = true;
+ while (running) {
+ running = ProcessSystemMessages();
+ if (!running) break;
+ // DoWork() will block if there is nothing to be done, meaning we are only
+ // going to handle message after commands are sent. It should happen at
+ // least once a frame, so it's "good enough".
+ // TODO: figure out a way to wait on the socket OR messages with
+ // MsgWaitForMultipleObjects. Asynchronous ("overlapped") read on the
+ // socket may let us do that on windows.
+ running = engine.DoWork();
+ }
+ g_gapi->Destroy();
+ nacl::Close(htp_handle);
+ NaClNrdAllModulesFini();
+}
+
+} // namespace command_buffer
+} // namespace o3d
+
+#ifdef OS_WIN
+int big_test_main(int argc, wchar_t **argv) {
+#else
+int big_test_main(int argc, char **argv) {
+#endif
+ o3d::command_buffer::BigTest();
+ return 0;
+}
diff --git a/o3d/command_buffer/service/cross/big_test_helpers.h b/o3d/command_buffer/service/cross/big_test_helpers.h
new file mode 100644
index 0000000..b32a577
--- /dev/null
+++ b/o3d/command_buffer/service/cross/big_test_helpers.h
@@ -0,0 +1,70 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains a few helper functions for running big tests.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_CROSS_BIG_TEST_HELPERS_H__
+#define O3D_COMMAND_BUFFER_SERVICE_CROSS_BIG_TEST_HELPERS_H__
+
+#include <build/build_config.h>
+#include "core/cross/types.h"
+#include "command_buffer/common/cross/gapi_interface.h"
+
+namespace o3d {
+namespace command_buffer {
+
+extern String *g_program_path;
+extern GAPIInterface *g_gapi;
+
+// very simple thread API (hides platform-specific implementations).
+typedef void (* ThreadFunc)(void *param);
+class Thread;
+
+// Creates and starts a thread.
+Thread *CreateThread(ThreadFunc func, void* param);
+
+// Joins (waits for) a thread, destroying it.
+void JoinThread(Thread *thread);
+
+// Processes system messages. Should be called once a frame at least.
+bool ProcessSystemMessages();
+
+} // namespace command_buffer
+} // namespace o3d
+
+#ifdef OS_WIN
+int big_test_main(int argc, wchar_t **argv);
+#else
+int big_test_main(int argc, char **argv);
+#endif
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_CROSS_BIG_TEST_HELPERS_H__
diff --git a/o3d/command_buffer/service/cross/buffer_rpc.cc b/o3d/command_buffer/service/cross/buffer_rpc.cc
new file mode 100644
index 0000000..c09f526
--- /dev/null
+++ b/o3d/command_buffer/service/cross/buffer_rpc.cc
@@ -0,0 +1,136 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file has the implementation of the Command Buffer Synchronous API RPC
+// glue.
+
+#include "command_buffer/common/cross/logging.h"
+#include "command_buffer/service/cross/buffer_rpc.h"
+
+namespace o3d {
+namespace command_buffer {
+
+// Implements the dispatch function, deciding on which function to call based
+// on the message ID, taking the arguments trivially serialized in the data
+// payload.
+BufferRPCImpl::ReturnValue BufferRPCImpl::DoCall(int message_id,
+ const void *data,
+ size_t size,
+ RPCHandle const *handles,
+ size_t handle_count) {
+ switch (message_id) {
+ case INIT_CONNECTION: {
+ DCHECK_EQ(0, size);
+ DCHECK_EQ(0, handle_count);
+ handler_->InitConnection();
+ return 0;
+ }
+ case CLOSE_CONNECTION:
+ DCHECK_EQ(0, size);
+ DCHECK_EQ(0, handle_count);
+ handler_->CloseConnection();
+ return 0;
+ case REGISTER_SHARED_MEMORY: {
+ DCHECK_EQ(sizeof(size_t), size); // NOLINT
+ DCHECK_EQ(1, handle_count);
+ RPCShmHandle buffer = static_cast<RPCShmHandle>(handles[0]);
+ return handler_->RegisterSharedMemory(buffer,
+ *static_cast<const size_t *>(data));
+ }
+ case UNREGISTER_SHARED_MEMORY: {
+ DCHECK_EQ(sizeof(unsigned int), size); // NOLINT
+ DCHECK_EQ(0, handle_count);
+ unsigned int shm_id = *(static_cast<const unsigned int *>(data));
+ handler_->UnregisterSharedMemory(shm_id);
+ return 0;
+ }
+ case SET_COMMAND_BUFFER: {
+ DCHECK_EQ(sizeof(SetCommandBufferStruct), size);
+ DCHECK_EQ(0, handle_count);
+ const SetCommandBufferStruct *params =
+ static_cast<const SetCommandBufferStruct *>(data);
+ handler_->SetCommandBuffer(params->shm_id,
+ params->offset,
+ params->size,
+ params->start_get);
+ return 0;
+ }
+ case PUT: {
+ DCHECK_EQ(sizeof(CommandBufferOffset), size);
+ DCHECK_EQ(0, handle_count);
+ CommandBufferOffset offset =
+ *(static_cast<const CommandBufferOffset *>(data));
+ handler_->Put(offset);
+ return 0;
+ }
+ case GET:
+ DCHECK_EQ(0, size);
+ DCHECK_EQ(0, handle_count);
+ return handler_->Get();
+ case GET_TOKEN:
+ DCHECK_EQ(0, size);
+ DCHECK_EQ(0, handle_count);
+ return handler_->GetToken();
+ case WAIT_GET_CHANGES: {
+ DCHECK_EQ(sizeof(CommandBufferOffset), size);
+ DCHECK_EQ(0, handle_count);
+ CommandBufferOffset current_value =
+ *(static_cast<const CommandBufferOffset *>(data));
+ return handler_->WaitGetChanges(current_value);
+ }
+ case SIGNAL_GET_CHANGES: {
+ DCHECK_EQ(sizeof(SignalGetChangesStruct), size);
+ DCHECK_EQ(0, handle_count);
+ const SignalGetChangesStruct *params =
+ static_cast<const SignalGetChangesStruct *>(data);
+ handler_->SignalGetChanges(params->current_value,
+ params->rpc_message_id);
+ return 0;
+ }
+ case GET_STATUS: {
+ DCHECK_EQ(0, size);
+ DCHECK_EQ(0, handle_count);
+ return handler_->GetStatus();
+ }
+ case GET_PARSE_ERROR: {
+ DCHECK_EQ(0, size);
+ DCHECK_EQ(0, handle_count);
+ return handler_->GetParseError();
+ }
+ default:
+ LOG(FATAL) << "unsupported RPC";
+ return 0;
+ }
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/cross/buffer_rpc.h b/o3d/command_buffer/service/cross/buffer_rpc.h
new file mode 100644
index 0000000..f49dc0d
--- /dev/null
+++ b/o3d/command_buffer/service/cross/buffer_rpc.h
@@ -0,0 +1,101 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file defines the RPC glue for the Command Buffer Synchronous API,
+// service side: an implementation of a RPC service, forwarding calls to
+// a Command Buffer API implemention (RPC -> API).
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_CROSS_BUFFER_RPC_H__
+#define O3D_COMMAND_BUFFER_SERVICE_CROSS_BUFFER_RPC_H__
+
+#include "command_buffer/common/cross/buffer_sync_api.h"
+#include "command_buffer/common/cross/rpc.h"
+
+namespace o3d {
+namespace command_buffer {
+
+// RPC service implementation, implementing the Command Buffer Synchronous API
+// RPC glue. This class is temporary, and will be replaced when the
+// NativeClient RPC mechanism will be available.
+//
+// The API exposed through the RPC mechanism maps 1-to-1 to BufferSyncInterface,
+// trivially serializing arguments.
+class BufferRPCImpl: public RPCImplInterface {
+ public:
+ explicit BufferRPCImpl(BufferSyncInterface *handler) : handler_(handler) {}
+ virtual ~BufferRPCImpl() {}
+
+ enum MessageId {
+ INIT_CONNECTION = RESPONSE_ID + 1,
+ CLOSE_CONNECTION,
+ REGISTER_SHARED_MEMORY,
+ UNREGISTER_SHARED_MEMORY,
+ SET_COMMAND_BUFFER,
+ PUT,
+ GET,
+ GET_TOKEN,
+ WAIT_GET_CHANGES,
+ SIGNAL_GET_CHANGES,
+ GET_STATUS,
+ GET_PARSE_ERROR,
+ };
+
+ // Structure describing the arguments for the SET_COMMAND_BUFFER RPC.
+ struct SetCommandBufferStruct {
+ unsigned int shm_id;
+ ptrdiff_t offset;
+ size_t size;
+ CommandBufferOffset start_get;
+ };
+
+ // Structure describing the arguments for the SIGNAL_GET_CHANGES RPC.
+ struct SignalGetChangesStruct {
+ CommandBufferOffset current_value;
+ int rpc_message_id;
+ };
+
+ // Implements the DoCall interface, interpreting the message with the
+ // parameters, and passing the calls with arguments to the handler.
+ virtual ReturnValue DoCall(int message_id,
+ const void * data,
+ size_t size,
+ RPCHandle const *handles,
+ size_t handle_count);
+
+ private:
+ BufferSyncInterface *handler_;
+};
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_CROSS_BUFFER_RPC_H__
diff --git a/o3d/command_buffer/service/cross/buffer_rpc_test.cc b/o3d/command_buffer/service/cross/buffer_rpc_test.cc
new file mode 100644
index 0000000..42122b9
--- /dev/null
+++ b/o3d/command_buffer/service/cross/buffer_rpc_test.cc
@@ -0,0 +1,171 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// Tests for the Command Buffer RPC glue.
+
+#include "tests/common/win/testing_common.h"
+#include "command_buffer/common/cross/mocks.h"
+#include "command_buffer/service/cross/buffer_rpc.h"
+#include "command_buffer/service/cross/mocks.h"
+
+namespace o3d {
+namespace command_buffer {
+
+using testing::Return;
+
+// Test fixture for BufferRPCImpl test. Creates a BufferSyncMock and a
+// BufferRPCImpl that uses it.
+class BufferRPCImplTest : public testing::Test {
+ protected:
+ virtual void SetUp() {
+ buffer_sync_mock_.reset(new BufferSyncMock);
+ buffer_rpc_impl_.reset(new BufferRPCImpl(buffer_sync_mock_.get()));
+ }
+ virtual void TearDown() {}
+
+ BufferSyncMock &buffer_sync_mock() { return *buffer_sync_mock_.get(); }
+ BufferRPCImpl *buffer_rpc_impl() { return buffer_rpc_impl_.get(); }
+ private:
+ scoped_ptr<BufferSyncMock> buffer_sync_mock_;
+ scoped_ptr<BufferRPCImpl> buffer_rpc_impl_;
+};
+
+// Checks that the INIT_CONNECTION RPC is properly parsed.
+TEST_F(BufferRPCImplTest, TestInitConnection) {
+ EXPECT_CALL(buffer_sync_mock(), InitConnection());
+ buffer_rpc_impl()->DoCall(BufferRPCImpl::INIT_CONNECTION, NULL, 0, NULL, 0);
+}
+
+// Checks that the CLOSE_CONNECTION RPC is properly parsed.
+TEST_F(BufferRPCImplTest, TestCloseConnection) {
+ EXPECT_CALL(buffer_sync_mock(), CloseConnection());
+ buffer_rpc_impl()->DoCall(BufferRPCImpl::CLOSE_CONNECTION, NULL, 0, NULL, 0);
+}
+
+// Checks that the REGISTER_SHARED_MEMORY RPC is properly parsed and that the
+// return value is properly forwarded.
+TEST_F(BufferRPCImplTest, TestRegisterSharedMemory) {
+ RPCShmHandle shm = reinterpret_cast<RPCShmHandle>(456);
+ size_t size = 789;
+ EXPECT_CALL(buffer_sync_mock(), RegisterSharedMemory(shm, size))
+ .WillOnce(Return(1234));
+ RPCHandle handles[1] = {shm};
+ EXPECT_EQ(1234, buffer_rpc_impl()->DoCall(
+ BufferRPCImpl::REGISTER_SHARED_MEMORY, &size, sizeof(size), handles, 1));
+}
+
+// Checks that the UNREGISTER_SHARED_MEMORY RPC is properly parsed.
+TEST_F(BufferRPCImplTest, TestUnregisterSharedMemory) {
+ unsigned int shm_id = 385;
+ EXPECT_CALL(buffer_sync_mock(), UnregisterSharedMemory(shm_id));
+ buffer_rpc_impl()->DoCall(BufferRPCImpl::UNREGISTER_SHARED_MEMORY, &shm_id,
+ sizeof(shm_id), NULL, 0);
+}
+
+// Checks that the SET_COMMAND_BUFFER RPC is properly parsed.
+TEST_F(BufferRPCImplTest, TestSetCommandBuffer) {
+ EXPECT_CALL(buffer_sync_mock(), SetCommandBuffer(93, 7878, 3434, 5151));
+ BufferRPCImpl::SetCommandBufferStruct param;
+ param.shm_id = 93;
+ param.offset = 7878;
+ param.size = 3434;
+ param.start_get = 5151;
+ buffer_rpc_impl()->DoCall(BufferRPCImpl::SET_COMMAND_BUFFER, &param,
+ sizeof(param), NULL, 0);
+}
+
+// Checks that the PUT RPC is properly parsed.
+TEST_F(BufferRPCImplTest, TestPut) {
+ CommandBufferOffset offset = 8765;
+ EXPECT_CALL(buffer_sync_mock(), Put(offset));
+ buffer_rpc_impl()->DoCall(BufferRPCImpl::PUT, &offset, sizeof(offset), NULL,
+ 0);
+}
+
+// Checks that the GET RPC is properly parsed and that the return value is
+// properly forwarded.
+TEST_F(BufferRPCImplTest, TestGet) {
+ EXPECT_CALL(buffer_sync_mock(), Get()).WillOnce(Return(9375));
+ EXPECT_EQ(9375, buffer_rpc_impl()->DoCall(BufferRPCImpl::GET, NULL, 0, NULL,
+ 0));
+}
+
+// Checks that the GET_TOKEN RPC is properly parsed and that the return value is
+// properly forwarded.
+TEST_F(BufferRPCImplTest, TestGetToken) {
+ EXPECT_CALL(buffer_sync_mock(), GetToken()).WillOnce(Return(1618));
+ EXPECT_EQ(1618, buffer_rpc_impl()->DoCall(BufferRPCImpl::GET_TOKEN, NULL, 0,
+ NULL, 0));
+}
+
+// Checks that the WAIT_GET_CHANGES RPC is properly parsed and that the return
+// value is properly forwarded.
+TEST_F(BufferRPCImplTest, TestWaitGetChanges) {
+ CommandBufferOffset value = 339;
+ EXPECT_CALL(buffer_sync_mock(), WaitGetChanges(value))
+ .WillOnce(Return(16180));
+ EXPECT_EQ(16180, buffer_rpc_impl()->DoCall(BufferRPCImpl::WAIT_GET_CHANGES,
+ &value, sizeof(value), NULL, 0));
+}
+
+// Checks that the SIGNAL_GET_CHANGES RPC is properly parsed.
+TEST_F(BufferRPCImplTest, TestSignalGetChanges) {
+ EXPECT_CALL(buffer_sync_mock(), SignalGetChanges(34, 21));
+ BufferRPCImpl::SignalGetChangesStruct param;
+ param.current_value = 34;
+ param.rpc_message_id = 21;
+ buffer_rpc_impl()->DoCall(BufferRPCImpl::SIGNAL_GET_CHANGES, &param,
+ sizeof(param), NULL, 0);
+}
+
+// Checks that the GET_STATUS RPC is properly parsed and that the return value
+// is properly forwarded.
+TEST_F(BufferRPCImplTest, TestGetStatus) {
+ EXPECT_CALL(buffer_sync_mock(), GetStatus())
+ .WillOnce(Return(BufferSyncInterface::PARSE_ERROR));
+ EXPECT_EQ(BufferSyncInterface::PARSE_ERROR,
+ buffer_rpc_impl()->DoCall(BufferRPCImpl::GET_STATUS, NULL, 0, NULL,
+ 0));
+}
+
+// Checks that the GET_STATUS RPC is properly parsed and that the return value
+// is properly forwarded.
+TEST_F(BufferRPCImplTest, TestGetParseError) {
+ EXPECT_CALL(buffer_sync_mock(), GetParseError())
+ .WillOnce(Return(BufferSyncInterface::PARSE_OUT_OF_BOUNDS));
+ EXPECT_EQ(BufferSyncInterface::PARSE_OUT_OF_BOUNDS,
+ buffer_rpc_impl()->DoCall(BufferRPCImpl::GET_PARSE_ERROR, NULL, 0,
+ NULL, 0));
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/cross/cmd_buffer_engine.cc b/o3d/command_buffer/service/cross/cmd_buffer_engine.cc
new file mode 100644
index 0000000..d96c63d
--- /dev/null
+++ b/o3d/command_buffer/service/cross/cmd_buffer_engine.cc
@@ -0,0 +1,290 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the implementation of the command buffer engine.
+
+#include "command_buffer/service/cross/buffer_rpc.h"
+#include "command_buffer/service/cross/cmd_buffer_engine.h"
+
+namespace o3d {
+namespace command_buffer {
+
+// Creates a RPC implementation using 'this' as the handler, and a RPC server
+// for it.
+CommandBufferEngine::CommandBufferEngine(AsyncAPIInterface *handler)
+ : buffer_rpc_impl_(),
+ process_interface_(NULL),
+ parser_(),
+ handler_(handler),
+ client_rpc_(NULL),
+ token_(0),
+ status_(NOT_CONNECTED),
+ signal_change_(false),
+ signal_rpc_message_id_(0),
+ parse_error_(PARSE_NO_ERROR) {
+ buffer_rpc_impl_.reset(new BufferRPCImpl(this));
+}
+
+CommandBufferEngine::~CommandBufferEngine() {}
+
+// Inits the connection. Registers the client RPC service.
+void CommandBufferEngine::InitConnection() {
+ status_ = NO_BUFFER;
+}
+
+// Closes the connection. Executes all remaining commands.
+void CommandBufferEngine::CloseConnection() {
+ FinishParsing();
+ status_ = NOT_CONNECTED;
+ parser_.reset(NULL);
+}
+
+// Adds the shared memory buffer somewhere into the list, return the index in
+// the list as the handle. Either find a hole in the list, or add it at the
+// end. We don't want to invalidate exiting indices.
+unsigned int CommandBufferEngine::RegisterSharedMemory(
+ RPCShmHandle handle,
+ size_t size) {
+ void *address = MapShm(handle, size);
+ if (!address) return kInvalidSharedMemoryId;
+ MemoryMapping mapping = {address, size};
+ for (unsigned int i = 0; i < shared_memory_buffers_.size(); ++i) {
+ if (shared_memory_buffers_[i].address == NULL) {
+ shared_memory_buffers_[i] = mapping;
+ return i;
+ }
+ }
+ shared_memory_buffers_.push_back(mapping);
+ return static_cast<unsigned int>(shared_memory_buffers_.size() - 1);
+}
+
+// Sets the list entry for the shared memory buffer to NULL. Don't erase() the
+// entry, We don't want to invalidate exiting indices.
+void CommandBufferEngine::UnregisterSharedMemory(unsigned int shm_id) {
+ if ((shm_id >= shared_memory_buffers_.size()) ||
+ !shared_memory_buffers_[shm_id].size) {
+ LOG(ERROR) << "Trying to unregister a non-registered shared memory";
+ return;
+ }
+ MemoryMapping &mapping = shared_memory_buffers_[shm_id];
+ UnmapShm(mapping.address, mapping.size);
+ mapping.address = NULL;
+ mapping.size = 0;
+}
+
+// Sets the command buffer. Executes all remaining commands in the old buffer
+// (if any) and creates a new command parser.
+void CommandBufferEngine::SetCommandBuffer(unsigned int shm_id,
+ ptrdiff_t offset,
+ size_t size,
+ CommandBufferOffset start_get) {
+ if ((shm_id >= shared_memory_buffers_.size()) ||
+ !shared_memory_buffers_[shm_id].size) {
+ LOG(ERROR) << "Trying to set the command buffer from a non-registered "
+ << "shared memory";
+ return;
+ }
+ if (status_ == NOT_CONNECTED) return;
+ FinishParsing();
+ parser_.reset(new CommandParser(shared_memory_buffers_[shm_id].address,
+ shared_memory_buffers_[shm_id].size, offset,
+ size, start_get, handler_));
+ status_ = PARSING;
+ parse_error_ = PARSE_NO_ERROR;
+}
+
+// Changes the put value.
+void CommandBufferEngine::Put(CommandBufferOffset offset) {
+ if (parser_.get()) {
+ parser_->set_put(offset);
+ }
+}
+
+// Retrieves the get value. This returns -1 if there is no current parser.
+CommandBufferOffset CommandBufferEngine::Get() {
+ if (parser_.get()) {
+ return parser_->get();
+ } else {
+ return -1;
+ }
+}
+
+// Retrieves the current token value.
+unsigned int CommandBufferEngine::GetToken() {
+ return token_;
+}
+
+// Executes commands until get is different from the value passed in. It will
+// return immediately if the get value is already different, or if the engine
+// is not in the PARSING status, or if the buffer is empty. It will return -1
+// if there is no current buffer.
+CommandBufferOffset CommandBufferEngine::WaitGetChanges(
+ CommandBufferOffset current_value) {
+ if (parser_.get()) {
+ while (status_ == PARSING &&
+ parser_->get() == current_value &&
+ !parser_->IsEmpty()) {
+ ProcessOneCommand();
+ }
+ return parser_->get();
+ } else {
+ return -1;
+ }
+}
+
+// Signals the client when get gets different from the value passed in. If get
+// is already different, or if the engine is not in the PARSING status, that
+// will happen immediately, otherwise it will happen when commands get
+// executed, moving the get pointer.
+void CommandBufferEngine::SignalGetChanges(CommandBufferOffset current_value,
+ int rpc_message_id) {
+ if (status_ != PARSING || parser_->get() != current_value) {
+ DoSignalChangedGet(rpc_message_id);
+ } else {
+ signal_change_ = true;
+ signal_rpc_message_id_ = rpc_message_id;
+ }
+}
+
+// Gets the memory address from the list entry.
+void *CommandBufferEngine::GetSharedMemoryAddress(unsigned int shm_id) {
+ if ((shm_id >= shared_memory_buffers_.size()) ||
+ !shared_memory_buffers_[shm_id].size) {
+ LOG(ERROR) << "Trying to get the address of a non-registered shared memory";
+ return NULL;
+ }
+ return shared_memory_buffers_[shm_id].address;
+}
+
+// Gets the memory size from the list entry.
+size_t CommandBufferEngine::GetSharedMemorySize(unsigned int shm_id) {
+ if ((shm_id >= shared_memory_buffers_.size()) ||
+ !shared_memory_buffers_[shm_id].size) {
+ LOG(ERROR) << "Trying to get the size of a non-registered shared memory";
+ return 0;
+ }
+ return shared_memory_buffers_[shm_id].size;
+}
+
+// Gets the status.
+BufferSyncInterface::ParserStatus CommandBufferEngine::GetStatus() {
+ return status_;
+}
+
+// Gets the current parse error, reset it to PARSE_NO_ERROR.
+BufferSyncInterface::ParseError CommandBufferEngine::GetParseError() {
+ ParseError error = parse_error_;
+ parse_error_ = PARSE_NO_ERROR;
+ return error;
+}
+
+// Finishes parsing, executing all the commands until the buffer is empty, or a
+// parsing error occurs.
+void CommandBufferEngine::FinishParsing() {
+ // terminates current parsing, that is, execute all the commands
+ // NOTE: status_ == PARSING implies parser_ != NULL
+ while (status_ == PARSING && !parser_->IsEmpty()) {
+ ProcessOneCommand();
+ }
+}
+
+// Processes one command from the command buffer. This must only be called when
+// in the PARSING status.
+// This will update the status_ and the parse_error_ fields if an error occurs.
+void CommandBufferEngine::ProcessOneCommand() {
+ DCHECK_EQ(PARSING, status_);
+ DCHECK(parser_.get());
+ ParseError result = parser_->ProcessCommand();
+ switch (result) {
+ case PARSE_NO_ERROR:
+ break;
+ case PARSE_OUT_OF_BOUNDS:
+ case PARSE_INVALID_SIZE:
+ status_ = PARSE_ERROR;
+ // Always override the error, to properly signal the stopping condition.
+ parse_error_ = result;
+ break;
+ case PARSE_INVALID_ARGUMENTS:
+ case PARSE_UNKNOWN_COMMAND:
+ // Only set the error if it is not set already.
+ if (parse_error_ == PARSE_NO_ERROR) {
+ parse_error_ = result;
+ }
+ break;
+ }
+ // get has changed, signal the client if needed.
+ if (signal_change_) {
+ DoSignalChangedGet(signal_rpc_message_id_);
+ signal_change_ = false;
+ }
+}
+
+// Executes the main loop. While there are commands in the buffer, processes
+// them one by one, checking for RPC messages between each of them (executing
+// all of them). If the buffer is empty, block until a RPC message comes.
+void CommandBufferEngine::DoMainLoop() {
+ while (DoWork()) { }
+ // Clean up if needed: execute all pending commands, then close the
+ // connection.
+ if (status_ != NOT_CONNECTED) CloseConnection();
+}
+
+bool CommandBufferEngine::HasWork() {
+ return (status_ == PARSING && !parser_->IsEmpty()) ||
+ process_interface_->HasMessage();
+}
+
+bool CommandBufferEngine::DoWork() {
+ if (status_ == PARSING && !parser_->IsEmpty()) {
+ bool running = true;
+ // process as many messages as available but do not block.
+ while (process_interface_->HasMessage()) {
+ running = process_interface_->ProcessMessage();
+ }
+ if (running) ProcessOneCommand();
+ return running;
+ } else {
+ // call ProcessMessage, always blocking. We have nothing else to do.
+ return process_interface_->ProcessMessage();
+ }
+}
+
+// Signals that get has changed, sending a RPC message back to the client. It
+// will send -1 if there is no current buffer.
+void CommandBufferEngine::DoSignalChangedGet(int rpc_message_id) {
+ DCHECK(client_rpc_);
+ CommandBufferOffset get = parser_.get() ? parser_->get() : -1;
+ client_rpc_->SendCall(rpc_message_id, &get, sizeof(get), NULL, 0);
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/cross/cmd_buffer_engine.h b/o3d/command_buffer/service/cross/cmd_buffer_engine.h
new file mode 100644
index 0000000..66cab72
--- /dev/null
+++ b/o3d/command_buffer/service/cross/cmd_buffer_engine.h
@@ -0,0 +1,214 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file defines the CommandBufferEngine class, providing the main loop for
+// the service, exposing the RPC API, managing the command parser.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_CROSS_CMD_BUFFER_ENGINE_H__
+#define O3D_COMMAND_BUFFER_SERVICE_CROSS_CMD_BUFFER_ENGINE_H__
+
+#include <vector>
+#include "base/scoped_ptr.h"
+#include "command_buffer/common/cross/buffer_sync_api.h"
+#include "command_buffer/service/cross/cmd_parser.h"
+
+namespace o3d {
+namespace command_buffer {
+
+class BufferRPCImpl;
+
+class CommandBufferEngine : public BufferSyncInterface {
+ public:
+ explicit CommandBufferEngine(AsyncAPIInterface *handler);
+ virtual ~CommandBufferEngine();
+
+ // Initializes the connection with the client.
+ virtual void InitConnection();
+
+ // Closes the connection with the client.
+ virtual void CloseConnection();
+
+ // Registers a shared memory buffer. While a buffer is registered, it can be
+ // accessed by the service, including the underlying asynchronous API,
+ // through a single identifier.
+ // Parameters:
+ // buffer: the shared memory buffer handle.
+ // Returns:
+ // an identifier for the shared memory.
+ virtual unsigned int RegisterSharedMemory(RPCShmHandle buffer, size_t size);
+
+ // Unregisters a shared memory buffer.
+ // Parameters:
+ // shm_id: the identifier for the shared memory buffer.
+ virtual void UnregisterSharedMemory(unsigned int shm_id);
+
+ // Initializes the command buffer.
+ // Parameters:
+ // buffer: the memory buffer descriptor in which the command buffer
+ // resides.
+ // offset: the offset of the command buffer, relative to the memory
+ // buffer.
+ // size: the size of the command buffer.
+ // start_get: the inital value for the Get pointer, relative to the
+ // command buffer, where the parser will start interpreting
+ // commands. Put is also initialized to that value.
+ virtual void SetCommandBuffer(unsigned int shm_id,
+ ptrdiff_t offset,
+ size_t size,
+ CommandBufferOffset start_get);
+
+ // Sets the value of the Put pointer.
+ // Parameters:
+ // offset: the new value of the Put pointer, as an offset into the command
+ // buffer.
+ virtual void Put(CommandBufferOffset offset);
+
+ // Gets the value of the Get pointer.
+ // Returns:
+ // the current value of the Get pointer, as an offset into the command
+ // buffer.
+ virtual CommandBufferOffset Get();
+
+ // Gets the current token value.
+ // Returns:
+ // the current token value.
+ virtual unsigned int GetToken();
+
+ // Waits until Get changes from the currently known value.
+ // Parameters:
+ // current_value: the currently known value. This call will block until
+ // Get is different from that value (returning immediately
+ // if it is already different).
+ // Returns:
+ // the current (changed) value of Get.
+ virtual CommandBufferOffset WaitGetChanges(
+ CommandBufferOffset current_value);
+
+ // Asks the service to signal the client when Get changes from the currently
+ // known value. This is a non-blocking version of WaitGetChanges.
+ // Parameters:
+ // current_value: the currently known value of Get.
+ // rpc_message_id: the RPC message ID to call on the client when Get is
+ // different from current_value. That RPC is called
+ // immediately if Get is already different from
+ // current_value.
+ virtual void SignalGetChanges(CommandBufferOffset current_value,
+ int rpc_message_id);
+
+ // Gets the status of the service.
+ // Returns:
+ // The status of the service.
+ virtual ParserStatus GetStatus();
+
+ // Gets the current parse error. The current parse error is set when the
+ // service is in the PARSE_ERROR status. It may also be set while in the
+ // PARSING state, if a recoverable error (like PARSE_UNKNOWN_METHOD) was
+ // encountered. Getting the error resets it to PARSE_NO_ERROR.
+ // Returns:
+ // The current parse error.
+ virtual ParseError GetParseError();
+
+ // Gets the base address of a registered shared memory buffer.
+ // Parameters:
+ // shm_id: the identifier for the shared memory buffer.
+ void *GetSharedMemoryAddress(unsigned int shm_id);
+
+ // Gets the size of a registered shared memory buffer.
+ // Parameters:
+ // shm_id: the identifier for the shared memory buffer.
+ size_t GetSharedMemorySize(unsigned int shm_id);
+
+ // Executes the main loop: parse commands and execute RPC calls until the
+ // server is killed.
+ void DoMainLoop();
+
+ // Returns whether or not the engine has work to do (process synchronous or
+ // asynchronous commands).
+ bool HasWork();
+
+ // Does some work (process synchronous or asynchronous commands). It will not
+ // block if HasWork() returns true.
+ // Returns:
+ // true if the engine should keep running, false if it has been sent a
+ // command to terminate.
+ bool DoWork();
+
+ // Gets the RPC server.
+ BufferRPCImpl *rpc_impl() const { return buffer_rpc_impl_.get(); }
+
+ // Sets the RPC processing interface.
+ void set_process_interface(RPCProcessInterface *iface) {
+ process_interface_ = iface;
+ }
+
+ // Sets the RPC processing interface.
+ void set_client_rpc(RPCSendInterface *iface) {
+ client_rpc_ = iface;
+ }
+
+ // Gets the command parser.
+ CommandParser *parser() const { return parser_.get(); }
+
+ // Sets the token value.
+ void set_token(unsigned int token) { token_ = token; }
+ private:
+
+ // Processes one command from the command buffer.
+ void ProcessOneCommand();
+
+ // Sends the signal that get has changed to the client.
+ void DoSignalChangedGet(int rpc_message_id);
+
+ // Finish parsing and executing all the commands in the buffer.
+ void FinishParsing();
+
+ scoped_ptr<BufferRPCImpl> buffer_rpc_impl_;
+ RPCProcessInterface *process_interface_;
+ scoped_ptr<CommandParser> parser_;
+ AsyncAPIInterface *handler_;
+ RPCSendInterface *client_rpc_;
+ unsigned int token_;
+ ParserStatus status_;
+ bool signal_change_;
+ int signal_rpc_message_id_;
+ ParseError parse_error_;
+ struct MemoryMapping {
+ void *address;
+ size_t size;
+ };
+ std::vector<MemoryMapping> shared_memory_buffers_;
+};
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_CROSS_CMD_BUFFER_ENGINE_H__
diff --git a/o3d/command_buffer/service/cross/cmd_buffer_engine_test.cc b/o3d/command_buffer/service/cross/cmd_buffer_engine_test.cc
new file mode 100644
index 0000000..14f56ec
--- /dev/null
+++ b/o3d/command_buffer/service/cross/cmd_buffer_engine_test.cc
@@ -0,0 +1,626 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// Tests for the Command Buffer Engine.
+
+#include "tests/common/win/testing_common.h"
+#include "command_buffer/service/cross/cmd_buffer_engine.h"
+#include "command_buffer/service/cross/mocks.h"
+
+namespace o3d {
+namespace command_buffer {
+
+using testing::AnyNumber;
+using testing::Return;
+using testing::Mock;
+using testing::Truly;
+using testing::Sequence;
+using testing::_;
+
+// Test fixture for CommandBufferEngine test - Creates a CommandBufferEngine,
+// using a mock AsyncAPIInterface.
+class CommandBufferEngineTest : public testing::Test {
+ protected:
+ CommandBufferEngineTest()
+ : shm_(kRPCInvalidHandle),
+ shm_id_(BufferSyncInterface::kInvalidSharedMemoryId) {}
+
+ virtual void SetUp() {
+ api_mock_.reset(new AsyncAPIMock);
+ engine_.reset(new CommandBufferEngine(api_mock_.get()));
+ process_mock_.reset(new RPCProcessMock());
+ engine_->set_process_interface(process_mock_.get());
+ }
+ virtual void TearDown() {
+ }
+
+ // Creates a shared memory buffer for the command buffer, and pass it to the
+ // engine.
+ CommandBufferEntry *InitCommandBuffer(size_t entries, unsigned int start) {
+ CHECK(shm_ == kRPCInvalidHandle);
+ CHECK(shm_id_ == BufferSyncInterface::kInvalidSharedMemoryId);
+ const size_t kShmSize = entries * sizeof(CommandBufferEntry); // NOLINT
+ shm_ = CreateShm(kShmSize);
+ EXPECT_NE(kRPCInvalidHandle, shm_);
+ if (kRPCInvalidHandle == shm_) return NULL;
+ shm_id_ = engine()->RegisterSharedMemory(shm_, kShmSize);
+ EXPECT_NE(BufferSyncInterface::kInvalidSharedMemoryId, shm_id_);
+ if (shm_id_ == BufferSyncInterface::kInvalidSharedMemoryId) {
+ DestroyShm(shm_);
+ shm_ = kRPCInvalidHandle;
+ return NULL;
+ }
+ engine()->SetCommandBuffer(shm_id_, 0, kShmSize, start);
+ return static_cast<CommandBufferEntry *>(
+ engine()->GetSharedMemoryAddress(shm_id_));
+ }
+
+ // Destroys the command buffer.
+ void DestroyCommandBuffer() {
+ engine()->UnregisterSharedMemory(shm_id_);
+ shm_id_ = BufferSyncInterface::kInvalidSharedMemoryId;
+ DestroyShm(shm_);
+ shm_ = kRPCInvalidHandle;
+ }
+
+ // Adds a command to the buffer, while adding it as an expected call on the
+ // API mock.
+ unsigned int AddCommandWithExpect(CommandBufferEntry *buffer,
+ BufferSyncInterface::ParseError _return,
+ unsigned int command,
+ unsigned int arg_count,
+ CommandBufferEntry *args) {
+ unsigned int put = 0;
+ CommandHeader header;
+ header.size = arg_count + 1;
+ header.command = command;
+ buffer[put++].value_header = header;
+ for (unsigned int i = 0; i < arg_count; ++i) {
+ buffer[put++].value_uint32 = args[i].value_uint32;
+ }
+ EXPECT_CALL(*api_mock(), DoCommand(command, arg_count,
+ Truly(AsyncAPIMock::IsArgs(arg_count, args))))
+ .InSequence(sequence_)
+ .WillOnce(Return(_return));
+ return put;
+ }
+
+ CommandBufferEngine *engine() { return engine_.get(); }
+ RPCProcessMock *process_mock() { return process_mock_.get(); }
+ AsyncAPIMock *api_mock() { return api_mock_.get(); }
+ private:
+ scoped_ptr<AsyncAPIMock> api_mock_;
+ scoped_ptr<CommandBufferEngine> engine_;
+ scoped_ptr<RPCProcessMock> process_mock_;
+ // handles for the command buffer.
+ RPCShmHandle shm_;
+ unsigned int shm_id_;
+ Sequence sequence_;
+};
+
+// Tests the initialization/termination sequence, checking that it sets the
+// engine in the correct states.
+TEST_F(CommandBufferEngineTest, TestInitialization) {
+ // Check initial state
+ EXPECT_TRUE(engine()->rpc_impl() != NULL);
+ EXPECT_TRUE(engine()->parser() == NULL);
+ EXPECT_EQ(BufferSyncInterface::NOT_CONNECTED, engine()->GetStatus());
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, engine()->GetParseError());
+ EXPECT_EQ(-1, engine()->Get());
+ EXPECT_EQ(0, engine()->GetToken());
+
+ engine()->InitConnection();
+ EXPECT_EQ(BufferSyncInterface::NO_BUFFER, engine()->GetStatus());
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, engine()->GetParseError());
+ EXPECT_EQ(-1, engine()->Get());
+
+ CommandBufferEntry *entries = InitCommandBuffer(25, 5);
+ ASSERT_TRUE(entries != NULL);
+
+ EXPECT_EQ(BufferSyncInterface::PARSING, engine()->GetStatus());
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, engine()->GetParseError());
+ EXPECT_EQ(5, engine()->Get());
+ EXPECT_TRUE(engine()->parser() != NULL);
+
+ engine()->set_token(5678);
+ EXPECT_EQ(5678, engine()->GetToken());
+
+ engine()->CloseConnection();
+ DestroyCommandBuffer();
+
+ EXPECT_EQ(BufferSyncInterface::NOT_CONNECTED, engine()->GetStatus());
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, engine()->GetParseError());
+ EXPECT_EQ(-1, engine()->Get());
+ EXPECT_TRUE(engine()->parser() == NULL);
+}
+
+// Checks that shared memory registration works.
+TEST_F(CommandBufferEngineTest, TestSharedMemoryRegistration) {
+ // Create and register a first shared memory buffer.
+ const size_t kShmSize1 = 10;
+ RPCShmHandle shm1 = CreateShm(kShmSize1);
+ ASSERT_NE(kRPCInvalidHandle, shm1);
+ unsigned int shm_id1 = engine()->RegisterSharedMemory(shm1, kShmSize1);
+ EXPECT_NE(BufferSyncInterface::kInvalidSharedMemoryId, shm_id1);
+ EXPECT_TRUE(engine()->GetSharedMemoryAddress(shm_id1) != NULL);
+ EXPECT_EQ(kShmSize1, engine()->GetSharedMemorySize(shm_id1));
+
+ // Create and register a second shared memory buffer, check that it has a
+ // different memory location than the first one.
+ const size_t kShmSize2 = 25;
+ RPCShmHandle shm2 = CreateShm(kShmSize2);
+ ASSERT_NE(kRPCInvalidHandle, shm2);
+ unsigned int shm_id2 = engine()->RegisterSharedMemory(shm2, kShmSize2);
+ EXPECT_NE(BufferSyncInterface::kInvalidSharedMemoryId, shm_id2);
+ EXPECT_TRUE(engine()->GetSharedMemoryAddress(shm_id2) != NULL);
+ EXPECT_EQ(kShmSize2, engine()->GetSharedMemorySize(shm_id2));
+ EXPECT_NE(shm_id1, shm_id2);
+ EXPECT_NE(engine()->GetSharedMemoryAddress(shm_id1),
+ engine()->GetSharedMemoryAddress(shm_id2));
+
+ // Create and register a third shared memory buffer, check that it has a
+ // different memory location than the first and second ones.
+ const size_t kShmSize3 = 33;
+ RPCShmHandle shm3 = CreateShm(kShmSize3);
+ ASSERT_NE(kRPCInvalidHandle, shm3);
+ unsigned int shm_id3 = engine()->RegisterSharedMemory(shm3, kShmSize3);
+ EXPECT_NE(BufferSyncInterface::kInvalidSharedMemoryId, shm_id3);
+ EXPECT_TRUE(engine()->GetSharedMemoryAddress(shm_id3) != NULL);
+ EXPECT_EQ(kShmSize3, engine()->GetSharedMemorySize(shm_id3));
+ EXPECT_NE(shm_id1, shm_id3);
+ EXPECT_NE(shm_id2, shm_id3);
+ EXPECT_NE(engine()->GetSharedMemoryAddress(shm_id1),
+ engine()->GetSharedMemoryAddress(shm_id3));
+ EXPECT_NE(engine()->GetSharedMemoryAddress(shm_id2),
+ engine()->GetSharedMemoryAddress(shm_id3));
+
+ engine()->UnregisterSharedMemory(shm_id1);
+ EXPECT_EQ(NULL, engine()->GetSharedMemoryAddress(shm_id1));
+ EXPECT_EQ(0UL, engine()->GetSharedMemorySize(shm_id1));
+ DestroyShm(shm1);
+
+ engine()->UnregisterSharedMemory(shm_id2);
+ EXPECT_EQ(NULL, engine()->GetSharedMemoryAddress(shm_id2));
+ EXPECT_EQ(0UL, engine()->GetSharedMemorySize(shm_id2));
+ DestroyShm(shm2);
+
+ engine()->UnregisterSharedMemory(shm_id3);
+ EXPECT_EQ(NULL, engine()->GetSharedMemoryAddress(shm_id2));
+ EXPECT_EQ(0UL, engine()->GetSharedMemorySize(shm_id2));
+ DestroyShm(shm3);
+}
+
+// Checks that commands in the buffer are properly executed, and that the
+// status/error stay valid.
+TEST_F(CommandBufferEngineTest, TestCommandProcessing) {
+ engine()->InitConnection();
+ CommandBufferEntry *entries = InitCommandBuffer(10, 0);
+ ASSERT_TRUE(entries != NULL);
+
+ CommandBufferOffset get = engine()->Get();
+ CommandBufferOffset put = get;
+
+ // Create a command buffer with 3 commands
+ put += AddCommandWithExpect(entries + put,
+ BufferSyncInterface::PARSE_NO_ERROR,
+ 0,
+ 0,
+ NULL);
+
+ CommandBufferEntry args1[2];
+ args1[0].value_uint32 = 3;
+ args1[1].value_float = 4.f;
+ put += AddCommandWithExpect(entries + put,
+ BufferSyncInterface::PARSE_NO_ERROR,
+ 1,
+ 2,
+ args1);
+
+ CommandBufferEntry args2[2];
+ args2[0].value_uint32 = 5;
+ args2[1].value_float = 6.f;
+ put += AddCommandWithExpect(entries + put,
+ BufferSyncInterface::PARSE_NO_ERROR,
+ 2,
+ 2,
+ args2);
+
+ engine()->Put(put);
+ while (get != put) {
+ // Check that the parsing progresses, and that no error occurs.
+ CommandBufferOffset new_get = engine()->WaitGetChanges(get);
+ EXPECT_NE(get, new_get);
+ ASSERT_EQ(BufferSyncInterface::PARSING, engine()->GetStatus());
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, engine()->GetParseError());
+ EXPECT_EQ(new_get, engine()->Get());
+ get = new_get;
+ }
+ // Check that the commands did happen.
+ Mock::VerifyAndClearExpectations(api_mock());
+
+ engine()->CloseConnection();
+ DestroyCommandBuffer();
+}
+
+// Checks that commands in the buffer are properly executed when wrapping the
+// buffer, and that the status/error stay valid.
+TEST_F(CommandBufferEngineTest, TestCommandWrapping) {
+ engine()->InitConnection();
+ CommandBufferEntry *entries = InitCommandBuffer(10, 6);
+ ASSERT_TRUE(entries != NULL);
+
+ CommandBufferOffset get = engine()->Get();
+ CommandBufferOffset put = get;
+
+ // Create a command buffer with 3 commands
+ put += AddCommandWithExpect(entries + put,
+ BufferSyncInterface::PARSE_NO_ERROR,
+ 0,
+ 0,
+ NULL);
+
+ CommandBufferEntry args1[2];
+ args1[0].value_uint32 = 3;
+ args1[1].value_float = 4.f;
+ put += AddCommandWithExpect(entries + put,
+ BufferSyncInterface::PARSE_NO_ERROR,
+ 1,
+ 2,
+ args1);
+ DCHECK_EQ(10, put);
+ put = 0;
+
+ CommandBufferEntry args2[2];
+ args2[0].value_uint32 = 5;
+ args2[1].value_float = 6.f;
+ put += AddCommandWithExpect(entries + put,
+ BufferSyncInterface::PARSE_NO_ERROR,
+ 2,
+ 2,
+ args2);
+
+ engine()->Put(put);
+ while (get != put) {
+ // Check that the parsing progresses, and that no error occurs.
+ CommandBufferOffset new_get = engine()->WaitGetChanges(get);
+ EXPECT_NE(get, new_get);
+ ASSERT_EQ(BufferSyncInterface::PARSING, engine()->GetStatus());
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, engine()->GetParseError());
+ EXPECT_EQ(new_get, engine()->Get());
+ get = new_get;
+ }
+ // Check that the commands did happen.
+ Mock::VerifyAndClearExpectations(api_mock());
+
+ engine()->CloseConnection();
+ DestroyCommandBuffer();
+}
+
+// Checks that commands in the buffer are properly executed when we change the
+// buffer, and when we close the connection.
+TEST_F(CommandBufferEngineTest, TestSetBufferAndClose) {
+ engine()->InitConnection();
+ CommandBufferEntry *entries = InitCommandBuffer(10, 0);
+ ASSERT_TRUE(entries != NULL);
+
+ CommandBufferOffset get = engine()->Get();
+ CommandBufferOffset put = get;
+
+ // Create a command buffer with 3 commands
+ put += AddCommandWithExpect(entries + put,
+ BufferSyncInterface::PARSE_NO_ERROR,
+ 0,
+ 0,
+ NULL);
+
+ CommandBufferEntry args1[2];
+ args1[0].value_uint32 = 3;
+ args1[1].value_float = 4.f;
+ put += AddCommandWithExpect(entries + put,
+ BufferSyncInterface::PARSE_NO_ERROR,
+ 1,
+ 2,
+ args1);
+
+ CommandBufferEntry args2[2];
+ args2[0].value_uint32 = 5;
+ args2[1].value_float = 6.f;
+ put += AddCommandWithExpect(entries + put,
+ BufferSyncInterface::PARSE_NO_ERROR,
+ 2,
+ 2,
+ args2);
+
+ engine()->Put(put);
+
+ // Setup a new buffer.
+ const size_t kShmSize = 10 * sizeof(CommandBufferEntry); // NOLINT
+ RPCShmHandle shm = CreateShm(kShmSize);
+ ASSERT_NE(kRPCInvalidHandle, shm);
+ unsigned int shm_id = engine()->RegisterSharedMemory(shm, kShmSize);
+ ASSERT_NE(BufferSyncInterface::kInvalidSharedMemoryId, shm_id);
+ CommandBufferEntry *entries2 = static_cast<CommandBufferEntry *>(
+ engine()->GetSharedMemoryAddress(shm_id));
+ engine()->SetCommandBuffer(shm_id, 0, kShmSize, 0);
+ EXPECT_EQ(BufferSyncInterface::PARSING, engine()->GetStatus());
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, engine()->GetParseError());
+ EXPECT_EQ(0, engine()->Get());
+
+ // Destroy the old command buffer.
+ DestroyCommandBuffer();
+
+ get = engine()->Get();
+ put = get;
+ put += AddCommandWithExpect(entries2 + put,
+ BufferSyncInterface::PARSE_NO_ERROR,
+ 1,
+ 2,
+ args1);
+
+ engine()->Put(put);
+
+ engine()->CloseConnection();
+ // Check that all the commands did happen.
+ Mock::VerifyAndClearExpectations(api_mock());
+
+ engine()->UnregisterSharedMemory(shm_id);
+ DestroyShm(shm);
+}
+
+// Checks that commands in the buffer are properly executed, even if they
+// generate a recoverable error. Check that the error status is properly set,
+// and reset when queried.
+TEST_F(CommandBufferEngineTest, TestRecoverableError) {
+ engine()->InitConnection();
+ CommandBufferEntry *entries = InitCommandBuffer(10, 0);
+ ASSERT_TRUE(entries != NULL);
+
+ CommandBufferOffset get = engine()->Get();
+ CommandBufferOffset put = get;
+
+ // Create a command buffer with 3 commands, 2 of them generating errors
+ put += AddCommandWithExpect(entries + put,
+ BufferSyncInterface::PARSE_NO_ERROR,
+ 0,
+ 0,
+ NULL);
+
+ CommandBufferEntry args1[2];
+ args1[0].value_uint32 = 3;
+ args1[1].value_float = 4.f;
+ put += AddCommandWithExpect(entries + put,
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS,
+ 1,
+ 2,
+ args1);
+
+ CommandBufferEntry args2[2];
+ args2[0].value_uint32 = 5;
+ args2[1].value_float = 6.f;
+ put += AddCommandWithExpect(entries + put,
+ BufferSyncInterface::PARSE_UNKNOWN_COMMAND,
+ 2,
+ 2,
+ args2);
+
+ engine()->Put(put);
+ while (get != put) {
+ // Check that the parsing progresses, and that no error occurs.
+ CommandBufferOffset new_get = engine()->WaitGetChanges(get);
+ EXPECT_NE(get, new_get);
+ ASSERT_EQ(BufferSyncInterface::PARSING, engine()->GetStatus());
+ EXPECT_EQ(new_get, engine()->Get());
+ get = new_get;
+ }
+ // Check that the commands did happen.
+ Mock::VerifyAndClearExpectations(api_mock());
+
+ // Check that the error status was set to the first error.
+ EXPECT_EQ(BufferSyncInterface::PARSE_INVALID_ARGUMENTS,
+ engine()->GetParseError());
+ // Check that the error status was reset after the query.
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, engine()->GetParseError());
+
+ engine()->CloseConnection();
+ DestroyCommandBuffer();
+}
+
+// Checks that commands in the buffer are properly executed up to the point
+// where a parsing error happened. Check that at that point the status and
+// error are properly set.
+TEST_F(CommandBufferEngineTest, TestNonRecoverableError) {
+ engine()->InitConnection();
+ // Create a buffer with 6 entries, starting at entry 1, but allocate enough
+ // memory so that we can add commands that cross over the limit.
+ const size_t kShmSize = 10 * sizeof(CommandBufferEntry); // NOLINT
+ RPCShmHandle shm = CreateShm(kShmSize);
+ ASSERT_NE(kRPCInvalidHandle, shm);
+ unsigned int shm_id = engine()->RegisterSharedMemory(shm, kShmSize);
+ ASSERT_NE(BufferSyncInterface::kInvalidSharedMemoryId, shm_id);
+ CommandBufferEntry *entries = static_cast<CommandBufferEntry *>(
+ engine()->GetSharedMemoryAddress(shm_id));
+ ASSERT_TRUE(entries != NULL);
+ engine()->SetCommandBuffer(shm_id, 0, 6 * sizeof(CommandBufferEntry), 1);
+
+ CommandBufferOffset get = engine()->Get();
+ CommandBufferOffset put = get;
+
+ // Create a command buffer with 3 commands, the last one overlapping the end
+ // of the buffer.
+ put += AddCommandWithExpect(entries + put,
+ BufferSyncInterface::PARSE_NO_ERROR,
+ 0,
+ 0,
+ NULL);
+
+ CommandBufferEntry args1[2];
+ args1[0].value_uint32 = 3;
+ args1[1].value_float = 4.f;
+ put += AddCommandWithExpect(entries + put,
+ BufferSyncInterface::PARSE_NO_ERROR,
+ 1,
+ 2,
+ args1);
+
+ CommandBufferOffset fail_get = put;
+ CommandHeader header;
+ header.size = 3;
+ header.command = 4;
+ entries[put++].value_header = header;
+ entries[put++].value_uint32 = 5;
+ entries[put++].value_uint32 = 6;
+
+ // we should be beyond the end of the buffer now.
+ DCHECK_LT(6, put);
+ put = 0;
+
+ engine()->Put(put);
+ while (get != put) {
+ // When the parsing stop progressing, break.
+ CommandBufferOffset new_get = engine()->WaitGetChanges(get);
+ if (new_get == get) {
+ EXPECT_EQ(new_get, engine()->Get());
+ break;
+ }
+ get = new_get;
+ }
+ // We should be in an error case now.
+ EXPECT_EQ(BufferSyncInterface::PARSE_ERROR, engine()->GetStatus());
+ // Check that the error status was set to the first error.
+ EXPECT_EQ(BufferSyncInterface::PARSE_OUT_OF_BOUNDS,
+ engine()->GetParseError());
+ // Check that the error status was reset after the query.
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, engine()->GetParseError());
+
+ // Check that the valid commands did happen.
+ Mock::VerifyAndClearExpectations(api_mock());
+
+ engine()->CloseConnection();
+ engine()->UnregisterSharedMemory(shm_id);
+ DestroyShm(shm);
+}
+
+// Checks that HasWork() and DoWork() have the correct semantics. If there is
+// work to do, DoWork should never block.
+TEST_F(CommandBufferEngineTest, TestDoWork) {
+ engine()->InitConnection();
+ CommandBufferEntry *entries = InitCommandBuffer(10, 0);
+ ASSERT_TRUE(entries != NULL);
+
+ CommandBufferOffset get = engine()->Get();
+ CommandBufferOffset put = get;
+
+ // Test that if we have no message and no command we will block.
+ process_mock()->Reset();
+ EXPECT_CALL(*process_mock(), HasMessage()).Times(AnyNumber());
+ EXPECT_FALSE(engine()->HasWork());
+ EXPECT_CALL(*process_mock(), ProcessMessage());
+ EXPECT_TRUE(engine()->DoWork());
+
+ EXPECT_TRUE(process_mock()->would_have_blocked());
+ Mock::VerifyAndClearExpectations(process_mock());
+
+ // Tests that messages get processed without blocking.
+ process_mock()->Reset();
+ EXPECT_CALL(*process_mock(), HasMessage()).Times(AnyNumber());
+ process_mock()->set_message_count(3);
+ EXPECT_TRUE(engine()->HasWork());
+
+ EXPECT_CALL(*process_mock(), ProcessMessage()).Times(3);
+ while (engine()->HasWork()) {
+ EXPECT_TRUE(engine()->DoWork());
+ }
+ EXPECT_EQ(0, process_mock()->message_count());
+ EXPECT_FALSE(process_mock()->would_have_blocked());
+ Mock::VerifyAndClearExpectations(process_mock());
+
+ // Test that if we have commands, we will process them without blocking.
+ // Create a command buffer with 3 commands
+ process_mock()->Reset();
+ EXPECT_CALL(*process_mock(), HasMessage()).Times(AnyNumber());
+ put += AddCommandWithExpect(entries + put,
+ BufferSyncInterface::PARSE_NO_ERROR,
+ 0,
+ 0,
+ NULL);
+
+ CommandBufferEntry args1[2];
+ args1[0].value_uint32 = 3;
+ args1[1].value_float = 4.f;
+ put += AddCommandWithExpect(entries + put,
+ BufferSyncInterface::PARSE_NO_ERROR,
+ 1,
+ 2,
+ args1);
+
+ CommandBufferEntry args2[2];
+ args2[0].value_uint32 = 5;
+ args2[1].value_float = 6.f;
+ put += AddCommandWithExpect(entries + put,
+ BufferSyncInterface::PARSE_NO_ERROR,
+ 2,
+ 2,
+ args2);
+
+ EXPECT_FALSE(engine()->HasWork()); // No work yet, until we change put
+
+ engine()->Put(put);
+ EXPECT_TRUE(engine()->HasWork());
+
+ EXPECT_CALL(*process_mock(), ProcessMessage()).Times(0);
+ while (engine()->HasWork()) {
+ EXPECT_TRUE(engine()->DoWork());
+ }
+ EXPECT_FALSE(process_mock()->would_have_blocked());
+ get = engine()->Get();
+ EXPECT_EQ(put, get); // once we're done, we should have executed everything.
+ ASSERT_EQ(BufferSyncInterface::PARSING, engine()->GetStatus());
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, engine()->GetParseError());
+ Mock::VerifyAndClearExpectations(process_mock());
+ Mock::VerifyAndClearExpectations(api_mock());
+
+ // Test that the engine stops if we send it a "kill" message.
+ process_mock()->Reset();
+ EXPECT_CALL(*process_mock(), HasMessage()).Times(AnyNumber());
+ process_mock()->set_message_count(1);
+ EXPECT_TRUE(engine()->HasWork());
+
+ EXPECT_CALL(*process_mock(), ProcessMessage()).WillOnce(Return(false));
+ EXPECT_FALSE(engine()->DoWork());
+ Mock::VerifyAndClearExpectations(process_mock());
+
+ engine()->CloseConnection();
+ DestroyCommandBuffer();
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/cross/cmd_parser.cc b/o3d/command_buffer/service/cross/cmd_parser.cc
new file mode 100644
index 0000000..a640d3d
--- /dev/null
+++ b/o3d/command_buffer/service/cross/cmd_parser.cc
@@ -0,0 +1,92 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the implementation of the command parser.
+
+#include "command_buffer/service/cross/cmd_parser.h"
+
+namespace o3d {
+namespace command_buffer {
+
+CommandParser::CommandParser(void *shm_address,
+ size_t shm_size,
+ ptrdiff_t offset,
+ size_t size,
+ CommandBufferOffset start_get,
+ AsyncAPIInterface *handler)
+ : get_(start_get),
+ put_(start_get),
+ handler_(handler) {
+ // check proper alignments.
+ DCHECK_EQ(0, (reinterpret_cast<intptr_t>(shm_address)) % 4);
+ DCHECK_EQ(0, offset % 4);
+ DCHECK_EQ(0, size % 4);
+ // check that the command buffer fits into the memory buffer.
+ DCHECK_GE(shm_size, offset + size);
+ char * buffer_begin = static_cast<char *>(shm_address) + offset;
+ buffer_ = reinterpret_cast<CommandBufferEntry *>(buffer_begin);
+ entry_count_ = size / 4;
+}
+
+// Process one command, reading the header from the command buffer, and
+// forwarding the command index and the arguments to the handler.
+// Note that:
+// - validation needs to happen on a copy of the data (to avoid race
+// conditions). This function only validates the header, leaving the arguments
+// validation to the handler, so it can pass a reference to them.
+// - get_ is modified *after* the command has been executed.
+BufferSyncInterface::ParseError CommandParser::ProcessCommand() {
+ CommandBufferOffset get = get_;
+ if (get == put_) return BufferSyncInterface::PARSE_NO_ERROR;
+
+ CommandHeader header = buffer_[get].value_header;
+ if (header.size == 0) return BufferSyncInterface::PARSE_INVALID_SIZE;
+ if (header.size + get > entry_count_)
+ return BufferSyncInterface::PARSE_OUT_OF_BOUNDS;
+ BufferSyncInterface::ParseError result = handler_->DoCommand(
+ header.command, header.size - 1, buffer_ + get + 1);
+ get_ = (get + header.size) % entry_count_;
+ return result;
+}
+
+// Processes all the commands, while the buffer is not empty. Stop if an error
+// is encountered.
+BufferSyncInterface::ParseError CommandParser::ProcessAllCommands() {
+ while (!IsEmpty()) {
+ BufferSyncInterface::ParseError error = ProcessCommand();
+ if (error) return error;
+ }
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/cross/cmd_parser.h b/o3d/command_buffer/service/cross/cmd_parser.h
new file mode 100644
index 0000000..84107c5
--- /dev/null
+++ b/o3d/command_buffer/service/cross/cmd_parser.h
@@ -0,0 +1,113 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the command parser class.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_CROSS_CMD_PARSER_H__
+#define O3D_COMMAND_BUFFER_SERVICE_CROSS_CMD_PARSER_H__
+
+#include "core/cross/types.h"
+#include "command_buffer/common/cross/rpc.h"
+#include "command_buffer/common/cross/buffer_sync_api.h"
+#include "command_buffer/common/cross/cmd_buffer_format.h"
+
+namespace o3d {
+namespace command_buffer {
+
+class AsyncAPIInterface;
+
+// Command parser class. This class parses commands from a shared memory
+// buffer, to implement some asynchronous RPC mechanism.
+class CommandParser {
+ public:
+ CommandParser(void *shm_address,
+ size_t shm_size,
+ ptrdiff_t offset,
+ size_t size,
+ CommandBufferOffset start_get,
+ AsyncAPIInterface *handler);
+
+ // Gets the "get" pointer. The get pointer is an index into the command
+ // buffer considered as an array of CommandBufferEntry.
+ CommandBufferOffset get() const { return get_; }
+
+ // Sets the "put" pointer. The put pointer is an index into the command
+ // buffer considered as an array of CommandBufferEntry.
+ void set_put(CommandBufferOffset put) { put_ = put; }
+
+ // Gets the "put" pointer. The put pointer is an index into the command
+ // buffer considered as an array of CommandBufferEntry.
+ CommandBufferOffset put() const { return put_; }
+
+ // Checks whether there are commands to process.
+ bool IsEmpty() const { return put_ == get_; }
+
+ // Processes one command, updating the get pointer. This will return an error
+ // if there are no commands in the buffer.
+ BufferSyncInterface::ParseError ProcessCommand();
+
+ // Processes all commands until get == put.
+ BufferSyncInterface::ParseError ProcessAllCommands();
+
+ private:
+ CommandBufferOffset get_;
+ CommandBufferOffset put_;
+ CommandBufferEntry *buffer_;
+ size_t entry_count_;
+ AsyncAPIInterface *handler_;
+};
+
+// This class defines the interface for an asynchronous API handler, that
+// is responsible for de-multiplexing commands and their arguments.
+class AsyncAPIInterface {
+ public:
+ AsyncAPIInterface() {}
+ virtual ~AsyncAPIInterface() {}
+
+ // Executes a command.
+ // Parameters:
+ // command: the command index.
+ // arg_count: the number of CommandBufferEntry arguments.
+ // args: the arguments.
+ // Returns:
+ // BufferSyncInterface::NO_ERROR if no error was found, one of
+ // BufferSyncInterface::ParseError otherwise.
+ virtual BufferSyncInterface::ParseError DoCommand(
+ unsigned int command,
+ unsigned int arg_count,
+ CommandBufferEntry *args) = 0;
+};
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_CROSS_CMD_PARSER_H__
diff --git a/o3d/command_buffer/service/cross/cmd_parser_test.cc b/o3d/command_buffer/service/cross/cmd_parser_test.cc
new file mode 100644
index 0000000..a63489c
--- /dev/null
+++ b/o3d/command_buffer/service/cross/cmd_parser_test.cc
@@ -0,0 +1,315 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// Tests for the command parser.
+
+#include "tests/common/win/testing_common.h"
+#include "command_buffer/service/cross/cmd_parser.h"
+#include "command_buffer/service/cross/mocks.h"
+
+namespace o3d {
+namespace command_buffer {
+
+using testing::Return;
+using testing::Mock;
+using testing::Truly;
+using testing::Sequence;
+using testing::_;
+
+// Test fixture for CommandParser test - Creates a mock AsyncAPIInterface, and
+// a fixed size memory buffer. Also provides a simple API to create a
+// CommandParser.
+class CommandParserTest : public testing::Test {
+ protected:
+ virtual void SetUp() {
+ api_mock_.reset(new AsyncAPIMock);
+ buffer_entry_count_ = 20;
+ buffer_.reset(new CommandBufferEntry[buffer_entry_count_]);
+ }
+ virtual void TearDown() {}
+
+ // Adds a DoCommand expectation in the mock.
+ void AddDoCommandExpect(BufferSyncInterface::ParseError _return,
+ unsigned int command,
+ unsigned int arg_count,
+ CommandBufferEntry *args) {
+ EXPECT_CALL(*api_mock(), DoCommand(command, arg_count,
+ Truly(AsyncAPIMock::IsArgs(arg_count, args))))
+ .InSequence(sequence_)
+ .WillOnce(Return(_return));
+ }
+
+ // Creates a parser, with a buffer of the specified size (in entries).
+ CommandParser *MakeParser(unsigned int entry_count) {
+ size_t shm_size = buffer_entry_count_ *
+ sizeof(CommandBufferEntry); // NOLINT
+ size_t command_buffer_size = entry_count *
+ sizeof(CommandBufferEntry); // NOLINT
+ DCHECK_LE(command_buffer_size, shm_size);
+ return new CommandParser(buffer(),
+ shm_size,
+ 0,
+ command_buffer_size,
+ 0,
+ api_mock());
+ }
+
+ unsigned int buffer_entry_count() { return 20; }
+ AsyncAPIMock *api_mock() { return api_mock_.get(); }
+ CommandBufferEntry *buffer() { return buffer_.get(); }
+ private:
+ unsigned int buffer_entry_count_;
+ scoped_ptr<AsyncAPIMock> api_mock_;
+ scoped_array<CommandBufferEntry> buffer_;
+ Sequence sequence_;
+};
+
+// Tests initialization conditions.
+TEST_F(CommandParserTest, TestInit) {
+ scoped_ptr<CommandParser> parser(MakeParser(10));
+ EXPECT_EQ(0, parser->get());
+ EXPECT_EQ(0, parser->put());
+ EXPECT_TRUE(parser->IsEmpty());
+}
+
+// Tests simple commands.
+TEST_F(CommandParserTest, TestSimple) {
+ scoped_ptr<CommandParser> parser(MakeParser(10));
+ CommandBufferOffset put = parser->put();
+ CommandHeader header;
+
+ // add a single command, no args
+ header.size = 1;
+ header.command = 123;
+ buffer()[put++].value_header = header;
+
+ parser->set_put(put);
+ EXPECT_EQ(put, parser->put());
+
+ AddDoCommandExpect(BufferSyncInterface::PARSE_NO_ERROR, 123, 0, NULL);
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, parser->ProcessCommand());
+ EXPECT_EQ(put, parser->get());
+ Mock::VerifyAndClearExpectations(api_mock());
+
+ // add a single command, 2 args
+ header.size = 3;
+ header.command = 456;
+ buffer()[put++].value_header = header;
+ buffer()[put++].value_int32 = 2134;
+ buffer()[put++].value_float = 1.f;
+
+ parser->set_put(put);
+ EXPECT_EQ(put, parser->put());
+
+ CommandBufferEntry param_array[2];
+ param_array[0].value_int32 = 2134;
+ param_array[1].value_float = 1.f;
+ AddDoCommandExpect(BufferSyncInterface::PARSE_NO_ERROR, 456, 2, param_array);
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, parser->ProcessCommand());
+ EXPECT_EQ(put, parser->get());
+ Mock::VerifyAndClearExpectations(api_mock());
+}
+
+// Tests having multiple commands in the buffer.
+TEST_F(CommandParserTest, TestMultipleCommands) {
+ scoped_ptr<CommandParser> parser(MakeParser(10));
+ CommandBufferOffset put = parser->put();
+ CommandHeader header;
+
+ // add 2 commands, test with single ProcessCommand()
+ header.size = 2;
+ header.command = 789;
+ buffer()[put++].value_header = header;
+ buffer()[put++].value_int32 = 5151;
+
+ CommandBufferOffset put_cmd2 = put;
+ header.size = 2;
+ header.command = 2121;
+ buffer()[put++].value_header = header;
+ buffer()[put++].value_int32 = 3434;
+
+ parser->set_put(put);
+ EXPECT_EQ(put, parser->put());
+
+ CommandBufferEntry param_array[2];
+ param_array[0].value_int32 = 5151;
+ AddDoCommandExpect(BufferSyncInterface::PARSE_NO_ERROR, 789, 1, param_array);
+ param_array[1].value_int32 = 3434;
+ AddDoCommandExpect(BufferSyncInterface::PARSE_NO_ERROR, 2121, 1,
+ param_array+1);
+
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, parser->ProcessCommand());
+ EXPECT_EQ(put_cmd2, parser->get());
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, parser->ProcessCommand());
+ EXPECT_EQ(put, parser->get());
+ Mock::VerifyAndClearExpectations(api_mock());
+
+ // add 2 commands again, test with ProcessAllCommands()
+ header.size = 2;
+ header.command = 4545;
+ buffer()[put++].value_header = header;
+ buffer()[put++].value_int32 = 5656;
+
+ header.size = 2;
+ header.command = 6767;
+ buffer()[put++].value_header = header;
+ buffer()[put++].value_int32 = 7878;
+
+ parser->set_put(put);
+ EXPECT_EQ(put, parser->put());
+
+ param_array[0].value_int32 = 5656;
+ AddDoCommandExpect(BufferSyncInterface::PARSE_NO_ERROR, 4545, 1, param_array);
+ param_array[1].value_int32 = 7878;
+ AddDoCommandExpect(BufferSyncInterface::PARSE_NO_ERROR, 6767, 1,
+ param_array+1);
+
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, parser->ProcessAllCommands());
+ EXPECT_EQ(put, parser->get());
+ Mock::VerifyAndClearExpectations(api_mock());
+}
+
+// Tests that the parser will wrap correctly at the end of the buffer.
+TEST_F(CommandParserTest, TestWrap) {
+ scoped_ptr<CommandParser> parser(MakeParser(5));
+ CommandBufferOffset put = parser->put();
+ CommandHeader header;
+
+ // add 3 commands with no args (1 word each)
+ for (unsigned int i = 0; i < 3; ++i) {
+ header.size = 1;
+ header.command = i;
+ buffer()[put++].value_header = header;
+ AddDoCommandExpect(BufferSyncInterface::PARSE_NO_ERROR, i, 0, NULL);
+ }
+ parser->set_put(put);
+ EXPECT_EQ(put, parser->put());
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, parser->ProcessAllCommands());
+ EXPECT_EQ(put, parser->get());
+ Mock::VerifyAndClearExpectations(api_mock());
+
+ // add 1 command with 1 arg (2 words). That should put us at the end of the
+ // buffer.
+ header.size = 2;
+ header.command = 3;
+ buffer()[put++].value_header = header;
+ buffer()[put++].value_int32 = 5;
+ CommandBufferEntry param;
+ param.value_int32 = 5;
+ AddDoCommandExpect(BufferSyncInterface::PARSE_NO_ERROR, 3, 1, &param);
+
+ DCHECK_EQ(5, put);
+ put = 0;
+ parser->set_put(put);
+ EXPECT_EQ(put, parser->put());
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, parser->ProcessAllCommands());
+ EXPECT_EQ(put, parser->get());
+ Mock::VerifyAndClearExpectations(api_mock());
+
+ // add 1 command with 1 arg (2 words).
+ header.size = 2;
+ header.command = 4;
+ buffer()[put++].value_header = header;
+ buffer()[put++].value_int32 = 6;
+ param.value_int32 = 6;
+ AddDoCommandExpect(BufferSyncInterface::PARSE_NO_ERROR, 4, 1, &param);
+ parser->set_put(put);
+ EXPECT_EQ(put, parser->put());
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, parser->ProcessAllCommands());
+ EXPECT_EQ(put, parser->get());
+ Mock::VerifyAndClearExpectations(api_mock());
+}
+
+// Tests error conditions.
+TEST_F(CommandParserTest, TestError) {
+ scoped_ptr<CommandParser> parser(MakeParser(5));
+ CommandBufferOffset put = parser->put();
+ CommandHeader header;
+
+ // Generate a command with size 0.
+ header.size = 0;
+ header.command = 3;
+ buffer()[put++].value_header = header;
+
+ parser->set_put(put);
+ EXPECT_EQ(put, parser->put());
+ EXPECT_EQ(BufferSyncInterface::PARSE_INVALID_SIZE,
+ parser->ProcessAllCommands());
+ // check that no DoCommand call was made.
+ Mock::VerifyAndClearExpectations(api_mock());
+
+ parser.reset(MakeParser(5));
+ put = parser->put();
+
+ // Generate a command with size 6, extends beyond the end of the buffer.
+ header.size = 6;
+ header.command = 3;
+ buffer()[put++].value_header = header;
+
+ parser->set_put(put);
+ EXPECT_EQ(put, parser->put());
+ EXPECT_EQ(BufferSyncInterface::PARSE_OUT_OF_BOUNDS,
+ parser->ProcessAllCommands());
+ // check that no DoCommand call was made.
+ Mock::VerifyAndClearExpectations(api_mock());
+
+ parser.reset(MakeParser(5));
+ put = parser->put();
+
+ // Generates 2 commands.
+ header.size = 1;
+ header.command = 3;
+ buffer()[put++].value_header = header;
+ CommandBufferOffset put_post_fail = put;
+ header.size = 1;
+ header.command = 4;
+ buffer()[put++].value_header = header;
+
+ parser->set_put(put);
+ EXPECT_EQ(put, parser->put());
+ // have the first command fail to parse.
+ AddDoCommandExpect(BufferSyncInterface::PARSE_UNKNOWN_COMMAND, 3, 0, NULL);
+ EXPECT_EQ(BufferSyncInterface::PARSE_UNKNOWN_COMMAND,
+ parser->ProcessAllCommands());
+ // check that only one command was executed, and that get reflects that
+ // correctly.
+ EXPECT_EQ(put_post_fail, parser->get());
+ Mock::VerifyAndClearExpectations(api_mock());
+ // make the second one succeed, and check that the parser recovered fine.
+ AddDoCommandExpect(BufferSyncInterface::PARSE_NO_ERROR, 4, 0, NULL);
+ EXPECT_EQ(BufferSyncInterface::PARSE_NO_ERROR, parser->ProcessAllCommands());
+ EXPECT_EQ(put, parser->get());
+ Mock::VerifyAndClearExpectations(api_mock());
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/cross/effect_utils.cc b/o3d/command_buffer/service/cross/effect_utils.cc
new file mode 100644
index 0000000..91beb17
--- /dev/null
+++ b/o3d/command_buffer/service/cross/effect_utils.cc
@@ -0,0 +1,70 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file implements effect related utilities.
+
+#include "command_buffer/service/cross/effect_utils.h"
+
+namespace o3d {
+namespace command_buffer {
+
+bool ParseEffectData(unsigned int size,
+ const void *data,
+ String *vertex_program_entry,
+ String *fragment_program_entry,
+ String *effect_code) {
+ const char *data_char = static_cast<const char *>(data);
+ unsigned int index = 0;
+
+ for (; index < size && data_char[index]; ++index) { }
+ if (index >= size) return false;
+ *vertex_program_entry = String(data_char, index);
+ ++index; // skip \0
+ unsigned int fragment_program_entry_begin = index;
+
+ for (; index < size && data_char[index]; ++index) { }
+ if (index >= size) return false;
+ *fragment_program_entry = String(data_char + fragment_program_entry_begin,
+ index - fragment_program_entry_begin);
+ ++index; // skip \0
+ unsigned int effect_code_begin = index;
+
+ // text doesn't have to be 0-terminated, but look for one so that we don't
+ // construct a std::string with a '\0' in it.
+ for (; index < size && data_char[index]; ++index) { }
+ *effect_code = String(data_char + effect_code_begin,
+ index - effect_code_begin);
+ return true;
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/cross/effect_utils.h b/o3d/command_buffer/service/cross/effect_utils.h
new file mode 100644
index 0000000..2402254
--- /dev/null
+++ b/o3d/command_buffer/service/cross/effect_utils.h
@@ -0,0 +1,56 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file declares some effect related utilities.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_CROSS_EFFECT_UTILS_H_
+#define O3D_COMMAND_BUFFER_SERVICE_CROSS_EFFECT_UTILS_H_
+
+#include "command_buffer/common/cross/types.h"
+
+namespace o3d {
+namespace command_buffer {
+
+// This function parses the data passed to the CREATE_EFFECT commands, which
+// follows the following format:
+// vertex_program_entry \0 fragment_program_entry \0 effect_code
+// It returns the various components.
+bool ParseEffectData(unsigned int size,
+ const void *data,
+ String *vertex_program_entry,
+ String *fragment_program_entry,
+ String *effect_code);
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_CROSS_EFFECT_UTILS_H_
diff --git a/o3d/command_buffer/service/cross/effect_utils_test.cc b/o3d/command_buffer/service/cross/effect_utils_test.cc
new file mode 100644
index 0000000..d37db9d
--- /dev/null
+++ b/o3d/command_buffer/service/cross/effect_utils_test.cc
@@ -0,0 +1,86 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the unit tests for the effect utilities.
+
+#include "tests/common/win/testing_common.h"
+#include "command_buffer/service/cross/effect_utils.h"
+
+namespace o3d {
+namespace command_buffer {
+
+TEST(ParseEffectDataTest, ValidData) {
+ // Tests well-formed data.
+ const char kEffect[] = "vertex_entry\0fragment_entry\0effect code";
+ String vertex_program_entry;
+ String fragment_program_entry;
+ String effect_code;
+ EXPECT_TRUE(ParseEffectData(sizeof(kEffect), kEffect, &vertex_program_entry,
+ &fragment_program_entry, &effect_code));
+ EXPECT_EQ(vertex_program_entry, "vertex_entry");
+ EXPECT_EQ(fragment_program_entry, "fragment_entry");
+ EXPECT_EQ(effect_code, "effect code");
+
+ // The terminal \0 isn't needed, check that we parse correctly without it.
+ EXPECT_TRUE(ParseEffectData(sizeof(kEffect)-1, kEffect,
+ &vertex_program_entry, &fragment_program_entry,
+ &effect_code));
+ EXPECT_EQ(vertex_program_entry, "vertex_entry");
+ EXPECT_EQ(fragment_program_entry, "fragment_entry");
+ EXPECT_EQ(effect_code, "effect code");
+}
+
+TEST(ParseEffectDataTest, InvalidData) {
+ const char kEffect[] = "vertex_entry\0fragment_entry\0effect code";
+ String vertex_program_entry;
+ String fragment_program_entry;
+ String effect_code;
+ // 0-size
+ EXPECT_FALSE(ParseEffectData(0, kEffect,
+ &vertex_program_entry, &fragment_program_entry,
+ &effect_code));
+ // Only vertex_entry, no \0
+ EXPECT_FALSE(ParseEffectData(strlen("vertex_entry"), kEffect,
+ &vertex_program_entry, &fragment_program_entry,
+ &effect_code));
+ // Only vertex_entry\0
+ EXPECT_FALSE(ParseEffectData(strlen("vertex_entry") + 1, kEffect,
+ &vertex_program_entry, &fragment_program_entry,
+ &effect_code));
+ // Only vertex_entry\0fragment_entry, no \0
+ EXPECT_FALSE(ParseEffectData(strlen("vertex_entry.fragment_entry"), kEffect,
+ &vertex_program_entry, &fragment_program_entry,
+ &effect_code));
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/cross/gapi_decoder.cc b/o3d/command_buffer/service/cross/gapi_decoder.cc
new file mode 100644
index 0000000..efd3bc5
--- /dev/null
+++ b/o3d/command_buffer/service/cross/gapi_decoder.cc
@@ -0,0 +1,881 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This class contains the implementation of the GAPI decoder class, decoding
+// GAPI commands into calls to a GAPIInterface class.
+
+#include "base/cross/bits.h"
+#include "command_buffer/common/cross/gapi_interface.h"
+#include "command_buffer/service/cross/gapi_decoder.h"
+#include "command_buffer/service/cross/cmd_buffer_engine.h"
+
+namespace o3d {
+namespace command_buffer {
+
+// Decode command with its arguments, and call the corresponding GAPIInterface
+// method.
+// Note: args is a pointer to the command buffer. As such, it could be changed
+// by a (malicious) client at any time, so if validation has to happen, it
+// should operate on a copy of them.
+BufferSyncInterface::ParseError GAPIDecoder::DoCommand(
+ unsigned int command,
+ unsigned int arg_count,
+ CommandBufferEntry *args) {
+ switch (command) {
+ case NOOP:
+ return BufferSyncInterface::PARSE_NO_ERROR;
+ case SET_TOKEN:
+ if (arg_count == 1) {
+ engine_->set_token(args[0].value_uint32);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case BEGIN_FRAME:
+ if (arg_count == 0) {
+ gapi_->BeginFrame();
+ return BufferSyncInterface::PARSE_NO_ERROR;
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case END_FRAME:
+ if (arg_count == 0) {
+ gapi_->EndFrame();
+ return BufferSyncInterface::PARSE_NO_ERROR;
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case CLEAR:
+ if (arg_count == 7) {
+ unsigned int buffers = args[0].value_uint32;
+ if (buffers & ~GAPIInterface::ALL_BUFFERS)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ RGBA rgba;
+ rgba.red = args[1].value_float;
+ rgba.green = args[2].value_float;
+ rgba.blue = args[3].value_float;
+ rgba.alpha = args[4].value_float;
+ float depth = args[5].value_float;
+ unsigned int stencil = args[6].value_uint32;
+ gapi_->Clear(buffers, rgba, depth, stencil);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case SET_VIEWPORT:
+ if (arg_count == 6) {
+ gapi_->SetViewport(args[0].value_uint32,
+ args[1].value_uint32,
+ args[2].value_uint32,
+ args[3].value_uint32,
+ args[4].value_float,
+ args[5].value_float);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case CREATE_VERTEX_BUFFER:
+ if (arg_count == 3) {
+ return gapi_->CreateVertexBuffer(args[0].value_uint32,
+ args[1].value_uint32,
+ args[2].value_uint32);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case DESTROY_VERTEX_BUFFER:
+ if (arg_count == 1) {
+ return gapi_->DestroyVertexBuffer(args[0].value_uint32);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case SET_VERTEX_BUFFER_DATA_IMMEDIATE: {
+ if (arg_count < 2) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ ResourceID id = args[0].value_uint32;
+ unsigned int offset = args[1].value_uint32;
+ unsigned int size = (arg_count - 2) * sizeof(args[0]);
+ return gapi_->SetVertexBufferData(id, offset, size, args + 2);
+ }
+ case SET_VERTEX_BUFFER_DATA:
+ if (arg_count == 5) {
+ ResourceID id = args[0].value_uint32;
+ unsigned int offset = args[1].value_uint32;
+ unsigned int size = args[2].value_uint32;
+ void *data = GetAddressAndCheckSize(args[3].value_uint32,
+ args[4].value_uint32,
+ size);
+ if (!data) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return gapi_->SetVertexBufferData(id, offset, size, data);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case GET_VERTEX_BUFFER_DATA:
+ if (arg_count == 5) {
+ ResourceID id = args[0].value_uint32;
+ unsigned int offset = args[1].value_uint32;
+ unsigned int size = args[2].value_uint32;
+ void *data = GetAddressAndCheckSize(args[3].value_uint32,
+ args[4].value_uint32,
+ size);
+ if (!data) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return gapi_->GetVertexBufferData(id, offset, size, data);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case CREATE_INDEX_BUFFER:
+ if (arg_count == 3) {
+ return gapi_->CreateIndexBuffer(args[0].value_uint32,
+ args[1].value_uint32,
+ args[2].value_uint32);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case DESTROY_INDEX_BUFFER:
+ if (arg_count == 1) {
+ return gapi_->DestroyIndexBuffer(args[0].value_uint32);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case SET_INDEX_BUFFER_DATA_IMMEDIATE: {
+ if (arg_count < 2) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ ResourceID id = args[0].value_uint32;
+ unsigned int offset = args[1].value_uint32;
+ unsigned int size = (arg_count - 2) * sizeof(args[0]);
+ return gapi_->SetIndexBufferData(id, offset, size, args + 2);
+ }
+ case SET_INDEX_BUFFER_DATA:
+ if (arg_count == 5) {
+ ResourceID id = args[0].value_uint32;
+ unsigned int offset = args[1].value_uint32;
+ unsigned int size = args[2].value_uint32;
+ void *data = GetAddressAndCheckSize(args[3].value_uint32,
+ args[4].value_uint32,
+ size);
+ if (!data) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return gapi_->SetIndexBufferData(id, offset, size, data);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case GET_INDEX_BUFFER_DATA:
+ if (arg_count == 5) {
+ ResourceID id = args[0].value_uint32;
+ unsigned int offset = args[1].value_uint32;
+ unsigned int size = args[2].value_uint32;
+ void *data = GetAddressAndCheckSize(args[3].value_uint32,
+ args[4].value_uint32,
+ size);
+ if (!data) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return gapi_->GetIndexBufferData(id, offset, size, data);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case CREATE_VERTEX_STRUCT:
+ if (arg_count == 2) {
+ return gapi_->CreateVertexStruct(args[0].value_uint32,
+ args[1].value_uint32);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case DESTROY_VERTEX_STRUCT:
+ if (arg_count == 1) {
+ return gapi_->DestroyVertexStruct(args[0].value_uint32);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case SET_VERTEX_INPUT:
+ return DecodeSetVertexInput(arg_count, args);
+ case SET_VERTEX_STRUCT:
+ if (arg_count == 1) {
+ return gapi_->SetVertexStruct(args[0].value_uint32);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case DRAW:
+ if (arg_count == 3) {
+ unsigned int primitive_type = args[0].value_uint32;
+ if (primitive_type >= GAPIInterface::MAX_PRIMITIVE_TYPE)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ unsigned int first = args[1].value_uint32;
+ unsigned int count = args[2].value_uint32;
+ return gapi_->Draw(
+ static_cast<GAPIInterface::PrimitiveType>(primitive_type),
+ first, count);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case DRAW_INDEXED:
+ if (arg_count == 6) {
+ unsigned int primitive_type = args[0].value_uint32;
+ if (primitive_type >= GAPIInterface::MAX_PRIMITIVE_TYPE)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ ResourceID index_buffer = args[1].value_uint32;
+ unsigned int first = args[2].value_uint32;
+ unsigned int count = args[3].value_uint32;
+ unsigned int min_index = args[4].value_uint32;
+ unsigned int max_index = args[5].value_uint32;
+ return gapi_->DrawIndexed(
+ static_cast<GAPIInterface::PrimitiveType>(primitive_type),
+ index_buffer, first, count, min_index, max_index);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case CREATE_EFFECT:
+ if (arg_count == 4) {
+ ResourceID id = args[0].value_uint32;
+ unsigned int size = args[1].value_uint32;
+ void *data = GetAddressAndCheckSize(args[2].value_uint32,
+ args[3].value_uint32,
+ size);
+ if (!data) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return gapi_->CreateEffect(id, size, data);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case CREATE_EFFECT_IMMEDIATE:
+ if (arg_count > 2) {
+ ResourceID id = args[0].value_uint32;
+ unsigned int size = args[1].value_uint32;
+ if (size > (arg_count-2) * sizeof(args[0]))
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return gapi_->CreateEffect(id, size, args + 2);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case DESTROY_EFFECT:
+ if (arg_count == 1) {
+ return gapi_->DestroyEffect(args[0].value_uint32);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case SET_EFFECT:
+ if (arg_count == 1) {
+ return gapi_->SetEffect(args[0].value_uint32);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case GET_PARAM_COUNT:
+ if (arg_count == 4) {
+ ResourceID id = args[0].value_uint32;
+ unsigned int size = args[1].value_uint32;
+ void *data = GetAddressAndCheckSize(args[2].value_uint32,
+ args[3].value_uint32,
+ size);
+ if (!data) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return gapi_->GetParamCount(id, size, data);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case CREATE_PARAM:
+ if (arg_count == 3) {
+ ResourceID param_id = args[0].value_uint32;
+ ResourceID effect_id = args[1].value_uint32;
+ unsigned int index = args[2].value_uint32;
+ return gapi_->CreateParam(param_id, effect_id, index);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case CREATE_PARAM_BY_NAME:
+ if (arg_count == 5) {
+ ResourceID param_id = args[0].value_uint32;
+ ResourceID effect_id = args[1].value_uint32;
+ unsigned int size = args[2].value_uint32;
+ void *data = GetAddressAndCheckSize(args[3].value_uint32,
+ args[4].value_uint32,
+ size);
+ if (!data)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return gapi_->CreateParamByName(param_id, effect_id, size, data);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case CREATE_PARAM_BY_NAME_IMMEDIATE:
+ if (arg_count > 3) {
+ ResourceID param_id = args[0].value_uint32;
+ ResourceID effect_id = args[1].value_uint32;
+ unsigned int size = args[2].value_uint32;
+ if (size > (arg_count-1) * sizeof(args[0]))
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return gapi_->CreateParamByName(param_id, effect_id, size, args + 3);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case DESTROY_PARAM:
+ if (arg_count == 1) {
+ return gapi_->DestroyParam(args[0].value_uint32);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case SET_PARAM_DATA:
+ if (arg_count == 4) {
+ ResourceID id = args[0].value_uint32;
+ unsigned int size = args[1].value_uint32;
+ void *data = GetAddressAndCheckSize(args[2].value_uint32,
+ args[3].value_uint32,
+ size);
+ if (!data) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return gapi_->SetParamData(id, size, data);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case SET_PARAM_DATA_IMMEDIATE:
+ if (arg_count > 2) {
+ ResourceID id = args[0].value_uint32;
+ unsigned int size = args[1].value_uint32;
+ if (size > (arg_count-2) * sizeof(args[0]))
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return gapi_->SetParamData(id, size, args + 2);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case GET_PARAM_DESC:
+ if (arg_count == 4) {
+ ResourceID id = args[0].value_uint32;
+ unsigned int size = args[1].value_uint32;
+ void *data = GetAddressAndCheckSize(args[2].value_uint32,
+ args[3].value_uint32,
+ size);
+ if (!data) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return gapi_->GetParamDesc(id, size, data);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case DESTROY_TEXTURE:
+ if (arg_count == 1) {
+ return gapi_->DestroyTexture(args[0].value_uint32);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case CREATE_TEXTURE_2D:
+ return DecodeCreateTexture2D(arg_count, args);
+ case CREATE_TEXTURE_3D:
+ return DecodeCreateTexture3D(arg_count, args);
+ case CREATE_TEXTURE_CUBE:
+ return DecodeCreateTextureCube(arg_count, args);
+ case SET_TEXTURE_DATA:
+ return DecodeSetTextureData(arg_count, args);
+ case SET_TEXTURE_DATA_IMMEDIATE:
+ return DecodeSetTextureDataImmediate(arg_count, args);
+ case GET_TEXTURE_DATA:
+ return DecodeGetTextureData(arg_count, args);
+ case CREATE_SAMPLER:
+ if (arg_count == 1) {
+ return gapi_->CreateSampler(args[0].value_uint32);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case DESTROY_SAMPLER:
+ if (arg_count == 1) {
+ return gapi_->DestroySampler(args[0].value_uint32);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case SET_SAMPLER_STATES:
+ return DecodeSetSamplerStates(arg_count, args);
+ case SET_SAMPLER_BORDER_COLOR:
+ if (arg_count == 5) {
+ RGBA rgba;
+ rgba.red = args[1].value_float;
+ rgba.green = args[2].value_float;
+ rgba.blue = args[3].value_float;
+ rgba.alpha = args[4].value_float;
+ return gapi_->SetSamplerBorderColor(args[0].value_uint32, rgba);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case SET_SAMPLER_TEXTURE:
+ if (arg_count == 2) {
+ return gapi_->SetSamplerTexture(args[0].value_uint32,
+ args[1].value_uint32);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case SET_SCISSOR:
+ if (arg_count == 2) {
+ namespace cmd = set_scissor;
+ Uint32 x_y_enable = args[0].value_uint32;
+ if (cmd::Unused::Get(x_y_enable) != 0)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ unsigned int x = cmd::X::Get(x_y_enable);
+ unsigned int y = cmd::Y::Get(x_y_enable);
+ bool enable = cmd::Enable::Get(x_y_enable) != 0;
+ Uint32 width_height = args[1].value_uint32;
+ unsigned int width = cmd::Width::Get(width_height);
+ unsigned int height = cmd::Height::Get(width_height);
+ gapi_->SetScissor(enable, x, y, width, height);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case SET_POLYGON_OFFSET:
+ if (arg_count == 2) {
+ gapi_->SetPolygonOffset(args[0].value_float, args[1].value_float);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case SET_POINT_LINE_RASTER:
+ if (arg_count == 2) {
+ namespace cmd = set_point_line_raster;
+ Uint32 enables = args[0].value_uint32;
+ if (cmd::Unused::Get(enables) != 0)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ bool line_smooth = cmd::LineSmoothEnable::Get(enables);
+ bool point_sprite = cmd::PointSpriteEnable::Get(enables);
+ float point_size = args[1].value_float;
+ gapi_->SetPointLineRaster(line_smooth, point_sprite, point_size);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case SET_POLYGON_RASTER:
+ if (arg_count == 1) {
+ namespace cmd = set_polygon_raster;
+ Uint32 fill_cull = args[0].value_uint32;
+ unsigned int fill_value = cmd::FillMode::Get(fill_cull);
+ unsigned int cull_value = cmd::CullMode::Get(fill_cull);
+ if (cmd::Unused::Get(fill_cull) != 0 ||
+ fill_value >= GAPIInterface::NUM_POLYGON_MODE ||
+ cull_value >= GAPIInterface::NUM_FACE_CULL_MODE)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ gapi_->SetPolygonRaster(
+ static_cast<GAPIInterface::PolygonMode>(fill_value),
+ static_cast<GAPIInterface::FaceCullMode>(cull_value));
+ return BufferSyncInterface::PARSE_NO_ERROR;
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case SET_ALPHA_TEST:
+ if (arg_count == 2) {
+ namespace cmd = set_alpha_test;
+ Uint32 func_enable = args[0].value_uint32;
+ if (cmd::Unused::Get(func_enable) != 0)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ // Check that the bitmask get cannot generate values outside of the
+ // allowed range.
+ COMPILE_ASSERT(cmd::Func::kMask < GAPIInterface::NUM_COMPARISON,
+ set_alpha_test_Func_may_produce_invalid_values);
+ GAPIInterface::Comparison comp =
+ static_cast<GAPIInterface::Comparison>(cmd::Func::Get(func_enable));
+ bool enable = cmd::Enable::Get(func_enable) != 0;
+ gapi_->SetAlphaTest(enable, args[1].value_float, comp);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case SET_DEPTH_TEST:
+ if (arg_count == 1) {
+ namespace cmd = set_depth_test;
+ Uint32 func_enable = args[0].value_uint32;
+ if (cmd::Unused::Get(func_enable) != 0)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ // Check that the bitmask get cannot generate values outside of the
+ // allowed range.
+ COMPILE_ASSERT(cmd::Func::kMask < GAPIInterface::NUM_COMPARISON,
+ set_alpha_test_Func_may_produce_invalid_values);
+ GAPIInterface::Comparison comp =
+ static_cast<GAPIInterface::Comparison>(cmd::Func::Get(func_enable));
+ bool write_enable = cmd::WriteEnable::Get(func_enable) != 0;
+ bool enable = cmd::Enable::Get(func_enable) != 0;
+ gapi_->SetDepthTest(enable, write_enable, comp);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case SET_STENCIL_TEST:
+ return DecodeSetStencilTest(arg_count, args);
+ case SET_COLOR_WRITE:
+ if (arg_count == 1) {
+ namespace cmd = set_color_write;
+ Uint32 enables = args[0].value_uint32;
+ if (cmd::Unused::Get(enables) != 0)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ bool red = cmd::RedMask::Get(enables) != 0;
+ bool green = cmd::GreenMask::Get(enables) != 0;
+ bool blue = cmd::BlueMask::Get(enables) != 0;
+ bool alpha = cmd::AlphaMask::Get(enables) != 0;
+ bool dither = cmd::DitherEnable::Get(enables) != 0;
+ gapi_->SetColorWrite(red, green, blue, alpha, dither);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ case SET_BLENDING:
+ return DecodeSetBlending(arg_count, args);
+ case SET_BLENDING_COLOR:
+ if (arg_count == 4) {
+ RGBA rgba;
+ rgba.red = args[0].value_float;
+ rgba.green = args[1].value_float;
+ rgba.blue = args[2].value_float;
+ rgba.alpha = args[3].value_float;
+ gapi_->SetBlendingColor(rgba);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ default:
+ return BufferSyncInterface::PARSE_UNKNOWN_COMMAND;
+ }
+}
+
+// Decodes the SET_VERTEX_INPUT command.
+BufferSyncInterface::ParseError GAPIDecoder::DecodeSetVertexInput(
+ unsigned int arg_count,
+ CommandBufferEntry *args) {
+ namespace cmd = set_vertex_input_cmd;
+ if (arg_count != 5) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ ResourceID vertex_struct_id = args[0].value_uint32;
+ unsigned int input_index = args[1].value_uint32;
+ ResourceID vertex_buffer_id = args[2].value_uint32;
+ unsigned int offset = args[3].value_uint32;
+ unsigned int type_stride_semantic = args[4].value_uint32;
+ unsigned int semantic_index = cmd::SemanticIndex::Get(type_stride_semantic);
+ unsigned int semantic = cmd::Semantic::Get(type_stride_semantic);
+ unsigned int type = cmd::Type::Get(type_stride_semantic);
+ unsigned int stride = cmd::Stride::Get(type_stride_semantic);
+ if (semantic >= vertex_struct::NUM_SEMANTICS ||
+ type >= vertex_struct::NUM_TYPES || stride == 0)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return gapi_->SetVertexInput(vertex_struct_id, input_index, vertex_buffer_id,
+ offset, stride,
+ static_cast<vertex_struct::Type>(type),
+ static_cast<vertex_struct::Semantic>(semantic),
+ semantic_index);
+}
+
+// Decodes the CREATE_TEXTURE_2D command.
+BufferSyncInterface::ParseError GAPIDecoder::DecodeCreateTexture2D(
+ unsigned int arg_count,
+ CommandBufferEntry *args) {
+ if (arg_count == 3) {
+ namespace cmd = create_texture_2d_cmd;
+ unsigned int id = args[0].value_uint32;
+ unsigned int width_height = args[1].value_uint32;
+ unsigned int levels_format_flags = args[2].value_uint32;
+ unsigned int width = cmd::Width::Get(width_height);
+ unsigned int height = cmd::Height::Get(width_height);
+ unsigned int levels = cmd::Levels::Get(levels_format_flags);
+ unsigned int unused = cmd::Unused::Get(levels_format_flags);
+ unsigned int format = cmd::Format::Get(levels_format_flags);
+ unsigned int flags = cmd::Flags::Get(levels_format_flags);
+ unsigned int max_levels =
+ 1 + base::bits::Log2Ceiling(std::max(width, height));
+ if ((width == 0) || (height == 0) || (levels > max_levels) ||
+ (unused != 0) || (format >= texture::NUM_FORMATS))
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ if (levels == 0) levels = max_levels;
+ return gapi_->CreateTexture2D(id, width, height, levels,
+ static_cast<texture::Format>(format), flags);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+}
+
+// Decodes the CREATE_TEXTURE_3D command.
+BufferSyncInterface::ParseError GAPIDecoder::DecodeCreateTexture3D(
+ unsigned int arg_count,
+ CommandBufferEntry *args) {
+ if (arg_count == 4) {
+ namespace cmd = create_texture_3d_cmd;
+ unsigned int id = args[0].value_uint32;
+ unsigned int width_height = args[1].value_uint32;
+ unsigned int depth_unused = args[2].value_uint32;
+ unsigned int levels_format_flags = args[3].value_uint32;
+ unsigned int width = cmd::Width::Get(width_height);
+ unsigned int height = cmd::Height::Get(width_height);
+ unsigned int depth = cmd::Depth::Get(depth_unused);
+ unsigned int unused1 = cmd::Unused1::Get(depth_unused);
+ unsigned int levels = cmd::Levels::Get(levels_format_flags);
+ unsigned int unused2 = cmd::Unused2::Get(levels_format_flags);
+ unsigned int format = cmd::Format::Get(levels_format_flags);
+ unsigned int flags = cmd::Flags::Get(levels_format_flags);
+ unsigned int max_levels =
+ 1 + base::bits::Log2Ceiling(std::max(depth, std::max(width, height)));
+ if ((width == 0) || (height == 0) || (depth == 0) ||
+ (levels > max_levels) || (unused1 != 0) || (unused2 != 0) ||
+ (format >= texture::NUM_FORMATS))
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ if (levels == 0) levels = max_levels;
+ return gapi_->CreateTexture3D(id, width, height, depth, levels,
+ static_cast<texture::Format>(format), flags);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+}
+
+// Decodes the CREATE_TEXTURE_CUBE command.
+BufferSyncInterface::ParseError GAPIDecoder::DecodeCreateTextureCube(
+ unsigned int arg_count,
+ CommandBufferEntry *args) {
+ if (arg_count == 3) {
+ namespace cmd = create_texture_cube_cmd;
+ unsigned int id = args[0].value_uint32;
+ unsigned int side_unused = args[1].value_uint32;
+ unsigned int levels_format_flags = args[2].value_uint32;
+ unsigned int side = cmd::Side::Get(side_unused);
+ unsigned int unused1 = cmd::Unused1::Get(side_unused);
+ unsigned int levels = cmd::Levels::Get(levels_format_flags);
+ unsigned int unused2 = cmd::Unused2::Get(levels_format_flags);
+ unsigned int format = cmd::Format::Get(levels_format_flags);
+ unsigned int flags = cmd::Flags::Get(levels_format_flags);
+ unsigned int max_levels = 1 + base::bits::Log2Ceiling(side);
+ if ((side == 0) || (levels > max_levels) || (unused1 != 0) ||
+ (unused2 != 0) || (format >= texture::NUM_FORMATS))
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ if (levels == 0) levels = max_levels;
+ return gapi_->CreateTextureCube(id, side, levels,
+ static_cast<texture::Format>(format),
+ flags);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+}
+
+// Decodes the SET_TEXTURE_DATA command.
+BufferSyncInterface::ParseError GAPIDecoder::DecodeSetTextureData(
+ unsigned int arg_count,
+ CommandBufferEntry *args) {
+ if (arg_count == 10) {
+ namespace cmd = set_texture_data_cmd;
+ unsigned int id = args[0].value_uint32;
+ unsigned int x_y = args[1].value_uint32;
+ unsigned int width_height = args[2].value_uint32;
+ unsigned int z_depth = args[3].value_uint32;
+ unsigned int level_face = args[4].value_uint32;
+ unsigned int row_pitch = args[5].value_uint32;
+ unsigned int slice_pitch = args[6].value_uint32;
+ unsigned int size = args[7].value_uint32;
+ unsigned int shm_id = args[8].value_uint32;
+ unsigned int offset = args[9].value_uint32;
+ unsigned int x = cmd::X::Get(x_y);
+ unsigned int y = cmd::Y::Get(x_y);
+ unsigned int width = cmd::Width::Get(width_height);
+ unsigned int height = cmd::Height::Get(width_height);
+ unsigned int z = cmd::Z::Get(z_depth);
+ unsigned int depth = cmd::Depth::Get(z_depth);
+ unsigned int level = cmd::Level::Get(level_face);
+ unsigned int face = cmd::Face::Get(level_face);
+ unsigned int unused = cmd::Unused::Get(level_face);
+ const void *data = GetAddressAndCheckSize(shm_id, offset, size);
+ if (face >= 6 || unused != 0 || !data)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return gapi_->SetTextureData(id, x, y, z, width, height, depth, level,
+ static_cast<texture::Face>(face), row_pitch,
+ slice_pitch, size, data);
+
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+}
+
+// Decodes the SET_TEXTURE_DATA_IMMEDIATE command.
+BufferSyncInterface::ParseError GAPIDecoder::DecodeSetTextureDataImmediate(
+ unsigned int arg_count,
+ CommandBufferEntry *args) {
+ if (arg_count > 8) {
+ namespace cmd = set_texture_data_immediate_cmd;
+ unsigned int id = args[0].value_uint32;
+ unsigned int x_y = args[1].value_uint32;
+ unsigned int width_height = args[2].value_uint32;
+ unsigned int z_depth = args[3].value_uint32;
+ unsigned int level_face = args[4].value_uint32;
+ unsigned int row_pitch = args[5].value_uint32;
+ unsigned int slice_pitch = args[6].value_uint32;
+ unsigned int size = args[7].value_uint32;
+ unsigned int x = cmd::X::Get(x_y);
+ unsigned int y = cmd::Y::Get(x_y);
+ unsigned int width = cmd::Width::Get(width_height);
+ unsigned int height = cmd::Height::Get(width_height);
+ unsigned int z = cmd::Z::Get(z_depth);
+ unsigned int depth = cmd::Depth::Get(z_depth);
+ unsigned int level = cmd::Level::Get(level_face);
+ unsigned int face = cmd::Face::Get(level_face);
+ unsigned int unused = cmd::Unused::Get(level_face);
+ if (face >= 6 || unused != 0 ||
+ size > (arg_count - 5) * sizeof(args[0]))
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return gapi_->SetTextureData(id, x, y, z, width, height, depth, level,
+ static_cast<texture::Face>(face), row_pitch,
+ slice_pitch, size, args + 8);
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+}
+
+// Decodes the GET_TEXTURE_DATA command.
+BufferSyncInterface::ParseError GAPIDecoder::DecodeGetTextureData(
+ unsigned int arg_count,
+ CommandBufferEntry *args) {
+ if (arg_count == 10) {
+ namespace cmd = get_texture_data_cmd;
+ unsigned int id = args[0].value_uint32;
+ unsigned int x_y = args[1].value_uint32;
+ unsigned int width_height = args[2].value_uint32;
+ unsigned int z_depth = args[3].value_uint32;
+ unsigned int level_face = args[4].value_uint32;
+ unsigned int row_pitch = args[5].value_uint32;
+ unsigned int slice_pitch = args[6].value_uint32;
+ unsigned int size = args[7].value_uint32;
+ unsigned int shm_id = args[8].value_uint32;
+ unsigned int offset = args[9].value_uint32;
+ unsigned int x = cmd::X::Get(x_y);
+ unsigned int y = cmd::Y::Get(x_y);
+ unsigned int width = cmd::Width::Get(width_height);
+ unsigned int height = cmd::Height::Get(width_height);
+ unsigned int z = cmd::Z::Get(z_depth);
+ unsigned int depth = cmd::Depth::Get(z_depth);
+ unsigned int level = cmd::Level::Get(level_face);
+ unsigned int face = cmd::Face::Get(level_face);
+ unsigned int unused = cmd::Unused::Get(level_face);
+ void *data = GetAddressAndCheckSize(shm_id, offset, size);
+ if (face >= 6 || unused != 0 || !data)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return gapi_->GetTextureData(id, x, y, z, width, height, depth, level,
+ static_cast<texture::Face>(face), row_pitch,
+ slice_pitch, size, data);
+
+ } else {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+}
+
+// Decodes the SET_SAMPLER_STATES command.
+BufferSyncInterface::ParseError GAPIDecoder::DecodeSetSamplerStates(
+ unsigned int arg_count,
+ CommandBufferEntry *args) {
+ namespace cmd = set_sampler_states;
+ if (arg_count != 2)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ ResourceID id = args[0].value_uint32;
+ Uint32 arg = args[1].value_uint32;
+ if (cmd::Unused::Get(arg) != 0)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ unsigned int address_u_value = cmd::AddressingU::Get(arg);
+ unsigned int address_v_value = cmd::AddressingV::Get(arg);
+ unsigned int address_w_value = cmd::AddressingW::Get(arg);
+ unsigned int mag_filter_value = cmd::MagFilter::Get(arg);
+ unsigned int min_filter_value = cmd::MinFilter::Get(arg);
+ unsigned int mip_filter_value = cmd::MipFilter::Get(arg);
+ unsigned int max_anisotropy = cmd::MaxAnisotropy::Get(arg);
+ if (address_u_value >= sampler::NUM_ADDRESSING_MODE ||
+ address_v_value >= sampler::NUM_ADDRESSING_MODE ||
+ address_w_value >= sampler::NUM_ADDRESSING_MODE ||
+ mag_filter_value >= sampler::NUM_FILTERING_MODE ||
+ min_filter_value >= sampler::NUM_FILTERING_MODE ||
+ mip_filter_value >= sampler::NUM_FILTERING_MODE ||
+ mag_filter_value == sampler::NONE ||
+ min_filter_value == sampler::NONE ||
+ max_anisotropy == 0) {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ gapi_->SetSamplerStates(
+ id,
+ static_cast<sampler::AddressingMode>(address_u_value),
+ static_cast<sampler::AddressingMode>(address_v_value),
+ static_cast<sampler::AddressingMode>(address_w_value),
+ static_cast<sampler::FilteringMode>(mag_filter_value),
+ static_cast<sampler::FilteringMode>(min_filter_value),
+ static_cast<sampler::FilteringMode>(mip_filter_value),
+ max_anisotropy);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+// Decodes the SET_STENCIL_TEST command.
+BufferSyncInterface::ParseError GAPIDecoder::DecodeSetStencilTest(
+ unsigned int arg_count,
+ CommandBufferEntry *args) {
+ namespace cmd = set_stencil_test;
+ if (arg_count != 2)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ Uint32 arg0 = args[0].value_uint32;
+ Uint32 arg1 = args[1].value_uint32;
+ if (cmd::Unused0::Get(arg0) != 0 ||
+ cmd::Unused1::Get(arg1) != 0 ||
+ cmd::Unused2::Get(arg1) != 0)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ unsigned int write_mask = cmd::WriteMask::Get(arg0);
+ unsigned int compare_mask = cmd::CompareMask::Get(arg0);
+ unsigned int ref = cmd::ReferenceValue::Get(arg0);
+ bool enable = cmd::Enable::Get(arg0) != 0;
+ bool separate_ccw = cmd::SeparateCCW::Get(arg0) != 0;
+ gapi_->SetStencilTest(enable, separate_ccw, write_mask, compare_mask, ref,
+ arg1);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+// Decodes the SET_BLENDING command.
+BufferSyncInterface::ParseError GAPIDecoder::DecodeSetBlending(
+ unsigned int arg_count,
+ CommandBufferEntry *args) {
+ namespace cmd = set_blending;
+ if (arg_count != 1)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ Uint32 arg = args[0].value_uint32;
+ bool enable = cmd::Enable::Get(arg) != 0;
+ bool separate_alpha = cmd::SeparateAlpha::Get(arg) != 0;
+ unsigned int color_eq = cmd::ColorEq::Get(arg);
+ unsigned int color_src = cmd::ColorSrcFunc::Get(arg);
+ unsigned int color_dst = cmd::ColorDstFunc::Get(arg);
+ unsigned int alpha_eq = cmd::AlphaEq::Get(arg);
+ unsigned int alpha_src = cmd::AlphaSrcFunc::Get(arg);
+ unsigned int alpha_dst = cmd::AlphaDstFunc::Get(arg);
+ if (cmd::Unused0::Get(arg) != 0 ||
+ cmd::Unused1::Get(arg) != 0 ||
+ color_eq >= GAPIInterface::NUM_BLEND_EQ ||
+ color_src >= GAPIInterface::NUM_BLEND_FUNC ||
+ color_dst >= GAPIInterface::NUM_BLEND_FUNC ||
+ alpha_eq >= GAPIInterface::NUM_BLEND_EQ ||
+ alpha_src >= GAPIInterface::NUM_BLEND_FUNC ||
+ alpha_dst >= GAPIInterface::NUM_BLEND_FUNC)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ gapi_->SetBlending(enable,
+ separate_alpha,
+ static_cast<GAPIInterface::BlendEq>(color_eq),
+ static_cast<GAPIInterface::BlendFunc>(color_src),
+ static_cast<GAPIInterface::BlendFunc>(color_dst),
+ static_cast<GAPIInterface::BlendEq>(alpha_eq),
+ static_cast<GAPIInterface::BlendFunc>(alpha_src),
+ static_cast<GAPIInterface::BlendFunc>(alpha_dst));
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+void *GAPIDecoder::GetAddressAndCheckSize(unsigned int shm_id,
+ unsigned int offset,
+ unsigned int size) {
+ void * shm_addr = engine_->GetSharedMemoryAddress(shm_id);
+ if (!shm_addr) return NULL;
+ size_t shm_size = engine_->GetSharedMemorySize(shm_id);
+ if (offset + size > shm_size) return NULL;
+ return static_cast<char *>(shm_addr) + offset;
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/cross/gapi_decoder.h b/o3d/command_buffer/service/cross/gapi_decoder.h
new file mode 100644
index 0000000..615bef6
--- /dev/null
+++ b/o3d/command_buffer/service/cross/gapi_decoder.h
@@ -0,0 +1,131 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the GAPI decoder class.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_CROSS_GAPI_DECODER_H__
+#define O3D_COMMAND_BUFFER_SERVICE_CROSS_GAPI_DECODER_H__
+
+#include "core/cross/types.h"
+#include "command_buffer/service/cross/cmd_parser.h"
+
+namespace o3d {
+namespace command_buffer {
+
+class GAPIInterface;
+class CommandBufferEngine;
+
+// This class implements the AsyncAPIInterface interface, decoding GAPI
+// commands and sending them to a GAPI interface.
+class GAPIDecoder : public AsyncAPIInterface {
+ public:
+ typedef BufferSyncInterface::ParseError ParseError;
+
+ explicit GAPIDecoder(GAPIInterface *gapi) : gapi_(gapi), engine_(NULL) {}
+ virtual ~GAPIDecoder() {}
+ // Executes a command.
+ // Parameters:
+ // command: the command index.
+ // arg_count: the number of CommandBufferEntry arguments.
+ // args: the arguments.
+ // Returns:
+ // BufferSyncInterface::NO_ERROR if no error was found, one of
+ // BufferSyncInterface::ParseError otherwise.
+ virtual ParseError DoCommand(unsigned int command,
+ unsigned int arg_count,
+ CommandBufferEntry *args);
+
+ // Sets the engine, to get shared memory buffers from, and to set the token
+ // to.
+ void set_engine(CommandBufferEngine *engine) { engine_ = engine; }
+ private:
+ // Decodes the SET_VERTEX_INPUT command.
+ ParseError DecodeSetVertexInput(unsigned int arg_count,
+ CommandBufferEntry *args);
+
+ // Decodes the CREATE_TEXTURE_2D command.
+ ParseError DecodeCreateTexture2D(unsigned int arg_count,
+ CommandBufferEntry *args);
+
+ // Decodes the CREATE_TEXTURE_3D command.
+ ParseError DecodeCreateTexture3D(unsigned int arg_count,
+ CommandBufferEntry *args);
+
+ // Decodes the CREATE_TEXTURE_CUBE command.
+ ParseError DecodeCreateTextureCube(unsigned int arg_count,
+ CommandBufferEntry *args);
+
+ // Decodes the SET_TEXTURE_DATA command.
+ ParseError DecodeSetTextureData(unsigned int arg_count,
+ CommandBufferEntry *args);
+
+ // Decodes the SET_TEXTURE_DATA_IMMEDIATE command.
+ ParseError DecodeSetTextureDataImmediate(unsigned int arg_count,
+ CommandBufferEntry *args);
+
+ // Decodes the GET_TEXTURE_DATA command.
+ ParseError DecodeGetTextureData(unsigned int arg_count,
+ CommandBufferEntry *args);
+
+ // Decodes the SET_SAMPLER_STATES command.
+ ParseError DecodeSetSamplerStates(unsigned int arg_count,
+ CommandBufferEntry *args);
+
+ // Decodes the SET_STENCIL_TEST command.
+ ParseError DecodeSetStencilTest(unsigned int arg_count,
+ CommandBufferEntry *args);
+
+ // Decodes the SET_BLENDING command.
+ ParseError DecodeSetBlending(unsigned int arg_count,
+ CommandBufferEntry *args);
+
+ // Gets the address of shared memory data, given a shared memory ID and an
+ // offset. Also checks that the size is consistent with the shared memory
+ // size.
+ // Parameters:
+ // shm_id: the id of the shared memory buffer.
+ // offset: the offset of the data in the shared memory buffer.
+ // size: the size of the data.
+ // Returns:
+ // NULL if shm_id isn't a valid shared memory buffer ID or if the size
+ // check fails. Return a pointer to the data otherwise.
+ void *GetAddressAndCheckSize(unsigned int shm_id,
+ unsigned int offset,
+ unsigned int size);
+ GAPIInterface *gapi_;
+ CommandBufferEngine *engine_;
+};
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_CROSS_GAPI_DECODER_H__
diff --git a/o3d/command_buffer/service/cross/gl/effect_gl.cc b/o3d/command_buffer/service/cross/gl/effect_gl.cc
new file mode 100644
index 0000000..9343d96
--- /dev/null
+++ b/o3d/command_buffer/service/cross/gl/effect_gl.cc
@@ -0,0 +1,633 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the implementation of the EffectParamGL and EffectGL
+// classes, as well as the effect-related GAPI functions on GL.
+
+#include <map>
+
+#include "base/cross/std_functional.h"
+#include "command_buffer/service/cross/gl/gapi_gl.h"
+#include "command_buffer/service/cross/effect_utils.h"
+
+namespace o3d {
+namespace command_buffer {
+
+EffectParamGL::EffectParamGL(effect_param::DataType data_type,
+ EffectGL *effect,
+ unsigned int param_index)
+ : EffectParam(data_type),
+ effect_(effect),
+ low_level_param_index_(param_index) {
+ DCHECK(effect_);
+ effect_->LinkParam(this);
+}
+
+EffectParamGL::~EffectParamGL() {
+ if (effect_)
+ effect_->UnlinkParam(this);
+}
+
+static effect_param::DataType CgTypeToCBType(CGtype cg_type) {
+ switch (cg_type) {
+ case CG_FLOAT:
+ case CG_FLOAT1:
+ return effect_param::FLOAT1;
+ case CG_FLOAT2:
+ return effect_param::FLOAT2;
+ case CG_FLOAT3:
+ return effect_param::FLOAT3;
+ case CG_FLOAT4:
+ return effect_param::FLOAT4;
+ case CG_INT:
+ case CG_INT1:
+ return effect_param::INT;
+ case CG_BOOL:
+ case CG_BOOL1:
+ return effect_param::BOOL;
+ case CG_FLOAT4x4:
+ return effect_param::MATRIX4;
+ case CG_SAMPLER:
+ case CG_SAMPLER1D:
+ case CG_SAMPLER2D:
+ case CG_SAMPLER3D:
+ case CG_SAMPLERCUBE:
+ return effect_param::SAMPLER;
+ default : {
+ DLOG(INFO) << "Cannot convert CGtype "
+ << cgGetTypeString(cg_type)
+ << " to a Param type.";
+ return effect_param::UNKNOWN;
+ }
+ }
+}
+
+EffectParamGL *EffectParamGL::Create(EffectGL *effect,
+ unsigned int index) {
+ DCHECK(effect);
+ const EffectGL::LowLevelParam &low_level_param =
+ effect->low_level_params_[index];
+ CGtype cg_type =
+ cgGetParameterType(EffectGL::GetEitherCgParameter(low_level_param));
+ effect_param::DataType type = CgTypeToCBType(cg_type);
+ if (type == effect_param::UNKNOWN) return NULL;
+ return new EffectParamGL(type, effect, index);
+}
+
+// Fills the Desc structure, appending name and semantic if any, and if enough
+// room is available in the buffer.
+bool EffectParamGL::GetDesc(unsigned int size, void *data) {
+ using effect_param::Desc;
+ if (size < sizeof(Desc)) // NOLINT
+ return false;
+ if (!effect_)
+ return false;
+ const EffectGL::LowLevelParam &low_level_param =
+ effect_->low_level_params_[low_level_param_index_];
+ CGparameter cg_param = EffectGL::GetEitherCgParameter(low_level_param);
+ const char *name = low_level_param.name;
+ const char* semantic = cgGetParameterSemantic(cg_param);
+ unsigned int name_size =
+ name ? static_cast<unsigned int>(strlen(name)) + 1 : 0;
+ unsigned int semantic_size = semantic ?
+ static_cast<unsigned int>(strlen(semantic)) + 1 : 0;
+ unsigned int total_size = sizeof(Desc) + name_size + semantic_size; // NOLINT
+
+ Desc *desc = static_cast<Desc *>(data);
+ memset(desc, 0, sizeof(*desc));
+ desc->size = total_size;
+ desc->data_type = data_type();
+ desc->data_size = GetDataSize(data_type());
+ desc->name_offset = 0;
+ desc->name_size = name_size;
+ desc->semantic_offset = 0;
+ desc->semantic_size = semantic_size;
+ unsigned int current_offset = sizeof(Desc);
+ if (name && current_offset + name_size <= size) {
+ desc->name_offset = current_offset;
+ memcpy(static_cast<char *>(data) + current_offset, name, name_size);
+ current_offset += name_size;
+ }
+ if (semantic && current_offset + semantic_size <= size) {
+ desc->semantic_offset = current_offset;
+ memcpy(static_cast<char *>(data) + current_offset, semantic, semantic_size);
+ current_offset += semantic_size;
+ }
+ return true;
+}
+
+// Sets the data into the Cg effect parameter, using the appropriate Cg call.
+bool EffectParamGL::SetData(GAPIGL *gapi,
+ unsigned int size,
+ const void * data) {
+ if (!effect_)
+ return false;
+
+ EffectGL::LowLevelParam &low_level_param =
+ effect_->low_level_params_[low_level_param_index_];
+ CGparameter vp_param = low_level_param.vp_param;
+ CGparameter fp_param = low_level_param.fp_param;
+ effect_param::DataType type = data_type();
+ if (size < effect_param::GetDataSize(type))
+ return false;
+
+ switch (type) {
+ case effect_param::FLOAT1:
+ if (vp_param)
+ cgSetParameter1f(vp_param, *static_cast<const float *>(data));
+ if (fp_param)
+ cgSetParameter1f(fp_param, *static_cast<const float *>(data));
+ break;
+ case effect_param::FLOAT2:
+ if (vp_param)
+ cgSetParameter2fv(vp_param, static_cast<const float *>(data));
+ if (fp_param)
+ cgSetParameter2fv(fp_param, static_cast<const float *>(data));
+ break;
+ case effect_param::FLOAT3:
+ if (vp_param)
+ cgSetParameter3fv(vp_param, static_cast<const float *>(data));
+ if (fp_param)
+ cgSetParameter3fv(fp_param, static_cast<const float *>(data));
+ break;
+ case effect_param::FLOAT4:
+ if (vp_param)
+ cgSetParameter4fv(vp_param, static_cast<const float *>(data));
+ if (fp_param)
+ cgSetParameter4fv(fp_param, static_cast<const float *>(data));
+ break;
+ case effect_param::MATRIX4:
+ if (vp_param)
+ cgSetMatrixParameterfr(vp_param, static_cast<const float *>(data));
+ if (fp_param)
+ cgSetMatrixParameterfr(fp_param, static_cast<const float *>(data));
+ break;
+ case effect_param::INT:
+ if (vp_param) cgSetParameter1i(vp_param, *static_cast<const int *>(data));
+ if (fp_param) cgSetParameter1i(fp_param, *static_cast<const int *>(data));
+ break;
+ case effect_param::BOOL: {
+ int bool_value = *static_cast<const bool *>(data)?1:0;
+ if (vp_param) cgSetParameter1i(vp_param, bool_value);
+ if (fp_param) cgSetParameter1i(fp_param, bool_value);
+ break;
+ }
+ case effect_param::SAMPLER: {
+ low_level_param.sampler_id = *static_cast<const ResourceID *>(data);
+ if (effect_ == gapi->current_effect()) {
+ gapi->DirtyEffect();
+ }
+ break;
+ }
+ default:
+ DLOG(ERROR) << "Invalid parameter type.";
+ return false;
+ }
+ return true;
+}
+EffectGL::EffectGL(CGprogram vertex_program,
+ CGprogram fragment_program)
+ : vertex_program_(vertex_program),
+ fragment_program_(fragment_program),
+ update_samplers_(true) {
+}
+
+EffectGL::~EffectGL() {
+ for (ParamResourceList::iterator it = resource_params_.begin();
+ it != resource_params_.end(); ++it) {
+ (*it)->ResetEffect();
+ }
+}
+
+void EffectGL::LinkParam(EffectParamGL *param) {
+ resource_params_.push_back(param);
+}
+
+void EffectGL::UnlinkParam(EffectParamGL *param) {
+ std::remove(resource_params_.begin(), resource_params_.end(), param);
+}
+
+// Rewrites vertex program assembly code to match GL semantics for clipping.
+// This parses the source, breaking it down into pieces:
+// - declaration ("!!ARBvp1.0")
+// - comments (that contain the parameter information)
+// - instructions
+// - "END" token.
+// Then it rewrites the instructions so that 'result.position' doesn't get
+// written directly, instead it is written to a temporary variable. Then a
+// transformation is done on that variable before outputing to
+// 'result.position':
+// - offset x an y by half a pixel (times w).
+// - remap z from [0..w] to [-w..w].
+//
+// Note that for the 1/2 pixel offset, we need a parameter that depends on the
+// current viewport. This is done through 'program.env[0]' which is shared
+// across all programs (so we only have to update it once when we change the
+// viewport), because Cg won't use them currently (it uses 'program.local'
+// instead).
+static bool RewriteVertexProgramSource(String *source) {
+ String::size_type pos = source->find('\n');
+ if (pos == String::npos) {
+ DLOG(ERROR) << "could not find program declaration";
+ return false;
+ }
+ String decl(*source, 0, pos + 1);
+ String::size_type start_comments = pos + 1;
+ // skip the comments that contain the parameters etc.
+ for (; pos < source->size(); pos = source->find('\n', pos)) {
+ ++pos;
+ if (pos >= source->size())
+ break;
+ if ((*source)[pos] != '#')
+ break;
+ }
+ if (pos >= source->size()) {
+ // we only found comments.
+ return false;
+ }
+ String comments(*source, start_comments, pos - start_comments);
+
+ String::size_type end_token = source->find("\nEND", pos + 1);
+ if (end_token == String::npos) {
+ DLOG(ERROR) << "Compiled shader doesn't have an END token";
+ return false;
+ }
+ String instructions(*source, pos, end_token + 1 - pos);
+
+ // Replace accesses to 'result.position' by accesses to our temp variable
+ // '$O3D_HPOS'.
+ // '$' is a valid symbol for identifiers, but Cg doesn't seem to be using
+ // it, so we can use it to ensure we don't have name conflicts.
+ static const char kOutPositionRegister[] = "result.position";
+ for (String::size_type i = instructions.find(kOutPositionRegister);
+ i < String::npos; i = instructions.find(kOutPositionRegister, i)) {
+ instructions.replace(i, strlen(kOutPositionRegister), "$O3D_HPOS");
+ }
+
+ *source = decl +
+ comments +
+ // .x = 1/viewport.width; .y = 1/viewport.height; .z = 2.0;
+ "PARAM $O3D_HELPER = program.env[0];\n"
+ "TEMP $O3D_HPOS;\n" +
+ instructions +
+ // hpos.x <- hpos.x + hpos.w / viewport.width;
+ // hpos.y <- hpos.y - hpos.w / viewport.height;
+ "MAD $O3D_HPOS.xy, $O3D_HELPER.xyyy, $O3D_HPOS.w, $O3D_HPOS.xyyy;\n"
+ // hpos.z <- hpos.z * 2 - hpos.w
+ "MAD $O3D_HPOS.z, $O3D_HPOS.z, $O3D_HELPER.z, -$O3D_HPOS.w;\n"
+ "MOV result.position, $O3D_HPOS;\n"
+ "END\n";
+ return true;
+}
+
+EffectGL *EffectGL::Create(GAPIGL *gapi,
+ const String& effect_code,
+ const String& vertex_program_entry,
+ const String& fragment_program_entry) {
+ CGcontext context = gapi->cg_context();
+ // Compile the original vertex program once, to get the ARBVP1 assembly code.
+ CGprogram original_vp = cgCreateProgram(
+ context, CG_SOURCE, effect_code.c_str(), CG_PROFILE_ARBVP1,
+ vertex_program_entry.c_str(), NULL);
+ const char* listing = cgGetLastListing(context);
+ if (original_vp == NULL) {
+ DLOG(ERROR) << "Effect Compile Error: " << cgGetErrorString(cgGetError())
+ << " : " << listing;
+ return NULL;
+ }
+
+ if (listing && listing[0] != 0) {
+ DLOG(WARNING) << "Effect Compile Warnings: " << listing;
+ }
+
+ String vp_assembly = cgGetProgramString(original_vp, CG_COMPILED_PROGRAM);
+ cgDestroyProgram(original_vp);
+ if (!RewriteVertexProgramSource(&vp_assembly)) {
+ return NULL;
+ }
+ CGprogram vertex_program = cgCreateProgram(
+ context, CG_OBJECT, vp_assembly.c_str(), CG_PROFILE_ARBVP1,
+ vertex_program_entry.c_str(), NULL);
+ listing = cgGetLastListing(context);
+ if (vertex_program == NULL) {
+ DLOG(ERROR) << "Effect post-rewrite Compile Error: "
+ << cgGetErrorString(cgGetError()) << " : " << listing;
+ return false;
+ }
+
+ if (listing && listing[0] != 0) {
+ DLOG(WARNING) << "Effect post-rewrite compile warnings: " << listing;
+ }
+
+ CHECK_GL_ERROR();
+
+ // If the program rewrite introduced some syntax or semantic errors, we won't
+ // know it until we load the program (through a GL error).
+ // So flush all GL errors first...
+ do {} while (glGetError() != GL_NO_ERROR);
+
+ // ... Then load the program ...
+ cgGLLoadProgram(vertex_program);
+
+ // ... And check for GL errors.
+ if (glGetError() != GL_NO_ERROR) {
+ DLOG(ERROR) << "Effect post-rewrite GL Error: "
+ << glGetString(GL_PROGRAM_ERROR_STRING_ARB)
+ << "\nSource: \n"
+ << vp_assembly;
+ return NULL;
+ }
+
+ CGprogram fragment_program = cgCreateProgram(
+ context, CG_SOURCE, effect_code.c_str(), CG_PROFILE_ARBFP1,
+ fragment_program_entry.c_str(), NULL);
+ listing = cgGetLastListing(context);
+ if (fragment_program == NULL) {
+ DLOG(ERROR) << "Effect Compile Error: "
+ << cgGetErrorString(cgGetError()) << " : "
+ << listing;
+ return NULL;
+ }
+
+ if (listing && listing[0] != 0) {
+ DLOG(WARNING) << "Effect Compile Warnings: " << listing;
+ }
+
+ cgGLLoadProgram(fragment_program);
+
+ // Also check for GL errors, in case Cg managed to compile, but generated a
+ // bad program.
+ if (glGetError() != GL_NO_ERROR) {
+ DLOG(ERROR) << "Effect GL Error: "
+ << glGetString(GL_PROGRAM_ERROR_STRING_ARB);
+ return false;
+ }
+ EffectGL *effect = new EffectGL(vertex_program, fragment_program);
+ effect->Initialize();
+ return effect;
+}
+
+int EffectGL::GetLowLevelParamIndexByName(const char *name) {
+ DCHECK(name);
+ for (unsigned int index = 0; index < low_level_params_.size(); ++index) {
+ if (!strcmp(name, low_level_params_[index].name)) {
+ return index;
+ }
+ }
+ return -1;
+}
+
+void EffectGL::AddLowLevelParams(CGparameter cg_param,
+ bool vp) {
+ // Loop over all *leaf* parameters, visiting only CGparameters that have
+ // had storage allocated to them.
+ for (; cg_param != NULL; cg_param = cgGetNextLeafParameter(cg_param)) {
+ CGenum variability = cgGetParameterVariability(cg_param);
+ if (variability != CG_UNIFORM)
+ continue;
+ CGenum direction = cgGetParameterDirection(cg_param);
+ if (direction != CG_IN)
+ continue;
+ const char *name = cgGetParameterName(cg_param);
+ if (!name)
+ continue;
+
+ int index = GetLowLevelParamIndexByName(name);
+ if (index < 0) {
+ LowLevelParam param = {name, NULL, NULL, kInvalidResource};
+ index = low_level_params_.size();
+ low_level_params_.push_back(param);
+ CGtype cg_type = cgGetParameterType(cg_param);
+ if (cg_type == CG_SAMPLER ||
+ cg_type == CG_SAMPLER1D ||
+ cg_type == CG_SAMPLER2D ||
+ cg_type == CG_SAMPLER3D ||
+ cg_type == CG_SAMPLERCUBE) {
+ sampler_params_.push_back(index);
+ }
+ }
+ if (vp) {
+ low_level_params_[index].vp_param = cg_param;
+ } else {
+ low_level_params_[index].fp_param = cg_param;
+ }
+ }
+}
+
+void EffectGL::Initialize() {
+ AddLowLevelParams(cgGetFirstLeafParameter(vertex_program_, CG_PROGRAM), true);
+ AddLowLevelParams(cgGetFirstLeafParameter(vertex_program_, CG_GLOBAL), true);
+ AddLowLevelParams(cgGetFirstLeafParameter(fragment_program_, CG_PROGRAM),
+ false);
+ AddLowLevelParams(cgGetFirstLeafParameter(fragment_program_, CG_GLOBAL),
+ false);
+}
+
+// Begins rendering with the effect, setting all the appropriate states.
+bool EffectGL::Begin(GAPIGL *gapi) {
+ cgGLBindProgram(vertex_program_);
+ cgGLBindProgram(fragment_program_);
+ // sampler->ApplyStates will mess with the texture binding on unit 0, so we
+ // do 2 passes.
+ // First to set the sampler states on the texture
+ for (unsigned int i = 0; i < sampler_params_.size(); ++i) {
+ unsigned int param_index = sampler_params_[i];
+ ResourceID id = low_level_params_[param_index].sampler_id;
+ if (id != kInvalidResource) {
+ SamplerGL *sampler = gapi->GetSampler(id);
+ if (!sampler->ApplyStates(gapi)) {
+ return false;
+ }
+ }
+ }
+ // Second to enable/disable the sampler params.
+ for (unsigned int i = 0; i < sampler_params_.size(); ++i) {
+ unsigned int param_index = sampler_params_[i];
+ const LowLevelParam &ll_param = low_level_params_[param_index];
+ ResourceID id = ll_param.sampler_id;
+ if (id != kInvalidResource) {
+ SamplerGL *sampler = gapi->GetSampler(id);
+ GLuint gl_texture = sampler->gl_texture();
+ cgGLSetTextureParameter(ll_param.fp_param, gl_texture);
+ cgGLEnableTextureParameter(ll_param.fp_param);
+ } else {
+ cgGLSetTextureParameter(ll_param.fp_param, 0);
+ cgGLDisableTextureParameter(ll_param.fp_param);
+ }
+ }
+ return true;
+}
+
+// Terminates rendering with the effect, resetting all the appropriate states.
+void EffectGL::End(GAPIGL *gapi) {
+}
+
+// Gets the parameter count from the list.
+unsigned int EffectGL::GetParamCount() const {
+ return low_level_params_.size();
+}
+
+// Gets a handle to the selected parameter, and wraps it into an
+// EffectParamGL if successful.
+EffectParamGL *EffectGL::CreateParam(unsigned int index) {
+ if (index < low_level_params_.size()) return NULL;
+ return EffectParamGL::Create(this, index);
+}
+
+// Gets a handle to the selected parameter, and wraps it into an
+// EffectParamGL if successful.
+EffectParamGL *EffectGL::CreateParamByName(const char *name) {
+ int index = GetLowLevelParamIndexByName(name);
+ if (index < 0) return NULL;
+ EffectParamGL::Create(this, index);
+}
+
+BufferSyncInterface::ParseError GAPIGL::CreateEffect(ResourceID id,
+ unsigned int size,
+ const void *data) {
+ if (id == current_effect_id_) DirtyEffect();
+ // Even though Assign would Destroy the effect at id, we do it explicitly in
+ // case the creation fails.
+ effects_.Destroy(id);
+ // Data is vp_main \0 fp_main \0 effect_text.
+ String vertex_program_entry;
+ String fragment_program_entry;
+ String effect_code;
+ if (!ParseEffectData(size, data,
+ &vertex_program_entry,
+ &fragment_program_entry,
+ &effect_code)) {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ EffectGL * effect = EffectGL::Create(this, effect_code,
+ vertex_program_entry,
+ fragment_program_entry);
+ if (!effect) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ effects_.Assign(id, effect);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+BufferSyncInterface::ParseError GAPIGL::DestroyEffect(ResourceID id) {
+ if (id == current_effect_id_) DirtyEffect();
+ return effects_.Destroy(id) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+BufferSyncInterface::ParseError GAPIGL::SetEffect(ResourceID id) {
+ DirtyEffect();
+ current_effect_id_ = id;
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+BufferSyncInterface::ParseError GAPIGL::GetParamCount(ResourceID id,
+ unsigned int size,
+ void *data) {
+ EffectGL *effect = effects_.Get(id);
+ if (!effect || size < sizeof(Uint32)) // NOLINT
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ *static_cast<Uint32 *>(data) = effect->GetParamCount();
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+BufferSyncInterface::ParseError GAPIGL::CreateParam(ResourceID param_id,
+ ResourceID effect_id,
+ unsigned int index) {
+ EffectGL *effect = effects_.Get(effect_id);
+ if (!effect) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ EffectParamGL *param = effect->CreateParam(index);
+ if (!param) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ effect_params_.Assign(param_id, param);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+BufferSyncInterface::ParseError GAPIGL::CreateParamByName(ResourceID param_id,
+ ResourceID effect_id,
+ unsigned int size,
+ const void *name) {
+ EffectGL *effect = effects_.Get(effect_id);
+ if (!effect) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ std::string string_name(static_cast<const char *>(name), size);
+ EffectParamGL *param = effect->CreateParamByName(string_name.c_str());
+ if (!param) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ effect_params_.Assign(param_id, param);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+BufferSyncInterface::ParseError GAPIGL::DestroyParam(ResourceID id) {
+ return effect_params_.Destroy(id) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+BufferSyncInterface::ParseError GAPIGL::SetParamData(ResourceID id,
+ unsigned int size,
+ const void *data) {
+ EffectParamGL *param = effect_params_.Get(id);
+ if (!param) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return param->SetData(this, size, data) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+BufferSyncInterface::ParseError GAPIGL::GetParamDesc(ResourceID id,
+ unsigned int size,
+ void *data) {
+ EffectParamGL *param = effect_params_.Get(id);
+ if (!param) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return param->GetDesc(size, data) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+// If the current effect is valid, call End on it, and tag for revalidation.
+void GAPIGL::DirtyEffect() {
+ if (validate_effect_) return;
+ DCHECK(current_effect_);
+ current_effect_->End(this);
+ current_effect_ = NULL;
+ validate_effect_ = true;
+}
+
+// Gets the current effect, and calls Begin on it (if successful).
+// Should only be called if the current effect is not valid.
+bool GAPIGL::ValidateEffect() {
+ DCHECK(validate_effect_);
+ DCHECK(!current_effect_);
+ current_effect_ = effects_.Get(current_effect_id_);
+ if (!current_effect_) return false;
+ validate_effect_ = false;
+ return current_effect_->Begin(this);
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/cross/gl/effect_gl.h b/o3d/command_buffer/service/cross/gl/effect_gl.h
new file mode 100644
index 0000000..fc29bf3
--- /dev/null
+++ b/o3d/command_buffer/service/cross/gl/effect_gl.h
@@ -0,0 +1,151 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the declaration of the EffectParamGL and EffectGL classes.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_CROSS_GL_EFFECT_GL_H_
+#define O3D_COMMAND_BUFFER_SERVICE_CROSS_GL_EFFECT_GL_H_
+
+#include <vector>
+#include "command_buffer/common/cross/gapi_interface.h"
+#include "command_buffer/service/cross/resource.h"
+#include "command_buffer/service/cross/gl/gl_utils.h"
+
+namespace o3d {
+namespace command_buffer {
+
+class GAPIGL;
+class EffectGL;
+
+// GL version of EffectParam.
+class EffectParamGL: public EffectParam {
+ public:
+ EffectParamGL(effect_param::DataType data_type,
+ EffectGL *effect,
+ unsigned int param_index);
+ virtual ~EffectParamGL();
+
+ // Sets the data into the GL effect parameter.
+ bool SetData(GAPIGL *gapi, unsigned int size, const void * data);
+
+ // Gets the description of the parameter.
+ bool GetDesc(unsigned int size, void *data);
+
+ // Resets the effect back-pointer. This is called when the effect gets
+ // destroyed, to invalidate the parameter.
+ void ResetEffect() { effect_ = NULL; }
+
+ // Creates an EffectParamGL from the EffectGL, by index.
+ static EffectParamGL *Create(EffectGL *effect,
+ unsigned int index);
+ private:
+ EffectGL *effect_;
+ unsigned int low_level_param_index_;
+ DISALLOW_COPY_AND_ASSIGN(EffectParamGL);
+};
+
+// GL version of Effect.
+class EffectGL : public Effect {
+ public:
+ EffectGL(CGprogram vertex_program,
+ CGprogram fragment_program);
+ virtual ~EffectGL();
+
+ // Compiles and creates an effect from source code.
+ static EffectGL *Create(GAPIGL *gapi,
+ const String &effect_code,
+ const String &vertex_program_entry,
+ const String &fragment_program_entry);
+
+ // Applies the effect states (vertex shader, pixel shader) to GL.
+ bool Begin(GAPIGL *gapi);
+
+ // Resets the effect states (vertex shader, pixel shader) to GL.
+ void End(GAPIGL *gapi);
+
+ // Gets the number of parameters in the effect.
+ unsigned int GetParamCount() const;
+
+ // Creates an effect parameter with the specified index.
+ EffectParamGL *CreateParam(unsigned int index);
+
+ // Creates an effect parameter of the specified name.
+ EffectParamGL *CreateParamByName(const char *name);
+
+ private:
+ struct LowLevelParam {
+ const char *name;
+ CGparameter vp_param;
+ CGparameter fp_param;
+ ResourceID sampler_id;
+ };
+ typedef std::vector<LowLevelParam> LowLevelParamList;
+ typedef std::vector<EffectParamGL *> ParamResourceList;
+
+ static CGparameter GetEitherCgParameter(
+ const LowLevelParam &low_level_param) {
+ return low_level_param.vp_param ?
+ low_level_param.vp_param : low_level_param.fp_param;
+ }
+
+ int GetLowLevelParamIndexByName(const char *name);
+ void AddLowLevelParams(CGparameter cg_param, bool vp);
+
+ // Creates the low level structures.
+ void Initialize();
+
+ // Links a param into this effect.
+ void LinkParam(EffectParamGL *param);
+
+ // Unlinks a param into this effect.
+ void UnlinkParam(EffectParamGL *param);
+
+ CGprogram vertex_program_;
+ CGprogram fragment_program_;
+ // List of all the Param resources created.
+ ParamResourceList resource_params_;
+ // List of all the Cg parameters present in either the vertex program or the
+ // fragment program.
+ LowLevelParamList low_level_params_;
+ // List of the indices of the low level params that are samplers.
+ std::vector<unsigned int> sampler_params_;
+ bool update_samplers_;
+
+ friend class EffectParamGL;
+ DISALLOW_COPY_AND_ASSIGN(EffectGL);
+};
+
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_CROSS_GL_EFFECT_GL_H_
diff --git a/o3d/command_buffer/service/cross/gl/gapi_gl.cc b/o3d/command_buffer/service/cross/gl/gapi_gl.cc
new file mode 100644
index 0000000..a09375a
--- /dev/null
+++ b/o3d/command_buffer/service/cross/gl/gapi_gl.cc
@@ -0,0 +1,141 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file implements the GAPIGL class.
+
+#include <build/build_config.h>
+#include "command_buffer/service/cross/gl/gl_utils.h"
+#include "command_buffer/service/cross/gl/gapi_gl.h"
+
+#ifdef OS_LINUX
+#include "command_buffer/service/linux/x_utils.h"
+#endif // OS_LINUX
+
+namespace o3d {
+namespace command_buffer {
+
+GAPIGL::GAPIGL()
+#ifdef OS_LINUX
+ : window_(NULL),
+#endif
+ cg_context_(NULL),
+ current_vertex_struct_(kInvalidResource),
+ validate_streams_(true),
+ max_vertices_(0),
+ current_effect_id_(kInvalidResource),
+ validate_effect_(true),
+ current_effect_(NULL) {
+}
+
+GAPIGL::~GAPIGL() {
+}
+
+bool GAPIGL::Initialize() {
+#ifdef OS_LINUX
+ DCHECK(window_);
+ if (!window_->Initialize())
+ return false;
+ if (!window_->MakeCurrent())
+ return false;
+ InitCommon();
+ CHECK_GL_ERROR();
+ return true;
+#else
+ return false;
+#endif
+}
+
+void GAPIGL::InitCommon() {
+ cg_context_ = cgCreateContext();
+ // Set up all Cg State Assignments for OpenGL.
+ cgGLRegisterStates(cg_context_);
+ cgGLSetDebugMode(CG_FALSE);
+ // Enable the profiles we use.
+ cgGLEnableProfile(CG_PROFILE_ARBVP1);
+ cgGLEnableProfile(CG_PROFILE_ARBFP1);
+ // Initialize global GL settings.
+ // Tell GL that texture buffers can be single-byte aligned.
+ glPixelStorei(GL_PACK_ALIGNMENT, 1);
+ glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
+ glHint(GL_PERSPECTIVE_CORRECTION_HINT, GL_NICEST);
+ glHint(GL_POLYGON_SMOOTH_HINT, GL_NICEST);
+
+ // Get the initial viewport (set to the window size) to set up the helper
+ // constant.
+ GLint viewport[4];
+ glGetIntegerv(GL_VIEWPORT, viewport);
+ SetViewport(viewport[0], viewport[1], viewport[2], viewport[3], 0.f, 1.f);
+ CHECK_GL_ERROR();
+}
+
+void GAPIGL::Destroy() {
+ vertex_buffers_.DestroyAllResources();
+ index_buffers_.DestroyAllResources();
+ vertex_structs_.DestroyAllResources();
+ effects_.DestroyAllResources();
+ effect_params_.DestroyAllResources();
+ // textures_.DestroyAllResources();
+ // samplers_.DestroyAllResources();
+ cgDestroyContext(cg_context_);
+ cg_context_ = NULL;
+#ifdef OS_LINUX
+ DCHECK(window_);
+ window_->Destroy();
+#endif
+}
+
+void GAPIGL::BeginFrame() {
+}
+
+void GAPIGL::EndFrame() {
+#ifdef OS_LINUX
+ DCHECK(window_);
+ window_->SwapBuffers();
+#endif
+ CHECK_GL_ERROR();
+}
+
+void GAPIGL::Clear(unsigned int buffers,
+ const RGBA &color,
+ float depth,
+ unsigned int stencil) {
+ glClearColor(color.red, color.green, color.blue, color.alpha);
+ glClearDepth(depth);
+ glClearStencil(stencil);
+ glClear((buffers & COLOR ? GL_COLOR_BUFFER_BIT : 0) |
+ (buffers & DEPTH ? GL_DEPTH_BUFFER_BIT : 0) |
+ (buffers & STENCIL ? GL_STENCIL_BUFFER_BIT : 0));
+ CHECK_GL_ERROR();
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/cross/gl/gapi_gl.h b/o3d/command_buffer/service/cross/gl/gapi_gl.h
new file mode 100644
index 0000000..8d71800
--- /dev/null
+++ b/o3d/command_buffer/service/cross/gl/gapi_gl.h
@@ -0,0 +1,395 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the GAPIGL class, implementing the GAPI interface for
+// GL.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_CROSS_GL_GAPI_GL_H__
+#define O3D_COMMAND_BUFFER_SERVICE_CROSS_GL_GAPI_GL_H__
+
+#include <build/build_config.h>
+#include "command_buffer/common/cross/gapi_interface.h"
+#include "command_buffer/service/cross/gl/gl_utils.h"
+#include "command_buffer/service/cross/gl/effect_gl.h"
+#include "command_buffer/service/cross/gl/geometry_gl.h"
+#include "command_buffer/service/cross/gl/sampler_gl.h"
+#include "command_buffer/service/cross/gl/texture_gl.h"
+
+namespace o3d {
+namespace command_buffer {
+#if defined(OS_LINUX)
+class XWindowWrapper;
+#endif // defined(OS_LINUX)
+
+// This class implements the GAPI interface for GL.
+class GAPIGL : public GAPIInterface {
+ public:
+ GAPIGL();
+ virtual ~GAPIGL();
+
+#if defined(OS_LINUX)
+ void set_window_wrapper(XWindowWrapper *window) { window_ = window; }
+#elif defined(OS_WIN)
+ void set_hwnd(HWND hwnd) { hwnd_ = hwnd; }
+#endif
+
+ // Initializes the graphics context, bound to a window.
+ // Returns:
+ // true if successful.
+ virtual bool Initialize();
+
+ // Destroys the graphics context.
+ virtual void Destroy();
+
+ // Implements the BeginFrame function for GL.
+ virtual void BeginFrame();
+
+ // Implements the EndFrame function for GL.
+ virtual void EndFrame();
+
+ // Implements the Clear function for GL.
+ virtual void Clear(unsigned int buffers,
+ const RGBA &color,
+ float depth,
+ unsigned int stencil);
+
+ // Implements the SetViewport function for GL.
+ virtual void SetViewport(unsigned int x,
+ unsigned int y,
+ unsigned int width,
+ unsigned int height,
+ float z_min,
+ float z_max);
+
+ // Implements the CreateVertexBuffer function for GL.
+ virtual ParseError CreateVertexBuffer(ResourceID id,
+ unsigned int size,
+ unsigned int flags);
+
+ // Implements the DestroyVertexBuffer function for GL.
+ virtual ParseError DestroyVertexBuffer(ResourceID id);
+
+ // Implements the SetVertexBufferData function for GL.
+ virtual ParseError SetVertexBufferData(ResourceID id,
+ unsigned int offset,
+ unsigned int size,
+ const void *data);
+
+ // Implements the GetVertexBufferData function for GL.
+ virtual ParseError GetVertexBufferData(ResourceID id,
+ unsigned int offset,
+ unsigned int size,
+ void *data);
+
+ // Implements the CreateIndexBuffer function for GL.
+ virtual ParseError CreateIndexBuffer(ResourceID id,
+ unsigned int size,
+ unsigned int flags);
+
+ // Implements the DestroyIndexBuffer function for GL.
+ virtual ParseError DestroyIndexBuffer(ResourceID id);
+
+ // Implements the SetIndexBufferData function for GL.
+ virtual ParseError SetIndexBufferData(ResourceID id,
+ unsigned int offset,
+ unsigned int size,
+ const void *data);
+
+ // Implements the GetIndexBufferData function for GL.
+ virtual ParseError GetIndexBufferData(ResourceID id,
+ unsigned int offset,
+ unsigned int size,
+ void *data);
+
+ // Implements the CreateVertexStruct function for GL.
+ virtual ParseError CreateVertexStruct(ResourceID id,
+ unsigned int input_count);
+
+ // Implements the DestroyVertexStruct function for GL.
+ virtual ParseError DestroyVertexStruct(ResourceID id);
+
+ // Implements the SetVertexInput function for GL.
+ virtual ParseError SetVertexInput(ResourceID vertex_struct_id,
+ unsigned int input_index,
+ ResourceID vertex_buffer_id,
+ unsigned int offset,
+ unsigned int stride,
+ vertex_struct::Type type,
+ vertex_struct::Semantic semantic,
+ unsigned int semantic_index);
+
+ // Implements the SetVertexStruct function for GL.
+ virtual ParseError SetVertexStruct(ResourceID id);
+
+ // Implements the Draw function for GL.
+ virtual ParseError Draw(PrimitiveType primitive_type,
+ unsigned int first,
+ unsigned int count);
+
+ // Implements the DrawIndexed function for GL.
+ virtual ParseError DrawIndexed(PrimitiveType primitive_type,
+ ResourceID index_buffer_id,
+ unsigned int first,
+ unsigned int count,
+ unsigned int min_index,
+ unsigned int max_index);
+
+ // Implements the CreateEffect function for GL.
+ virtual ParseError CreateEffect(ResourceID id,
+ unsigned int size,
+ const void *data);
+
+ // Implements the DestroyEffect function for GL.
+ virtual ParseError DestroyEffect(ResourceID id);
+
+ // Implements the SetEffect function for GL.
+ virtual ParseError SetEffect(ResourceID id);
+
+ // Implements the GetParamCount function for GL.
+ virtual ParseError GetParamCount(ResourceID id,
+ unsigned int size,
+ void *data);
+
+ // Implements the CreateParam function for GL.
+ virtual ParseError CreateParam(ResourceID param_id,
+ ResourceID effect_id,
+ unsigned int index);
+
+ // Implements the CreateParamByName function for GL.
+ virtual ParseError CreateParamByName(ResourceID param_id,
+ ResourceID effect_id,
+ unsigned int size,
+ const void *name);
+
+ // Implements the DestroyParam function for GL.
+ virtual ParseError DestroyParam(ResourceID id);
+
+ // Implements the SetParamData function for GL.
+ virtual ParseError SetParamData(ResourceID id,
+ unsigned int size,
+ const void *data);
+
+ // Implements the GetParamDesc function for GL.
+ virtual ParseError GetParamDesc(ResourceID id,
+ unsigned int size,
+ void *data);
+
+ // Implements the CreateTexture2D function for GL.
+ virtual ParseError CreateTexture2D(ResourceID id,
+ unsigned int width,
+ unsigned int height,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags);
+
+ // Implements the CreateTexture3D function for GL.
+ virtual ParseError CreateTexture3D(ResourceID id,
+ unsigned int width,
+ unsigned int height,
+ unsigned int depth,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags);
+
+ // Implements the CreateTextureCube function for GL.
+ virtual ParseError CreateTextureCube(ResourceID id,
+ unsigned int side,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags);
+
+ // Implements the SetTextureData function for GL.
+ virtual ParseError SetTextureData(ResourceID id,
+ unsigned int x,
+ unsigned int y,
+ unsigned int z,
+ unsigned int width,
+ unsigned int height,
+ unsigned int depth,
+ unsigned int level,
+ texture::Face face,
+ unsigned int pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ const void *data);
+
+ // Implements the GetTextureData function for GL.
+ virtual ParseError GetTextureData(ResourceID id,
+ unsigned int x,
+ unsigned int y,
+ unsigned int z,
+ unsigned int width,
+ unsigned int height,
+ unsigned int depth,
+ unsigned int level,
+ texture::Face face,
+ unsigned int pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ void *data);
+
+ // Implements the DestroyTexture function for GL.
+ virtual ParseError DestroyTexture(ResourceID id);
+
+ // Implements the CreateSampler function for GL.
+ virtual ParseError CreateSampler(ResourceID id);
+
+ // Implements the DestroySampler function for GL.
+ virtual ParseError DestroySampler(ResourceID id);
+
+ // Implements the SetSamplerStates function for GL.
+ virtual ParseError SetSamplerStates(ResourceID id,
+ sampler::AddressingMode addressing_u,
+ sampler::AddressingMode addressing_v,
+ sampler::AddressingMode addressing_w,
+ sampler::FilteringMode mag_filter,
+ sampler::FilteringMode min_filter,
+ sampler::FilteringMode mip_filter,
+ unsigned int max_anisotropy);
+
+ // Implements the SetSamplerBorderColor function for GL.
+ virtual ParseError SetSamplerBorderColor(ResourceID id, const RGBA &color);
+
+ // Implements the SetSamplerTexture function for GL.
+ virtual ParseError SetSamplerTexture(ResourceID id, ResourceID texture_id);
+
+ // Implements the SetScissor function for GL.
+ virtual void SetScissor(bool enable,
+ unsigned int x,
+ unsigned int y,
+ unsigned int width,
+ unsigned int height);
+
+ // Implements the SetPointLineRaster function for GL.
+ virtual void SetPointLineRaster(bool line_smooth,
+ bool point_sprite,
+ float point_size);
+
+ // Implements the SetPolygonOffset function for GL.
+ virtual void SetPolygonOffset(float slope_factor, float units);
+
+ // Implements the SetPolygonRaster function for GL.
+ virtual void SetPolygonRaster(PolygonMode fill_mode,
+ FaceCullMode cull_mode);
+
+ // Implements the SetAlphaTest function for GL.
+ virtual void SetAlphaTest(bool enable,
+ float reference,
+ Comparison comp);
+
+ // Implements the SetDepthTest function for GL.
+ virtual void SetDepthTest(bool enable,
+ bool write_enable,
+ Comparison comp);
+
+ // Implements the SetStencilTest function for GL.
+ virtual void SetStencilTest(bool enable,
+ bool separate_ccw,
+ unsigned int write_mask,
+ unsigned int compare_mask,
+ unsigned int ref,
+ Uint32 func_ops);
+
+ // Implements the SetColorWritefunction for GL.
+ virtual void SetColorWrite(bool red,
+ bool green,
+ bool blue,
+ bool alpha,
+ bool dither);
+
+ // Implements the SetBlending function for GL.
+ virtual void SetBlending(bool enable,
+ bool separate_alpha,
+ BlendEq color_eq,
+ BlendFunc color_src_func,
+ BlendFunc color_dst_func,
+ BlendEq alpha_eq,
+ BlendFunc alpha_src_func,
+ BlendFunc alpha_dst_func);
+
+ // Implements the SetBlendingColor function for GL.
+ virtual void SetBlendingColor(const RGBA &color);
+
+ // Gets a vertex buffer by resource ID.
+ VertexBufferGL *GetVertexBuffer(ResourceID id) {
+ return vertex_buffers_.Get(id);
+ }
+
+ // Gets a texture by resource ID.
+ TextureGL *GetTexture(ResourceID id) {
+ return textures_.Get(id);
+ }
+
+ // Gets a sampler by resource ID.
+ SamplerGL *GetSampler(ResourceID id) {
+ return samplers_.Get(id);
+ }
+
+ CGcontext cg_context() const { return cg_context_; }
+
+ EffectGL *current_effect() const { return current_effect_; }
+ // "Dirty" the current effect. This resets the vertex and fragment program,
+ // and requires ValidateEffect() to be called before further draws occur.
+ void DirtyEffect();
+ private:
+ void InitCommon();
+ // Validates the current vertex struct to GL, setting the vertex attributes.
+ bool ValidateStreams();
+ // Validates the current effect to GL. This sets the vertex and fragment
+ // programs, and updates parameters if needed.
+ bool ValidateEffect();
+
+#ifdef OS_LINUX
+ XWindowWrapper *window_;
+#endif
+ CGcontext cg_context_;
+
+ ResourceID current_vertex_struct_;
+ bool validate_streams_;
+ unsigned int max_vertices_;
+ ResourceID current_effect_id_;
+ bool validate_effect_;
+ EffectGL *current_effect_;
+
+ ResourceMap<VertexBufferGL> vertex_buffers_;
+ ResourceMap<IndexBufferGL> index_buffers_;
+ ResourceMap<VertexStructGL> vertex_structs_;
+ ResourceMap<EffectGL> effects_;
+ ResourceMap<EffectParamGL> effect_params_;
+ ResourceMap<TextureGL> textures_;
+ ResourceMap<SamplerGL> samplers_;
+};
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_CROSS_GL_GAPI_GL_H__
diff --git a/o3d/command_buffer/service/cross/gl/geometry_gl.cc b/o3d/command_buffer/service/cross/gl/geometry_gl.cc
new file mode 100644
index 0000000..2dd8758
--- /dev/null
+++ b/o3d/command_buffer/service/cross/gl/geometry_gl.cc
@@ -0,0 +1,548 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the implementation of the VertexBufferGL, IndexBufferGL
+// and VertexStructGL classes, as well as the geometry-related GAPI functions.
+
+#include "command_buffer/service/cross/gl/gapi_gl.h"
+#include "command_buffer/service/cross/gl/geometry_gl.h"
+
+namespace o3d {
+namespace command_buffer {
+
+VertexBufferGL::~VertexBufferGL() {
+ glDeleteBuffers(1, &gl_buffer_);
+ CHECK_GL_ERROR();
+}
+
+// Creates the GL buffer object.
+void VertexBufferGL::Create() {
+ glGenBuffers(1, &gl_buffer_);
+ glBindBuffer(GL_ARRAY_BUFFER, gl_buffer_);
+ GLenum usage =
+ (flags() & vertex_buffer::DYNAMIC) ? GL_DYNAMIC_DRAW : GL_STATIC_DRAW;
+ glBufferData(GL_ARRAY_BUFFER, size(), NULL, usage);
+ CHECK_GL_ERROR();
+}
+
+// Sets the data into the GL buffer object.
+bool VertexBufferGL::SetData(unsigned int offset,
+ unsigned int size,
+ const void *data) {
+ if (!gl_buffer_) {
+ LOG(ERROR) << "Calling SetData on a non-initialized VertexBufferGL.";
+ return false;
+ }
+ if ((offset >= this->size()) || (offset + size > this->size())) {
+ LOG(ERROR) << "Invalid size or offset on VertexBufferGL::SetData.";
+ return false;
+ }
+ glBindBuffer(GL_ARRAY_BUFFER, gl_buffer_);
+ glBufferSubData(GL_ARRAY_BUFFER, offset, size, data);
+ CHECK_GL_ERROR();
+ return true;
+}
+
+// Gets the data from the GL buffer object.
+bool VertexBufferGL::GetData(unsigned int offset,
+ unsigned int size,
+ void *data) {
+ if (!gl_buffer_) {
+ LOG(ERROR) << "Calling GetData on a non-initialized VertexBufferGL.";
+ return false;
+ }
+ if ((offset >= this->size()) || (offset + size > this->size())) {
+ LOG(ERROR) << "Invalid size or offset on VertexBufferGL::GetData.";
+ return false;
+ }
+ glBindBuffer(GL_ARRAY_BUFFER, gl_buffer_);
+ glGetBufferSubData(GL_ARRAY_BUFFER, offset, size, data);
+ CHECK_GL_ERROR();
+ return true;
+}
+
+IndexBufferGL::~IndexBufferGL() {
+ glDeleteBuffers(1, &gl_buffer_);
+}
+
+// Creates the GL buffer object.
+void IndexBufferGL::Create() {
+ glGenBuffers(1, &gl_buffer_);
+ glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, gl_buffer_);
+ GLenum usage =
+ (flags() & vertex_buffer::DYNAMIC) ? GL_DYNAMIC_DRAW : GL_STATIC_DRAW;
+ glBufferData(GL_ELEMENT_ARRAY_BUFFER, size(), NULL, usage);
+ CHECK_GL_ERROR();
+}
+
+// Sets the data into the GL buffer object.
+bool IndexBufferGL::SetData(unsigned int offset,
+ unsigned int size,
+ const void *data) {
+ if (!gl_buffer_) {
+ LOG(ERROR) << "Calling SetData on a non-initialized IndexBufferGL.";
+ return false;
+ }
+ if ((offset >= this->size()) || (offset + size > this->size())) {
+ LOG(ERROR) << "Invalid size or offset on IndexBufferGL::SetData.";
+ return false;
+ }
+ glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, gl_buffer_);
+ glBufferSubData(GL_ELEMENT_ARRAY_BUFFER, offset, size, data);
+ CHECK_GL_ERROR();
+ return true;
+}
+
+// Gets the data from the GL buffer object.
+bool IndexBufferGL::GetData(unsigned int offset,
+ unsigned int size,
+ void *data) {
+ if (!gl_buffer_) {
+ LOG(ERROR) << "Calling GetData on a non-initialized IndexBufferGL.";
+ return false;
+ }
+ if ((offset >= this->size()) || (offset + size > this->size())) {
+ LOG(ERROR) << "Invalid size or offset on IndexBufferGL::GetData.";
+ return false;
+ }
+ glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, gl_buffer_);
+ glGetBufferSubData(GL_ELEMENT_ARRAY_BUFFER, offset, size, data);
+ CHECK_GL_ERROR();
+ return true;
+}
+
+// Sets the input element in the VertexStruct resource.
+void VertexStructGL::SetInput(unsigned int input_index,
+ ResourceID vertex_buffer_id,
+ unsigned int offset,
+ unsigned int stride,
+ vertex_struct::Type type,
+ vertex_struct::Semantic semantic,
+ unsigned int semantic_index) {
+ Element &element = GetElement(input_index);
+ element.vertex_buffer = vertex_buffer_id;
+ element.offset = offset;
+ element.stride = stride;
+ element.type = type;
+ element.semantic = semantic;
+ element.semantic_index = semantic_index;
+ dirty_ = true;
+}
+
+namespace {
+
+inline const GLvoid *OffsetToPtr(GLintptr offset) {
+ return static_cast<char *>(NULL) + offset;
+}
+
+} // anonymous namespace
+
+unsigned int VertexStructGL::SetStreams(GAPIGL *gapi) {
+ if (dirty_) Compile();
+ unsigned int max_vertices = UINT_MAX;
+ for (unsigned int i = 0; i < kMaxAttribs; ++i) {
+ const AttribDesc &attrib = attribs_[i];
+ if (attrib.vertex_buffer_id == kInvalidResource) {
+ if (i == 0) {
+ glDisableClientState(GL_VERTEX_ARRAY);
+ } else {
+ glDisableVertexAttribArray(i);
+ }
+ } else {
+ glEnableVertexAttribArray(i);
+ VertexBufferGL *vertex_buffer =
+ gapi->GetVertexBuffer(attrib.vertex_buffer_id);
+ if (!vertex_buffer) {
+ glDisableVertexAttribArray(i);
+ max_vertices = 0;
+ continue;
+ }
+ DCHECK_NE(vertex_buffer->gl_buffer(), 0);
+ glBindBuffer(GL_ARRAY_BUFFER, vertex_buffer->gl_buffer());
+ glVertexAttribPointer(i, attrib.size, attrib.type, attrib.normalized,
+ attrib.stride, OffsetToPtr(attrib.offset));
+ max_vertices = std::min(max_vertices,
+ vertex_buffer->size() / attrib.stride);
+ }
+ }
+ CHECK_GL_ERROR();
+ return max_vertices;
+}
+
+namespace {
+
+// from the ARB_vertex_program extension, at
+// http://www.opengl.org/registry/specs/ARB/vertex_program.txt
+//
+// Generic
+// Attribute Conventional Attribute Conventional Attribute Command
+// --------- ------------------------ ------------------------------
+// 0 vertex position Vertex
+// 1 vertex weights 0-3 WeightARB, VertexWeightEXT
+// 2 normal Normal
+// 3 primary color Color
+// 4 secondary color SecondaryColorEXT
+// 5 fog coordinate FogCoordEXT
+// 6 - -
+// 7 - -
+// 8 texture coordinate set 0 MultiTexCoord(TEXTURE0, ...)
+// 9 texture coordinate set 1 MultiTexCoord(TEXTURE1, ...)
+// 10 texture coordinate set 2 MultiTexCoord(TEXTURE2, ...)
+// 11 texture coordinate set 3 MultiTexCoord(TEXTURE3, ...)
+// 12 texture coordinate set 4 MultiTexCoord(TEXTURE4, ...)
+// 13 texture coordinate set 5 MultiTexCoord(TEXTURE5, ...)
+// 14 texture coordinate set 6 MultiTexCoord(TEXTURE6, ...)
+// 15 texture coordinate set 7 MultiTexCoord(TEXTURE7, ...)
+// 8+n texture coordinate set n MultiTexCoord(TEXTURE0+n, ...)
+//
+// Note: we only accept at most 8 texture coordinates for maximum compatibility
+// with DirectX.
+
+inline unsigned int GetAttribIndex(vertex_struct::Semantic semantic,
+ unsigned int semantic_index) {
+ switch (semantic) {
+ case vertex_struct::POSITION:
+ DCHECK_EQ(semantic_index, 0);
+ return 0;
+ case vertex_struct::NORMAL:
+ DCHECK_EQ(semantic_index, 0);
+ return 2;
+ case vertex_struct::COLOR:
+ DCHECK_LT(semantic_index, 2);
+ return 3 + semantic_index;
+ case vertex_struct::TEX_COORD:
+ DCHECK_LT(semantic_index, 8);
+ return 8 + semantic_index;
+ default:
+ DLOG(FATAL) << "Not reached.";
+ break;
+ }
+}
+
+inline void ExtractSizeTypeNormalized(vertex_struct::Type type,
+ GLint *size,
+ GLenum *gl_type,
+ GLboolean *normalized) {
+ switch (type) {
+ case vertex_struct::FLOAT1:
+ case vertex_struct::FLOAT2:
+ case vertex_struct::FLOAT3:
+ case vertex_struct::FLOAT4:
+ *size = type - vertex_struct::FLOAT1 + 1;
+ *gl_type = GL_FLOAT;
+ *normalized = false;
+ break;
+ case vertex_struct::UCHAR4N:
+ *size = 4;
+ *gl_type = GL_UNSIGNED_BYTE;
+ *normalized = true;
+ break;
+ default:
+ DLOG(FATAL) << "Not reached.";
+ break;
+ }
+}
+
+} // anonymous namespace
+
+#ifndef COMPILER_MSVC
+// Although required by the spec, this causes problems on MSVC. It is needed by
+// GCC.
+const unsigned int VertexStructGL::kMaxAttribs;
+#endif
+
+void VertexStructGL::Compile() {
+ DCHECK(dirty_);
+ for (unsigned int i = 0; i < kMaxAttribs; ++i) {
+ attribs_[i].vertex_buffer_id = kInvalidResource;
+ }
+ for (unsigned int i = 0; i < count_ ; ++i) {
+ const Element &element = GetElement(i);
+ unsigned int index = GetAttribIndex(element.semantic,
+ element.semantic_index);
+ DCHECK_LT(index, kMaxAttribs);
+ AttribDesc &attrib = attribs_[index];
+ attrib.vertex_buffer_id = element.vertex_buffer;
+ ExtractSizeTypeNormalized(element.type, &attrib.size, &attrib.type,
+ &attrib.normalized);
+ attrib.stride = element.stride;
+ attrib.offset = element.offset;
+ }
+ dirty_ = false;
+}
+
+BufferSyncInterface::ParseError GAPIGL::CreateVertexBuffer(ResourceID id,
+ unsigned int size,
+ unsigned int flags) {
+ VertexBufferGL *vertex_buffer = new VertexBufferGL(size, flags);
+ vertex_buffer->Create();
+ vertex_buffers_.Assign(id, vertex_buffer);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+BufferSyncInterface::ParseError GAPIGL::DestroyVertexBuffer(ResourceID id) {
+ return vertex_buffers_.Destroy(id) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+BufferSyncInterface::ParseError GAPIGL::SetVertexBufferData(ResourceID id,
+ unsigned int offset,
+ unsigned int size,
+ const void *data) {
+ VertexBufferGL *vertex_buffer = vertex_buffers_.Get(id);
+ if (!vertex_buffer) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return vertex_buffer->SetData(offset, size, data) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+BufferSyncInterface::ParseError GAPIGL::GetVertexBufferData(ResourceID id,
+ unsigned int offset,
+ unsigned int size,
+ void *data) {
+ VertexBufferGL *vertex_buffer = vertex_buffers_.Get(id);
+ if (!vertex_buffer) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return vertex_buffer->GetData(offset, size, data) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+BufferSyncInterface::ParseError GAPIGL::CreateIndexBuffer(ResourceID id,
+ unsigned int size,
+ unsigned int flags) {
+ IndexBufferGL *index_buffer = new IndexBufferGL(size, flags);
+ index_buffer->Create();
+ index_buffers_.Assign(id, index_buffer);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+BufferSyncInterface::ParseError GAPIGL::DestroyIndexBuffer(ResourceID id) {
+ return vertex_buffers_.Destroy(id) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+BufferSyncInterface::ParseError GAPIGL::SetIndexBufferData(ResourceID id,
+ unsigned int offset,
+ unsigned int size,
+ const void *data) {
+ IndexBufferGL *index_buffer = index_buffers_.Get(id);
+ if (!index_buffer) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return index_buffer->SetData(offset, size, data) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+BufferSyncInterface::ParseError GAPIGL::GetIndexBufferData(ResourceID id,
+ unsigned int offset,
+ unsigned int size,
+ void *data) {
+ IndexBufferGL *index_buffer = index_buffers_.Get(id);
+ if (!index_buffer) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return index_buffer->GetData(offset, size, data) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+BufferSyncInterface::ParseError GAPIGL::CreateVertexStruct(
+ ResourceID id,
+ unsigned int input_count) {
+ if (id == current_vertex_struct_) validate_streams_ = true;
+ VertexStructGL *vertex_struct = new VertexStructGL(input_count);
+ vertex_structs_.Assign(id, vertex_struct);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+BufferSyncInterface::ParseError GAPIGL::DestroyVertexStruct(ResourceID id) {
+ if (id == current_vertex_struct_) validate_streams_ = true;
+ return vertex_structs_.Destroy(id) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+BufferSyncInterface::ParseError GAPIGL::SetVertexInput(
+ ResourceID vertex_struct_id,
+ unsigned int input_index,
+ ResourceID vertex_buffer_id,
+ unsigned int offset,
+ unsigned int stride,
+ vertex_struct::Type type,
+ vertex_struct::Semantic semantic,
+ unsigned int semantic_index) {
+ switch (semantic) {
+ case vertex_struct::POSITION:
+ if (semantic_index != 0) {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ break;
+ case vertex_struct::NORMAL:
+ if (semantic_index != 0) {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ break;
+ case vertex_struct::COLOR:
+ if (semantic_index >= 2) {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ break;
+ case vertex_struct::TEX_COORD:
+ if (semantic_index >= 8) {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ break;
+ default:
+ DLOG(FATAL) << "Not reached.";
+ break;
+ }
+ if (vertex_buffer_id == current_vertex_struct_) validate_streams_ = true;
+ VertexStructGL *vertex_struct = vertex_structs_.Get(vertex_struct_id);
+ if (!vertex_struct || input_index >= vertex_struct->count())
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ vertex_struct->SetInput(input_index, vertex_buffer_id, offset, stride, type,
+ semantic, semantic_index);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+BufferSyncInterface::ParseError GAPIGL::SetVertexStruct(ResourceID id) {
+ current_vertex_struct_ = id;
+ validate_streams_ = true;
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+bool GAPIGL::ValidateStreams() {
+ DCHECK(validate_streams_);
+ VertexStructGL *vertex_struct = vertex_structs_.Get(current_vertex_struct_);
+ if (!vertex_struct) {
+ LOG(ERROR) << "Drawing with invalid streams.";
+ return false;
+ }
+ max_vertices_ = vertex_struct->SetStreams(this);
+ validate_streams_ = false;
+ return max_vertices_ > 0;
+}
+
+namespace {
+
+void PrimitiveTypeToGL(GAPIInterface::PrimitiveType primitive_type,
+ GLenum *gl_mode,
+ unsigned int *count) {
+ switch (primitive_type) {
+ case GAPIInterface::POINTS:
+ *gl_mode = GL_POINTS;
+ break;
+ case GAPIInterface::LINES:
+ *gl_mode = GL_LINES;
+ *count *= 2;
+ break;
+ case GAPIInterface::LINE_STRIPS:
+ *gl_mode = GL_LINE_STRIP;
+ ++*count;
+ break;
+ case GAPIInterface::TRIANGLES:
+ *gl_mode = GL_TRIANGLES;
+ *count *= 3;
+ break;
+ case GAPIInterface::TRIANGLE_STRIPS:
+ *gl_mode = GL_TRIANGLE_STRIP;
+ *count += 2;
+ break;
+ case GAPIInterface::TRIANGLE_FANS:
+ *gl_mode = GL_TRIANGLE_FAN;
+ *count += 2;
+ break;
+ default:
+ LOG(FATAL) << "Invalid primitive type";
+ break;
+ }
+}
+
+} // anonymous namespace
+
+BufferSyncInterface::ParseError GAPIGL::Draw(PrimitiveType primitive_type,
+ unsigned int first,
+ unsigned int count) {
+ if (validate_effect_ && !ValidateEffect()) {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ DCHECK(current_effect_);
+ if (validate_streams_ && !ValidateStreams()) {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ GLenum gl_mode = GL_POINTS;
+ PrimitiveTypeToGL(primitive_type, &gl_mode, &count);
+ if (first + count > max_vertices_) {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ glDrawArrays(gl_mode, first, count);
+ CHECK_GL_ERROR();
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+BufferSyncInterface::ParseError GAPIGL::DrawIndexed(
+ PrimitiveType primitive_type,
+ ResourceID index_buffer_id,
+ unsigned int first,
+ unsigned int count,
+ unsigned int min_index,
+ unsigned int max_index) {
+ IndexBufferGL *index_buffer = index_buffers_.Get(index_buffer_id);
+ if (!index_buffer) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ if (validate_effect_ && !ValidateEffect()) {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ DCHECK(current_effect_);
+ if (validate_streams_ && !ValidateStreams()) {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ if ((min_index >= max_vertices_) || (max_index > max_vertices_)) {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ GLenum gl_mode = GL_POINTS;
+ PrimitiveTypeToGL(primitive_type, &gl_mode, &count);
+ glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, index_buffer->gl_buffer());
+ GLenum index_type = (index_buffer->flags() & index_buffer::INDEX_32BIT) ?
+ GL_UNSIGNED_INT : GL_UNSIGNED_SHORT;
+ GLuint index_size = (index_buffer->flags() & index_buffer::INDEX_32BIT) ?
+ sizeof(GLuint) : sizeof(GLushort); // NOLINT
+ GLuint offset = first * index_size;
+ if (offset + count * index_size > index_buffer->size()) {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ glDrawRangeElements(gl_mode, min_index, max_index, count, index_type,
+ OffsetToPtr(offset));
+ CHECK_GL_ERROR();
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/cross/gl/geometry_gl.h b/o3d/command_buffer/service/cross/gl/geometry_gl.h
new file mode 100644
index 0000000..580f14d
--- /dev/null
+++ b/o3d/command_buffer/service/cross/gl/geometry_gl.h
@@ -0,0 +1,144 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file declares the VertexBufferGL, IndexBufferGL and VertexStructGL
+// classes.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_CROSS_GL_GEOMETRY_GL_H_
+#define O3D_COMMAND_BUFFER_SERVICE_CROSS_GL_GEOMETRY_GL_H_
+
+#include "command_buffer/common/cross/gapi_interface.h"
+#include "command_buffer/service/cross/resource.h"
+#include "command_buffer/service/cross/gl/gl_utils.h"
+
+namespace o3d {
+namespace command_buffer {
+
+class GAPIGL;
+
+// GL version of VertexBuffer.
+class VertexBufferGL : public VertexBuffer {
+ public:
+ VertexBufferGL(unsigned int size, unsigned int flags)
+ : VertexBuffer(size, flags),
+ gl_buffer_(0) {}
+ virtual ~VertexBufferGL();
+
+ // Creates the GL vertex buffer.
+ void Create();
+
+ // Sets the data into the GL vertex buffer.
+ bool SetData(unsigned int offset, unsigned int size, const void *data);
+
+ // Gets the data from the GL vertex buffer.
+ bool GetData(unsigned int offset, unsigned int size, void *data);
+
+ // Gets the GL vertex buffer.
+ GLuint gl_buffer() const { return gl_buffer_; }
+
+ private:
+ GLuint gl_buffer_;
+ DISALLOW_COPY_AND_ASSIGN(VertexBufferGL);
+};
+
+// GL version of IndexBuffer.
+class IndexBufferGL : public IndexBuffer {
+ public:
+ IndexBufferGL(unsigned int size, unsigned int flags)
+ : IndexBuffer(size, flags),
+ gl_buffer_(0) {}
+ virtual ~IndexBufferGL();
+
+ // Creates the GL index buffer.
+ void Create();
+
+ // Sets the data into the GL index buffer.
+ bool SetData(unsigned int offset, unsigned int size, const void *data);
+
+ // Gets the data from the GL index buffer.
+ bool GetData(unsigned int offset, unsigned int size, void *data);
+
+ // Gets the GL index buffer.
+ GLuint gl_buffer() const { return gl_buffer_; }
+
+ private:
+ GLuint gl_buffer_;
+ DISALLOW_COPY_AND_ASSIGN(IndexBufferGL);
+};
+
+// GL version of VertexStruct.
+class VertexStructGL : public VertexStruct {
+ public:
+ explicit VertexStructGL(unsigned int count)
+ : VertexStruct(count),
+ dirty_(true) {}
+ virtual ~VertexStructGL() {}
+
+ // Adds an input to the vertex struct.
+ void SetInput(unsigned int input_index,
+ ResourceID vertex_buffer_id,
+ unsigned int offset,
+ unsigned int stride,
+ vertex_struct::Type type,
+ vertex_struct::Semantic semantic,
+ unsigned int semantic_index);
+
+ // Sets the input streams to GL.
+ unsigned int SetStreams(GAPIGL *gapi);
+
+ private:
+ static const unsigned int kMaxAttribs = 16;
+
+ // This struct describes the parameters that are passed to
+ // glVertexAttribPointer.
+ struct AttribDesc {
+ ResourceID vertex_buffer_id;
+ GLint size;
+ GLenum type;
+ GLboolean normalized;
+ GLsizei stride;
+ GLintptr offset;
+ };
+
+ // Compiles the vertex declaration into the attribute array.
+ void Compile();
+
+ bool dirty_;
+ AttribDesc attribs_[kMaxAttribs];
+ DISALLOW_COPY_AND_ASSIGN(VertexStructGL);
+};
+
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_CROSS_GL_GEOMETRY_GL_H_
diff --git a/o3d/command_buffer/service/cross/gl/gl_utils.h b/o3d/command_buffer/service/cross/gl/gl_utils.h
new file mode 100644
index 0000000..b41a4cf
--- /dev/null
+++ b/o3d/command_buffer/service/cross/gl/gl_utils.h
@@ -0,0 +1,63 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file includes all the necessary GL/Cg headers and implement some useful
+// utilities.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_CROSS_GL_GL_UTILS_H_
+#define O3D_COMMAND_BUFFER_SERVICE_CROSS_GL_GL_UTILS_H_
+
+#include <build/build_config.h>
+#define GL_GLEXT_PROTOTYPES
+#if defined(OS_WIN)
+#include <GL/gl.h>
+#elif defined(OS_MACOSX)
+#include <OpenGL/OpenGL.h>
+#include <AGL/agl.h>
+#elif defined(OS_LINUX)
+#include <GL/gl.h>
+#endif // defined(PLATFORM)
+#include <Cg/cg.h>
+#include <Cg/cgGL.h>
+
+// Define this for extra GL error debugging (slower).
+// #define GL_ERROR_DEBUGGING
+#ifdef GL_ERROR_DEBUGGING
+#define CHECK_GL_ERROR() do { \
+ GLenum gl_error = glGetError(); \
+ LOG_IF(ERROR, gl_error != GL_NO_ERROR) << "GL Error :" << gl_error; \
+ } while (0)
+#else // GL_ERROR_DEBUGGING
+#define CHECK_GL_ERROR() void(0)
+#endif // GL_ERROR_DEBUGGING
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_CROSS_GL_GL_UTILS_H_
diff --git a/o3d/command_buffer/service/cross/gl/sampler_gl.cc b/o3d/command_buffer/service/cross/gl/sampler_gl.cc
new file mode 100644
index 0000000..2262e6c
--- /dev/null
+++ b/o3d/command_buffer/service/cross/gl/sampler_gl.cc
@@ -0,0 +1,235 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file implements the sampler-related GAPI functions on GL.
+
+#include "command_buffer/service/cross/gl/gapi_gl.h"
+#include "command_buffer/service/cross/gl/sampler_gl.h"
+
+namespace o3d {
+namespace command_buffer {
+
+namespace {
+
+// Gets the GL enum corresponding to an addressing mode.
+GLenum GLAddressMode(sampler::AddressingMode o3d_mode) {
+ switch (o3d_mode) {
+ case sampler::WRAP:
+ return GL_REPEAT;
+ case sampler::MIRROR_REPEAT:
+ return GL_MIRRORED_REPEAT;
+ case sampler::CLAMP_TO_EDGE:
+ return GL_CLAMP_TO_EDGE;
+ case sampler::CLAMP_TO_BORDER:
+ return GL_CLAMP_TO_BORDER;
+ default:
+ DLOG(FATAL) << "Not Reached";
+ return GL_REPEAT;
+ }
+}
+
+// Gets the GL enum for the minification filter based on the command buffer min
+// and mip filtering modes.
+GLenum GLMinFilter(sampler::FilteringMode min_filter,
+ sampler::FilteringMode mip_filter) {
+ switch (min_filter) {
+ case sampler::POINT:
+ if (mip_filter == sampler::NONE)
+ return GL_NEAREST;
+ else if (mip_filter == sampler::POINT)
+ return GL_NEAREST_MIPMAP_NEAREST;
+ else if (mip_filter == sampler::LINEAR)
+ return GL_NEAREST_MIPMAP_LINEAR;
+ case sampler::LINEAR:
+ if (mip_filter == sampler::NONE)
+ return GL_LINEAR;
+ else if (mip_filter == sampler::POINT)
+ return GL_LINEAR_MIPMAP_NEAREST;
+ else if (mip_filter == sampler::LINEAR)
+ return GL_LINEAR_MIPMAP_LINEAR;
+ default:
+ DLOG(FATAL) << "Not Reached";
+ return GL_LINEAR_MIPMAP_NEAREST;
+ }
+}
+
+// Gets the GL enum for the magnification filter based on the command buffer mag
+// filtering mode.
+GLenum GLMagFilter(sampler::FilteringMode mag_filter) {
+ switch (mag_filter) {
+ case sampler::POINT:
+ return GL_NEAREST;
+ case sampler::LINEAR:
+ return GL_LINEAR;
+ default:
+ DLOG(FATAL) << "Not Reached";
+ return GL_LINEAR;
+ }
+}
+
+// Gets the GL enum representing the GL target based on the texture type.
+GLenum GLTextureTarget(texture::Type type) {
+ switch (type) {
+ case texture::TEXTURE_2D:
+ return GL_TEXTURE_2D;
+ case texture::TEXTURE_3D:
+ return GL_TEXTURE_3D;
+ case texture::TEXTURE_CUBE:
+ return GL_TEXTURE_CUBE_MAP;
+ }
+}
+
+} // anonymous namespace
+
+SamplerGL::SamplerGL()
+ : texture_id_(kInvalidResource),
+ gl_texture_(0) {
+ SetStates(sampler::CLAMP_TO_EDGE,
+ sampler::CLAMP_TO_EDGE,
+ sampler::CLAMP_TO_EDGE,
+ sampler::LINEAR,
+ sampler::LINEAR,
+ sampler::POINT,
+ 1);
+ RGBA black = {0, 0, 0, 1};
+ SetBorderColor(black);
+}
+
+bool SamplerGL::ApplyStates(GAPIGL *gapi) {
+ DCHECK(gapi);
+ TextureGL *texture = gapi->GetTexture(texture_id_);
+ if (!texture) {
+ gl_texture_ = 0;
+ return false;
+ }
+ GLenum target = GLTextureTarget(texture->type());
+ gl_texture_ = texture->gl_texture();
+ glBindTexture(target, gl_texture_);
+ glTexParameteri(target, GL_TEXTURE_WRAP_S, gl_wrap_s_);
+ glTexParameteri(target, GL_TEXTURE_WRAP_T, gl_wrap_t_);
+ glTexParameteri(target, GL_TEXTURE_WRAP_R, gl_wrap_r_);
+ glTexParameteri(target, GL_TEXTURE_MIN_FILTER, gl_min_filter_);
+ glTexParameteri(target, GL_TEXTURE_MAG_FILTER, gl_mag_filter_);
+ glTexParameteri(target, GL_TEXTURE_MAX_ANISOTROPY_EXT, gl_max_anisotropy_);
+ glTexParameterfv(target, GL_TEXTURE_BORDER_COLOR, gl_border_color_);
+ return true;
+}
+
+void SamplerGL::SetStates(sampler::AddressingMode addressing_u,
+ sampler::AddressingMode addressing_v,
+ sampler::AddressingMode addressing_w,
+ sampler::FilteringMode mag_filter,
+ sampler::FilteringMode min_filter,
+ sampler::FilteringMode mip_filter,
+ unsigned int max_anisotropy) {
+ // These are validated in GAPIDecoder.cc
+ DCHECK_NE(mag_filter, sampler::NONE);
+ DCHECK_NE(min_filter, sampler::NONE);
+ DCHECK_GT(max_anisotropy, 0);
+ gl_wrap_s_ = GLAddressMode(addressing_u);
+ gl_wrap_t_ = GLAddressMode(addressing_v);
+ gl_wrap_r_ = GLAddressMode(addressing_w);
+ gl_mag_filter_ = GLMagFilter(mag_filter);
+ gl_min_filter_ = GLMinFilter(min_filter, mip_filter);
+ gl_max_anisotropy_ = max_anisotropy;
+}
+
+void SamplerGL::SetBorderColor(const RGBA &color) {
+ gl_border_color_[0] = color.red;
+ gl_border_color_[1] = color.green;
+ gl_border_color_[2] = color.blue;
+ gl_border_color_[3] = color.alpha;
+}
+
+BufferSyncInterface::ParseError GAPIGL::CreateSampler(
+ ResourceID id) {
+ // Dirty effect, because this sampler id may be used.
+ DirtyEffect();
+ samplers_.Assign(id, new SamplerGL());
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+// Destroys the Sampler resource.
+BufferSyncInterface::ParseError GAPIGL::DestroySampler(ResourceID id) {
+ // Dirty effect, because this sampler id may be used.
+ DirtyEffect();
+ return samplers_.Destroy(id) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+BufferSyncInterface::ParseError GAPIGL::SetSamplerStates(
+ ResourceID id,
+ sampler::AddressingMode addressing_u,
+ sampler::AddressingMode addressing_v,
+ sampler::AddressingMode addressing_w,
+ sampler::FilteringMode mag_filter,
+ sampler::FilteringMode min_filter,
+ sampler::FilteringMode mip_filter,
+ unsigned int max_anisotropy) {
+ SamplerGL *sampler = samplers_.Get(id);
+ if (!sampler)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ // Dirty effect, because this sampler id may be used.
+ DirtyEffect();
+ sampler->SetStates(addressing_u, addressing_v, addressing_w,
+ mag_filter, min_filter, mip_filter, max_anisotropy);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+BufferSyncInterface::ParseError GAPIGL::SetSamplerBorderColor(
+ ResourceID id,
+ const RGBA &color) {
+ SamplerGL *sampler = samplers_.Get(id);
+ if (!sampler)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ // Dirty effect, because this sampler id may be used.
+ DirtyEffect();
+ sampler->SetBorderColor(color);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+BufferSyncInterface::ParseError GAPIGL::SetSamplerTexture(
+ ResourceID id,
+ ResourceID texture_id) {
+ SamplerGL *sampler = samplers_.Get(id);
+ if (!sampler)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ // Dirty effect, because this sampler id may be used.
+ DirtyEffect();
+ sampler->SetTexture(texture_id);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/cross/gl/sampler_gl.h b/o3d/command_buffer/service/cross/gl/sampler_gl.h
new file mode 100644
index 0000000..dc84b4f
--- /dev/null
+++ b/o3d/command_buffer/service/cross/gl/sampler_gl.h
@@ -0,0 +1,87 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file declares the SamplerGL class.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_CROSS_GL_SAMPLER_GL_H_
+#define O3D_COMMAND_BUFFER_SERVICE_CROSS_GL_SAMPLER_GL_H_
+
+#include "command_buffer/common/cross/gapi_interface.h"
+#include "command_buffer/service/cross/gl/gl_utils.h"
+#include "command_buffer/service/cross/resource.h"
+
+namespace o3d {
+namespace command_buffer {
+
+class GAPIGL;
+
+// GL version of Sampler.
+class SamplerGL : public Sampler {
+ public:
+ SamplerGL();
+
+ // Applies sampler states to GL.
+ bool ApplyStates(GAPIGL *gapi);
+
+ // Sets sampler states.
+ void SetStates(sampler::AddressingMode addressing_u,
+ sampler::AddressingMode addressing_v,
+ sampler::AddressingMode addressing_w,
+ sampler::FilteringMode mag_filter,
+ sampler::FilteringMode min_filter,
+ sampler::FilteringMode mip_filter,
+ unsigned int max_anisotropy);
+
+ // Sets the border color states.
+ void SetBorderColor(const RGBA &color);
+
+ // Sets the texture.
+ void SetTexture(ResourceID texture) { texture_id_ = texture; }
+
+ GLuint gl_texture() const { return gl_texture_; }
+
+ private:
+ GLenum gl_wrap_s_;
+ GLenum gl_wrap_t_;
+ GLenum gl_wrap_r_;
+ GLenum gl_mag_filter_;
+ GLenum gl_min_filter_;
+ GLuint gl_max_anisotropy_;
+ GLfloat gl_border_color_[4];
+ GLuint gl_texture_;
+ ResourceID texture_id_;
+};
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_CROSS_GL_SAMPLER_GL_H_
diff --git a/o3d/command_buffer/service/cross/gl/states_gl.cc b/o3d/command_buffer/service/cross/gl/states_gl.cc
new file mode 100644
index 0000000..4acc96c
--- /dev/null
+++ b/o3d/command_buffer/service/cross/gl/states_gl.cc
@@ -0,0 +1,342 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file implements the render state-related GAPI functions on GL.
+
+#include "command_buffer/common/cross/cmd_buffer_format.h"
+#include "command_buffer/service/cross/gl/gapi_gl.h"
+
+namespace o3d {
+namespace command_buffer {
+
+namespace {
+
+GLenum kGLPolygonModes[] = {
+ GL_POINT,
+ GL_LINE,
+ GL_FILL,
+};
+COMPILE_ASSERT(GAPIInterface::NUM_POLYGON_MODE == arraysize(kGLPolygonModes),
+ kGLPolygonModes_does_not_match_GAPIInterface_PolygonMode);
+
+GLenum kGLComparison[] = {
+ GL_NEVER,
+ GL_LESS,
+ GL_EQUAL,
+ GL_LEQUAL,
+ GL_GREATER,
+ GL_NOTEQUAL,
+ GL_GEQUAL,
+ GL_ALWAYS,
+};
+COMPILE_ASSERT(GAPIInterface::NUM_COMPARISON == arraysize(kGLComparison),
+ kGLComparison_does_not_match_GAPIInterface_Comparison);
+
+GLenum kGLBlendFunc[] = {
+ GL_ZERO,
+ GL_ONE,
+ GL_SRC_COLOR,
+ GL_ONE_MINUS_SRC_COLOR,
+ GL_SRC_ALPHA,
+ GL_ONE_MINUS_SRC_ALPHA,
+ GL_DST_ALPHA,
+ GL_ONE_MINUS_DST_ALPHA,
+ GL_DST_COLOR,
+ GL_ONE_MINUS_DST_COLOR,
+ GL_SRC_ALPHA_SATURATE,
+ GL_CONSTANT_COLOR,
+ GL_ONE_MINUS_CONSTANT_COLOR,
+};
+COMPILE_ASSERT(GAPIInterface::NUM_BLEND_FUNC == arraysize(kGLBlendFunc),
+ kGLBlendFunc_does_not_match_GAPIInterface_BlendFunc);
+
+GLenum kGLBlendEq[] = {
+ GL_FUNC_ADD,
+ GL_FUNC_SUBTRACT,
+ GL_FUNC_REVERSE_SUBTRACT,
+ GL_MIN,
+ GL_MAX,
+};
+COMPILE_ASSERT(GAPIInterface::NUM_BLEND_EQ == arraysize(kGLBlendEq),
+ kGLBlendEq_does_not_match_GAPIInterface_BlendEq);
+
+GLenum kGLStencilOp[] = {
+ GL_KEEP,
+ GL_ZERO,
+ GL_REPLACE,
+ GL_INCR,
+ GL_DECR,
+ GL_INVERT,
+ GL_INCR_WRAP,
+ GL_DECR_WRAP,
+};
+COMPILE_ASSERT(GAPIInterface::NUM_STENCIL_OP == arraysize(kGLStencilOp),
+ kGLStencilOp_does_not_match_GAPIInterface_StencilOp);
+
+// Check that the definition of the counter-clockwise func/ops match the
+// clockwise ones, just shifted by 16 bits, so that we can use
+// DecodeStencilFuncOps on both of them.
+#define CHECK_CCW_MATCHES_CW(FIELD) \
+ COMPILE_ASSERT(set_stencil_test::CW ## FIELD::kLength == \
+ set_stencil_test::CCW ## FIELD::kLength, \
+ CCW ## FIELD ## _length_does_not_match_ ## CW ## FIELD); \
+ COMPILE_ASSERT(set_stencil_test::CW ## FIELD::kShift + 16 == \
+ set_stencil_test::CCW ## FIELD::kShift, \
+ CCW ## FIELD ## _shift_does_not_match_ ## CW ## FIELD)
+
+CHECK_CCW_MATCHES_CW(Func);
+CHECK_CCW_MATCHES_CW(PassOp);
+CHECK_CCW_MATCHES_CW(FailOp);
+CHECK_CCW_MATCHES_CW(ZFailOp);
+
+#undef CHECK_CCW_MATCHES_CW
+
+// Decodes stencil test function and operations from the bitfield.
+void DecodeStencilFuncOps(Uint32 params,
+ GLenum *func,
+ GLenum *pass,
+ GLenum *fail,
+ GLenum *zfail) {
+ namespace cmd = set_stencil_test;
+ // Sanity check. The value has already been tested in
+ // GAPIDecoder::DecodeSetStencilTest in gapi_decoder.cc.
+ DCHECK_EQ(cmd::Unused1::Get(params), 0);
+ // Check that the bitmask get cannot generate values outside of the allowed
+ // range.
+ COMPILE_ASSERT(cmd::CWFunc::kMask < GAPIInterface::NUM_COMPARISON,
+ set_stencil_test_CWFunc_may_produce_invalid_values);
+ *func = kGLComparison[cmd::CWFunc::Get(params)];
+
+ COMPILE_ASSERT(cmd::CWPassOp::kMask < GAPIInterface::NUM_STENCIL_OP,
+ set_stencil_test_CWPassOp_may_produce_invalid_values);
+ *pass = kGLStencilOp[cmd::CWPassOp::Get(params)];
+
+ COMPILE_ASSERT(cmd::CWFailOp::kMask < GAPIInterface::NUM_STENCIL_OP,
+ set_stencil_test_CWFailOp_may_produce_invalid_values);
+ *fail = kGLStencilOp[cmd::CWFailOp::Get(params)];
+
+ COMPILE_ASSERT(cmd::CWZFailOp::kMask < GAPIInterface::NUM_STENCIL_OP,
+ set_stencil_test_CWZFailOp_may_produce_invalid_values);
+ *zfail = kGLStencilOp[cmd::CWZFailOp::Get(params)];
+}
+
+} // anonymous namespace
+
+void GAPIGL::SetViewport(unsigned int x,
+ unsigned int y,
+ unsigned int width,
+ unsigned int height,
+ float z_min,
+ float z_max) {
+ glViewport(x, y, width, height);
+ glDepthRange(z_min, z_max);
+ // Update the helper constant used for the D3D -> GL remapping.
+ // See effect_gl.cc for details.
+ glProgramEnvParameter4fARB(GL_VERTEX_PROGRAM_ARB, 0,
+ 1.f / width, 1.f / height, 2.f, 0.f);
+ CHECK_GL_ERROR();
+}
+
+void GAPIGL::SetScissor(bool enable,
+ unsigned int x,
+ unsigned int y,
+ unsigned int width,
+ unsigned int height) {
+ if (enable) {
+ glEnable(GL_SCISSOR_TEST);
+ glScissor(x, y, width, height);
+ } else {
+ glDisable(GL_SCISSOR_TEST);
+ }
+}
+
+void GAPIGL::SetPointLineRaster(bool line_smooth,
+ bool point_sprite,
+ float point_size) {
+ if (line_smooth) {
+ glEnable(GL_LINE_SMOOTH);
+ } else {
+ glDisable(GL_LINE_SMOOTH);
+ }
+ if (point_sprite) {
+ glEnable(GL_POINT_SPRITE);
+ // TODO: check which TC gets affected by point sprites in D3D.
+ glActiveTextureARB(GL_TEXTURE0);
+ glTexEnvi(GL_POINT_SPRITE, GL_COORD_REPLACE, GL_TRUE);
+ } else {
+ glActiveTextureARB(GL_TEXTURE0);
+ glTexEnvi(GL_POINT_SPRITE, GL_COORD_REPLACE, GL_FALSE);
+ glDisable(GL_POINT_SPRITE);
+ }
+ glPointSize(point_size);
+}
+
+void GAPIGL::SetPolygonOffset(float slope_factor, float units) {
+ glPolygonOffset(slope_factor, units);
+}
+
+void GAPIGL::SetPolygonRaster(PolygonMode fill_mode,
+ FaceCullMode cull_mode) {
+ DCHECK_LT(fill_mode, NUM_POLYGON_MODE);
+ glPolygonMode(GL_FRONT_AND_BACK, kGLPolygonModes[fill_mode]);
+ DCHECK_LT(cull_mode, NUM_FACE_CULL_MODE);
+ switch (cull_mode) {
+ case CULL_CW:
+ glEnable(GL_CULL_FACE);
+ glCullFace(GL_BACK);
+ break;
+ case CULL_CCW:
+ glEnable(GL_CULL_FACE);
+ glCullFace(GL_FRONT);
+ break;
+ default:
+ glDisable(GL_CULL_FACE);
+ break;
+ }
+}
+
+void GAPIGL::SetAlphaTest(bool enable,
+ float reference,
+ Comparison comp) {
+ DCHECK_LT(comp, NUM_COMPARISON);
+ if (enable) {
+ glEnable(GL_ALPHA_TEST);
+ glAlphaFunc(kGLComparison[comp], reference);
+ } else {
+ glDisable(GL_ALPHA_TEST);
+ }
+}
+
+void GAPIGL::SetDepthTest(bool enable,
+ bool write_enable,
+ Comparison comp) {
+ DCHECK_LT(comp, NUM_COMPARISON);
+ if (enable) {
+ glEnable(GL_DEPTH_TEST);
+ glDepthFunc(kGLComparison[comp]);
+ } else {
+ glDisable(GL_DEPTH_TEST);
+ }
+ glDepthMask(write_enable);
+}
+
+void GAPIGL::SetStencilTest(bool enable,
+ bool separate_ccw,
+ unsigned int write_mask,
+ unsigned int compare_mask,
+ unsigned int ref,
+ Uint32 func_ops) {
+ if (enable) {
+ glEnable(GL_STENCIL_TEST);
+ glStencilMask(write_mask);
+
+ GLenum func;
+ GLenum pass;
+ GLenum fail;
+ GLenum zfail;
+ DecodeStencilFuncOps(func_ops, &func, &pass, &fail, &zfail);
+ if (separate_ccw) {
+ glStencilFuncSeparate(GL_FRONT, func, ref, compare_mask);
+ glStencilOpSeparate(GL_FRONT, pass, fail, zfail);
+ // Extract upper 16 bits.
+ Uint32 ccw_func_ops = BitField<16, 16>::Get(func_ops);
+ GLenum ccw_func;
+ GLenum ccw_pass;
+ GLenum ccw_fail;
+ GLenum ccw_zfail;
+ DecodeStencilFuncOps(ccw_func_ops, &ccw_func, &ccw_pass, &ccw_fail,
+ &ccw_zfail);
+ glStencilFuncSeparate(GL_BACK, ccw_func, ref, compare_mask);
+ glStencilOpSeparate(GL_BACK, ccw_pass, ccw_fail, ccw_zfail);
+ } else {
+ glStencilFunc(func, ref, compare_mask);
+ glStencilOp(pass, fail, zfail);
+ }
+ } else {
+ glDisable(GL_STENCIL_TEST);
+ }
+}
+
+void GAPIGL::SetColorWrite(bool red,
+ bool green,
+ bool blue,
+ bool alpha,
+ bool dither) {
+ glColorMask(red, green, blue, alpha);
+ if (dither) {
+ glEnable(GL_DITHER);
+ } else {
+ glDisable(GL_DITHER);
+ }
+}
+
+void GAPIGL::SetBlending(bool enable,
+ bool separate_alpha,
+ BlendEq color_eq,
+ BlendFunc color_src_func,
+ BlendFunc color_dst_func,
+ BlendEq alpha_eq,
+ BlendFunc alpha_src_func,
+ BlendFunc alpha_dst_func) {
+ DCHECK_LT(color_eq, NUM_BLEND_EQ);
+ DCHECK_LT(color_src_func, NUM_BLEND_FUNC);
+ DCHECK_LT(color_dst_func, NUM_BLEND_FUNC);
+ DCHECK_LT(alpha_eq, NUM_BLEND_EQ);
+ DCHECK_LT(alpha_src_func, NUM_BLEND_FUNC);
+ DCHECK_LT(alpha_dst_func, NUM_BLEND_FUNC);
+ if (enable) {
+ glEnable(GL_BLEND);
+ GLenum gl_color_eq = kGLBlendEq[color_eq];
+ GLenum gl_color_src_func = kGLBlendFunc[color_src_func];
+ GLenum gl_color_dst_func = kGLBlendFunc[color_dst_func];
+ if (separate_alpha) {
+ GLenum gl_alpha_eq = kGLBlendEq[alpha_eq];
+ GLenum gl_alpha_src_func = kGLBlendFunc[alpha_src_func];
+ GLenum gl_alpha_dst_func = kGLBlendFunc[alpha_dst_func];
+ glBlendFuncSeparate(gl_color_src_func, gl_color_dst_func,
+ gl_alpha_src_func, gl_alpha_dst_func);
+ glBlendEquationSeparate(gl_color_eq, gl_alpha_eq);
+ } else {
+ glBlendFunc(gl_color_src_func, gl_color_dst_func);
+ glBlendEquation(gl_color_eq);
+ }
+ } else {
+ glDisable(GL_BLEND);
+ }
+}
+
+void GAPIGL::SetBlendingColor(const RGBA &color) {
+ glBlendColor(color.red, color.green, color.blue, color.alpha);
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/cross/gl/texture_gl.cc b/o3d/command_buffer/service/cross/gl/texture_gl.cc
new file mode 100644
index 0000000..9155dcc
--- /dev/null
+++ b/o3d/command_buffer/service/cross/gl/texture_gl.cc
@@ -0,0 +1,678 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file implements the texture-related GAPI functions on GL.
+
+#include "command_buffer/service/cross/gl/gapi_gl.h"
+#include "command_buffer/service/cross/gl/texture_gl.h"
+
+namespace o3d {
+namespace command_buffer {
+
+namespace {
+
+// Gets the GL internal format, format and type corresponding to a command
+// buffer texture format.
+bool GetGLFormatType(texture::Format format,
+ GLenum *internal_format,
+ GLenum *gl_format,
+ GLenum *gl_type) {
+ switch (format) {
+ case texture::XRGB8:
+ *internal_format = GL_RGB;
+ *gl_format = GL_BGRA;
+ *gl_type = GL_UNSIGNED_BYTE;
+ return true;
+ case texture::ARGB8:
+ *internal_format = GL_RGBA;
+ *gl_format = GL_BGRA;
+ *gl_type = GL_UNSIGNED_BYTE;
+ return true;
+ case texture::ABGR16F:
+ *internal_format = GL_RGBA16F_ARB;
+ *gl_format = GL_RGBA;
+ *gl_type = GL_HALF_FLOAT_ARB;
+ return true;
+ case texture::DXT1:
+ *internal_format = GL_COMPRESSED_RGBA_S3TC_DXT1_EXT;
+ *gl_format = 0;
+ *gl_type = 0;
+ return true;
+ default:
+ return false;
+ }
+}
+
+// Helper class used to prepare image data to match the layout that
+// glTexImage* and glCompressedTexImage* expect.
+class SetImageHelper {
+ public:
+ SetImageHelper()
+ : buffer_(NULL),
+ image_data_(NULL),
+ image_size_(0) {
+ }
+
+ // Initializes the helper with the input data, re-using the input buffer if
+ // possible, or copying it into a temporary one.
+ bool Initialize(const MipLevelInfo &mip_info,
+ const Volume& volume,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int src_size,
+ const void *data) {
+ TransferInfo src_transfer_info;
+ MakeTransferInfo(&src_transfer_info, mip_info, volume, row_pitch,
+ slice_pitch);
+ if (!CheckVolume(mip_info, volume) ||
+ src_size < src_transfer_info.total_size)
+ return false;
+ if (!src_transfer_info.packed) {
+ TransferInfo dst_transfer_info;
+ MakePackedTransferInfo(&dst_transfer_info, mip_info, volume);
+ buffer_.reset(new unsigned char[dst_transfer_info.total_size]);
+ TransferVolume(volume, mip_info, dst_transfer_info, buffer_.get(),
+ src_transfer_info, data);
+ image_data_ = buffer_.get();
+ image_size_ = dst_transfer_info.total_size;
+ } else {
+ image_data_ = data;
+ image_size_ = src_transfer_info.total_size;
+ }
+ return true;
+ }
+
+ // Gets the buffer that contains the data in the GL format.
+ const void *image_data() { return image_data_; }
+ // Gets the size of the buffer as GL expects it.
+ unsigned int image_size() { return image_size_; }
+
+ private:
+ scoped_array<unsigned char> buffer_;
+ const void *image_data_;
+ unsigned int image_size_;
+ DISALLOW_COPY_AND_ASSIGN(SetImageHelper);
+};
+
+// Helper class used to retrieve image data to match the layout that
+// glGetTexImage and glGetCompressedTexImage expect.
+class GetImageHelper {
+ public:
+ GetImageHelper()
+ : dst_data_(NULL),
+ buffer_(NULL),
+ image_data_(NULL) {
+ memset(&mip_info_, 0, sizeof(mip_info_));
+ memset(&volume_, 0, sizeof(volume_));
+ memset(&dst_transfer_info_, 0, sizeof(dst_transfer_info_));
+ memset(&src_transfer_info_, 0, sizeof(src_transfer_info_));
+ }
+
+ // Initialize the helper to make available a buffer to get the data from GL.
+ // It will re-use the passed in buffer if the layout matches GL, or allocate
+ // a temporary one.
+ bool Initialize(const MipLevelInfo &mip_info,
+ const Volume& volume,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int dst_size,
+ void *dst_data) {
+ mip_info_ = mip_info;
+ volume_ = volume;
+ dst_data_ = dst_data;
+ MakeTransferInfo(&dst_transfer_info_, mip_info, volume, row_pitch,
+ slice_pitch);
+ if (!CheckVolume(mip_info, volume) ||
+ dst_size < dst_transfer_info_.total_size)
+ return false;
+
+ if (!IsFullVolume(mip_info, volume) || !dst_transfer_info_.packed) {
+ // We can only retrieve the full image from GL.
+ Volume full_volume = {
+ 0, 0, 0,
+ mip_info.width, mip_info.height, mip_info.depth
+ };
+ MakePackedTransferInfo(&src_transfer_info_, mip_info, full_volume);
+ buffer_.reset(new unsigned char[src_transfer_info_.total_size]);
+ image_data_ = buffer_.get();
+ } else {
+ image_data_ = dst_data;
+ }
+ return true;
+ }
+
+ // Finalize the helper, copying the data into the final buffer if needed.
+ void Finalize() {
+ if (!buffer_.get()) return;
+ unsigned int offset =
+ volume_.x / mip_info_.block_size_x * mip_info_.block_bpp +
+ volume_.y / mip_info_.block_size_y * src_transfer_info_.row_pitch +
+ volume_.z * src_transfer_info_.slice_pitch;
+ src_transfer_info_.row_size = dst_transfer_info_.row_size;
+ TransferVolume(volume_, mip_info_, dst_transfer_info_, dst_data_,
+ src_transfer_info_, buffer_.get() + offset);
+ }
+
+ // Gets the buffer that can receive the data from GL.
+ void *image_data() { return image_data_; }
+
+ private:
+ MipLevelInfo mip_info_;
+ Volume volume_;
+ TransferInfo dst_transfer_info_;
+ TransferInfo src_transfer_info_;
+ void *dst_data_;
+ scoped_array<unsigned char> buffer_;
+ void *image_data_;
+ DISALLOW_COPY_AND_ASSIGN(GetImageHelper);
+};
+
+} // anonymous namespace
+
+TextureGL::~TextureGL() {
+ glDeleteTextures(1, &gl_texture_);
+ CHECK_GL_ERROR();
+}
+
+Texture2DGL *Texture2DGL::Create(unsigned int width,
+ unsigned int height,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags) {
+ DCHECK_GT(width, 0);
+ DCHECK_GT(height, 0);
+ DCHECK_GT(levels, 0);
+ GLenum gl_internal_format = 0;
+ GLenum gl_format = 0;
+ GLenum gl_type = 0;
+ bool r = GetGLFormatType(format, &gl_internal_format, &gl_format, &gl_type);
+ DCHECK(r); // Was checked in the decoder.
+ GLuint gl_texture = 0;
+ glGenTextures(1, &gl_texture);
+ glBindTexture(GL_TEXTURE_2D, gl_texture);
+ glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAX_LEVEL, levels - 1);
+ // glCompressedTexImage2D does't accept NULL as a parameter, so we need
+ // to pass in some data.
+ scoped_array<unsigned char> buffer;
+ if (!gl_format) {
+ MipLevelInfo mip_info;
+ MakeMipLevelInfo(&mip_info, format, width, height, 1, 0);
+ unsigned int size = GetMipLevelSize(mip_info);
+ buffer.reset(new unsigned char[size]);
+ }
+ unsigned int mip_width = width;
+ unsigned int mip_height = height;
+ for (unsigned int i = 0; i < levels; ++i) {
+ if (gl_format) {
+ glTexImage2D(GL_TEXTURE_2D, i, gl_internal_format, mip_width, mip_height,
+ 0, gl_format, gl_type, NULL);
+ } else {
+ MipLevelInfo mip_info;
+ MakeMipLevelInfo(&mip_info, format, width, height, 1, i);
+ unsigned int size = GetMipLevelSize(mip_info);
+ glCompressedTexImage2D(GL_TEXTURE_2D, i, gl_internal_format, mip_width,
+ mip_height, 0, size, buffer.get());
+ }
+ mip_width = std::max(1U, mip_width >> 1);
+ mip_height = std::max(1U, mip_height >> 1);
+ }
+ return new Texture2DGL(levels, format, flags, width, height, gl_texture);
+}
+
+// Sets data into a 2D texture resource.
+bool Texture2DGL::SetData(const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ const void *data) {
+ if (level >= levels())
+ return false;
+ MipLevelInfo mip_info;
+ MakeMipLevelInfo(&mip_info, format(), width_, height_, 1, level);
+ SetImageHelper helper;
+ if (!helper.Initialize(mip_info, volume, row_pitch, slice_pitch, size, data))
+ return false;
+
+ GLenum gl_internal_format = 0;
+ GLenum gl_format = 0;
+ GLenum gl_type = 0;
+ bool r = GetGLFormatType(format(), &gl_internal_format, &gl_format, &gl_type);
+ DCHECK(r);
+ glBindTexture(GL_TEXTURE_2D, gl_texture_);
+ if (gl_format) {
+ glTexSubImage2D(GL_TEXTURE_2D, level, volume.x, volume.y, volume.width,
+ volume.height, gl_format, gl_type, helper.image_data());
+ } else {
+ glCompressedTexSubImage2D(GL_TEXTURE_2D, level, volume.x, volume.y,
+ volume.width, volume.height, gl_internal_format,
+ helper.image_size(), helper.image_data());
+ }
+ return true;
+}
+
+bool Texture2DGL::GetData(const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ void *data) {
+ if (level >= levels())
+ return false;
+ MipLevelInfo mip_info;
+ MakeMipLevelInfo(&mip_info, format(), width_, height_, 1, level);
+ GetImageHelper helper;
+ if (!helper.Initialize(mip_info, volume, row_pitch, slice_pitch, size, data))
+ return false;
+
+ GLenum gl_internal_format = 0;
+ GLenum gl_format = 0;
+ GLenum gl_type = 0;
+ bool r = GetGLFormatType(format(), &gl_internal_format, &gl_format, &gl_type);
+ DCHECK(r);
+ glBindTexture(GL_TEXTURE_2D, gl_texture_);
+ if (gl_format) {
+ glGetTexImage(GL_TEXTURE_2D, level, gl_format, gl_type,
+ helper.image_data());
+ } else {
+ glGetCompressedTexImage(GL_TEXTURE_2D, level, helper.image_data());
+ }
+
+ helper.Finalize();
+ return true;
+}
+
+Texture3DGL *Texture3DGL::Create(unsigned int width,
+ unsigned int height,
+ unsigned int depth,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags) {
+ DCHECK_GT(width, 0);
+ DCHECK_GT(height, 0);
+ DCHECK_GT(depth, 0);
+ DCHECK_GT(levels, 0);
+ GLenum gl_internal_format = 0;
+ GLenum gl_format = 0;
+ GLenum gl_type = 0;
+ bool r = GetGLFormatType(format, &gl_internal_format, &gl_format, &gl_type);
+ DCHECK(r); // Was checked in the decoder.
+ GLuint gl_texture = 0;
+ glGenTextures(1, &gl_texture);
+ glBindTexture(GL_TEXTURE_3D, gl_texture);
+ glTexParameteri(GL_TEXTURE_3D, GL_TEXTURE_MAX_LEVEL, levels - 1);
+ // glCompressedTexImage3D does't accept NULL as a parameter, so we need
+ // to pass in some data.
+ scoped_array<unsigned char> buffer;
+ if (!gl_format) {
+ MipLevelInfo mip_info;
+ MakeMipLevelInfo(&mip_info, format, width, height, depth, 0);
+ unsigned int size = GetMipLevelSize(mip_info);
+ buffer.reset(new unsigned char[size]);
+ }
+ unsigned int mip_width = width;
+ unsigned int mip_height = height;
+ unsigned int mip_depth = depth;
+ for (unsigned int i = 0; i < levels; ++i) {
+ if (gl_format) {
+ glTexImage3D(GL_TEXTURE_3D, i, gl_internal_format, mip_width, mip_height,
+ mip_depth, 0, gl_format, gl_type, NULL);
+ } else {
+ MipLevelInfo mip_info;
+ MakeMipLevelInfo(&mip_info, format, width, height, depth, i);
+ unsigned int size = GetMipLevelSize(mip_info);
+ glCompressedTexImage3D(GL_TEXTURE_3D, i, gl_internal_format, mip_width,
+ mip_height, mip_depth, 0, size, buffer.get());
+ }
+ mip_width = std::max(1U, mip_width >> 1);
+ mip_height = std::max(1U, mip_height >> 1);
+ mip_depth = std::max(1U, mip_depth >> 1);
+ }
+ return new Texture3DGL(levels, format, flags, width, height, depth,
+ gl_texture);
+}
+
+bool Texture3DGL::SetData(const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ const void *data) {
+ if (level >= levels())
+ return false;
+ MipLevelInfo mip_info;
+ MakeMipLevelInfo(&mip_info, format(), width_, height_, depth_, level);
+ SetImageHelper helper;
+ if (!helper.Initialize(mip_info, volume, row_pitch, slice_pitch, size, data))
+ return false;
+
+ GLenum gl_internal_format = 0;
+ GLenum gl_format = 0;
+ GLenum gl_type = 0;
+ bool r = GetGLFormatType(format(), &gl_internal_format, &gl_format, &gl_type);
+ DCHECK(r);
+ glBindTexture(GL_TEXTURE_3D, gl_texture_);
+ if (gl_format) {
+ glTexSubImage3D(GL_TEXTURE_3D, level, volume.x, volume.y, volume.z,
+ volume.width, volume.height, volume.depth,
+ gl_format, gl_type, helper.image_data());
+ } else {
+ glCompressedTexSubImage3D(GL_TEXTURE_3D, level, volume.x, volume.y,
+ volume.z, volume.width, volume.height,
+ volume.depth, gl_internal_format,
+ helper.image_size(), helper.image_data());
+ }
+ return true;
+}
+
+bool Texture3DGL::GetData(const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ void *data) {
+ if (level >= levels())
+ return false;
+ MipLevelInfo mip_info;
+ MakeMipLevelInfo(&mip_info, format(), width_, height_, depth_, level);
+ GetImageHelper helper;
+ if (!helper.Initialize(mip_info, volume, row_pitch, slice_pitch, size, data))
+ return false;
+
+ GLenum gl_internal_format = 0;
+ GLenum gl_format = 0;
+ GLenum gl_type = 0;
+ bool r = GetGLFormatType(format(), &gl_internal_format, &gl_format, &gl_type);
+ DCHECK(r);
+ glBindTexture(GL_TEXTURE_3D, gl_texture_);
+ if (gl_format) {
+ glGetTexImage(GL_TEXTURE_3D, level, gl_format, gl_type,
+ helper.image_data());
+ } else {
+ glGetCompressedTexImage(GL_TEXTURE_3D, level, helper.image_data());
+ }
+
+ helper.Finalize();
+ return true;
+}
+
+TextureCubeGL *TextureCubeGL::Create(unsigned int side,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags) {
+ DCHECK_GT(side, 0);
+ DCHECK_GT(levels, 0);
+ GLenum gl_internal_format = 0;
+ GLenum gl_format = 0;
+ GLenum gl_type = 0;
+ bool r = GetGLFormatType(format, &gl_internal_format, &gl_format, &gl_type);
+ DCHECK(r); // Was checked in the decoder.
+ GLuint gl_texture = 0;
+ glGenTextures(1, &gl_texture);
+ glBindTexture(GL_TEXTURE_CUBE_MAP, gl_texture);
+ glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_MAX_LEVEL,
+ levels-1);
+ // glCompressedTexImage2D does't accept NULL as a parameter, so we need
+ // to pass in some data.
+ scoped_array<unsigned char> buffer;
+ if (!gl_format) {
+ MipLevelInfo mip_info;
+ MakeMipLevelInfo(&mip_info, format, side, side, 1, 0);
+ unsigned int size = GetMipLevelSize(mip_info);
+ buffer.reset(new unsigned char[size]);
+ }
+ unsigned int mip_side = side;
+ for (unsigned int i = 0; i < levels; ++i) {
+ if (gl_format) {
+ for (unsigned int face = 0; face < 6; ++face) {
+ glTexImage2D(GL_TEXTURE_CUBE_MAP_POSITIVE_X + face, i,
+ gl_internal_format, mip_side, mip_side,
+ 0, gl_format, gl_type, NULL);
+ }
+ } else {
+ MipLevelInfo mip_info;
+ MakeMipLevelInfo(&mip_info, format, side, side, 1, i);
+ unsigned int size = GetMipLevelSize(mip_info);
+ for (unsigned int face = 0; face < 6; ++face) {
+ glCompressedTexImage2D(GL_TEXTURE_CUBE_MAP_POSITIVE_X + face, i,
+ gl_internal_format, mip_side, mip_side, 0, size,
+ buffer.get());
+ }
+ }
+ mip_side = std::max(1U, mip_side >> 1);
+ }
+ return new TextureCubeGL(levels, format, flags, side, gl_texture);
+}
+
+// Check that GL_TEXTURE_CUBE_MAP_POSITIVE_X + face yields the correct GLenum.
+COMPILE_ASSERT(GL_TEXTURE_CUBE_MAP_POSITIVE_X + texture::FACE_POSITIVE_X ==
+ GL_TEXTURE_CUBE_MAP_POSITIVE_X, POSITIVE_X_ENUMS_DO_NOT_MATCH);
+COMPILE_ASSERT(GL_TEXTURE_CUBE_MAP_POSITIVE_X + texture::FACE_NEGATIVE_X ==
+ GL_TEXTURE_CUBE_MAP_NEGATIVE_X, NEGATIVE_X_ENUMS_DO_NOT_MATCH);
+COMPILE_ASSERT(GL_TEXTURE_CUBE_MAP_POSITIVE_X + texture::FACE_POSITIVE_Y ==
+ GL_TEXTURE_CUBE_MAP_POSITIVE_Y, POSITIVE_Y_ENUMS_DO_NOT_MATCH);
+COMPILE_ASSERT(GL_TEXTURE_CUBE_MAP_POSITIVE_X + texture::FACE_NEGATIVE_Y ==
+ GL_TEXTURE_CUBE_MAP_NEGATIVE_Y, NEGATIVE_Y_ENUMS_DO_NOT_MATCH);
+COMPILE_ASSERT(GL_TEXTURE_CUBE_MAP_POSITIVE_X + texture::FACE_POSITIVE_Z ==
+ GL_TEXTURE_CUBE_MAP_POSITIVE_Z, POSITIVE_Z_ENUMS_DO_NOT_MATCH);
+COMPILE_ASSERT(GL_TEXTURE_CUBE_MAP_POSITIVE_X + texture::FACE_NEGATIVE_Z ==
+ GL_TEXTURE_CUBE_MAP_NEGATIVE_Z, NEGATIVE_Z_ENUMS_DO_NOT_MATCH);
+
+bool TextureCubeGL::SetData(const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ const void *data) {
+ if (level >= levels())
+ return false;
+ MipLevelInfo mip_info;
+ MakeMipLevelInfo(&mip_info, format(), side_, side_, 1, level);
+ SetImageHelper helper;
+ if (!helper.Initialize(mip_info, volume, row_pitch, slice_pitch, size, data))
+ return false;
+
+ GLenum gl_internal_format = 0;
+ GLenum gl_format = 0;
+ GLenum gl_type = 0;
+ bool r = GetGLFormatType(format(), &gl_internal_format, &gl_format, &gl_type);
+ DCHECK(r);
+ glBindTexture(GL_TEXTURE_CUBE_MAP, gl_texture_);
+ GLenum face_target = GL_TEXTURE_CUBE_MAP_POSITIVE_X + face;
+ if (gl_format) {
+ glTexSubImage2D(face_target, level, volume.x, volume.y, volume.width,
+ volume.height, gl_format, gl_type, helper.image_data());
+ } else {
+ glCompressedTexSubImage2D(face_target, level, volume.x, volume.y,
+ volume.width, volume.height, gl_internal_format,
+ helper.image_size(), helper.image_data());
+ }
+ return true;
+}
+
+bool TextureCubeGL::GetData(const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ void *data) {
+ if (level >= levels())
+ return false;
+ MipLevelInfo mip_info;
+ MakeMipLevelInfo(&mip_info, format(), side_, side_, 1, level);
+ GetImageHelper helper;
+ if (!helper.Initialize(mip_info, volume, row_pitch, slice_pitch, size, data))
+ return false;
+
+ GLenum gl_internal_format = 0;
+ GLenum gl_format = 0;
+ GLenum gl_type = 0;
+ bool r = GetGLFormatType(format(), &gl_internal_format, &gl_format, &gl_type);
+ DCHECK(r);
+ glBindTexture(GL_TEXTURE_CUBE_MAP, gl_texture_);
+ GLenum face_target = GL_TEXTURE_CUBE_MAP_POSITIVE_X + face;
+ if (gl_format) {
+ glGetTexImage(face_target, level, gl_format, gl_type,
+ helper.image_data());
+ } else {
+ glGetCompressedTexImage(face_target, level, helper.image_data());
+ }
+
+ helper.Finalize();
+ return true;
+}
+
+// GAPIGL functions.
+
+// Destroys a texture resource.
+BufferSyncInterface::ParseError GAPIGL::DestroyTexture(ResourceID id) {
+ // Dirty effect, because this texture id may be used.
+ DirtyEffect();
+ return textures_.Destroy(id) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+// Creates a 2D texture resource.
+BufferSyncInterface::ParseError GAPIGL::CreateTexture2D(
+ ResourceID id,
+ unsigned int width,
+ unsigned int height,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags) {
+ Texture2DGL *texture = Texture2DGL::Create(width, height, levels, format,
+ flags);
+ if (!texture) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ // Dirty effect, because this texture id may be used.
+ DirtyEffect();
+ textures_.Assign(id, texture);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+// Creates a 3D texture resource.
+BufferSyncInterface::ParseError GAPIGL::CreateTexture3D(
+ ResourceID id,
+ unsigned int width,
+ unsigned int height,
+ unsigned int depth,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags) {
+ Texture3DGL *texture = Texture3DGL::Create(width, height, depth, levels,
+ format, flags);
+ if (!texture) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ // Dirty effect, because this texture id may be used.
+ DirtyEffect();
+ textures_.Assign(id, texture);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+// Creates a cube map texture resource.
+BufferSyncInterface::ParseError GAPIGL::CreateTextureCube(
+ ResourceID id,
+ unsigned int side,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags) {
+ TextureCubeGL *texture = TextureCubeGL::Create(side, levels, format, flags);
+ if (!texture) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ // Dirty effect, because this texture id may be used.
+ DirtyEffect();
+ textures_.Assign(id, texture);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+// Copies the data into a texture resource.
+BufferSyncInterface::ParseError GAPIGL::SetTextureData(
+ ResourceID id,
+ unsigned int x,
+ unsigned int y,
+ unsigned int z,
+ unsigned int width,
+ unsigned int height,
+ unsigned int depth,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ const void *data) {
+ TextureGL *texture = textures_.Get(id);
+ if (!texture)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ Volume volume = {x, y, z, width, height, depth};
+ // Dirty effect: SetData may need to call glBindTexture which will mess up the
+ // sampler parameters.
+ DirtyEffect();
+ return texture->SetData(volume, level, face, row_pitch, slice_pitch,
+ size, data) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+// Copies the data from a texture resource.
+BufferSyncInterface::ParseError GAPIGL::GetTextureData(
+ ResourceID id,
+ unsigned int x,
+ unsigned int y,
+ unsigned int z,
+ unsigned int width,
+ unsigned int height,
+ unsigned int depth,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ void *data) {
+ TextureGL *texture = textures_.Get(id);
+ if (!texture)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ Volume volume = {x, y, z, width, height, depth};
+ // Dirty effect: GetData may need to call glBindTexture which will mess up the
+ // sampler parameters.
+ DirtyEffect();
+ return texture->GetData(volume, level, face, row_pitch, slice_pitch,
+ size, data) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/cross/gl/texture_gl.h b/o3d/command_buffer/service/cross/gl/texture_gl.h
new file mode 100644
index 0000000..bc6d9c2
--- /dev/null
+++ b/o3d/command_buffer/service/cross/gl/texture_gl.h
@@ -0,0 +1,223 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file declares the TextureGL, Texture2DGL, Texture3DGL and TextureCubeGL
+// classes.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_CROSS_GL_TEXTURE_GL_H_
+#define O3D_COMMAND_BUFFER_SERVICE_CROSS_GL_TEXTURE_GL_H_
+
+#include "command_buffer/common/cross/gapi_interface.h"
+#include "command_buffer/service/cross/gl/gl_utils.h"
+#include "command_buffer/service/cross/resource.h"
+#include "command_buffer/service/cross/texture_utils.h"
+
+namespace o3d {
+namespace command_buffer {
+
+// The base class for a GL texture resource, providing access to the base GL
+// texture that can be assigned to an effect parameter or a sampler unit.
+class TextureGL : public Texture {
+ public:
+ TextureGL(texture::Type type,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags,
+ GLuint gl_texture)
+ : Texture(type, levels, format, flags),
+ gl_texture_(gl_texture) {}
+ virtual ~TextureGL();
+
+ // Gets the GL texture object.
+ GLuint gl_texture() const { return gl_texture_; }
+
+ // Sets data into a texture resource.
+ virtual bool SetData(const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ const void *data) = 0;
+
+ // Gets data from a texture resource.
+ virtual bool GetData(const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ void *data) = 0;
+
+ protected:
+ const GLuint gl_texture_;
+
+ private:
+ DISALLOW_COPY_AND_ASSIGN(TextureGL);
+};
+
+// A 2D texture resource for GL.
+class Texture2DGL : public TextureGL {
+ public:
+ Texture2DGL(unsigned int levels,
+ texture::Format format,
+ unsigned int flags,
+ unsigned int width,
+ unsigned int height,
+ GLuint gl_texture)
+ : TextureGL(texture::TEXTURE_2D, levels, format, flags, gl_texture),
+ width_(width),
+ height_(height) {}
+
+ // Creates a 2D texture resource.
+ static Texture2DGL *Create(unsigned int width,
+ unsigned int height,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags);
+
+ // Sets data into a 2D texture resource.
+ virtual bool SetData(const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ const void *data);
+
+ // Gets data from a 2D texture resource.
+ virtual bool GetData(const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ void *data);
+
+ private:
+ unsigned int width_;
+ unsigned int height_;
+ DISALLOW_COPY_AND_ASSIGN(Texture2DGL);
+};
+
+// A 3D texture resource for GL.
+class Texture3DGL : public TextureGL {
+ public:
+ Texture3DGL(unsigned int levels,
+ texture::Format format,
+ unsigned int flags,
+ unsigned int width,
+ unsigned int height,
+ unsigned int depth,
+ GLuint gl_texture)
+ : TextureGL(texture::TEXTURE_2D, levels, format, flags, gl_texture),
+ width_(width),
+ height_(height),
+ depth_(depth) {}
+
+ // Creates a 3D texture resource.
+ static Texture3DGL *Create(unsigned int width,
+ unsigned int height,
+ unsigned int depth,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags);
+
+ // Sets data into a 3D texture resource.
+ virtual bool SetData(const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ const void *data);
+
+ // Gets data from a 3D texture resource.
+ virtual bool GetData(const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ void *data);
+
+ private:
+ unsigned int width_;
+ unsigned int height_;
+ unsigned int depth_;
+ DISALLOW_COPY_AND_ASSIGN(Texture3DGL);
+};
+
+// A cube map texture resource for GL.
+class TextureCubeGL : public TextureGL {
+ public:
+ TextureCubeGL(unsigned int levels,
+ texture::Format format,
+ unsigned int flags,
+ unsigned int side,
+ GLuint gl_texture)
+ : TextureGL(texture::TEXTURE_CUBE, levels, format, flags, gl_texture),
+ side_(side) {}
+
+ // Creates a cube map texture resource.
+ static TextureCubeGL *Create(unsigned int side,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags);
+
+ // Sets data into a cube map texture resource.
+ virtual bool SetData(const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ const void *data);
+
+ // Gets data from a cube map texture resource.
+ virtual bool GetData(const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ void *data);
+
+ private:
+ unsigned int side_;
+ DISALLOW_COPY_AND_ASSIGN(TextureCubeGL);
+};
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_CROSS_GL_TEXTURE_GL_H_
diff --git a/o3d/command_buffer/service/cross/mocks.h b/o3d/command_buffer/service/cross/mocks.h
new file mode 100644
index 0000000..cb245c0
--- /dev/null
+++ b/o3d/command_buffer/service/cross/mocks.h
@@ -0,0 +1,149 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains definitions for mock objects, used for testing.
+
+// TODO: This file "manually" defines some mock objects. Using gMock
+// would be definitely preferable, unfortunately it doesn't work on Windows
+// yet.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_CROSS_MOCKS_H__
+#define O3D_COMMAND_BUFFER_SERVICE_CROSS_MOCKS_H__
+
+#include <vector>
+#include "gmock/gmock.h"
+#include "command_buffer/service/cross/cmd_parser.h"
+#include "command_buffer/service/cross/cmd_buffer_engine.h"
+
+namespace o3d {
+namespace command_buffer {
+
+// Mocks an AsyncAPIInterface, using GMock.
+class AsyncAPIMock : public AsyncAPIInterface {
+ public:
+ AsyncAPIMock() {
+ testing::DefaultValue<BufferSyncInterface::ParseError>::Set(
+ BufferSyncInterface::PARSE_NO_ERROR);
+ }
+
+ // Predicate that matches args passed to DoCommand, by looking at the values.
+ class IsArgs {
+ public:
+ IsArgs(unsigned int arg_count, CommandBufferEntry *args)
+ : arg_count_(arg_count),
+ args_(args) { }
+
+ bool operator() (CommandBufferEntry *args) const {
+ for (unsigned int i = 0; i < arg_count_; ++i) {
+ if (args[i].value_uint32 != args_[i].value_uint32) return false;
+ }
+ return true;
+ }
+
+ private:
+ unsigned int arg_count_;
+ CommandBufferEntry *args_;
+ };
+
+ MOCK_METHOD3(DoCommand, BufferSyncInterface::ParseError(
+ unsigned int command,
+ unsigned int arg_count,
+ CommandBufferEntry *args));
+
+ // Sets the engine, to forward SET_TOKEN commands to it.
+ void set_engine(CommandBufferEngine *engine) { engine_ = engine; }
+
+ // Forwards the SetToken commands to the engine.
+ void SetToken(unsigned int command,
+ unsigned int arg_count,
+ CommandBufferEntry *args) {
+ DCHECK(engine_);
+ DCHECK_EQ(1, command);
+ DCHECK_EQ(1, arg_count);
+ engine_->set_token(args[0].value_uint32);
+ }
+ private:
+ CommandBufferEngine *engine_;
+};
+
+class RPCProcessMock : public RPCProcessInterface {
+ public:
+ RPCProcessMock()
+ : would_have_blocked_(false),
+ message_count_(0) {
+ ON_CALL(*this, ProcessMessage()).WillByDefault(
+ testing::Invoke(this, &RPCProcessMock::DefaultProcessMessage));
+ ON_CALL(*this, HasMessage()).WillByDefault(
+ testing::Invoke(this, &RPCProcessMock::DefaultHasMessage));
+ }
+ MOCK_METHOD0(ProcessMessage, bool());
+ MOCK_METHOD0(HasMessage, bool());
+
+ void Reset() {
+ would_have_blocked_ = false;
+ message_count_ = 0;
+ }
+
+ bool DefaultProcessMessage() {
+ if (message_count_ > 0) {
+ --message_count_;
+ } else {
+ would_have_blocked_ = true;
+ }
+ return true;
+ }
+
+ bool DefaultHasMessage() {
+ return message_count_ > 0;
+ }
+
+ bool AddMessage() {
+ ++message_count_;
+ }
+
+ bool would_have_blocked() { return would_have_blocked_; }
+ void set_would_have_blocked(bool would_have_blocked) {
+ would_have_blocked_ = would_have_blocked;
+ }
+
+ unsigned int message_count() { return message_count_; }
+ void set_message_count(unsigned int count) { message_count_ = count; }
+ private:
+ bool would_have_blocked_;
+ unsigned int message_count_;
+};
+
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_CROSS_MOCKS_H__
diff --git a/o3d/command_buffer/service/cross/plugin.cc b/o3d/command_buffer/service/cross/plugin.cc
new file mode 100644
index 0000000..651713d
--- /dev/null
+++ b/o3d/command_buffer/service/cross/plugin.cc
@@ -0,0 +1,439 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the command renderer service (renderer) plug-in.
+// NOTE: this is only implemented on Windows currently.
+// TODO: other platforms.
+
+#include <npupp.h>
+#include <build/build_config.h>
+#ifdef OS_WIN
+#include <windows.h>
+#endif
+
+#include "command_buffer/common/cross/gapi_interface.h"
+#include "command_buffer/common/cross/rpc_imc.h"
+#include "command_buffer/service/cross/buffer_rpc.h"
+#include "command_buffer/service/cross/cmd_buffer_engine.h"
+#include "command_buffer/service/cross/gapi_decoder.h"
+#ifdef OS_WIN
+#include "command_buffer/service/win/d3d9/gapi_d3d9.h"
+#endif
+#include "native_client/service_runtime/nrd_xfer_lib/nrd_all_modules.h"
+#include "tools/idlglue/ng/static_glue/npapi/npn_api.h"
+
+namespace o3d {
+namespace command_buffer {
+
+// The Plugin class implements the plug-in instance. It derives from NPObject
+// to be the scriptable object as well.
+class Plugin : public NPObject {
+ public:
+ // Sets the window used by the plug-in.
+ NPError SetWindow(NPWindow *window);
+
+ // Gets the NPClass representing the NPAPI entrypoints to the object.
+ static NPClass *GetNPClass() {
+ return &class_;
+ }
+
+ private:
+ explicit Plugin(NPP npp);
+ ~Plugin();
+
+ // Creates the renderer using the IMC socket. This is called from Javascript
+ // using the create() function.
+ void Create(nacl::HtpHandle socket);
+
+ // Destroys the renderer. This is called from Javascript using the destroy()
+ // function.
+ void Destroy();
+
+ // NPAPI bindings.
+ static NPObject *Allocate(NPP npp, NPClass *npclass);
+ static void Deallocate(NPObject *object);
+ static bool HasMethod(NPObject *header, NPIdentifier name);
+ static bool Invoke(NPObject *header, NPIdentifier name,
+ const NPVariant *args, uint32_t argCount,
+ NPVariant *result);
+ static bool InvokeDefault(NPObject *header, const NPVariant *args,
+ uint32_t argCount, NPVariant *result);
+ static bool HasProperty(NPObject *header, NPIdentifier name);
+ static bool GetProperty(NPObject *header, NPIdentifier name,
+ NPVariant *variant);
+ static bool SetProperty(NPObject *header, NPIdentifier name,
+ const NPVariant *variant);
+ static bool Enumerate(NPObject *header, NPIdentifier **value,
+ uint32_t *count);
+
+#ifdef OS_WIN
+ static DWORD WINAPI ThreadMain(void *param) {
+ static_cast<Plugin *>(param)->DoThread();
+ return 0;
+ }
+#endif
+ // Executes the main renderer thread.
+ void DoThread();
+
+ NPP npp_;
+ static NPClass class_;
+ NPIdentifier create_id_;
+ NPIdentifier destroy_id_;
+ NPIdentifier handle_id_;
+
+#ifdef OS_WIN
+ HWND hwnd_;
+ HANDLE thread_;
+ DWORD thread_id_;
+#endif
+
+ nacl::HtpHandle handle_;
+ scoped_ptr<GAPIInterface> gapi_;
+};
+
+
+Plugin::Plugin(NPP npp)
+ : npp_(npp),
+#ifdef OS_WIN
+ hwnd_(NULL),
+ thread_(NULL),
+ thread_id_(0),
+#endif
+ handle_(nacl::kInvalidHtpHandle) {
+ const char *names[3] = {"create", "destroy", "handle"};
+ NPIdentifier ids[3];
+ NPN_GetStringIdentifiers(names, 3, ids);
+ create_id_ = ids[0];
+ destroy_id_ = ids[1];
+ handle_id_ = ids[2];
+}
+
+Plugin::~Plugin() {
+ if (gapi_.get()) Destroy();
+}
+
+NPError Plugin::SetWindow(NPWindow *window) {
+#ifdef OS_WIN
+ HWND hWnd = window ? static_cast<HWND>(window->window) : NULL;
+ hwnd_ = hWnd;
+ return NPERR_NO_ERROR;
+#endif // OS_WIN
+}
+
+// Creates the renderer. This spawns a thread that answers requests (the D3D
+// context is created in that other thread, so that we don't need to enable
+// multi-threading on it).
+void Plugin::Create(nacl::HtpHandle handle) {
+ if (gapi_.get()) return;
+ handle_ = handle;
+#ifdef OS_WIN
+ if (!hwnd_) return;
+ GAPID3D9 *gapi_d3d = new GAPID3D9;
+ gapi_d3d->set_hwnd(hwnd_);
+ gapi_.reset(gapi_d3d);
+ // TODO: use chrome/base threads.
+ thread_ = ::CreateThread(NULL, 0, ThreadMain, this, 0, &thread_id_);
+#endif
+}
+
+// Destroys the renderer. This terminates the renderer thread, and waits until
+// it is finished.
+void Plugin::Destroy() {
+ if (!gapi_.get()) return;
+#ifdef OS_WIN
+ ::PostThreadMessage(thread_id_, WM_USER, 0, 0);
+ ::WaitForSingleObject(thread_, INFINITE);
+ ::CloseHandle(thread_);
+#endif
+ gapi_.reset(NULL);
+}
+
+// Executes the main renderer thread: answers requests, executes commands.
+void Plugin::DoThread() {
+ scoped_ptr<GAPIDecoder> decoder(new GAPIDecoder(gapi_.get()));
+ scoped_ptr<CommandBufferEngine> engine(
+ new CommandBufferEngine(decoder.get()));
+ decoder->set_engine(engine.get());
+
+ IMCMessageProcessor processor(handle_, engine->rpc_impl());
+ engine->set_process_interface(&processor);
+ IMCSender sender(handle_);
+ engine->set_client_rpc(&sender);
+
+ gapi_->Initialize();
+ while (true) {
+ bool done = false;
+#ifdef OS_WIN
+ MSG msg;
+ while (::PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) {
+ ::TranslateMessage(&msg);
+ ::DispatchMessage(&msg);
+ if (msg.message == WM_USER) {
+ done = true;
+ break;
+ }
+ }
+#endif
+ if (done) break;
+ // NOTE: DoWork will block when there is nothing to do. This can be an
+ // issue at termination if the browser tries to kill the plug-in before the
+ // NaCl module, because then this thread won't terminate, and it will block
+ // the main (browser) thread. Workaround: kill the NaCl module (kill the
+ // sel_ldr window).
+ // TODO: Fix this. It needs select()/poll() or a timeout in the
+ // IMC library, and that doesn't exist currently. We could use non-blocking,
+ // e.g. HasWork() and sleep if there is nothing to do, but that would
+ // translate into unacceptable latencies - 10ms per call.
+ if (!engine->DoWork()) break;
+ }
+ gapi_->Destroy();
+}
+
+NPClass Plugin::class_ = {
+ NP_CLASS_STRUCT_VERSION,
+ Plugin::Allocate,
+ Plugin::Deallocate,
+ 0,
+ Plugin::HasMethod,
+ Plugin::Invoke,
+ 0,
+ Plugin::HasProperty,
+ Plugin::GetProperty,
+ Plugin::SetProperty,
+ 0,
+ Plugin::Enumerate,
+};
+
+NPObject *Plugin::Allocate(NPP npp, NPClass *npclass) {
+ return new Plugin(npp);
+}
+
+void Plugin::Deallocate(NPObject *object) {
+ delete static_cast<Plugin *>(object);
+}
+
+bool Plugin::HasMethod(NPObject *header, NPIdentifier name) {
+ Plugin *plugin = static_cast<Plugin *>(header);
+ // 2 methods supported: create(handle) and destroy().
+ return (name == plugin->create_id_ ||
+ name == plugin->destroy_id_);
+}
+
+bool Plugin::Invoke(NPObject *header, NPIdentifier name,
+ const NPVariant *args, uint32_t argCount,
+ NPVariant *result) {
+ Plugin *plugin = static_cast<Plugin *>(header);
+ VOID_TO_NPVARIANT(*result);
+ if (name == plugin->create_id_ && argCount == 1 &&
+ NPVARIANT_IS_OBJECT(args[0])) {
+ // create(handle) was called.
+ //
+ // Temporary ugly hack: the NPObject is a wrapper around a HtpHandle, but
+ // to get that handle we need to get the "handle" property on it which is a
+ // string that represents the address in memory of that HtpHandle.
+ NPObject *object = NPVARIANT_TO_OBJECT(args[0]);
+
+ NPVariant handle_prop;
+ bool result = NPN_GetProperty(plugin->npp_, object, plugin->handle_id_,
+ &handle_prop);
+ if (!result || !NPVARIANT_IS_STRING(handle_prop))
+ return false;
+ String handle_string(NPVARIANT_TO_STRING(handle_prop).utf8characters,
+ NPVARIANT_TO_STRING(handle_prop).utf8length);
+ intptr_t handle_value = strtol(handle_string.c_str(), NULL, 0);
+ nacl::HtpHandle handle = reinterpret_cast<nacl::HtpHandle>(handle_value);
+ plugin->Create(handle);
+ return true;
+ } else if (name == plugin->destroy_id_ && argCount == 0) {
+ // destroy() was called.
+ plugin->Destroy();
+ return true;
+ } else {
+ return false;
+ }
+}
+
+bool Plugin::InvokeDefault(NPObject *header, const NPVariant *args,
+ uint32_t argCount, NPVariant *result) {
+ return false;
+}
+
+bool Plugin::HasProperty(NPObject *header, NPIdentifier name) {
+ return false;
+}
+
+bool Plugin::GetProperty(NPObject *header, NPIdentifier name,
+ NPVariant *variant) {
+ return false;
+}
+
+bool Plugin::SetProperty(NPObject *header, NPIdentifier name,
+ const NPVariant *variant) {
+ return false;
+}
+
+bool Plugin::Enumerate(NPObject *header, NPIdentifier **value,
+ uint32_t *count) {
+ Plugin *plugin = static_cast<Plugin *>(header);
+ *count = 2;
+ NPIdentifier *ids = static_cast<NPIdentifier *>(
+ NPN_MemAlloc(*count * sizeof(NPIdentifier)));
+ ids[0] = plugin->create_id_;
+ ids[1] = plugin->destroy_id_;
+ *value = ids;
+ return true;
+}
+
+} // namespace command_buffer
+} // namespace o3d
+
+using o3d::command_buffer::Plugin;
+
+extern "C" {
+// NPAPI entry points.
+
+char *NP_GetMIMEDescription(void) {
+ return "application/vnd.cmdbuf::CommandBuffer MIME";
+}
+
+NPError OSCALL NP_Initialize(NPNetscapeFuncs *browserFuncs) {
+ NPError retval = InitializeNPNApi(browserFuncs);
+ if (retval != NPERR_NO_ERROR) return retval;
+ return NPERR_NO_ERROR;
+}
+
+NPError OSCALL NP_GetEntryPoints(NPPluginFuncs *pluginFuncs) {
+ pluginFuncs->version = 11;
+ pluginFuncs->size = sizeof(pluginFuncs);
+ pluginFuncs->newp = NPP_New;
+ pluginFuncs->destroy = NPP_Destroy;
+ pluginFuncs->setwindow = NPP_SetWindow;
+ pluginFuncs->newstream = NPP_NewStream;
+ pluginFuncs->destroystream = NPP_DestroyStream;
+ pluginFuncs->asfile = NPP_StreamAsFile;
+ pluginFuncs->writeready = NPP_WriteReady;
+ pluginFuncs->write = NPP_Write;
+ pluginFuncs->print = NPP_Print;
+ pluginFuncs->event = NPP_HandleEvent;
+ pluginFuncs->urlnotify = NPP_URLNotify;
+ pluginFuncs->getvalue = NPP_GetValue;
+ pluginFuncs->setvalue = NPP_SetValue;
+
+ return NPERR_NO_ERROR;
+}
+
+NPError OSCALL NP_Shutdown(void) {
+ return NPERR_NO_ERROR;
+}
+
+// Creates a plugin instance.
+NPError NPP_New(NPMIMEType pluginType, NPP instance, uint16 mode, int16 argc,
+ char *argn[], char *argv[], NPSavedData *saved) {
+ NPObject *object = NPN_CreateObject(instance, Plugin::GetNPClass());
+ if (object == NULL) {
+ return NPERR_OUT_OF_MEMORY_ERROR;
+ }
+ instance->pdata = object;
+ return NPERR_NO_ERROR;
+}
+
+// Destroys a plugin instance.
+NPError NPP_Destroy(NPP instance, NPSavedData **save) {
+ Plugin *obj = static_cast<Plugin*>(instance->pdata);
+ if (obj) {
+ obj->SetWindow(NULL);
+ NPN_ReleaseObject(obj);
+ instance->pdata = NULL;
+ }
+
+ return NPERR_NO_ERROR;
+}
+
+// Sets the window used by the plugin instance.
+NPError NPP_SetWindow(NPP instance, NPWindow *window) {
+ Plugin *obj = static_cast<Plugin*>(instance->pdata);
+ obj->SetWindow(window);
+ return NPERR_NO_ERROR;
+}
+
+// Gets the scriptable object for the plug-in instance.
+NPError NPP_GetValue(NPP instance, NPPVariable variable, void *value) {
+ switch (variable) {
+ case NPPVpluginScriptableNPObject: {
+ void **v = static_cast<void **>(value);
+ Plugin *obj = static_cast<Plugin*>(instance->pdata);
+ NPN_RetainObject(obj);
+ *v = obj;
+ return NPERR_NO_ERROR;
+ }
+ default:
+ break;
+ }
+ return NPERR_GENERIC_ERROR;
+}
+
+NPError NPP_NewStream(NPP instance, NPMIMEType type, NPStream *stream,
+ NPBool seekable, uint16 *stype) {
+ return NPERR_NO_ERROR;
+}
+
+NPError NPP_DestroyStream(NPP instance, NPStream *stream, NPReason reason) {
+ return NPERR_NO_ERROR;
+}
+
+int32 NPP_WriteReady(NPP instance, NPStream *stream) {
+ return 4096;
+}
+
+int32 NPP_Write(NPP instance, NPStream *stream, int32 offset, int32 len,
+ void *buffer) {
+ return len;
+}
+
+void NPP_StreamAsFile(NPP instance, NPStream *stream, const char *fname) {
+}
+
+void NPP_Print(NPP instance, NPPrint *platformPrint) {
+}
+
+int16 NPP_HandleEvent(NPP instance, void *event) {
+ return 0;
+}
+
+void NPP_URLNotify(NPP instance, const char *url, NPReason reason,
+ void *notifyData) {
+}
+
+NPError NPP_SetValue(NPP instance, NPNVariable variable, void *value) {
+ return NPERR_GENERIC_ERROR;
+}
+} // extern "C"
diff --git a/o3d/command_buffer/service/cross/resource.cc b/o3d/command_buffer/service/cross/resource.cc
new file mode 100644
index 0000000..00a9f86
--- /dev/null
+++ b/o3d/command_buffer/service/cross/resource.cc
@@ -0,0 +1,102 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the implementation of ResourceMapBase.
+
+#include "command_buffer/service/cross/resource.h"
+
+namespace o3d {
+namespace command_buffer {
+
+// Assigns a resource to a resource ID, by setting it at the right location
+// into the list, resizing the list if necessary, and destroying an existing
+// resource if one existed already.
+void ResourceMapBase::Assign(ResourceID id, Resource *resource) {
+ if (id >= resources_.size()) {
+ resources_.resize(id + 1, NULL);
+ } else {
+ Resource *&entry = resources_[id];
+ if (entry) {
+ delete entry;
+ entry = NULL;
+ }
+ }
+ DCHECK(resources_[id] == NULL);
+ resources_[id] = resource;
+}
+
+// Destroys a resource contained in the map, setting its entry to NULL. If
+// necessary, this will trim the list.
+bool ResourceMapBase::Destroy(ResourceID id) {
+ if (id >= resources_.size()) {
+ return false;
+ }
+ Resource *&entry = resources_[id];
+ if (entry) {
+ delete entry;
+ entry = NULL;
+
+ // Removing the last element, we can trim the list.
+ // TODO: this may not be optimal to do every time. Investigate if it
+ // becomes an issue, and add a threshold before we resize.
+ if (id == resources_.size() - 1) {
+ size_t last_valid = resources_.max_size();
+ for (unsigned int i = id; i < resources_.size(); --i) {
+ if (resources_[i]) {
+ last_valid = i;
+ break;
+ }
+ }
+ if (last_valid == resources_.max_size()) {
+ resources_.clear();
+ } else {
+ resources_.resize(last_valid + 1);
+ }
+ }
+ return true;
+ }
+ return false;
+}
+
+// Goes over all non-NULL entries in the list, destroying them, then clears the
+// list.
+void ResourceMapBase::DestroyAllResources() {
+ for (Container::iterator i = resources_.begin(); i != resources_.end(); ++i) {
+ if (*i) {
+ delete *i;
+ }
+ }
+ resources_.clear();
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/cross/resource.h b/o3d/command_buffer/service/cross/resource.h
new file mode 100644
index 0000000..5c6e6ed
--- /dev/null
+++ b/o3d/command_buffer/service/cross/resource.h
@@ -0,0 +1,249 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the definition for resource classes and the resource map.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_CROSS_RESOURCE_H__
+#define O3D_COMMAND_BUFFER_SERVICE_CROSS_RESOURCE_H__
+
+#include <vector>
+#include "base/scoped_ptr.h"
+#include "core/cross/types.h"
+#include "command_buffer/common/cross/resource.h"
+
+namespace o3d {
+namespace command_buffer {
+
+// Base class for resources, just providing a common Destroy function.
+class Resource {
+ public:
+ Resource() {}
+ virtual ~Resource() {}
+ private:
+ DISALLOW_COPY_AND_ASSIGN(Resource);
+};
+
+// VertexBuffer class, representing a vertex buffer resource.
+class VertexBuffer: public Resource {
+ public:
+ VertexBuffer(unsigned int size, unsigned int flags)
+ : size_(size),
+ flags_(flags) {}
+ virtual ~VertexBuffer() {}
+
+ // Returns the vertex buffer flags.
+ unsigned int flags() const { return flags_; }
+ // Sets the vertex buffer flags.
+ void set_flags(unsigned int flags) { flags_ = flags; }
+ // Returns the vertex buffer size.
+ unsigned int size() const { return size_; }
+ // Sets the vertex buffer size.
+ void set_size(unsigned int size) { size_ = size; }
+ protected:
+ unsigned int size_;
+ unsigned int flags_;
+ private:
+ DISALLOW_COPY_AND_ASSIGN(VertexBuffer);
+};
+
+// IndexBuffer class, representing an index buffer resource.
+class IndexBuffer: public Resource {
+ public:
+ IndexBuffer(unsigned int size, unsigned int flags)
+ : size_(size),
+ flags_(flags) {}
+ virtual ~IndexBuffer() {}
+
+ // Returns the index buffer flags.
+ unsigned int flags() const { return flags_; }
+ // Sets the index buffer flags.
+ void set_flags(unsigned int flags) { flags_ = flags; }
+ // Returns the index buffer size.
+ unsigned int size() const { return size_; }
+ // Sets the index buffer size.
+ void set_size(unsigned int size) { size_ = size; }
+ protected:
+ unsigned int size_;
+ unsigned int flags_;
+ private:
+ DISALLOW_COPY_AND_ASSIGN(IndexBuffer);
+};
+
+// VertexStruct class, representing a vertex struct resource.
+class VertexStruct: public Resource {
+ public:
+ // The representation of an input data stream.
+ struct Element {
+ ResourceID vertex_buffer;
+ unsigned int offset;
+ unsigned int stride;
+ vertex_struct::Type type;
+ vertex_struct::Semantic semantic;
+ unsigned int semantic_index;
+ };
+
+ explicit VertexStruct(unsigned int count)
+ : count_(count),
+ elements_(new Element[count]) {
+ memset(elements_.get(), 0, count * sizeof(Element)); // NOLINT
+ }
+
+ // Returns the number of inputs in this struct.
+ unsigned int count() const { return count_; }
+ // Returns an element by index.
+ Element &GetElement(unsigned int i) {
+ DCHECK_GT(count_, i);
+ return elements_[i];
+ }
+ protected:
+ unsigned int count_;
+ scoped_array<Element> elements_;
+ private:
+ DISALLOW_COPY_AND_ASSIGN(VertexStruct);
+};
+
+// Effect class, representing an effect.
+class Effect: public Resource {
+ public:
+ Effect() {}
+ private:
+ DISALLOW_COPY_AND_ASSIGN(Effect);
+};
+
+// EffectParam class, representing an effect parameter.
+class EffectParam: public Resource {
+ public:
+ explicit EffectParam(effect_param::DataType data_type)
+ : data_type_(data_type) {
+ }
+
+ // Gets the data type of this parameter.
+ effect_param::DataType data_type() const { return data_type_; }
+ private:
+ effect_param::DataType data_type_;
+ DISALLOW_COPY_AND_ASSIGN(EffectParam);
+};
+
+// Texture class, representing a texture resource.
+class Texture: public Resource {
+ public:
+ Texture(texture::Type type,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags)
+ : type_(type),
+ levels_(levels),
+ format_(format),
+ flags_(flags) {}
+ virtual ~Texture() {}
+
+ // Returns the type of the texture.
+ texture::Type type() const { return type_; }
+ // Returns the texture flags.
+ unsigned int flags() const { return flags_; }
+ // Returns the texture format.
+ texture::Format format() const { return format_; }
+ // Returns the number of mipmap levels in the texture.
+ unsigned int levels() const { return levels_; }
+ private:
+ texture::Type type_;
+ unsigned int levels_;
+ texture::Format format_;
+ unsigned int flags_;
+ DISALLOW_COPY_AND_ASSIGN(Texture);
+};
+
+// Texture class, representing a sampler resource.
+class Sampler: public Resource {
+ public:
+ Sampler() {}
+ private:
+ DISALLOW_COPY_AND_ASSIGN(Sampler);
+};
+
+// Base of ResourceMap. Contains most of the implementation of ResourceMap, to
+// avoid template bloat.
+class ResourceMapBase {
+ public:
+ ResourceMapBase() : resources_() {}
+ ~ResourceMapBase() {}
+
+ // Assigns a resource to a resource ID. Assigning a resource to an ID that
+ // already has an existing resource will destroy that existing resource. The
+ // map takes ownership of the resource.
+ void Assign(ResourceID id, Resource* resource);
+ // Destroys a resource.
+ bool Destroy(ResourceID id);
+ // Destroy all resources.
+ void DestroyAllResources();
+ // Gets a resource by ID.
+ Resource *Get(ResourceID id) {
+ return (id < resources_.size()) ? resources_[id] : NULL;
+ }
+ private:
+ typedef std::vector<Resource *> Container;
+ Container resources_;
+};
+
+// Resource Map class, allowing resource ID <-> Resource association. This is a
+// dense map, optimized for retrieval (O(1)).
+template<class T> class ResourceMap {
+ public:
+ ResourceMap() : container_() {}
+ ~ResourceMap() {}
+
+ // Assigns a resource to a resource ID. Assigning a resource to an ID that
+ // already has an existing resource will destroy that existing resource. The
+ // map takes ownership of the resource.
+ void Assign(ResourceID id, T* resource) {
+ container_.Assign(id, resource);
+ }
+ // Destroys a resource.
+ bool Destroy(ResourceID id) {
+ return container_.Destroy(id);
+ }
+ // Destroy all resources.
+ void DestroyAllResources() {
+ return container_.DestroyAllResources();
+ }
+ // Gets a resource by ID.
+ T *Get(ResourceID id) {
+ return down_cast<T*>(container_.Get(id));
+ }
+ private:
+ ResourceMapBase container_;
+};
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_CROSS_RESOURCE_H__
diff --git a/o3d/command_buffer/service/cross/resource_test.cc b/o3d/command_buffer/service/cross/resource_test.cc
new file mode 100644
index 0000000..e8fedb0
--- /dev/null
+++ b/o3d/command_buffer/service/cross/resource_test.cc
@@ -0,0 +1,128 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// Tests for the ResourceMap.
+
+#include "tests/common/win/testing_common.h"
+#include "command_buffer/service/cross/resource.h"
+
+namespace o3d {
+namespace command_buffer {
+
+// Mock resource implementation that checks for leaks.
+class ResourceMock : public Resource {
+ public:
+ ResourceMock() : Resource() {
+ ++instance_count_;
+ }
+ virtual ~ResourceMock() {
+ --instance_count_;
+ }
+
+ // Returns the instance count. The instance count is increased in the
+ // constructor and decreased in the destructor, to track leaks. The reason is
+ // that we can't mock the destructor, though we want to make sure the mock is
+ // destroyed.
+ static int instance_count() { return instance_count_; }
+ private:
+ static int instance_count_;
+ DISALLOW_COPY_AND_ASSIGN(ResourceMock);
+};
+int ResourceMock::instance_count_ = 0;
+
+// Test fixture for ResourceMap test. Creates a ResourceMap using a mock
+// Resource, and checks for ResourceMock leaks.
+class ResourceMapTest : public testing::Test {
+ protected:
+ typedef ResourceMap<ResourceMock> Map;
+ virtual void SetUp() {
+ instance_count_ = ResourceMock::instance_count();
+ map_.reset(new Map());
+ }
+ virtual void TearDown() {
+ CheckLeaks();
+ }
+
+ // Makes sure we didn't leak any ResourceMock object.
+ void CheckLeaks() {
+ EXPECT_EQ(instance_count_, ResourceMock::instance_count());
+ }
+
+ Map *map() const { return map_.get(); }
+ private:
+ int instance_count_;
+ scoped_ptr<Map> map_;
+};
+
+TEST_F(ResourceMapTest, TestMap) {
+ // check that initial mapping is empty.
+ EXPECT_EQ(NULL, map()->Get(0));
+ EXPECT_EQ(NULL, map()->Get(1));
+ EXPECT_EQ(NULL, map()->Get(392));
+
+ // create a new resource, assign it to an ID.
+ ResourceMock *resource = new ResourceMock();
+ map()->Assign(123, resource);
+ EXPECT_EQ(resource, map()->Get(123));
+
+ // Destroy the resource, making sure the object is deleted.
+ EXPECT_EQ(true, map()->Destroy(123));
+ EXPECT_EQ(false, map()->Destroy(123)); // destroying again should fail.
+ resource = NULL;
+ CheckLeaks();
+
+ // create a new resource, add it to the map, and make sure it gets deleted
+ // when we assign a new resource to that ID.
+ resource = new ResourceMock();
+ map()->Assign(1, resource);
+ resource = new ResourceMock();
+ map()->Assign(1, resource);
+ EXPECT_EQ(resource, map()->Get(1)); // check that we have the new resource.
+ EXPECT_EQ(true, map()->Destroy(1));
+ CheckLeaks();
+
+ // Adds 3 resources, then call DestroyAllResources().
+ resource = new ResourceMock();
+ map()->Assign(1, resource);
+ resource = new ResourceMock();
+ map()->Assign(2, resource);
+ resource = new ResourceMock();
+ map()->Assign(3, resource);
+ map()->DestroyAllResources();
+ EXPECT_EQ(NULL, map()->Get(1));
+ EXPECT_EQ(NULL, map()->Get(2));
+ EXPECT_EQ(NULL, map()->Get(3));
+ CheckLeaks();
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/cross/texture_utils.cc b/o3d/command_buffer/service/cross/texture_utils.cc
new file mode 100644
index 0000000..926f6db
--- /dev/null
+++ b/o3d/command_buffer/service/cross/texture_utils.cc
@@ -0,0 +1,104 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the implementation of some utilities for textures.
+
+#include <stdlib.h>
+#include "command_buffer/service/cross/texture_utils.h"
+
+namespace o3d {
+namespace command_buffer {
+
+void MakeTransferInfo(TransferInfo *transfer_info,
+ const MipLevelInfo &mip_level,
+ const Volume &volume,
+ unsigned int row_pitch,
+ unsigned int slice_pitch) {
+ transfer_info->row_pitch = row_pitch;
+ transfer_info->slice_pitch = slice_pitch;
+ transfer_info->row_size =
+ volume.width / mip_level.block_size_x * mip_level.block_bpp;
+ transfer_info->slice_size = transfer_info->row_size +
+ (volume.height / mip_level.block_size_y - 1) * row_pitch;
+ transfer_info->total_size = transfer_info->slice_size +
+ (volume.depth - 1) * slice_pitch;
+ transfer_info->packed = (transfer_info->row_size == row_pitch) &&
+ (volume.depth == 1 || transfer_info->slice_size == slice_pitch);
+}
+
+void MakePackedTransferInfo(TransferInfo *transfer_info,
+ const MipLevelInfo &mip_level,
+ const Volume &volume) {
+ transfer_info->row_size =
+ volume.width / mip_level.block_size_x * mip_level.block_bpp;
+ transfer_info->row_pitch = transfer_info->row_size;
+ transfer_info->slice_size =
+ volume.height / mip_level.block_size_y * transfer_info->row_pitch;
+ transfer_info->slice_pitch = transfer_info->slice_size;
+ transfer_info->total_size = volume.depth * transfer_info->slice_pitch;
+ transfer_info->packed = true;
+}
+
+// Transfers a volume of texels.
+void TransferVolume(const Volume &volume,
+ const MipLevelInfo &mip_level,
+ const TransferInfo &dst_transfer_info,
+ void *dst_data,
+ const TransferInfo &src_transfer_info,
+ const void *src_data) {
+ DCHECK_EQ(src_transfer_info.row_size, dst_transfer_info.row_size);
+ if (src_transfer_info.packed && dst_transfer_info.packed) {
+ // fast path
+ DCHECK_EQ(src_transfer_info.total_size, dst_transfer_info.total_size);
+ DCHECK_EQ(src_transfer_info.row_pitch, dst_transfer_info.row_pitch);
+ DCHECK_EQ(src_transfer_info.slice_pitch, dst_transfer_info.slice_pitch);
+ memcpy(dst_data, src_data, src_transfer_info.total_size);
+ } else {
+ const char *src = static_cast<const char *>(src_data);
+ char *dst = static_cast<char *>(dst_data);
+ for (unsigned int slice = 0; slice < volume.depth; ++slice) {
+ const char *row_src = src;
+ char *row_dst = dst;
+ for (unsigned int row = 0; row < volume.height;
+ row += mip_level.block_size_y) {
+ memcpy(row_dst, row_src, src_transfer_info.row_size);
+ row_src += src_transfer_info.row_pitch;
+ row_dst += dst_transfer_info.row_pitch;
+ }
+ src += src_transfer_info.slice_pitch;
+ dst += dst_transfer_info.slice_pitch;
+ }
+ }
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/cross/texture_utils.h b/o3d/command_buffer/service/cross/texture_utils.h
new file mode 100644
index 0000000..aa186b5
--- /dev/null
+++ b/o3d/command_buffer/service/cross/texture_utils.h
@@ -0,0 +1,158 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file declares some utilities for textures, in particular to deal with
+// in-memory texture data and layout.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_CROSS_TEXTURE_UTILS_H_
+#define O3D_COMMAND_BUFFER_SERVICE_CROSS_TEXTURE_UTILS_H_
+
+#include "command_buffer/common/cross/logging.h"
+#include "command_buffer/common/cross/resource.h"
+
+namespace o3d {
+namespace command_buffer {
+
+// Structure describing a volume of pixels.
+struct Volume {
+ unsigned int x;
+ unsigned int y;
+ unsigned int z;
+ unsigned int width;
+ unsigned int height;
+ unsigned int depth;
+};
+
+// Structure describing the dimensions and structure of a mip level.
+struct MipLevelInfo {
+ unsigned int block_bpp;
+ unsigned int block_size_x;
+ unsigned int block_size_y;
+ unsigned int width;
+ unsigned int height;
+ unsigned int depth;
+};
+
+// Structure describing a memory layout for transfers.
+struct TransferInfo {
+ unsigned int row_size; // size in bytes of a row of blocks.
+ unsigned int row_pitch; // number of bytes between 2 successive rows.
+ unsigned int slice_size; // size in bytes of a slice of data.
+ unsigned int slice_pitch; // number of bytes between 2 successive slices.
+ unsigned int total_size; // total size of the data.
+ bool packed; // indicates whether the data is tightly packed.
+};
+
+// Round a value up, so that it is divisible by the block size.
+static inline unsigned int RoundToBlockSize(unsigned int base,
+ unsigned int block) {
+ DCHECK_GT(base, 0);
+ DCHECK_GT(block, 0);
+ return block + base - 1 - (base - 1) % block;
+}
+
+// Fills a MipLevelInfo structure from the base texture dimensions.
+static inline void MakeMipLevelInfo(MipLevelInfo *mip_info,
+ texture::Format format,
+ unsigned int base_width,
+ unsigned int base_height,
+ unsigned int base_depth,
+ unsigned int level) {
+ mip_info->block_bpp = texture::GetBytesPerBlock(format);
+ mip_info->block_size_x = texture::GetBlockSizeX(format);
+ mip_info->block_size_y = texture::GetBlockSizeY(format);
+ mip_info->width = RoundToBlockSize(
+ texture::GetMipMapDimension(base_width, level), mip_info->block_size_x);
+ mip_info->height = RoundToBlockSize(
+ texture::GetMipMapDimension(base_height, level), mip_info->block_size_y);
+ mip_info->depth = texture::GetMipMapDimension(base_depth, level);
+}
+
+// Gets the size in bytes of a mip level.
+static inline unsigned int GetMipLevelSize(const MipLevelInfo &mip_info) {
+ return mip_info.block_bpp * mip_info.width / mip_info.block_size_x *
+ mip_info.height / mip_info.block_size_y * mip_info.depth;
+}
+
+// Checks that [x .. x+width] is contained in [0 .. mip_width], and that both x
+// and width are divisible by block_size, and that width is positive.
+static inline bool CheckDimension(unsigned int x,
+ unsigned int width,
+ unsigned int mip_width,
+ unsigned int block_size) {
+ return x < mip_width && x+width <= mip_width && x % block_size == 0 &&
+ width % block_size == 0 && width > 0;
+}
+
+// Checks that given volume fits into a mip level.
+static inline bool CheckVolume(const MipLevelInfo &mip_info,
+ const Volume &volume) {
+ return CheckDimension(volume.x, volume.width, mip_info.width,
+ mip_info.block_size_x) &&
+ CheckDimension(volume.y, volume.height, mip_info.height,
+ mip_info.block_size_y) &&
+ CheckDimension(volume.z, volume.depth, mip_info.depth, 1);
+}
+
+// Checks whether a volume fully maps a mip level.
+static inline bool IsFullVolume(const MipLevelInfo &mip_info,
+ const Volume &volume) {
+ return (volume.x == 0) && (volume.y == 0) && (volume.z == 0) &&
+ (volume.width == mip_info.width) &&
+ (volume.height == mip_info.height) &&
+ (volume.depth == mip_info.depth);
+}
+
+// Makes a transfer info from a mip level, a volume and row/slice pitches.
+void MakeTransferInfo(TransferInfo *transfer_info,
+ const MipLevelInfo &mip_level,
+ const Volume &volume,
+ unsigned int row_pitch,
+ unsigned int slice_pitch);
+
+// Makes a transfer info from a mip level and a volume, considering packed data.
+void MakePackedTransferInfo(TransferInfo *transfer_info,
+ const MipLevelInfo &mip_level,
+ const Volume &volume);
+
+// Transfers a volume of texels.
+void TransferVolume(const Volume &volume,
+ const MipLevelInfo &mip_level,
+ const TransferInfo &dst_transfer_info,
+ void *dst_data,
+ const TransferInfo &src_transfer_info,
+ const void *src_data);
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_CROSS_TEXTURE_UTILS_H_
diff --git a/o3d/command_buffer/service/linux/big_test_main.cc b/o3d/command_buffer/service/linux/big_test_main.cc
new file mode 100644
index 0000000..21ffecd
--- /dev/null
+++ b/o3d/command_buffer/service/linux/big_test_main.cc
@@ -0,0 +1,125 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the entry to the big test program, for linux.
+
+#include "command_buffer/service/cross/big_test_helpers.h"
+#include "command_buffer/service/cross/gl/gapi_gl.h"
+#include "command_buffer/service/linux/x_utils.h"
+
+namespace o3d {
+namespace command_buffer {
+
+String *g_program_path = NULL;
+GAPIInterface *g_gapi = NULL;
+
+bool ProcessSystemMessages() {
+ return true;
+}
+
+} // namespace command_buffer
+} // namespace o3d
+
+using o3d::String;
+using o3d::command_buffer::g_program_path;
+using o3d::command_buffer::g_gapi;
+using o3d::command_buffer::GAPIGL;
+using o3d::command_buffer::XWindowWrapper;
+
+
+// Creates a GL-compatible window of specified dimensions.
+Window CreateWindow(Display *display, unsigned int width, unsigned int height) {
+ int attribs[] = {
+ GLX_RGBA,
+ GLX_DOUBLEBUFFER,
+ GLX_RED_SIZE, 1,
+ GLX_GREEN_SIZE, 1,
+ GLX_BLUE_SIZE, 1,
+ None
+ };
+ XVisualInfo *visualInfo = glXChooseVisual(display,
+ DefaultScreen(display),
+ attribs);
+ Window root_window = RootWindow(display, visualInfo->screen);
+ Colormap colorMap = XCreateColormap(display, root_window, visualInfo->visual,
+ AllocNone);
+
+ XSetWindowAttributes windowAttributes;
+ windowAttributes.colormap = colorMap;
+ windowAttributes.border_pixel = 0;
+ windowAttributes.event_mask = StructureNotifyMask;
+ Window window = XCreateWindow(display, root_window,
+ 0, 0, width, height, 0, visualInfo->depth,
+ InputOutput, visualInfo->visual,
+ CWBorderPixel|CWColormap|CWEventMask,
+ &windowAttributes);
+ if (!window) return 0;
+ XMapWindow(display, window);
+ XSync(display, True);
+ return window;
+}
+
+// Creates a window, initializes the GAPI instance.
+int main(int argc, char *argv[]) {
+ String program_path = argv[0];
+
+ // Remove all characters starting with last '/'.
+ size_t backslash_pos = program_path.rfind('/');
+ if (backslash_pos != String::npos) {
+ program_path.erase(backslash_pos);
+ }
+ g_program_path = &program_path;
+
+ GAPIGL gl_gapi;
+ g_gapi = &gl_gapi;
+
+ Display *display = XOpenDisplay(0);
+ if (!display) {
+ LOG(FATAL) << "Could not open the display.";
+ return 1;
+ }
+
+ Window window = CreateWindow(display, 300, 300);
+ if (!window) {
+ LOG(FATAL) << "Could not create a window.";
+ return 1;
+ }
+
+ XWindowWrapper wrapper(display, window);
+ gl_gapi.set_window_wrapper(&wrapper);
+
+ int ret = big_test_main(argc, argv);
+
+ g_gapi = NULL;
+ g_program_path = NULL;
+ return ret;
+}
diff --git a/o3d/command_buffer/service/linux/x_utils.cc b/o3d/command_buffer/service/linux/x_utils.cc
new file mode 100644
index 0000000..3ee0e7d
--- /dev/null
+++ b/o3d/command_buffer/service/linux/x_utils.cc
@@ -0,0 +1,93 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This class implements the XWindowWrapper class.
+
+#include "command_buffer/common/cross/logging.h"
+#include "command_buffer/service/linux/x_utils.h"
+
+namespace o3d {
+namespace command_buffer {
+
+bool XWindowWrapper::Initialize() {
+ XWindowAttributes attributes;
+ XGetWindowAttributes(display_, window_, &attributes);
+ XVisualInfo visual_info_template;
+ visual_info_template.visualid = XVisualIDFromVisual(attributes.visual);
+ int visual_info_count = 0;
+ XVisualInfo *visual_info_list = XGetVisualInfo(display_, VisualIDMask,
+ &visual_info_template,
+ &visual_info_count);
+ DCHECK(visual_info_list);
+ DCHECK_GT(visual_info_count, 0);
+ context_ = 0;
+ for (int i = 0; i < visual_info_count; ++i) {
+ context_ = glXCreateContext(display_, visual_info_list + i, 0,
+ True);
+ if (context_) break;
+ }
+ XFree(visual_info_list);
+ if (!context_) {
+ DLOG(ERROR) << "Couldn't create GL context.";
+ return false;
+ }
+ return true;
+}
+
+bool XWindowWrapper::MakeCurrent() {
+ if (glXMakeCurrent(display_, window_, context_) != True) {
+ glXDestroyContext(display_, context_);
+ context_ = 0;
+ DLOG(ERROR) << "Couldn't make context current.";
+ return false;
+ }
+ return true;
+}
+
+void XWindowWrapper::Destroy() {
+ Bool result = glXMakeCurrent(display_, 0, 0);
+ // glXMakeCurrent isn't supposed to fail when unsetting the context, unless
+ // we have pending draws on an invalid window - which shouldn't be the case
+ // here.
+ DCHECK(result);
+ if (context_) {
+ glXDestroyContext(display_, context_);
+ context_ = 0;
+ }
+}
+
+void XWindowWrapper::SwapBuffers() {
+ glXSwapBuffers(display_, window_);
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/linux/x_utils.h b/o3d/command_buffer/service/linux/x_utils.h
new file mode 100644
index 0000000..4cf618e
--- /dev/null
+++ b/o3d/command_buffer/service/linux/x_utils.h
@@ -0,0 +1,78 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file declares the XWindowWrapper class.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_LINUX_X_UTILS_H_
+#define O3D_COMMAND_BUFFER_SERVICE_LINUX_X_UTILS_H_
+
+#include <GL/glx.h>
+#include "base/basictypes.h"
+#include "command_buffer/common/cross/logging.h"
+
+namespace o3d {
+namespace command_buffer {
+
+// This class is a wrapper around an X Window and associated GL context. It is
+// useful to isolate intrusive X headers, since it can be forward declared
+// (Window and GLXContext can't).
+class XWindowWrapper {
+ public:
+ XWindowWrapper(Display *display, Window window)
+ : display_(display),
+ window_(window) {
+ DCHECK(display_);
+ DCHECK(window_);
+ }
+ // Initializes the GL context.
+ bool Initialize();
+
+ // Destroys the GL context.
+ void Destroy();
+
+ // Makes the GL context current on the current thread.
+ bool MakeCurrent();
+
+ // Swaps front and back buffers.
+ void SwapBuffers();
+
+ private:
+ Display *display_;
+ Window window_;
+ GLXContext context_;
+ DISALLOW_COPY_AND_ASSIGN(XWindowWrapper);
+};
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_LINUX_X_UTILS_H_
diff --git a/o3d/command_buffer/service/win/big_test_main.cc b/o3d/command_buffer/service/win/big_test_main.cc
new file mode 100644
index 0000000..8a21b72
--- /dev/null
+++ b/o3d/command_buffer/service/win/big_test_main.cc
@@ -0,0 +1,163 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+#include <windows.h>
+#include <Shellapi.h>
+#include "command_buffer/service/cross/big_test_helpers.h"
+#include "command_buffer/service/win/d3d9/gapi_d3d9.h"
+#include "core/cross/types.h"
+
+namespace o3d {
+namespace command_buffer {
+
+String *g_program_path = NULL;
+GAPIInterface *g_gapi = NULL;
+
+class Thread {
+ public:
+ Thread(ThreadFunc func, void *data)
+ : handle_(NULL),
+ func_(func),
+ data_(data) {
+ }
+
+ ~Thread() {}
+
+ HANDLE handle() const { return handle_; }
+ void set_handle(HANDLE handle) { handle_ = handle; }
+
+ void * data() const { return data_; }
+ ThreadFunc func() const { return func_; }
+
+ private:
+ HANDLE handle_;
+ ThreadFunc func_;
+ void * data_;
+};
+
+DWORD WINAPI ThreadMain(LPVOID lpParam) {
+ Thread *thread = static_cast<Thread *>(lpParam);
+ ThreadFunc func = thread->func();
+ func(thread->data());
+ return 0;
+}
+
+Thread *CreateThread(ThreadFunc func, void* param) {
+ Thread *thread = new Thread(func, param);
+ HANDLE handle = ::CreateThread(NULL, 0, ThreadMain, thread, 0, NULL);
+ return thread;
+}
+
+void JoinThread(Thread *thread) {
+ ::WaitForSingleObject(thread->handle(), INFINITE);
+ ::CloseHandle(thread->handle());
+ delete thread;
+}
+
+bool ProcessSystemMessages() {
+ MSG msg;
+ while (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) {
+ if (msg.message == WM_QUIT) {
+ return false;
+ }
+ TranslateMessage(&msg);
+ DispatchMessage(&msg);
+ }
+ return true;
+}
+
+} // namespace command_buffer
+} // namespace o3d
+
+using o3d::String;
+using o3d::command_buffer::g_program_path;
+using o3d::command_buffer::g_gapi;
+using o3d::command_buffer::GAPID3D9;
+
+LRESULT CALLBACK WindowProc(HWND hWnd,
+ UINT msg,
+ WPARAM wParam,
+ LPARAM lParam) {
+ switch (msg) {
+ case WM_CLOSE:
+ PostQuitMessage(0);
+ break;
+ case WM_DESTROY:
+ PostQuitMessage(0);
+ break;
+ default:
+ return DefWindowProc(hWnd, msg, wParam, lParam);
+ }
+ return 0;
+}
+
+int main(int argc, char *argv[]) {
+ WNDCLASSEX wc = {
+ sizeof(WNDCLASSEX), CS_CLASSDC, WindowProc, 0L, 0L, GetModuleHandle(NULL),
+ NULL, NULL, NULL, NULL, L"O3D big test", NULL
+ };
+ RegisterClassEx(&wc);
+
+ // Create the application's window.
+ HWND hWnd = CreateWindow(L"O3D big test", L"O3D Big Test",
+ WS_OVERLAPPEDWINDOW | WS_VISIBLE, 100, 100, 300,
+ 300, GetDesktopWindow(), NULL, wc.hInstance, NULL);
+ UpdateWindow(hWnd);
+
+ GAPID3D9 d3d9_gapi;
+ d3d9_gapi.set_hwnd(hWnd);
+ g_gapi = &d3d9_gapi;
+
+ wchar_t program_filename[512];
+ GetModuleFileName(NULL, program_filename, sizeof(program_filename));
+ program_filename[511] = 0;
+
+ String program_path = WideToUTF8(std::wstring(program_filename));
+
+ // Remove all characters starting with last '\'.
+ size_t backslash_pos = program_path.rfind('\\');
+ if (backslash_pos != String::npos) {
+ program_path.erase(backslash_pos);
+ }
+ g_program_path = &program_path;
+
+ // Convert the command line arguments to an argc, argv format.
+ LPWSTR *arg_list = NULL;
+ int arg_count;
+ arg_list = CommandLineToArgvW(GetCommandLineW(), &arg_count);
+
+ int ret = big_test_main(arg_count, arg_list);
+
+ g_gapi = NULL;
+ g_program_path = NULL;
+ return ret;
+}
diff --git a/o3d/command_buffer/service/win/d3d9/d3d9_utils.h b/o3d/command_buffer/service/win/d3d9/d3d9_utils.h
new file mode 100644
index 0000000..647d984
--- /dev/null
+++ b/o3d/command_buffer/service/win/d3d9/d3d9_utils.h
@@ -0,0 +1,102 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file defines a few utilities for Direct3D.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_WIN_D3D9_D3D9_UTILS_H__
+#define O3D_COMMAND_BUFFER_SERVICE_WIN_D3D9_D3D9_UTILS_H__
+
+#ifndef NOMINMAX
+// windows.h defines min() and max() as macros, conflicting with std::min and
+// std::max unless NOMINMAX is defined.
+#define NOMINMAX
+#endif
+#include <windows.h>
+#include <d3d9.h>
+#include <d3dx9.h>
+#include <dxerr.h>
+#include <algorithm>
+#include "command_buffer/common/cross/gapi_interface.h"
+
+#if defined (_DEBUG)
+
+#ifndef HR
+#define HR(x) { \
+ HRESULT hr = x; \
+ if (FAILED(hr)) { \
+ LOG(ERROR) << "DirectX error at " << __FILE__ << ":" << __LINE__ \
+ << " when calling " << #x << ": " << DXGetErrorStringA(hr); \
+ } \
+ }
+#endif
+
+#else // _DEBUG
+
+#ifndef HR
+#define HR(x) x;
+#endif
+
+#endif // _DEBUG
+
+namespace o3d {
+namespace command_buffer {
+
+union FloatAndDWORD {
+ float float_value;
+ DWORD dword_value;
+};
+
+// Bit casts a float into a DWORD. That's what D3D expects for some values.
+inline DWORD FloatAsDWORD(float value) {
+ volatile FloatAndDWORD float_and_dword;
+ float_and_dword.float_value = value;
+ return float_and_dword.dword_value;
+}
+
+// Clamps a float to [0 .. 1] and maps it to [0 .. 255]
+inline unsigned int FloatToClampedByte(float value) {
+ value = std::min(1.f, std::max(0.f, value));
+ return static_cast<unsigned int>(value * 255);
+}
+
+// Converts a RGBA color into a D3DCOLOR
+inline D3DCOLOR RGBAToD3DCOLOR(const RGBA &color) {
+ return D3DCOLOR_RGBA(FloatToClampedByte(color.red),
+ FloatToClampedByte(color.green),
+ FloatToClampedByte(color.blue),
+ FloatToClampedByte(color.alpha));
+}
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_WIN_D3D9_D3D9_UTILS_H__
diff --git a/o3d/command_buffer/service/win/d3d9/effect_d3d9.cc b/o3d/command_buffer/service/win/d3d9/effect_d3d9.cc
new file mode 100644
index 0000000..a472e79
--- /dev/null
+++ b/o3d/command_buffer/service/win/d3d9/effect_d3d9.cc
@@ -0,0 +1,569 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the implementation of the D3D9 versions of the
+// Effect resource.
+// This file also contains the related GAPID3D9 function implementations.
+
+#include <algorithm>
+#include "command_buffer/service/win/d3d9/d3d9_utils.h"
+#include "command_buffer/service/win/d3d9/geometry_d3d9.h"
+#include "command_buffer/service/win/d3d9/gapi_d3d9.h"
+#include "command_buffer/service/win/d3d9/effect_d3d9.h"
+#include "command_buffer/service/win/d3d9/sampler_d3d9.h"
+#include "command_buffer/service/cross/effect_utils.h"
+
+// TODO: remove link-dependency on D3DX.
+
+namespace o3d {
+namespace command_buffer {
+
+// Logs the D3D effect error, from either the buffer, or GetLastError().
+static void LogFXError(LPD3DXBUFFER error_buffer) {
+ if (error_buffer) {
+ LPVOID compile_errors = error_buffer->GetBufferPointer();
+ LOG(ERROR) << "Failed to compile effect: "
+ << static_cast<char *>(compile_errors);
+ } else {
+ HLOCAL hLocal = NULL;
+ FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM|FORMAT_MESSAGE_ALLOCATE_BUFFER,
+ NULL,
+ GetLastError(),
+ 0,
+ reinterpret_cast<wchar_t*>(&hLocal),
+ 0,
+ NULL);
+ wchar_t* msg = reinterpret_cast<wchar_t*>(LocalLock(hLocal));
+ LOG(ERROR) << "Failed to compile effect: " << msg;
+ LocalFree(hLocal);
+ }
+}
+
+EffectD3D9::EffectD3D9(ID3DXEffect *d3d_effect,
+ ID3DXConstantTable *fs_constant_table)
+ : d3d_effect_(d3d_effect),
+ fs_constant_table_(fs_constant_table),
+ sync_parameters_(false) {
+ for (unsigned int i = 0; i < kMaxSamplerUnits; ++i) {
+ samplers_[i] = kInvalidResource;
+ }
+}
+// Releases the D3D effect.
+EffectD3D9::~EffectD3D9() {
+ for (ParamList::iterator it = params_.begin(); it != params_.end(); ++it) {
+ (*it)->ResetEffect();
+ }
+ DCHECK(d3d_effect_);
+ d3d_effect_->Release();
+ DCHECK(fs_constant_table_);
+ fs_constant_table_->Release();
+}
+
+// Compiles the effect, and checks that the effect conforms to what we expect
+// (no extra technique or pass in the effect code, since one is implicitly added
+// using the program entry points) and that it validates. If successful, wrap
+// the D3D effect into a new EffectD3D9.
+EffectD3D9 *EffectD3D9::Create(GAPID3D9 *gapi,
+ const String& effect_code,
+ const String& vertex_program_entry,
+ const String& fragment_program_entry) {
+ String prepared_effect = effect_code +
+ "technique Shaders { "
+ " pass p0 { "
+ " VertexShader = compile vs_2_0 " + vertex_program_entry + "();"
+ " PixelShader = compile ps_2_0 " + fragment_program_entry + "();"
+ " }"
+ "};";
+ ID3DXEffect *d3d_effect = NULL;
+ LPD3DXBUFFER error_buffer;
+ IDirect3DDevice9 *device = gapi->d3d_device();
+ if (D3DXCreateEffect(device,
+ prepared_effect.c_str(),
+ prepared_effect.size(),
+ NULL,
+ NULL,
+ 0,
+ NULL,
+ &d3d_effect,
+ &error_buffer) != D3D_OK) {
+ LogFXError(error_buffer);
+ return NULL;
+ }
+ // check that .
+ D3DXEFFECT_DESC effect_desc;
+ HR(d3d_effect->GetDesc(&effect_desc));
+ if (effect_desc.Techniques != 1) {
+ LOG(ERROR) << "Only 1 technique is allowed in an effect.";
+ d3d_effect->Release();
+ return NULL;
+ }
+ D3DXHANDLE technique = d3d_effect->GetTechnique(0);
+ DCHECK(technique);
+ if (d3d_effect->ValidateTechnique(technique) != D3D_OK) {
+ LOG(ERROR) << "Technique doesn't validate.";
+ d3d_effect->Release();
+ return NULL;
+ }
+ D3DXTECHNIQUE_DESC technique_desc;
+ HR(d3d_effect->GetTechniqueDesc(technique, &technique_desc));
+ if (technique_desc.Passes != 1) {
+ LOG(ERROR) << "Only 1 pass is allowed in an effect.";
+ d3d_effect->Release();
+ return NULL;
+ }
+ d3d_effect->SetTechnique(technique);
+ D3DXHANDLE pass = d3d_effect->GetPass(technique, 0);
+ D3DXPASS_DESC pass_desc;
+ HR(d3d_effect->GetPassDesc(pass, &pass_desc));
+ ID3DXConstantTable *table = NULL;
+ HR(D3DXGetShaderConstantTable(pass_desc.pPixelShaderFunction,
+ &table));
+ if (!table) {
+ LOG(ERROR) << "Could not get the constant table.";
+ d3d_effect->Release();
+ return NULL;
+ }
+ return new EffectD3D9(d3d_effect, table);
+}
+
+// Begins rendering with the effect, setting all the appropriate states.
+bool EffectD3D9::Begin(GAPID3D9 *gapi) {
+ UINT numpasses;
+ HR(d3d_effect_->Begin(&numpasses, 0));
+ HR(d3d_effect_->BeginPass(0));
+ sync_parameters_ = false;
+ return SetSamplers(gapi);
+}
+
+// Terminates rendering with the effect, resetting all the appropriate states.
+void EffectD3D9::End(GAPID3D9 *gapi) {
+ HR(d3d_effect_->EndPass());
+ HR(d3d_effect_->End());
+}
+
+// Gets the parameter count from the D3D effect description.
+unsigned int EffectD3D9::GetParamCount() {
+ D3DXEFFECT_DESC effect_desc;
+ HR(d3d_effect_->GetDesc(&effect_desc));
+ return effect_desc.Parameters;
+}
+
+// Retrieves the matching DataType from a D3D parameter description.
+static effect_param::DataType GetDataTypeFromD3D(
+ const D3DXPARAMETER_DESC &desc) {
+ switch (desc.Type) {
+ case D3DXPT_FLOAT:
+ switch (desc.Class) {
+ case D3DXPC_SCALAR:
+ return effect_param::FLOAT1;
+ case D3DXPC_VECTOR:
+ switch (desc.Columns) {
+ case 2:
+ return effect_param::FLOAT2;
+ case 3:
+ return effect_param::FLOAT3;
+ case 4:
+ return effect_param::FLOAT4;
+ default:
+ return effect_param::UNKNOWN;
+ }
+ case D3DXPC_MATRIX_ROWS:
+ case D3DXPC_MATRIX_COLUMNS:
+ if (desc.Columns == 4 && desc.Rows == 4) {
+ return effect_param::MATRIX4;
+ } else {
+ return effect_param::UNKNOWN;
+ }
+ default:
+ return effect_param::UNKNOWN;
+ }
+ case D3DXPT_INT:
+ if (desc.Class == D3DXPC_SCALAR) {
+ return effect_param::INT;
+ } else {
+ return effect_param::UNKNOWN;
+ }
+ case D3DXPT_BOOL:
+ if (desc.Class == D3DXPC_SCALAR) {
+ return effect_param::BOOL;
+ } else {
+ return effect_param::UNKNOWN;
+ }
+ case D3DXPT_SAMPLER:
+ case D3DXPT_SAMPLER2D:
+ case D3DXPT_SAMPLER3D:
+ case D3DXPT_SAMPLERCUBE:
+ if (desc.Class == D3DXPC_OBJECT) {
+ return effect_param::SAMPLER;
+ } else {
+ return effect_param::UNKNOWN;
+ }
+ default:
+ return effect_param::UNKNOWN;
+ }
+}
+
+// Gets a handle to the selected parameter, and wraps it into an
+// EffectParamD3D9 if successful.
+EffectParamD3D9 *EffectD3D9::CreateParam(unsigned int index) {
+ D3DXHANDLE handle = d3d_effect_->GetParameter(NULL, index);
+ if (!handle) return NULL;
+ return EffectParamD3D9::Create(this, handle);
+}
+
+// Gets a handle to the selected parameter, and wraps it into an
+// EffectParamD3D9 if successful.
+EffectParamD3D9 *EffectD3D9::CreateParamByName(const char *name) {
+ D3DXHANDLE handle = d3d_effect_->GetParameterByName(NULL, name);
+ if (!handle) return NULL;
+ return EffectParamD3D9::Create(this, handle);
+}
+
+bool EffectD3D9::CommitParameters(GAPID3D9 *gapi) {
+ if (sync_parameters_) {
+ sync_parameters_ = false;
+ d3d_effect_->CommitChanges();
+ return SetSamplers(gapi);
+ } else {
+ return true;
+ }
+}
+
+bool EffectD3D9::SetSamplers(GAPID3D9 *gapi) {
+ IDirect3DDevice9 *d3d_device = gapi->d3d_device();
+ bool result = true;
+ for (unsigned int i = 0; i < kMaxSamplerUnits; ++i) {
+ SamplerD3D9 *sampler = gapi->GetSampler(samplers_[i]);
+ if (sampler) {
+ result &= sampler->ApplyStates(gapi, i);
+ } else {
+ HR(d3d_device->SetTexture(i, NULL));
+ }
+ }
+ return result;
+}
+
+void EffectD3D9::LinkParam(EffectParamD3D9 *param) {
+ params_.push_back(param);
+}
+
+void EffectD3D9::UnlinkParam(EffectParamD3D9 *param) {
+ std::remove(params_.begin(), params_.end(), param);
+}
+
+EffectParamD3D9::EffectParamD3D9(effect_param::DataType data_type,
+ EffectD3D9 *effect,
+ D3DXHANDLE handle)
+ : EffectParam(data_type),
+ effect_(effect),
+ handle_(handle),
+ sampler_units_(NULL),
+ sampler_unit_count_(0) {
+ DCHECK(effect_);
+ effect_->LinkParam(this);
+}
+
+EffectParamD3D9::~EffectParamD3D9() {
+ if (effect_) effect_->UnlinkParam(this);
+}
+
+EffectParamD3D9 *EffectParamD3D9::Create(EffectD3D9 *effect,
+ D3DXHANDLE handle) {
+ DCHECK(effect);
+ D3DXPARAMETER_DESC desc;
+ HR(effect->d3d_effect_->GetParameterDesc(handle, &desc));
+ effect_param::DataType data_type = GetDataTypeFromD3D(desc);
+ EffectParamD3D9 *param = new EffectParamD3D9(data_type, effect, handle);
+ if (data_type == effect_param::SAMPLER) {
+ ID3DXConstantTable *table = effect->fs_constant_table_;
+ DCHECK(table);
+ D3DXHANDLE sampler_handle = table->GetConstantByName(NULL, desc.Name);
+ if (sampler_handle) {
+ D3DXCONSTANT_DESC desc_array[kMaxSamplerUnits];
+ unsigned int num_desc = kMaxSamplerUnits;
+ table->GetConstantDesc(sampler_handle, desc_array, &num_desc);
+ // We have no good way of querying how many descriptions would really be
+ // returned as we're capping the number to kMaxSamplerUnits (which should
+ // be more than sufficient). If however we do end up with the max number
+ // there's a chance that there were actually more so let's log it.
+ if (num_desc == kMaxSamplerUnits) {
+ DLOG(WARNING) << "Number of constant descriptions might have exceeded "
+ << "the maximum of " << kMaxSamplerUnits;
+ }
+ param->sampler_unit_count_ = 0;
+ if (num_desc > 0) {
+ param->sampler_units_.reset(new unsigned int[num_desc]);
+ for (unsigned int desc_index = 0; desc_index < num_desc; desc_index++) {
+ D3DXCONSTANT_DESC constant_desc = desc_array[desc_index];
+ if (constant_desc.Class == D3DXPC_OBJECT &&
+ (constant_desc.Type == D3DXPT_SAMPLER ||
+ constant_desc.Type == D3DXPT_SAMPLER2D ||
+ constant_desc.Type == D3DXPT_SAMPLER3D ||
+ constant_desc.Type == D3DXPT_SAMPLERCUBE)) {
+ param->sampler_units_[param->sampler_unit_count_++] =
+ constant_desc.RegisterIndex;
+ }
+ }
+ }
+ }
+ // if the sampler hasn't been found in the constant table, that means it
+ // isn't referenced, hence it doesn't use any sampler unit.
+ }
+ return param;
+}
+
+// Fills the Desc structure, appending name and semantic if any, and if enough
+// room is available in the buffer.
+bool EffectParamD3D9::GetDesc(unsigned int size, void *data) {
+ using effect_param::Desc;
+ if (size < sizeof(Desc)) // NOLINT
+ return false;
+ if (!effect_)
+ return false;
+ ID3DXEffect *d3d_effect = effect_->d3d_effect_;
+ D3DXPARAMETER_DESC d3d_desc;
+ HR(d3d_effect->GetParameterDesc(handle_, &d3d_desc));
+ unsigned int name_size =
+ d3d_desc.Name ? static_cast<unsigned int>(strlen(d3d_desc.Name)) + 1 : 0;
+ unsigned int semantic_size = d3d_desc.Semantic ?
+ static_cast<unsigned int>(strlen(d3d_desc.Semantic)) + 1 : 0;
+ unsigned int total_size = sizeof(Desc) + name_size + semantic_size; // NOLINT
+
+ Desc *desc = static_cast<Desc *>(data);
+ memset(desc, 0, sizeof(*desc));
+ desc->size = total_size;
+ desc->data_type = data_type();
+ desc->data_size = GetDataSize(data_type());
+ desc->name_offset = 0;
+ desc->name_size = name_size;
+ desc->semantic_offset = 0;
+ desc->semantic_size = semantic_size;
+ unsigned int current_offset = sizeof(Desc);
+ if (d3d_desc.Name && current_offset + name_size <= size) {
+ desc->name_offset = current_offset;
+ memcpy(static_cast<char *>(data) + current_offset,
+ d3d_desc.Name, name_size);
+ current_offset += name_size;
+ }
+ if (d3d_desc.Semantic && current_offset + semantic_size <= size) {
+ desc->semantic_offset = current_offset;
+ memcpy(static_cast<char *>(data) + current_offset,
+ d3d_desc.Semantic, semantic_size);
+ current_offset += semantic_size;
+ }
+ return true;
+}
+
+// Sets the data into the D3D effect parameter, using the appropriate D3D call.
+bool EffectParamD3D9::SetData(GAPID3D9 *gapi,
+ unsigned int size,
+ const void * data) {
+ if (!effect_)
+ return false;
+ ID3DXEffect *d3d_effect = effect_->d3d_effect_;
+ effect_param::DataType type = data_type();
+ if (size < effect_param::GetDataSize(type)) return false;
+ switch (type) {
+ case effect_param::FLOAT1:
+ HR(d3d_effect->SetFloat(handle_, *static_cast<const float *>(data)));
+ break;
+ case effect_param::FLOAT2:
+ HR(d3d_effect->SetFloatArray(handle_, static_cast<const float *>(data),
+ 2));
+ break;
+ case effect_param::FLOAT3:
+ HR(d3d_effect->SetFloatArray(handle_, static_cast<const float *>(data),
+ 3));
+ break;
+ case effect_param::FLOAT4:
+ HR(d3d_effect->SetFloatArray(handle_, static_cast<const float *>(data),
+ 4));
+ break;
+ case effect_param::MATRIX4:
+ HR(d3d_effect->SetMatrix(handle_,
+ reinterpret_cast<const D3DXMATRIX *>(data)));
+ break;
+ case effect_param::INT:
+ HR(d3d_effect->SetInt(handle_, *static_cast<const int *>(data)));
+ break;
+ case effect_param::BOOL:
+ HR(d3d_effect->SetBool(handle_, *static_cast<const bool *>(data)?1:0));
+ break;
+ case effect_param::SAMPLER: {
+ ResourceID id = *static_cast<const ResourceID *>(data);
+ for (unsigned int i = 0; i < sampler_unit_count_; ++i) {
+ effect_->samplers_[sampler_units_[i]] = id;
+ }
+ break;
+ }
+ default:
+ DLOG(ERROR) << "Invalid parameter type.";
+ return false;
+ }
+ if (effect_ == gapi->current_effect()) {
+ effect_->sync_parameters_ = true;
+ }
+ return true;
+}
+
+// Calls EffectD3D9::Create, and assign the result to the resource ID.
+// If changing the current effect, dirty it.
+BufferSyncInterface::ParseError GAPID3D9::CreateEffect(
+ ResourceID id,
+ unsigned int size,
+ const void *data) {
+ if (id == current_effect_id_) DirtyEffect();
+ // Even though Assign would Destroy the effect at id, we do it explicitly in
+ // case the creation fails.
+ effects_.Destroy(id);
+ // Data is vp_main \0 fp_main \0 effect_text.
+ String vertex_program_entry;
+ String fragment_program_entry;
+ String effect_code;
+ if (!ParseEffectData(size, data,
+ &vertex_program_entry,
+ &fragment_program_entry,
+ &effect_code)) {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ EffectD3D9 * effect = EffectD3D9::Create(this, effect_code,
+ vertex_program_entry,
+ fragment_program_entry);
+ if (!effect) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ effects_.Assign(id, effect);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+// Destroys the Effect resource.
+// If destroying the current effect, dirty it.
+BufferSyncInterface::ParseError GAPID3D9::DestroyEffect(ResourceID id) {
+ if (id == current_effect_id_) DirtyEffect();
+ return effects_.Destroy(id) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+// Sets the current effect ID, dirtying the current effect.
+BufferSyncInterface::ParseError GAPID3D9::SetEffect(ResourceID id) {
+ DirtyEffect();
+ current_effect_id_ = id;
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+// Gets the param count from the effect and store it in the memory buffer.
+BufferSyncInterface::ParseError GAPID3D9::GetParamCount(
+ ResourceID id,
+ unsigned int size,
+ void *data) {
+ EffectD3D9 *effect = effects_.Get(id);
+ if (!effect || size < sizeof(Uint32)) // NOLINT
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ *static_cast<Uint32 *>(data) = effect->GetParamCount();
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+BufferSyncInterface::ParseError GAPID3D9::CreateParam(
+ ResourceID param_id,
+ ResourceID effect_id,
+ unsigned int index) {
+ EffectD3D9 *effect = effects_.Get(effect_id);
+ if (!effect) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ EffectParamD3D9 *param = effect->CreateParam(index);
+ if (!param) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ effect_params_.Assign(param_id, param);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+BufferSyncInterface::ParseError GAPID3D9::CreateParamByName(
+ ResourceID param_id,
+ ResourceID effect_id,
+ unsigned int size,
+ const void *name) {
+ EffectD3D9 *effect = effects_.Get(effect_id);
+ if (!effect) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ std::string string_name(static_cast<const char *>(name), size);
+ EffectParamD3D9 *param = effect->CreateParamByName(string_name.c_str());
+ if (!param) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ effect_params_.Assign(param_id, param);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+BufferSyncInterface::ParseError GAPID3D9::DestroyParam(ResourceID id) {
+ return effect_params_.Destroy(id) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+BufferSyncInterface::ParseError GAPID3D9::SetParamData(
+ ResourceID id,
+ unsigned int size,
+ const void *data) {
+ EffectParamD3D9 *param = effect_params_.Get(id);
+ if (!param) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return param->SetData(this, size, data) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+BufferSyncInterface::ParseError GAPID3D9::GetParamDesc(
+ ResourceID id,
+ unsigned int size,
+ void *data) {
+ EffectParamD3D9 *param = effect_params_.Get(id);
+ if (!param) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return param->GetDesc(size, data) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+// If the current effect is valid, call End on it, and tag for revalidation.
+void GAPID3D9::DirtyEffect() {
+ if (validate_effect_) return;
+ DCHECK(current_effect_);
+ current_effect_->End(this);
+ current_effect_ = NULL;
+ validate_effect_ = true;
+}
+
+// Gets the current effect, and calls Begin on it (if successful).
+// Should only be called if the current effect is not valid.
+bool GAPID3D9::ValidateEffect() {
+ DCHECK(validate_effect_);
+ DCHECK(!current_effect_);
+ current_effect_ = effects_.Get(current_effect_id_);
+ if (!current_effect_) return false;
+ validate_effect_ = false;
+ return current_effect_->Begin(this);
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/win/d3d9/effect_d3d9.h b/o3d/command_buffer/service/win/d3d9/effect_d3d9.h
new file mode 100644
index 0000000..318598c
--- /dev/null
+++ b/o3d/command_buffer/service/win/d3d9/effect_d3d9.h
@@ -0,0 +1,127 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the definition of the D3D9 versions of effect-related
+// resource classes.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_WIN_D3D9_EFFECT_D3D9_H__
+#define O3D_COMMAND_BUFFER_SERVICE_WIN_D3D9_EFFECT_D3D9_H__
+
+#include <vector>
+#include "command_buffer/common/cross/gapi_interface.h"
+#include "command_buffer/service/win/d3d9/d3d9_utils.h"
+#include "command_buffer/service/cross/resource.h"
+
+namespace o3d {
+namespace command_buffer {
+
+class GAPID3D9;
+class EffectD3D9;
+
+// ps_2_0 limit
+static const unsigned int kMaxSamplerUnits = 16;
+
+// D3D version of EffectParam. This class keeps a reference to the D3D effect.
+class EffectParamD3D9: public EffectParam {
+ public:
+ EffectParamD3D9(effect_param::DataType data_type,
+ EffectD3D9 *effect,
+ D3DXHANDLE handle);
+ virtual ~EffectParamD3D9();
+
+ // Sets the data into the D3D effect parameter.
+ bool SetData(GAPID3D9 *gapi, unsigned int size, const void * data);
+
+ // Gets the description of the parameter.
+ bool GetDesc(unsigned int size, void *data);
+
+ // Resets the effect back-pointer. This is called when the effect gets
+ // destroyed, to invalidate the parameter.
+ void ResetEffect() { effect_ = NULL; }
+
+ static EffectParamD3D9 *Create(EffectD3D9 *effect, D3DXHANDLE handle);
+ private:
+ EffectD3D9 *effect_;
+ D3DXHANDLE handle_;
+ unsigned int sampler_unit_count_;
+ scoped_array<unsigned int> sampler_units_;
+};
+
+// D3D9 version of Effect.
+class EffectD3D9 : public Effect {
+ public:
+ EffectD3D9(ID3DXEffect *d3d_effect,
+ ID3DXConstantTable *fs_constant_table);
+ virtual ~EffectD3D9();
+ // Compiles and creates an effect from source code.
+ static EffectD3D9 *Create(GAPID3D9 *gapi,
+ const String &effect_code,
+ const String &vertex_program_entry,
+ const String &fragment_program_entry);
+ // Applies the effect states (vertex shader, pixel shader) to D3D.
+ bool Begin(GAPID3D9 *gapi);
+ // Resets the effect states (vertex shader, pixel shader) to D3D.
+ void End(GAPID3D9 *gapi);
+ // Commits parameters to D3D, if they were modified while the effect is
+ // active.
+ bool CommitParameters(GAPID3D9 *gapi);
+
+ // Gets the number of parameters in the effect.
+ unsigned int GetParamCount();
+ // Creates an effect parameter with the specified index.
+ EffectParamD3D9 *CreateParam(unsigned int index);
+ // Creates an effect parameter of the specified name.
+ EffectParamD3D9 *CreateParamByName(const char *name);
+ private:
+ typedef std::vector<EffectParamD3D9 *> ParamList;
+
+ // Links a param into this effect.
+ void LinkParam(EffectParamD3D9 *param);
+ // Unlinks a param into this effect.
+ void UnlinkParam(EffectParamD3D9 *param);
+ // Sets sampler states.
+ bool SetSamplers(GAPID3D9 *gapi);
+
+ ID3DXEffect *d3d_effect_;
+ ID3DXConstantTable *fs_constant_table_;
+ ParamList params_;
+ bool sync_parameters_;
+ ResourceID samplers_[kMaxSamplerUnits];
+
+ friend class EffectParamD3D9;
+ DISALLOW_COPY_AND_ASSIGN(EffectD3D9);
+};
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_WIN_D3D9_EFFECT_D3D9_H__
diff --git a/o3d/command_buffer/service/win/d3d9/gapi_d3d9.cc b/o3d/command_buffer/service/win/d3d9/gapi_d3d9.cc
new file mode 100644
index 0000000..f5fe0d4
--- /dev/null
+++ b/o3d/command_buffer/service/win/d3d9/gapi_d3d9.cc
@@ -0,0 +1,305 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the implementation of the GAPID3D9 class.
+
+#include "command_buffer/service/win/d3d9/gapi_d3d9.h"
+
+namespace o3d {
+namespace command_buffer {
+
+GAPID3D9::GAPID3D9()
+ : d3d_(NULL),
+ d3d_device_(NULL),
+ hwnd_(NULL),
+ current_vertex_struct_(0),
+ validate_streams_(true),
+ max_vertices_(0),
+ current_effect_id_(0),
+ validate_effect_(true),
+ current_effect_(NULL),
+ vertex_buffers_(),
+ index_buffers_(),
+ vertex_structs_() {}
+
+GAPID3D9::~GAPID3D9() {}
+
+// Initializes a D3D interface and device, and sets basic states.
+bool GAPID3D9::Initialize() {
+ d3d_ = Direct3DCreate9(D3D_SDK_VERSION);
+ if (NULL == d3d_) {
+ LOG(ERROR) << "Failed to create the initial D3D9 Interface";
+ return false;
+ }
+ d3d_device_ = NULL;
+
+ D3DDISPLAYMODE d3ddm;
+ d3d_->GetAdapterDisplayMode(D3DADAPTER_DEFAULT, &d3ddm);
+ // NOTE: make sure the backbuffer matches this format, as it is
+ // currently currently assumed to be 32-bit 8X8R8G8B
+
+ D3DPRESENT_PARAMETERS d3dpp;
+ ZeroMemory(&d3dpp, sizeof(d3dpp));
+ d3dpp.Windowed = TRUE;
+ d3dpp.SwapEffect = D3DSWAPEFFECT_DISCARD;
+ d3dpp.BackBufferFormat = d3ddm.Format;
+ d3dpp.EnableAutoDepthStencil = TRUE;
+ d3dpp.AutoDepthStencilFormat = D3DFMT_D24S8;
+ d3dpp.PresentationInterval = D3DPRESENT_INTERVAL_ONE; // wait for vsync
+ // Note: SwapEffect=DISCARD is req. for multisample to function
+ // Note: AutoDepthStencilFormat is 16-bit (not the usual 8-bit)
+
+ // query multisampling
+ const int kNumTypesToCheck = 4;
+ D3DMULTISAMPLE_TYPE multisample_types[] = { D3DMULTISAMPLE_5_SAMPLES,
+ D3DMULTISAMPLE_4_SAMPLES,
+ D3DMULTISAMPLE_2_SAMPLES,
+ D3DMULTISAMPLE_NONE };
+ DWORD multisample_quality = 0;
+ for (int i = 0; i < kNumTypesToCheck; ++i) {
+ // check back-buffer for multisampling at level "i";
+ // back buffer = 32-bit XRGB (i.e. no alpha)
+ if (SUCCEEDED(d3d_->CheckDeviceMultiSampleType(
+ D3DADAPTER_DEFAULT,
+ D3DDEVTYPE_HAL,
+ D3DFMT_X8R8G8B8,
+ true, // result is windowed
+ multisample_types[i],
+ &multisample_quality))) {
+ // back buffer succeeded, now check depth-buffer
+ // depth buffer = 24-bit, stencil = 8-bit
+ // NOTE: 8-bit not 16-bit like the D3DPRESENT_PARAMETERS
+ if (SUCCEEDED(d3d_->CheckDeviceMultiSampleType(
+ D3DADAPTER_DEFAULT,
+ D3DDEVTYPE_HAL,
+ D3DFMT_D24S8,
+ true, // result is windowed
+ multisample_types[i],
+ &multisample_quality))) {
+ d3dpp.MultiSampleType = multisample_types[i];
+ d3dpp.MultiSampleQuality = multisample_quality - 1;
+ break;
+ }
+ }
+ }
+ // D3DCREATE_FPU_PRESERVE is there because Firefox 3 relies on specific FPU
+ // flags for its UI rendering. Apparently Firefox 2 does not, though we don't
+ // currently propagate that info.
+ // TODO: check if FPU_PRESERVE has a significant perf hit, in which
+ // case find out if we can disable it for Firefox 2/other browsers, and/or if
+ // it makes sense to switch FPU flags before/after every DX call.
+ DWORD flags = D3DCREATE_HARDWARE_VERTEXPROCESSING | D3DCREATE_FPU_PRESERVE;
+ if (!SUCCEEDED(d3d_->CreateDevice(D3DADAPTER_DEFAULT,
+ D3DDEVTYPE_HAL,
+ hwnd_,
+ flags,
+ &d3dpp,
+ &d3d_device_))) {
+ LOG(ERROR) << "Failed to create the D3D Device";
+ return false;
+ }
+ // initialise the d3d graphics state.
+ HR(d3d_device_->SetRenderState(D3DRS_LIGHTING, FALSE));
+ HR(d3d_device_->SetRenderState(D3DRS_ZENABLE, TRUE));
+ HR(d3d_device_->SetRenderState(D3DRS_CULLMODE, D3DCULL_NONE));
+ return true;
+}
+
+// Deletes the D3D9 Device and releases the D3D interface.
+void GAPID3D9::Destroy() {
+ vertex_buffers_.DestroyAllResources();
+ index_buffers_.DestroyAllResources();
+ vertex_structs_.DestroyAllResources();
+ effects_.DestroyAllResources();
+ effect_params_.DestroyAllResources();
+ textures_.DestroyAllResources();
+ samplers_.DestroyAllResources();
+ if (d3d_device_) {
+ d3d_device_->Release();
+ d3d_device_ = NULL;
+ }
+ if (d3d_) {
+ d3d_->Release();
+ d3d_ = NULL;
+ }
+}
+
+// Begins the frame.
+void GAPID3D9::BeginFrame() {
+ HR(d3d_device_->BeginScene());
+}
+
+// Ends the frame, presenting the back buffer.
+void GAPID3D9::EndFrame() {
+ DirtyEffect();
+ HR(d3d_device_->EndScene());
+ HR(d3d_device_->Present(NULL, NULL, NULL, NULL));
+}
+
+// Clears the selected buffers.
+void GAPID3D9::Clear(unsigned int buffers,
+ const RGBA &color,
+ float depth,
+ unsigned int stencil) {
+ DWORD flags = (buffers & COLOR ? D3DCLEAR_TARGET : 0) |
+ (buffers & DEPTH ? D3DCLEAR_ZBUFFER : 0) |
+ (buffers & STENCIL ? D3DCLEAR_STENCIL : 0);
+ HR(d3d_device_->Clear(0,
+ NULL,
+ flags,
+ D3DCOLOR_COLORVALUE(color.red,
+ color.green,
+ color.blue,
+ color.alpha),
+ depth,
+ stencil));
+}
+
+// Sets the viewport.
+void GAPID3D9::SetViewport(unsigned int x,
+ unsigned int y,
+ unsigned int width,
+ unsigned int height,
+ float z_min,
+ float z_max) {
+ D3DVIEWPORT9 viewport = {x, y, width, height, z_min, z_max};
+ HR(d3d_device_->SetViewport(&viewport));
+}
+
+// Converts an unsigned int RGBA color into an unsigned int ARGB (DirectX)
+// color.
+static unsigned int RGBAToARGB(unsigned int rgba) {
+ return (rgba >> 8) | (rgba << 24);
+}
+
+// Sets the current VertexStruct. Just keep track of the ID.
+BufferSyncInterface::ParseError GAPID3D9::SetVertexStruct(ResourceID id) {
+ current_vertex_struct_ = id;
+ validate_streams_ = true;
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+// Sets in D3D the input streams of the current vertex struct.
+bool GAPID3D9::ValidateStreams() {
+ DCHECK(validate_streams_);
+ VertexStructD3D9 *vertex_struct = vertex_structs_.Get(current_vertex_struct_);
+ if (!vertex_struct) {
+ LOG(ERROR) << "Drawing with invalid streams.";
+ return false;
+ }
+ max_vertices_ = vertex_struct->SetStreams(this);
+ validate_streams_ = false;
+ return max_vertices_ > 0;
+}
+
+// Converts a GAPID3D9::PrimitiveType to a D3DPRIMITIVETYPE.
+static D3DPRIMITIVETYPE D3DPrimitive(GAPID3D9::PrimitiveType primitive_type) {
+ switch (primitive_type) {
+ case GAPID3D9::POINTS:
+ return D3DPT_POINTLIST;
+ case GAPID3D9::LINES:
+ return D3DPT_LINELIST;
+ case GAPID3D9::LINE_STRIPS:
+ return D3DPT_LINESTRIP;
+ case GAPID3D9::TRIANGLES:
+ return D3DPT_TRIANGLELIST;
+ case GAPID3D9::TRIANGLE_STRIPS:
+ return D3DPT_TRIANGLESTRIP;
+ case GAPID3D9::TRIANGLE_FANS:
+ return D3DPT_TRIANGLEFAN;
+ default:
+ LOG(FATAL) << "Invalid primitive type";
+ return D3DPT_POINTLIST;
+ }
+}
+
+// Draws with the current vertex struct.
+BufferSyncInterface::ParseError GAPID3D9::Draw(
+ PrimitiveType primitive_type,
+ unsigned int first,
+ unsigned int count) {
+ if (validate_streams_ && !ValidateStreams()) {
+ // TODO: add proper error management
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ if (validate_effect_ && !ValidateEffect()) {
+ // TODO: add proper error management
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ DCHECK(current_effect_);
+ if (!current_effect_->CommitParameters(this)) {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ if (first + count > max_vertices_) {
+ // TODO: add proper error management
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ HR(d3d_device_->DrawPrimitive(D3DPrimitive(primitive_type), first, count));
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+// Draws with the current vertex struct.
+BufferSyncInterface::ParseError GAPID3D9::DrawIndexed(
+ PrimitiveType primitive_type,
+ ResourceID index_buffer_id,
+ unsigned int first,
+ unsigned int count,
+ unsigned int min_index,
+ unsigned int max_index) {
+ IndexBufferD3D9 *index_buffer = index_buffers_.Get(index_buffer_id);
+ if (!index_buffer) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ if (validate_streams_ && !ValidateStreams()) {
+ // TODO: add proper error management
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ if (validate_effect_ && !ValidateEffect()) {
+ // TODO: add proper error management
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ DCHECK(current_effect_);
+ if (!current_effect_->CommitParameters(this)) {
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+ if ((min_index >= max_vertices_) || (max_index > max_vertices_)) {
+ // TODO: add proper error management
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ }
+
+ HR(d3d_device_->SetIndices(index_buffer->d3d_index_buffer()));
+ HR(d3d_device_->DrawIndexedPrimitive(D3DPrimitive(primitive_type), 0,
+ min_index, max_index - min_index + 1,
+ first, count));
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/win/d3d9/gapi_d3d9.h b/o3d/command_buffer/service/win/d3d9/gapi_d3d9.h
new file mode 100644
index 0000000..e94f35c
--- /dev/null
+++ b/o3d/command_buffer/service/win/d3d9/gapi_d3d9.h
@@ -0,0 +1,384 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the GAPID3D9 class, implementing the GAPI interface for
+// D3D9.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_WIN_D3D9_GAPI_D3D9_H__
+#define O3D_COMMAND_BUFFER_SERVICE_WIN_D3D9_GAPI_D3D9_H__
+
+#include "command_buffer/common/cross/gapi_interface.h"
+#include "command_buffer/service/win/d3d9/d3d9_utils.h"
+#include "command_buffer/service/win/d3d9/geometry_d3d9.h"
+#include "command_buffer/service/win/d3d9/effect_d3d9.h"
+#include "command_buffer/service/win/d3d9/texture_d3d9.h"
+#include "command_buffer/service/win/d3d9/sampler_d3d9.h"
+
+namespace o3d {
+namespace command_buffer {
+
+// This class implements the GAPI interface for D3D9.
+class GAPID3D9 : public GAPIInterface {
+ public:
+ GAPID3D9();
+ virtual ~GAPID3D9();
+
+ void set_hwnd(HWND hwnd) { hwnd_ = hwnd; }
+
+ // Initializes the graphics context, bound to a window.
+ // Returns:
+ // true if successful.
+ virtual bool Initialize();
+
+ // Destroys the graphics context.
+ virtual void Destroy();
+
+ // Implements the BeginFrame function for D3D9.
+ virtual void BeginFrame();
+
+ // Implements the EndFrame function for D3D9.
+ virtual void EndFrame();
+
+ // Implements the Clear function for D3D9.
+ virtual void Clear(unsigned int buffers,
+ const RGBA &color,
+ float depth,
+ unsigned int stencil);
+
+ // Implements the SetViewport function for D3D9.
+ virtual void SetViewport(unsigned int x,
+ unsigned int y,
+ unsigned int width,
+ unsigned int height,
+ float z_min,
+ float z_max);
+
+ // Implements the CreateVertexBuffer function for D3D9.
+ virtual ParseError CreateVertexBuffer(ResourceID id,
+ unsigned int size,
+ unsigned int flags);
+
+ // Implements the DestroyVertexBuffer function for D3D9.
+ virtual ParseError DestroyVertexBuffer(ResourceID id);
+
+ // Implements the SetVertexBufferData function for D3D9.
+ virtual ParseError SetVertexBufferData(ResourceID id,
+ unsigned int offset,
+ unsigned int size,
+ const void *data);
+
+ // Implements the GetVertexBufferData function for D3D9.
+ virtual ParseError GetVertexBufferData(ResourceID id,
+ unsigned int offset,
+ unsigned int size,
+ void *data);
+
+ // Implements the CreateIndexBuffer function for D3D9.
+ virtual ParseError CreateIndexBuffer(ResourceID id,
+ unsigned int size,
+ unsigned int flags);
+
+ // Implements the DestroyIndexBuffer function for D3D9.
+ virtual ParseError DestroyIndexBuffer(ResourceID id);
+
+ // Implements the SetIndexBufferData function for D3D9.
+ virtual ParseError SetIndexBufferData(ResourceID id,
+ unsigned int offset,
+ unsigned int size,
+ const void *data);
+
+ // Implements the GetIndexBufferData function for D3D9.
+ virtual ParseError GetIndexBufferData(ResourceID id,
+ unsigned int offset,
+ unsigned int size,
+ void *data);
+
+ // Implements the CreateVertexStruct function for D3D9.
+ virtual ParseError CreateVertexStruct(ResourceID id,
+ unsigned int input_count);
+
+ // Implements the DestroyVertexStruct function for D3D9.
+ virtual ParseError DestroyVertexStruct(ResourceID id);
+
+ // Implements the SetVertexInput function for D3D9.
+ virtual ParseError SetVertexInput(ResourceID vertex_struct_id,
+ unsigned int input_index,
+ ResourceID vertex_buffer_id,
+ unsigned int offset,
+ unsigned int stride,
+ vertex_struct::Type type,
+ vertex_struct::Semantic semantic,
+ unsigned int semantic_index);
+
+ // Implements the SetVertexStruct function for D3D9.
+ virtual ParseError SetVertexStruct(ResourceID id);
+
+ // Implements the Draw function for D3D9.
+ virtual ParseError Draw(PrimitiveType primitive_type,
+ unsigned int first,
+ unsigned int count);
+
+ // Implements the DrawIndexed function for D3D9.
+ virtual ParseError DrawIndexed(PrimitiveType primitive_type,
+ ResourceID index_buffer_id,
+ unsigned int first,
+ unsigned int count,
+ unsigned int min_index,
+ unsigned int max_index);
+
+ // Implements the CreateEffect function for D3D9.
+ virtual ParseError CreateEffect(ResourceID id,
+ unsigned int size,
+ const void *data);
+
+ // Implements the DestroyEffect function for D3D9.
+ virtual ParseError DestroyEffect(ResourceID id);
+
+ // Implements the SetEffect function for D3D9.
+ virtual ParseError SetEffect(ResourceID id);
+
+ // Implements the GetParamCount function for D3D9.
+ virtual ParseError GetParamCount(ResourceID id,
+ unsigned int size,
+ void *data);
+
+ // Implements the CreateParam function for D3D9.
+ virtual ParseError CreateParam(ResourceID param_id,
+ ResourceID effect_id,
+ unsigned int index);
+
+ // Implements the CreateParamByName function for D3D9.
+ virtual ParseError CreateParamByName(ResourceID param_id,
+ ResourceID effect_id,
+ unsigned int size,
+ const void *name);
+
+ // Implements the DestroyParam function for D3D9.
+ virtual ParseError DestroyParam(ResourceID id);
+
+ // Implements the SetParamData function for D3D9.
+ virtual ParseError SetParamData(ResourceID id,
+ unsigned int size,
+ const void *data);
+
+ // Implements the GetParamDesc function for D3D9.
+ virtual ParseError GetParamDesc(ResourceID id,
+ unsigned int size,
+ void *data);
+
+ // Implements the CreateTexture2D function for D3D9.
+ virtual ParseError CreateTexture2D(ResourceID id,
+ unsigned int width,
+ unsigned int height,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags);
+
+ // Implements the CreateTexture3D function for D3D9.
+ virtual ParseError CreateTexture3D(ResourceID id,
+ unsigned int width,
+ unsigned int height,
+ unsigned int depth,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags);
+
+ // Implements the CreateTextureCube function for D3D9.
+ virtual ParseError CreateTextureCube(ResourceID id,
+ unsigned int side,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags);
+
+ // Implements the SetTextureData function for D3D9.
+ virtual ParseError SetTextureData(ResourceID id,
+ unsigned int x,
+ unsigned int y,
+ unsigned int z,
+ unsigned int width,
+ unsigned int height,
+ unsigned int depth,
+ unsigned int level,
+ texture::Face face,
+ unsigned int pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ const void *data);
+
+ // Implements the GetTextureData function for D3D9.
+ virtual ParseError GetTextureData(ResourceID id,
+ unsigned int x,
+ unsigned int y,
+ unsigned int z,
+ unsigned int width,
+ unsigned int height,
+ unsigned int depth,
+ unsigned int level,
+ texture::Face face,
+ unsigned int pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ void *data);
+
+ // Implements the DestroyTexture function for D3D9.
+ virtual ParseError DestroyTexture(ResourceID id);
+
+ // Implements the CreateSampler function for D3D9.
+ virtual ParseError CreateSampler(ResourceID id);
+
+ // Implements the DestroySampler function for D3D9.
+ virtual ParseError DestroySampler(ResourceID id);
+
+ // Implements the SetSamplerStates function for D3D9.
+ virtual ParseError SetSamplerStates(ResourceID id,
+ sampler::AddressingMode addressing_u,
+ sampler::AddressingMode addressing_v,
+ sampler::AddressingMode addressing_w,
+ sampler::FilteringMode mag_filter,
+ sampler::FilteringMode min_filter,
+ sampler::FilteringMode mip_filter,
+ unsigned int max_anisotropy);
+
+ // Implements the SetSamplerBorderColor function for D3D9.
+ virtual ParseError SetSamplerBorderColor(ResourceID id, const RGBA &color);
+
+ // Implements the SetSamplerTexture function for D3D9.
+ virtual ParseError SetSamplerTexture(ResourceID id, ResourceID texture_id);
+
+ // Implements the SetScissor function for D3D9.
+ virtual void SetScissor(bool enable,
+ unsigned int x,
+ unsigned int y,
+ unsigned int width,
+ unsigned int height);
+
+ // Implements the SetPointLineRaster function for D3D9.
+ virtual void SetPointLineRaster(bool line_smooth,
+ bool point_sprite,
+ float point_size);
+
+ // Implements the SetPolygonOffset function for D3D9.
+ virtual void SetPolygonOffset(float slope_factor, float units);
+
+ // Implements the SetPolygonRaster function for D3D9.
+ virtual void SetPolygonRaster(PolygonMode fill_mode,
+ FaceCullMode cull_mode);
+
+ // Implements the SetAlphaTest function for D3D9.
+ virtual void SetAlphaTest(bool enable,
+ float reference,
+ Comparison comp);
+
+ // Implements the SetDepthTest function for D3D9.
+ virtual void SetDepthTest(bool enable,
+ bool write_enable,
+ Comparison comp);
+
+ // Implements the SetStencilTest function for D3D9.
+ virtual void SetStencilTest(bool enable,
+ bool separate_ccw,
+ unsigned int write_mask,
+ unsigned int compare_mask,
+ unsigned int ref,
+ Uint32 func_ops);
+
+ // Implements the SetColorWritefunction for D3D9.
+ virtual void SetColorWrite(bool red,
+ bool green,
+ bool blue,
+ bool alpha,
+ bool dither);
+
+ // Implements the SetBlending function for D3D9.
+ virtual void SetBlending(bool enable,
+ bool separate_alpha,
+ BlendEq color_eq,
+ BlendFunc color_src_func,
+ BlendFunc color_dst_func,
+ BlendEq alpha_eq,
+ BlendFunc alpha_src_func,
+ BlendFunc alpha_dst_func);
+
+ // Implements the SetBlendingColor function for D3D9.
+ virtual void SetBlendingColor(const RGBA &color);
+
+ // Gets the D3D9 device.
+ IDirect3DDevice9 *d3d_device() const { return d3d_device_; }
+
+ // Gets a vertex buffer by resource ID.
+ VertexBufferD3D9 *GetVertexBuffer(ResourceID id) {
+ return vertex_buffers_.Get(id);
+ }
+
+ // Gets a texture by resource ID.
+ TextureD3D9 *GetTexture(ResourceID id) {
+ return textures_.Get(id);
+ }
+
+ // Gets a sampler by resource ID.
+ SamplerD3D9 *GetSampler(ResourceID id) {
+ return samplers_.Get(id);
+ }
+
+ EffectD3D9 *current_effect() { return current_effect_; }
+ private:
+ // Validates the current vertex struct to D3D, setting the streams.
+ bool ValidateStreams();
+ // Validates the current effect to D3D. This sends the effect states to D3D.
+ bool ValidateEffect();
+ // "Dirty" the current effect. This resets the effect states to D3D, and
+ // requires ValidateEffect() to be called before further draws occur.
+ void DirtyEffect();
+
+ LPDIRECT3D9 d3d_;
+ LPDIRECT3DDEVICE9 d3d_device_;
+ HWND hwnd_;
+ ResourceID current_vertex_struct_;
+ bool validate_streams_;
+ unsigned int max_vertices_;
+ ResourceID current_effect_id_;
+ bool validate_effect_;
+ EffectD3D9 *current_effect_;
+
+ ResourceMap<VertexBufferD3D9> vertex_buffers_;
+ ResourceMap<IndexBufferD3D9> index_buffers_;
+ ResourceMap<VertexStructD3D9> vertex_structs_;
+ ResourceMap<EffectD3D9> effects_;
+ ResourceMap<EffectParamD3D9> effect_params_;
+ ResourceMap<TextureD3D9> textures_;
+ ResourceMap<SamplerD3D9> samplers_;
+};
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_WIN_D3D9_GAPI_D3D9_H__
diff --git a/o3d/command_buffer/service/win/d3d9/geometry_d3d9.cc b/o3d/command_buffer/service/win/d3d9/geometry_d3d9.cc
new file mode 100644
index 0000000..b81dcc3
--- /dev/null
+++ b/o3d/command_buffer/service/win/d3d9/geometry_d3d9.cc
@@ -0,0 +1,428 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the implementation of the D3D9 versions of the
+// VertexBuffer, IndexBuffer and VertexStruct resources.
+// This file also contains the related GAPID3D9 function implementations.
+
+#include <algorithm>
+#include "command_buffer/service/win/d3d9/d3d9_utils.h"
+#include "command_buffer/service/win/d3d9/geometry_d3d9.h"
+#include "command_buffer/service/win/d3d9/gapi_d3d9.h"
+
+namespace o3d {
+namespace command_buffer {
+
+// Destroys the D3D9 vertex buffer.
+VertexBufferD3D9::~VertexBufferD3D9() {
+ DCHECK(d3d_vertex_buffer_ != NULL);
+ if (d3d_vertex_buffer_) {
+ d3d_vertex_buffer_->Release();
+ d3d_vertex_buffer_ = NULL;
+ }
+}
+
+// Creates a D3D9 vertex buffer.
+void VertexBufferD3D9::Create(GAPID3D9 *gapi) {
+ DCHECK(d3d_vertex_buffer_ == NULL);
+
+ DWORD d3d_usage = (flags() & vertex_buffer::DYNAMIC) ? D3DUSAGE_DYNAMIC : 0;
+ D3DPOOL d3d_pool = D3DPOOL_MANAGED;
+ HR(gapi->d3d_device()->CreateVertexBuffer(size(), d3d_usage, 0, d3d_pool,
+ &d3d_vertex_buffer_, NULL));
+}
+
+// Sets the data into the D3D9 vertex buffer, using Lock() and memcpy.
+bool VertexBufferD3D9::SetData(unsigned int offset,
+ unsigned int size,
+ const void *data) {
+ if (!d3d_vertex_buffer_) {
+ LOG(ERROR) << "Calling SetData on a non-initialized VertexBufferD3D9.";
+ return false;
+ }
+ if ((offset >= this->size()) || (offset + size > this->size())) {
+ LOG(ERROR) << "Invalid size or offset on VertexBufferD3D9::SetData.";
+ return false;
+ }
+ void *ptr = NULL;
+ DWORD lock_flags = 0;
+ // If we are setting the full buffer, discard the old data. That's only
+ // possible to do for a dynamic d3d vertex buffer.
+ if ((offset == 0) && (size == this->size()) &&
+ (flags() & vertex_buffer::DYNAMIC))
+ lock_flags = D3DLOCK_DISCARD;
+ HR(d3d_vertex_buffer_->Lock(offset, size, &ptr, lock_flags));
+ memcpy(ptr, data, size);
+ HR(d3d_vertex_buffer_->Unlock());
+ return true;
+}
+
+// Gets the data from the D3D9 vertex buffer, using Lock() and memcpy.
+bool VertexBufferD3D9::GetData(unsigned int offset,
+ unsigned int size,
+ void *data) {
+ if (!d3d_vertex_buffer_) {
+ LOG(ERROR) << "Calling SetData on a non-initialized VertexBufferD3D9.";
+ return false;
+ }
+ if ((offset >= this->size()) || (offset + size > this->size())) {
+ LOG(ERROR) << "Invalid size or offset on VertexBufferD3D9::SetData.";
+ return false;
+ }
+ void *ptr = NULL;
+ DWORD lock_flags = D3DLOCK_READONLY;
+ HR(d3d_vertex_buffer_->Lock(offset, size, &ptr, lock_flags));
+ memcpy(data, ptr, size);
+ HR(d3d_vertex_buffer_->Unlock());
+ return true;
+}
+
+// Destroys the D3D9 index buffer.
+IndexBufferD3D9::~IndexBufferD3D9() {
+ DCHECK(d3d_index_buffer_ != NULL);
+ if (d3d_index_buffer_) {
+ d3d_index_buffer_->Release();
+ d3d_index_buffer_ = NULL;
+ }
+}
+
+// Creates a D3D9 index buffer.
+void IndexBufferD3D9::Create(GAPID3D9 *gapi) {
+ DCHECK(d3d_index_buffer_ == NULL);
+
+ DWORD d3d_usage = (flags() & index_buffer::DYNAMIC) ? D3DUSAGE_DYNAMIC : 0;
+ D3DFORMAT d3d_format =
+ (flags() & index_buffer::INDEX_32BIT) ? D3DFMT_INDEX32 : D3DFMT_INDEX16;
+ D3DPOOL d3d_pool = D3DPOOL_MANAGED;
+ HR(gapi->d3d_device()->CreateIndexBuffer(size(), d3d_usage, d3d_format,
+ d3d_pool, &d3d_index_buffer_,
+ NULL));
+}
+
+// Sets the data into the D3D9 index buffer, using Lock() and memcpy.
+bool IndexBufferD3D9::SetData(unsigned int offset,
+ unsigned int size,
+ const void *data) {
+ if (!d3d_index_buffer_) {
+ LOG(ERROR) << "Calling SetData on a non-initialized IndexBufferD3D9.";
+ return false;
+ }
+ if ((offset >= this->size()) || (offset + size > this->size())) {
+ LOG(ERROR) << "Invalid size or offset on IndexBufferD3D9::SetData.";
+ return false;
+ }
+ void *ptr = NULL;
+ DWORD lock_flags = 0;
+ // If we are setting the full buffer, discard the old data. That's only
+ // possible to do for a dynamic d3d index buffer.
+ if ((offset == 0) && (size == this->size()) &&
+ (flags() & index_buffer::DYNAMIC))
+ lock_flags = D3DLOCK_DISCARD;
+ HR(d3d_index_buffer_->Lock(offset, size, &ptr, lock_flags));
+ memcpy(ptr, data, size);
+ HR(d3d_index_buffer_->Unlock());
+ return true;
+}
+
+// Gets the data from the D3D9 index buffer, using Lock() and memcpy.
+bool IndexBufferD3D9::GetData(unsigned int offset,
+ unsigned int size,
+ void *data) {
+ if (!d3d_index_buffer_) {
+ LOG(ERROR) << "Calling SetData on a non-initialized IndexBufferD3D9.";
+ return false;
+ }
+ if ((offset >= this->size()) || (offset + size > this->size())) {
+ LOG(ERROR) << "Invalid size or offset on IndexBufferD3D9::SetData.";
+ return false;
+ }
+ void *ptr = NULL;
+ DWORD lock_flags = D3DLOCK_READONLY;
+ HR(d3d_index_buffer_->Lock(offset, size, &ptr, lock_flags));
+ memcpy(data, ptr, size);
+ HR(d3d_index_buffer_->Unlock());
+ return true;
+}
+
+// Sets the input element in the VertexStruct resource.
+void VertexStructD3D9::SetInput(unsigned int input_index,
+ ResourceID vertex_buffer_id,
+ unsigned int offset,
+ unsigned int stride,
+ vertex_struct::Type type,
+ vertex_struct::Semantic semantic,
+ unsigned int semantic_index) {
+ Element &element = GetElement(input_index);
+ element.vertex_buffer = vertex_buffer_id;
+ element.offset = offset;
+ element.stride = stride;
+ element.type = type;
+ element.semantic = semantic;
+ element.semantic_index = semantic_index;
+ dirty_ = true;
+}
+
+// Sets the vdecl and stream sources in D3D9. Compiles them if needed.
+unsigned int VertexStructD3D9::SetStreams(GAPID3D9 *gapi) {
+ IDirect3DDevice9 *d3d_device = gapi->d3d_device();
+ if (dirty_) Compile(d3d_device);
+ HR(d3d_device->SetVertexDeclaration(d3d_vertex_decl_));
+ unsigned int max_vertices = UINT_MAX;
+ for (unsigned int i = 0; i < streams_.size(); ++i) {
+ const StreamPair &pair = streams_[i];
+ VertexBufferD3D9 *vertex_buffer = gapi->GetVertexBuffer(pair.first);
+ if (!vertex_buffer) {
+ max_vertices = 0;
+ continue;
+ }
+ HR(d3d_device->SetStreamSource(i, vertex_buffer->d3d_vertex_buffer(), 0,
+ pair.second));
+ max_vertices = std::min(max_vertices, vertex_buffer->size()/pair.second);
+ }
+ return max_vertices;
+}
+
+// Converts a vertex_struct::Type to a D3DDECLTYPE.
+static D3DDECLTYPE D3DType(vertex_struct::Type type) {
+ switch (type) {
+ case vertex_struct::FLOAT1:
+ return D3DDECLTYPE_FLOAT1;
+ case vertex_struct::FLOAT2:
+ return D3DDECLTYPE_FLOAT2;
+ case vertex_struct::FLOAT3:
+ return D3DDECLTYPE_FLOAT3;
+ case vertex_struct::FLOAT4:
+ return D3DDECLTYPE_FLOAT4;
+ case vertex_struct::UCHAR4N:
+ return D3DDECLTYPE_UBYTE4N;
+ case vertex_struct::NUM_TYPES:
+ break;
+ }
+ LOG(FATAL) << "Invalid type";
+ return D3DDECLTYPE_FLOAT1;
+}
+
+// Converts a vertex_struct::Semantic to a D3DDECLUSAGE.
+static D3DDECLUSAGE D3DUsage(vertex_struct::Semantic semantic) {
+ switch (semantic) {
+ case vertex_struct::POSITION:
+ return D3DDECLUSAGE_POSITION;
+ case vertex_struct::NORMAL:
+ return D3DDECLUSAGE_NORMAL;
+ case vertex_struct::COLOR:
+ return D3DDECLUSAGE_COLOR;
+ case vertex_struct::TEX_COORD:
+ return D3DDECLUSAGE_TEXCOORD;
+ case vertex_struct::NUM_SEMANTICS:
+ break;
+ }
+ LOG(FATAL) << "Invalid type";
+ return D3DDECLUSAGE_POSITION;
+}
+
+// Destroys the d3d vertex declaration.
+VertexStructD3D9::~VertexStructD3D9() {
+ Destroy();
+}
+
+void VertexStructD3D9::Destroy() {
+ if (d3d_vertex_decl_) {
+ d3d_vertex_decl_->Release();
+ d3d_vertex_decl_ = NULL;
+ }
+ streams_.clear();
+}
+
+// Compiles a stream map and a d3d vertex declaration from the list of inputs.
+// 2 inputs that use the same vertex buffer and stride will use the same
+// d3d stream.
+void VertexStructD3D9::Compile(IDirect3DDevice9 *d3d_device) {
+ DCHECK(dirty_);
+ Destroy();
+ streams_.reserve(count_);
+ scoped_array<D3DVERTEXELEMENT9> d3d_elements(
+ new D3DVERTEXELEMENT9[count_ + 1]);
+ memset(d3d_elements.get(), 0, sizeof(D3DVERTEXELEMENT9) * (count_ + 1));
+ // build streams_ like a set, but the order matters.
+ for (unsigned int i = 0; i < count_ ; ++i) {
+ Element &element = GetElement(i);
+ D3DVERTEXELEMENT9 &d3d_element = d3d_elements[i];
+ StreamPair pair(element.vertex_buffer, element.stride);
+ std::vector<StreamPair>::iterator it =
+ std::find(streams_.begin(), streams_.end(), pair);
+ unsigned int stream_index = 0;
+ if (it == streams_.end()) {
+ streams_.push_back(pair);
+ stream_index = static_cast<unsigned int>(streams_.size() - 1);
+ } else {
+ stream_index = it - streams_.begin();
+ }
+ d3d_element.Stream = stream_index;
+ d3d_element.Offset = element.offset;
+ d3d_element.Type = D3DType(element.type);
+ d3d_element.Usage = D3DUsage(element.semantic);
+ d3d_element.UsageIndex = element.semantic_index;
+ }
+ D3DVERTEXELEMENT9 &end = d3d_elements[count_];
+ end.Stream = 0xFF;
+ end.Type = D3DDECLTYPE_UNUSED;
+ HR(d3d_device->CreateVertexDeclaration(d3d_elements.get(),
+ &d3d_vertex_decl_));
+ dirty_ = false;
+}
+
+// Creates and assigns a VertexBufferD3D9 resource.
+BufferSyncInterface::ParseError GAPID3D9::CreateVertexBuffer(
+ ResourceID id,
+ unsigned int size,
+ unsigned int flags) {
+ VertexBufferD3D9 *vertex_buffer = new VertexBufferD3D9(size, flags);
+ vertex_buffer->Create(this);
+ vertex_buffers_.Assign(id, vertex_buffer);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+// Destroys a VertexBufferD3D9 resource.
+BufferSyncInterface::ParseError GAPID3D9::DestroyVertexBuffer(ResourceID id) {
+ return vertex_buffers_.Destroy(id) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+// Copies the data into the VertexBufferD3D9 resource.
+BufferSyncInterface::ParseError GAPID3D9::SetVertexBufferData(
+ ResourceID id,
+ unsigned int offset,
+ unsigned int size,
+ const void *data) {
+ VertexBufferD3D9 *vertex_buffer = vertex_buffers_.Get(id);
+ if (!vertex_buffer) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return vertex_buffer->SetData(offset, size, data) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+// Copies the data from the VertexBufferD3D9 resource.
+BufferSyncInterface::ParseError GAPID3D9::GetVertexBufferData(
+ ResourceID id,
+ unsigned int offset,
+ unsigned int size,
+ void *data) {
+ VertexBufferD3D9 *vertex_buffer = vertex_buffers_.Get(id);
+ if (!vertex_buffer) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return vertex_buffer->GetData(offset, size, data) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+// Creates and assigns an IndexBufferD3D9 resource.
+BufferSyncInterface::ParseError GAPID3D9::CreateIndexBuffer(
+ ResourceID id,
+ unsigned int size,
+ unsigned int flags) {
+ IndexBufferD3D9 *index_buffer = new IndexBufferD3D9(size, flags);
+ index_buffer->Create(this);
+ index_buffers_.Assign(id, index_buffer);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+// Destroys an IndexBufferD3D9 resource.
+BufferSyncInterface::ParseError GAPID3D9::DestroyIndexBuffer(ResourceID id) {
+ return index_buffers_.Destroy(id) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+// Copies the data into the IndexBufferD3D9 resource.
+BufferSyncInterface::ParseError GAPID3D9::SetIndexBufferData(
+ ResourceID id,
+ unsigned int offset,
+ unsigned int size,
+ const void *data) {
+ IndexBufferD3D9 *index_buffer = index_buffers_.Get(id);
+ if (!index_buffer) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return index_buffer->SetData(offset, size, data) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+// Copies the data from the IndexBufferD3D9 resource.
+BufferSyncInterface::ParseError GAPID3D9::GetIndexBufferData(
+ ResourceID id,
+ unsigned int offset,
+ unsigned int size,
+ void *data) {
+ IndexBufferD3D9 *index_buffer = index_buffers_.Get(id);
+ if (!index_buffer) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ return index_buffer->GetData(offset, size, data) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+// Creates and assigns a VertexStructD3D9 resource.
+BufferSyncInterface::ParseError GAPID3D9::CreateVertexStruct(
+ ResourceID id, unsigned int input_count) {
+ if (id == current_vertex_struct_) validate_streams_ = true;
+ VertexStructD3D9 *vertex_struct = new VertexStructD3D9(input_count);
+ vertex_structs_.Assign(id, vertex_struct);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+// Destroys a VertexStructD3D9 resource.
+BufferSyncInterface::ParseError GAPID3D9::DestroyVertexStruct(ResourceID id) {
+ if (id == current_vertex_struct_) validate_streams_ = true;
+ return vertex_structs_.Destroy(id) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+// Sets an input into a VertexStructD3D9 resource.
+BufferSyncInterface::ParseError GAPID3D9::SetVertexInput(
+ ResourceID vertex_struct_id,
+ unsigned int input_index,
+ ResourceID vertex_buffer_id,
+ unsigned int offset,
+ unsigned int stride,
+ vertex_struct::Type type,
+ vertex_struct::Semantic semantic,
+ unsigned int semantic_index) {
+ if (vertex_buffer_id == current_vertex_struct_) validate_streams_ = true;
+ VertexStructD3D9 *vertex_struct = vertex_structs_.Get(vertex_struct_id);
+ if (!vertex_struct || input_index >= vertex_struct->count())
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ vertex_struct->SetInput(input_index, vertex_buffer_id, offset, stride, type,
+ semantic, semantic_index);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/win/d3d9/geometry_d3d9.h b/o3d/command_buffer/service/win/d3d9/geometry_d3d9.h
new file mode 100644
index 0000000..4182b5b
--- /dev/null
+++ b/o3d/command_buffer/service/win/d3d9/geometry_d3d9.h
@@ -0,0 +1,126 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the definition of the D3D9 versions of geometry-related
+// resource classes.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_WIN_D3D9_GEOMETRY_D3D9_H__
+#define O3D_COMMAND_BUFFER_SERVICE_WIN_D3D9_GEOMETRY_D3D9_H__
+
+#include <vector>
+#include <utility>
+#include "command_buffer/common/cross/gapi_interface.h"
+#include "command_buffer/service/win/d3d9/d3d9_utils.h"
+#include "command_buffer/service/cross/resource.h"
+
+namespace o3d {
+namespace command_buffer {
+
+class GAPID3D9;
+
+// D3D9 version of VertexBuffer.
+class VertexBufferD3D9 : public VertexBuffer {
+ public:
+ VertexBufferD3D9(unsigned int size, unsigned int flags)
+ : VertexBuffer(size, flags),
+ d3d_vertex_buffer_(NULL) {}
+ virtual ~VertexBufferD3D9();
+ // Creates the D3D vertex buffer.
+ void Create(GAPID3D9 *gapi);
+ // Sets the data into the D3D vertex buffer.
+ bool SetData(unsigned int offset, unsigned int size, const void *data);
+ // Gets the data from the D3D vertex buffer.
+ bool GetData(unsigned int offset, unsigned int size, void *data);
+
+ // Gets the D3D vertex buffer.
+ IDirect3DVertexBuffer9 * d3d_vertex_buffer() { return d3d_vertex_buffer_; }
+ private:
+ IDirect3DVertexBuffer9 *d3d_vertex_buffer_;
+ DISALLOW_COPY_AND_ASSIGN(VertexBufferD3D9);
+};
+
+// D3D9 version of IndexBuffer.
+class IndexBufferD3D9 : public IndexBuffer {
+ public:
+ IndexBufferD3D9(unsigned int size, unsigned int flags)
+ : IndexBuffer(size, flags),
+ d3d_index_buffer_(NULL) {}
+ virtual ~IndexBufferD3D9();
+ // Creates the D3D index buffer.
+ void Create(GAPID3D9 *gapi);
+ // Sets the data into the D3D index buffer.
+ bool SetData(unsigned int offset, unsigned int size, const void *data);
+ // Gets the data from the D3D index buffer.
+ bool GetData(unsigned int offset, unsigned int size, void *data);
+
+ // Gets the D3D index buffer.
+ IDirect3DIndexBuffer9 *d3d_index_buffer() const { return d3d_index_buffer_; }
+ private:
+ IDirect3DIndexBuffer9 *d3d_index_buffer_;
+ DISALLOW_COPY_AND_ASSIGN(IndexBufferD3D9);
+};
+
+// D3D9 version of VertexStruct.
+class VertexStructD3D9 : public VertexStruct {
+ public:
+ explicit VertexStructD3D9(unsigned int count)
+ : VertexStruct(count),
+ dirty_(true),
+ d3d_vertex_decl_(NULL) {}
+ virtual ~VertexStructD3D9();
+ // Adds an input to the vertex struct.
+ void SetInput(unsigned int input_index,
+ ResourceID vertex_buffer_id,
+ unsigned int offset,
+ unsigned int stride,
+ vertex_struct::Type type,
+ vertex_struct::Semantic semantic,
+ unsigned int semantic_index);
+ // Sets the input streams to D3D.
+ unsigned int SetStreams(GAPID3D9 *gapi);
+ private:
+ // Destroys the vertex declaration and stream map.
+ void Destroy();
+ // Compiles the vertex declaration and stream map.
+ void Compile(IDirect3DDevice9 *d3d_device);
+
+ bool dirty_;
+ typedef std::pair<ResourceID, unsigned int> StreamPair;
+ std::vector<StreamPair> streams_;
+ IDirect3DVertexDeclaration9 *d3d_vertex_decl_;
+ DISALLOW_COPY_AND_ASSIGN(VertexStructD3D9);
+};
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_WIN_D3D9_GEOMETRY_D3D9_H__
diff --git a/o3d/command_buffer/service/win/d3d9/sampler_d3d9.cc b/o3d/command_buffer/service/win/d3d9/sampler_d3d9.cc
new file mode 100644
index 0000000..928d577
--- /dev/null
+++ b/o3d/command_buffer/service/win/d3d9/sampler_d3d9.cc
@@ -0,0 +1,199 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the implementation of the SamplerD3D9 class.
+
+#include "command_buffer/service/win/d3d9/gapi_d3d9.h"
+#include "command_buffer/service/win/d3d9/sampler_d3d9.h"
+#include "command_buffer/service/win/d3d9/texture_d3d9.h"
+
+namespace o3d {
+namespace command_buffer {
+
+namespace {
+
+// Converts an addressing mode to corresponding D3D values.
+D3DTEXTUREADDRESS AddressModeToD3D(sampler::AddressingMode mode) {
+ switch (mode) {
+ case sampler::WRAP:
+ return D3DTADDRESS_WRAP;
+ case sampler::MIRROR_REPEAT:
+ return D3DTADDRESS_MIRROR;
+ case sampler::CLAMP_TO_EDGE:
+ return D3DTADDRESS_CLAMP;
+ case sampler::CLAMP_TO_BORDER:
+ return D3DTADDRESS_BORDER;
+ }
+ DLOG(FATAL) << "Not reached";
+ return D3DTADDRESS_WRAP;
+}
+
+// Converts a filtering mode to corresponding D3D values.
+D3DTEXTUREFILTERTYPE FilteringModeToD3D(sampler::FilteringMode mode) {
+ switch (mode) {
+ case sampler::NONE:
+ return D3DTEXF_NONE;
+ case sampler::POINT:
+ return D3DTEXF_POINT;
+ case sampler::LINEAR:
+ return D3DTEXF_LINEAR;
+ }
+ DLOG(FATAL) << "Not reached";
+ return D3DTEXF_POINT;
+}
+
+} // anonymous namespace
+
+SamplerD3D9::SamplerD3D9()
+ : texture_id_(kInvalidResource) {
+ SetStates(sampler::CLAMP_TO_EDGE,
+ sampler::CLAMP_TO_EDGE,
+ sampler::CLAMP_TO_EDGE,
+ sampler::LINEAR,
+ sampler::LINEAR,
+ sampler::POINT,
+ 1);
+ RGBA black = {0, 0, 0, 1};
+ SetBorderColor(black);
+}
+
+bool SamplerD3D9::ApplyStates(GAPID3D9 *gapi, unsigned int unit) const {
+ DCHECK(gapi);
+ TextureD3D9 *texture = gapi->GetTexture(texture_id_);
+ if (!texture) {
+ return false;
+ }
+ IDirect3DDevice9 *d3d_device = gapi->d3d_device();
+ HR(d3d_device->SetTexture(unit, texture->d3d_base_texture()));
+ HR(d3d_device->SetSamplerState(unit, D3DSAMP_ADDRESSU, d3d_address_u_));
+ HR(d3d_device->SetSamplerState(unit, D3DSAMP_ADDRESSV, d3d_address_v_));
+ HR(d3d_device->SetSamplerState(unit, D3DSAMP_ADDRESSW, d3d_address_w_));
+ HR(d3d_device->SetSamplerState(unit, D3DSAMP_MAGFILTER, d3d_mag_filter_));
+ HR(d3d_device->SetSamplerState(unit, D3DSAMP_MINFILTER, d3d_min_filter_));
+ HR(d3d_device->SetSamplerState(unit, D3DSAMP_MIPFILTER, d3d_mip_filter_));
+ HR(d3d_device->SetSamplerState(unit, D3DSAMP_MAXANISOTROPY,
+ d3d_max_anisotropy_));
+ HR(d3d_device->SetSamplerState(unit, D3DSAMP_BORDERCOLOR, d3d_border_color_));
+ return true;
+}
+
+void SamplerD3D9::SetStates(sampler::AddressingMode addressing_u,
+ sampler::AddressingMode addressing_v,
+ sampler::AddressingMode addressing_w,
+ sampler::FilteringMode mag_filter,
+ sampler::FilteringMode min_filter,
+ sampler::FilteringMode mip_filter,
+ unsigned int max_anisotropy) {
+ // These are validated in GAPIDecoder.cc
+ DCHECK_NE(mag_filter, sampler::NONE);
+ DCHECK_NE(min_filter, sampler::NONE);
+ DCHECK_GT(max_anisotropy, 0);
+ d3d_address_u_ = AddressModeToD3D(addressing_u);
+ d3d_address_v_ = AddressModeToD3D(addressing_v);
+ d3d_address_w_ = AddressModeToD3D(addressing_w);
+ d3d_mag_filter_ = FilteringModeToD3D(mag_filter);
+ d3d_min_filter_ = FilteringModeToD3D(min_filter);
+ d3d_mip_filter_ = FilteringModeToD3D(mip_filter);
+ if (max_anisotropy > 1) {
+ d3d_mag_filter_ = D3DTEXF_ANISOTROPIC;
+ d3d_min_filter_ = D3DTEXF_ANISOTROPIC;
+ }
+ d3d_max_anisotropy_ = max_anisotropy;
+}
+
+void SamplerD3D9::SetBorderColor(const RGBA &color) {
+ d3d_border_color_ = RGBAToD3DCOLOR(color);
+}
+
+BufferSyncInterface::ParseError GAPID3D9::CreateSampler(
+ ResourceID id) {
+ // Dirty effect, because this sampler id may be used
+ DirtyEffect();
+ samplers_.Assign(id, new SamplerD3D9());
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+// Destroys the Sampler resource.
+BufferSyncInterface::ParseError GAPID3D9::DestroySampler(ResourceID id) {
+ // Dirty effect, because this sampler id may be used
+ DirtyEffect();
+ return samplers_.Destroy(id) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+BufferSyncInterface::ParseError GAPID3D9::SetSamplerStates(
+ ResourceID id,
+ sampler::AddressingMode addressing_u,
+ sampler::AddressingMode addressing_v,
+ sampler::AddressingMode addressing_w,
+ sampler::FilteringMode mag_filter,
+ sampler::FilteringMode min_filter,
+ sampler::FilteringMode mip_filter,
+ unsigned int max_anisotropy) {
+ SamplerD3D9 *sampler = samplers_.Get(id);
+ if (!sampler)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ // Dirty effect, because this sampler id may be used
+ DirtyEffect();
+ sampler->SetStates(addressing_u, addressing_v, addressing_w,
+ mag_filter, min_filter, mip_filter, max_anisotropy);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+BufferSyncInterface::ParseError GAPID3D9::SetSamplerBorderColor(
+ ResourceID id,
+ const RGBA &color) {
+ SamplerD3D9 *sampler = samplers_.Get(id);
+ if (!sampler)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ // Dirty effect, because this sampler id may be used
+ DirtyEffect();
+ sampler->SetBorderColor(color);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+BufferSyncInterface::ParseError GAPID3D9::SetSamplerTexture(
+ ResourceID id,
+ ResourceID texture_id) {
+ SamplerD3D9 *sampler = samplers_.Get(id);
+ if (!sampler)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ // Dirty effect, because this sampler id may be used
+ DirtyEffect();
+ sampler->SetTexture(texture_id);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/win/d3d9/sampler_d3d9.h b/o3d/command_buffer/service/win/d3d9/sampler_d3d9.h
new file mode 100644
index 0000000..e029963
--- /dev/null
+++ b/o3d/command_buffer/service/win/d3d9/sampler_d3d9.h
@@ -0,0 +1,85 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the definition of the SamplerD3D9 class, implementing
+// samplers for D3D.
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_WIN_D3D9_SAMPLER_D3D9_H_
+#define O3D_COMMAND_BUFFER_SERVICE_WIN_D3D9_SAMPLER_D3D9_H_
+
+#include "command_buffer/common/cross/gapi_interface.h"
+#include "command_buffer/service/win/d3d9/d3d9_utils.h"
+#include "command_buffer/service/cross/resource.h"
+
+namespace o3d {
+namespace command_buffer {
+
+class GAPID3D9;
+
+// D3D9 version of Sampler.
+class SamplerD3D9 : public Sampler {
+ public:
+ SamplerD3D9();
+
+ // Applies sampler states to D3D.
+ bool ApplyStates(GAPID3D9 *gapi, unsigned int unit) const;
+
+ // Sets sampler states.
+ void SetStates(sampler::AddressingMode addressing_u,
+ sampler::AddressingMode addressing_v,
+ sampler::AddressingMode addressing_w,
+ sampler::FilteringMode mag_filter,
+ sampler::FilteringMode min_filter,
+ sampler::FilteringMode mip_filter,
+ unsigned int max_anisotropy);
+
+ // Sets the border color states.
+ void SetBorderColor(const RGBA &color);
+
+ // Sets the texture.
+ void SetTexture(ResourceID texture) { texture_id_ = texture; }
+ private:
+ D3DTEXTUREADDRESS d3d_address_u_;
+ D3DTEXTUREADDRESS d3d_address_v_;
+ D3DTEXTUREADDRESS d3d_address_w_;
+ D3DTEXTUREFILTERTYPE d3d_mag_filter_;
+ D3DTEXTUREFILTERTYPE d3d_min_filter_;
+ D3DTEXTUREFILTERTYPE d3d_mip_filter_;
+ DWORD d3d_max_anisotropy_;
+ D3DCOLOR d3d_border_color_;
+ ResourceID texture_id_;
+};
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_WIN_D3D9_SAMPLER_D3D9_H_
diff --git a/o3d/command_buffer/service/win/d3d9/states_d3d9.cc b/o3d/command_buffer/service/win/d3d9/states_d3d9.cc
new file mode 100644
index 0000000..4a28db9
--- /dev/null
+++ b/o3d/command_buffer/service/win/d3d9/states_d3d9.cc
@@ -0,0 +1,349 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file contains the implementation of the state-related GAPID3D9
+// functions.
+
+#include <algorithm>
+#include "command_buffer/common/cross/cmd_buffer_format.h"
+#include "command_buffer/service/win/d3d9/gapi_d3d9.h"
+
+namespace o3d {
+namespace command_buffer {
+
+namespace {
+
+// Checks that a GAPIInterface enum matches a D3D enum so that it can be
+// converted quickly.
+#define CHECK_GAPI_ENUM_MATCHES_D3D(GAPI_ENUM, D3D_ENUM) \
+ COMPILE_ASSERT(GAPIInterface::GAPI_ENUM + 1 == D3D_ENUM, \
+ GAPI_ENUM ## _plus_1_not_ ## D3D_ENUM)
+
+// Converts values from the PolygonMode enum to corresponding D3D values
+inline D3DFILLMODE PolygonModeToD3D(GAPIInterface::PolygonMode fill_mode) {
+ DCHECK_LT(fill_mode, GAPIInterface::NUM_POLYGON_MODE);
+
+ // Check that all acceptable values translate to D3D values by adding 1.
+
+ CHECK_GAPI_ENUM_MATCHES_D3D(POLYGON_MODE_POINTS, D3DFILL_POINT);
+ CHECK_GAPI_ENUM_MATCHES_D3D(POLYGON_MODE_LINES, D3DFILL_WIREFRAME);
+ CHECK_GAPI_ENUM_MATCHES_D3D(POLYGON_MODE_FILL, D3DFILL_SOLID);
+ return static_cast<D3DFILLMODE>(fill_mode + 1);
+}
+
+// Converts values from the FaceCullMode enum to corresponding D3D values
+inline D3DCULL FaceCullModeToD3D(GAPIInterface::FaceCullMode cull_mode) {
+ DCHECK_LT(cull_mode, GAPIInterface::NUM_FACE_CULL_MODE);
+
+ // Check that all acceptable values translate to D3D values by adding 1.
+ CHECK_GAPI_ENUM_MATCHES_D3D(CULL_NONE, D3DCULL_NONE);
+ CHECK_GAPI_ENUM_MATCHES_D3D(CULL_CW, D3DCULL_CW);
+ CHECK_GAPI_ENUM_MATCHES_D3D(CULL_CCW, D3DCULL_CCW);
+ return static_cast<D3DCULL>(cull_mode + 1);
+}
+
+// Converts values from the Comparison enum to corresponding D3D values
+inline D3DCMPFUNC ComparisonToD3D(GAPIInterface::Comparison comp) {
+ DCHECK_LT(comp, GAPIInterface::NUM_COMPARISON);
+
+ // Check that all acceptable values translate to D3D values by adding 1.
+ CHECK_GAPI_ENUM_MATCHES_D3D(NEVER, D3DCMP_NEVER);
+ CHECK_GAPI_ENUM_MATCHES_D3D(LESS, D3DCMP_LESS);
+ CHECK_GAPI_ENUM_MATCHES_D3D(EQUAL, D3DCMP_EQUAL);
+ CHECK_GAPI_ENUM_MATCHES_D3D(LEQUAL, D3DCMP_LESSEQUAL);
+ CHECK_GAPI_ENUM_MATCHES_D3D(GREATER, D3DCMP_GREATER);
+ CHECK_GAPI_ENUM_MATCHES_D3D(NOT_EQUAL, D3DCMP_NOTEQUAL);
+ CHECK_GAPI_ENUM_MATCHES_D3D(GEQUAL, D3DCMP_GREATEREQUAL);
+ CHECK_GAPI_ENUM_MATCHES_D3D(ALWAYS, D3DCMP_ALWAYS);
+ return static_cast<D3DCMPFUNC>(comp + 1);
+}
+
+// Converts values from the StencilOp enum to corresponding D3D values
+inline D3DSTENCILOP StencilOpToD3D(GAPIInterface::StencilOp stencil_op) {
+ DCHECK_LT(stencil_op, GAPIInterface::NUM_STENCIL_OP);
+
+ // Check that all acceptable values translate to D3D values by adding 1.
+ CHECK_GAPI_ENUM_MATCHES_D3D(KEEP, D3DSTENCILOP_KEEP);
+ CHECK_GAPI_ENUM_MATCHES_D3D(ZERO, D3DSTENCILOP_ZERO);
+ CHECK_GAPI_ENUM_MATCHES_D3D(REPLACE, D3DSTENCILOP_REPLACE);
+ CHECK_GAPI_ENUM_MATCHES_D3D(INC_NO_WRAP, D3DSTENCILOP_INCRSAT);
+ CHECK_GAPI_ENUM_MATCHES_D3D(DEC_NO_WRAP, D3DSTENCILOP_DECRSAT);
+ CHECK_GAPI_ENUM_MATCHES_D3D(INVERT, D3DSTENCILOP_INVERT);
+ CHECK_GAPI_ENUM_MATCHES_D3D(INC_WRAP, D3DSTENCILOP_INCR);
+ CHECK_GAPI_ENUM_MATCHES_D3D(DEC_WRAP, D3DSTENCILOP_DECR);
+ return static_cast<D3DSTENCILOP>(stencil_op + 1);
+}
+
+// Converts values from the BlendEq enum to corresponding D3D values
+inline D3DBLENDOP BlendEqToD3D(GAPIInterface::BlendEq blend_eq) {
+ DCHECK_LT(blend_eq, GAPIInterface::NUM_BLEND_EQ);
+ // Check that all acceptable values translate to D3D values by adding 1.
+ CHECK_GAPI_ENUM_MATCHES_D3D(BLEND_EQ_ADD, D3DBLENDOP_ADD);
+ CHECK_GAPI_ENUM_MATCHES_D3D(BLEND_EQ_SUB, D3DBLENDOP_SUBTRACT);
+ CHECK_GAPI_ENUM_MATCHES_D3D(BLEND_EQ_REV_SUB, D3DBLENDOP_REVSUBTRACT);
+ CHECK_GAPI_ENUM_MATCHES_D3D(BLEND_EQ_MIN, D3DBLENDOP_MIN);
+ CHECK_GAPI_ENUM_MATCHES_D3D(BLEND_EQ_MAX, D3DBLENDOP_MAX);
+ return static_cast<D3DBLENDOP>(blend_eq + 1);
+}
+
+// Converts values from the BlendFunc enum to corresponding D3D values
+D3DBLEND BlendFuncToD3D(GAPIInterface::BlendFunc blend_func) {
+ // The D3DBLEND enum values don't map 1-to-1 to BlendFunc, so we use a switch
+ // here.
+ switch (blend_func) {
+ case GAPIInterface::BLEND_FUNC_ZERO:
+ return D3DBLEND_ZERO;
+ case GAPIInterface::BLEND_FUNC_ONE:
+ return D3DBLEND_ONE;
+ case GAPIInterface::BLEND_FUNC_SRC_COLOR:
+ return D3DBLEND_SRCCOLOR;
+ case GAPIInterface::BLEND_FUNC_INV_SRC_COLOR:
+ return D3DBLEND_INVSRCCOLOR;
+ case GAPIInterface::BLEND_FUNC_SRC_ALPHA:
+ return D3DBLEND_SRCALPHA;
+ case GAPIInterface::BLEND_FUNC_INV_SRC_ALPHA:
+ return D3DBLEND_INVSRCALPHA;
+ case GAPIInterface::BLEND_FUNC_DST_ALPHA:
+ return D3DBLEND_DESTALPHA;
+ case GAPIInterface::BLEND_FUNC_INV_DST_ALPHA:
+ return D3DBLEND_INVDESTALPHA;
+ case GAPIInterface::BLEND_FUNC_DST_COLOR:
+ return D3DBLEND_DESTCOLOR;
+ case GAPIInterface::BLEND_FUNC_INV_DST_COLOR:
+ return D3DBLEND_INVDESTCOLOR;
+ case GAPIInterface::BLEND_FUNC_SRC_ALPHA_SATUTRATE:
+ return D3DBLEND_SRCALPHASAT;
+ case GAPIInterface::BLEND_FUNC_BLEND_COLOR:
+ return D3DBLEND_BLENDFACTOR;
+ case GAPIInterface::BLEND_FUNC_INV_BLEND_COLOR:
+ return D3DBLEND_INVBLENDFACTOR;
+ default:
+ DLOG(FATAL) << "Invalid BlendFunc";
+ return D3DBLEND_ZERO;
+ }
+}
+
+// Decodes stencil test function and operations from the bitfield.
+void DecodeStencilFuncOps(Uint32 params,
+ GAPIInterface::Comparison *func,
+ GAPIInterface::StencilOp *pass,
+ GAPIInterface::StencilOp *fail,
+ GAPIInterface::StencilOp *zfail) {
+ namespace cmd = set_stencil_test;
+ // Sanity check. The value has already been tested in
+ // GAPIDecoder::DecodeSetStencilTest in gapi_decoder.cc.
+ DCHECK_EQ(cmd::Unused1::Get(params), 0);
+ // Check that the bitmask get cannot generate values outside of the allowed
+ // range.
+ COMPILE_ASSERT(cmd::CWFunc::kMask < GAPIInterface::NUM_COMPARISON,
+ set_stencil_test_CWFunc_may_produce_invalid_values);
+ *func = static_cast<GAPIInterface::Comparison>(cmd::CWFunc::Get(params));
+
+ COMPILE_ASSERT(cmd::CWPassOp::kMask < GAPIInterface::NUM_STENCIL_OP,
+ set_stencil_test_CWPassOp_may_produce_invalid_values);
+ *pass = static_cast<GAPIInterface::StencilOp>(cmd::CWPassOp::Get(params));
+
+ COMPILE_ASSERT(cmd::CWFailOp::kMask < GAPIInterface::NUM_STENCIL_OP,
+ set_stencil_test_CWFailOp_may_produce_invalid_values);
+ *fail = static_cast<GAPIInterface::StencilOp>(cmd::CWFailOp::Get(params));
+
+ COMPILE_ASSERT(cmd::CWZFailOp::kMask < GAPIInterface::NUM_STENCIL_OP,
+ set_stencil_test_CWZFailOp_may_produce_invalid_values);
+ *zfail = static_cast<GAPIInterface::StencilOp>(cmd::CWZFailOp::Get(params));
+}
+
+} // anonymous namespace
+
+void GAPID3D9::SetScissor(bool enable,
+ unsigned int x,
+ unsigned int y,
+ unsigned int width,
+ unsigned int height) {
+ HR(d3d_device_->SetRenderState(D3DRS_SCISSORTESTENABLE,
+ enable ? TRUE : FALSE));
+ RECT rect = {x, y, x + width, y + height};
+ HR(d3d_device_->SetScissorRect(&rect));
+}
+
+void GAPID3D9::SetPolygonOffset(float slope_factor, float units) {
+ HR(d3d_device_->SetRenderState(D3DRS_SLOPESCALEDEPTHBIAS,
+ FloatAsDWORD(slope_factor)));
+ // TODO: this value is hard-coded currently because we only create a
+ // 24-bit depth buffer. Move this to a member of GAPID3D9 if this changes.
+ const float kUnitScale = 1.f / (1 << 24);
+ HR(d3d_device_->SetRenderState(D3DRS_DEPTHBIAS,
+ FloatAsDWORD(units * kUnitScale)));
+}
+
+void GAPID3D9::SetPointLineRaster(bool line_smooth,
+ bool point_sprite,
+ float point_size) {
+ HR(d3d_device_->SetRenderState(D3DRS_ANTIALIASEDLINEENABLE,
+ line_smooth ? TRUE : FALSE));
+ HR(d3d_device_->SetRenderState(D3DRS_POINTSPRITEENABLE,
+ point_sprite ? TRUE : FALSE));
+ HR(d3d_device_->SetRenderState(D3DRS_POINTSIZE,
+ FloatAsDWORD(point_size)));
+}
+
+void GAPID3D9::SetPolygonRaster(PolygonMode fill_mode,
+ FaceCullMode cull_mode) {
+ HR(d3d_device_->SetRenderState(D3DRS_FILLMODE, PolygonModeToD3D(fill_mode)));
+ HR(d3d_device_->SetRenderState(D3DRS_CULLMODE, FaceCullModeToD3D(cull_mode)));
+}
+
+void GAPID3D9::SetAlphaTest(bool enable,
+ float reference,
+ Comparison comp) {
+ HR(d3d_device_->SetRenderState(D3DRS_ALPHABLENDENABLE,
+ enable ? TRUE : FALSE));
+ HR(d3d_device_->SetRenderState(D3DRS_ALPHAREF,
+ FloatToClampedByte(reference)));
+ HR(d3d_device_->SetRenderState(D3DRS_ALPHAFUNC, ComparisonToD3D(comp)));
+}
+
+void GAPID3D9::SetDepthTest(bool enable,
+ bool write_enable,
+ Comparison comp) {
+ HR(d3d_device_->SetRenderState(D3DRS_ZENABLE,
+ enable ? TRUE : FALSE));
+ HR(d3d_device_->SetRenderState(D3DRS_ZWRITEENABLE,
+ write_enable ? TRUE : FALSE));
+ HR(d3d_device_->SetRenderState(D3DRS_ZFUNC, ComparisonToD3D(comp)));
+}
+
+void GAPID3D9::SetStencilTest(bool enable,
+ bool separate_ccw,
+ unsigned int write_mask,
+ unsigned int compare_mask,
+ unsigned int ref,
+ Uint32 func_ops) {
+ HR(d3d_device_->SetRenderState(D3DRS_STENCILENABLE,
+ enable ? TRUE : FALSE));
+ HR(d3d_device_->SetRenderState(D3DRS_STENCILWRITEMASK, write_mask));
+ HR(d3d_device_->SetRenderState(D3DRS_STENCILMASK, compare_mask));
+ HR(d3d_device_->SetRenderState(D3DRS_STENCILREF, ref));
+
+ Comparison func;
+ StencilOp pass;
+ StencilOp fail;
+ StencilOp zfail;
+ DecodeStencilFuncOps(func_ops, &func, &pass, &fail, &zfail);
+ HR(d3d_device_->SetRenderState(D3DRS_STENCILFUNC,
+ ComparisonToD3D(func)));
+ HR(d3d_device_->SetRenderState(D3DRS_STENCILPASS,
+ StencilOpToD3D(pass)));
+ HR(d3d_device_->SetRenderState(D3DRS_STENCILFAIL,
+ StencilOpToD3D(fail)));
+ HR(d3d_device_->SetRenderState(D3DRS_STENCILZFAIL,
+ StencilOpToD3D(zfail)));
+
+ if (separate_ccw) {
+ HR(d3d_device_->SetRenderState(D3DRS_TWOSIDEDSTENCILMODE, TRUE));
+ // Check that the definition of the counter-clockwise func/ops match the
+ // clockwise ones, just shifted by 16 bits, so that we can use
+ // DecodeStencilFuncOps on both of them.
+#define CHECK_CCW_MATCHES_CW(FIELD) \
+ COMPILE_ASSERT(set_stencil_test::CW ## FIELD::kLength == \
+ set_stencil_test::CCW ## FIELD::kLength, \
+ CCW ## FIELD ## _length_does_not_match_ ## CW ## FIELD); \
+ COMPILE_ASSERT(set_stencil_test::CW ## FIELD::kShift + 16 == \
+ set_stencil_test::CCW ## FIELD::kShift, \
+ CCW ## FIELD ## _shift_does_not_match_ ## CW ## FIELD)
+ CHECK_CCW_MATCHES_CW(Func);
+ CHECK_CCW_MATCHES_CW(PassOp);
+ CHECK_CCW_MATCHES_CW(FailOp);
+ CHECK_CCW_MATCHES_CW(ZFailOp);
+#undef CHECK_CCW_MATCHES_CW
+ // Extract upper 16 bits.
+ Uint32 ccw_func_ops = BitField<16, 16>::Get(func_ops);
+
+ DecodeStencilFuncOps(ccw_func_ops, &func, &pass, &fail, &zfail);
+ HR(d3d_device_->SetRenderState(D3DRS_CCW_STENCILFUNC,
+ ComparisonToD3D(func)));
+ HR(d3d_device_->SetRenderState(D3DRS_CCW_STENCILPASS,
+ StencilOpToD3D(pass)));
+ HR(d3d_device_->SetRenderState(D3DRS_CCW_STENCILFAIL,
+ StencilOpToD3D(fail)));
+ HR(d3d_device_->SetRenderState(D3DRS_CCW_STENCILZFAIL,
+ StencilOpToD3D(zfail)));
+ } else {
+ HR(d3d_device_->SetRenderState(D3DRS_TWOSIDEDSTENCILMODE, FALSE));
+ }
+}
+
+void GAPID3D9::SetColorWrite(bool red,
+ bool green,
+ bool blue,
+ bool alpha,
+ bool dither) {
+ Uint32 mask = red ? D3DCOLORWRITEENABLE_RED : 0;
+ mask |= green ? D3DCOLORWRITEENABLE_GREEN : 0;
+ mask |= blue ? D3DCOLORWRITEENABLE_BLUE : 0;
+ mask |= alpha ? D3DCOLORWRITEENABLE_ALPHA : 0;
+ HR(d3d_device_->SetRenderState(D3DRS_COLORWRITEENABLE, mask));
+ HR(d3d_device_->SetRenderState(D3DRS_DITHERENABLE, dither ? TRUE : FALSE));
+}
+
+void GAPID3D9::SetBlending(bool enable,
+ bool separate_alpha,
+ BlendEq color_eq,
+ BlendFunc color_src_func,
+ BlendFunc color_dst_func,
+ BlendEq alpha_eq,
+ BlendFunc alpha_src_func,
+ BlendFunc alpha_dst_func) {
+ HR(d3d_device_->SetRenderState(D3DRS_ALPHABLENDENABLE,
+ enable ? TRUE : FALSE));
+ HR(d3d_device_->SetRenderState(D3DRS_BLENDOP, BlendEqToD3D(color_eq)));
+ HR(d3d_device_->SetRenderState(D3DRS_SRCBLEND,
+ BlendFuncToD3D(color_src_func)));
+ HR(d3d_device_->SetRenderState(D3DRS_DESTBLEND,
+ BlendFuncToD3D(color_dst_func)));
+ if (separate_alpha) {
+ HR(d3d_device_->SetRenderState(D3DRS_SEPARATEALPHABLENDENABLE, TRUE));
+ HR(d3d_device_->SetRenderState(D3DRS_BLENDOP, BlendEqToD3D(alpha_eq)));
+ HR(d3d_device_->SetRenderState(D3DRS_SRCBLEND,
+ BlendFuncToD3D(alpha_src_func)));
+ HR(d3d_device_->SetRenderState(D3DRS_DESTBLEND,
+ BlendFuncToD3D(alpha_dst_func)));
+ } else {
+ HR(d3d_device_->SetRenderState(D3DRS_SEPARATEALPHABLENDENABLE, FALSE));
+ }
+}
+
+void GAPID3D9::SetBlendingColor(const RGBA &color) {
+ HR(d3d_device_->SetRenderState(D3DRS_BLENDFACTOR, RGBAToD3DCOLOR(color)));
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/win/d3d9/texture_d3d9.cc b/o3d/command_buffer/service/win/d3d9/texture_d3d9.cc
new file mode 100644
index 0000000..cbd8cf5
--- /dev/null
+++ b/o3d/command_buffer/service/win/d3d9/texture_d3d9.cc
@@ -0,0 +1,636 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+// This file implements the D3D9 versions of the texture resources, as well as
+// the related GAPID3D9 function implementations.
+
+#include "command_buffer/service/win/d3d9/gapi_d3d9.h"
+#include "command_buffer/service/win/d3d9/texture_d3d9.h"
+
+namespace o3d {
+namespace command_buffer {
+
+// Converts a texture format to a D3D texture format.
+static D3DFORMAT D3DFormat(texture::Format format) {
+ switch (format) {
+ case texture::XRGB8: return D3DFMT_X8R8G8B8;
+ case texture::ARGB8: return D3DFMT_A8R8G8B8;
+ case texture::ABGR16F: return D3DFMT_A16B16G16R16F;
+ case texture::DXT1: return D3DFMT_DXT1;
+ default: return D3DFMT_UNKNOWN;
+ };
+}
+
+// Converts a cube map face to a D3D face.
+static D3DCUBEMAP_FACES D3DFace(texture::Face face) {
+ switch (face) {
+ case texture::FACE_POSITIVE_X:
+ return D3DCUBEMAP_FACE_POSITIVE_X;
+ case texture::FACE_NEGATIVE_X:
+ return D3DCUBEMAP_FACE_NEGATIVE_X;
+ case texture::FACE_POSITIVE_Y:
+ return D3DCUBEMAP_FACE_POSITIVE_Y;
+ case texture::FACE_NEGATIVE_Y:
+ return D3DCUBEMAP_FACE_NEGATIVE_Y;
+ case texture::FACE_POSITIVE_Z:
+ return D3DCUBEMAP_FACE_POSITIVE_Z;
+ case texture::FACE_NEGATIVE_Z:
+ return D3DCUBEMAP_FACE_NEGATIVE_Z;
+ }
+ LOG(FATAL) << "Not reached.";
+ return D3DCUBEMAP_FACE_POSITIVE_X;
+}
+
+// Texture 2D functions
+
+// Destroys the 2D texture, releasing the D3D texture, and its shadow if any.
+Texture2DD3D9::~Texture2DD3D9() {
+ DCHECK(d3d_texture_);
+ d3d_texture_->Release();
+ d3d_texture_ = NULL;
+ if (d3d_shadow_) {
+ d3d_shadow_->Release();
+ d3d_shadow_ = NULL;
+ }
+}
+
+// Creates a 2D texture. For dynamic textures, create it in the default pool,
+// and a shadow version in the system memory pool (that we can lock). For
+// regular texture, simply create one in the managed pool.
+Texture2DD3D9 *Texture2DD3D9::Create(GAPID3D9 *gapi,
+ unsigned int width,
+ unsigned int height,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags) {
+ DCHECK_GT(width, 0);
+ DCHECK_GT(height, 0);
+ DCHECK_GT(levels, 0);
+ D3DFORMAT d3d_format = D3DFormat(format);
+ IDirect3DDevice9 *device = gapi->d3d_device();
+ if (flags & texture::DYNAMIC) {
+ IDirect3DTexture9 *d3d_texture = NULL;
+ HRESULT result = device->CreateTexture(width, height, levels,
+ D3DUSAGE_DYNAMIC, d3d_format,
+ D3DPOOL_DEFAULT, &d3d_texture,
+ NULL);
+ if (result != D3D_OK) {
+ LOG(ERROR) << "DirectX error when calling CreateTexture: "
+ << DXGetErrorStringA(result);
+ return NULL;
+ }
+ IDirect3DTexture9 *d3d_shadow = NULL;
+ result = device->CreateTexture(width, height, levels, D3DUSAGE_DYNAMIC,
+ d3d_format, D3DPOOL_SYSTEMMEM, &d3d_shadow,
+ NULL);
+ if (result != D3D_OK) {
+ LOG(ERROR) << "DirectX error when calling CreateTexture: "
+ << DXGetErrorStringA(result);
+ d3d_texture->Release();
+ return NULL;
+ }
+ return new Texture2DD3D9(levels, format, flags, width, height, d3d_texture,
+ d3d_shadow);
+ } else {
+ IDirect3DTexture9 *d3d_texture = NULL;
+ HRESULT result = device->CreateTexture(width, height, levels, 0, d3d_format,
+ D3DPOOL_MANAGED, &d3d_texture, NULL);
+ if (result != D3D_OK) {
+ LOG(ERROR) << "DirectX error when calling CreateTexture: "
+ << DXGetErrorStringA(result);
+ return NULL;
+ }
+ return new Texture2DD3D9(levels, format, flags, width, height, d3d_texture,
+ NULL);
+ }
+}
+
+// Sets the data in the texture, using LockRect()/UnlockRect(). For dynamic
+// textures, it copies the data to the shadow texture, then updates the actual
+// texture. For regular texture, it directly modifies the actual texture.
+bool Texture2DD3D9::SetData(GAPID3D9 *gapi,
+ const Volume &volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ const void *data) {
+ DCHECK(d3d_texture_);
+ IDirect3DTexture9 *lock_texture = d3d_shadow_ ? d3d_shadow_ : d3d_texture_;
+
+ MipLevelInfo mip_info;
+ MakeMipLevelInfo(&mip_info, format(), width_, height_, 1, level);
+ TransferInfo src_transfer_info;
+ MakeTransferInfo(&src_transfer_info, mip_info, volume, row_pitch,
+ slice_pitch);
+ if (!CheckVolume(mip_info, volume) || level >= levels() ||
+ size < src_transfer_info.total_size)
+ return false;
+
+ bool full_rect = IsFullVolume(mip_info, volume);
+ D3DLOCKED_RECT locked_rect;
+ RECT rect = {volume.x, volume.y, volume.x+volume.width,
+ volume.y+volume.height};
+ DWORD lock_flags =
+ full_rect && (flags() & texture::DYNAMIC) ? D3DLOCK_DISCARD : 0;
+ HR(lock_texture->LockRect(level, &locked_rect, full_rect ? NULL : &rect,
+ lock_flags));
+
+ TransferInfo dst_transfer_info;
+ MakeTransferInfo(&dst_transfer_info, mip_info, volume, locked_rect.Pitch,
+ slice_pitch);
+ TransferVolume(volume, mip_info, dst_transfer_info, locked_rect.pBits,
+ src_transfer_info, data);
+
+ HR(lock_texture->UnlockRect(level));
+ if (d3d_shadow_) {
+ HR(gapi->d3d_device()->UpdateTexture(d3d_shadow_, d3d_texture_));
+ }
+ return true;
+}
+
+// Gets the data from the texture, using LockRect()/UnlockRect(). For dynamic
+// textures, it gets the data from the shadow texture, For regular texture, it
+// gets it directly from the actual texture.
+bool Texture2DD3D9::GetData(GAPID3D9 *gapi,
+ const Volume &volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ void *data) {
+ DCHECK(d3d_texture_);
+ IDirect3DTexture9 *lock_texture = d3d_shadow_ ? d3d_shadow_ : d3d_texture_;
+
+ MipLevelInfo mip_info;
+ MakeMipLevelInfo(&mip_info, format(), width_, height_, 1, level);
+ TransferInfo dst_transfer_info;
+ MakeTransferInfo(&dst_transfer_info, mip_info, volume, row_pitch,
+ slice_pitch);
+ if (!CheckVolume(mip_info, volume) || level >= levels() ||
+ size < dst_transfer_info.total_size)
+ return false;
+
+ bool full_rect = IsFullVolume(mip_info, volume);
+ D3DLOCKED_RECT locked_rect;
+ RECT rect = {volume.x, volume.y, volume.x+volume.width,
+ volume.y+volume.height};
+ DWORD lock_flags = D3DLOCK_READONLY;
+ HR(lock_texture->LockRect(level, &locked_rect, full_rect ? NULL : &rect,
+ lock_flags));
+ TransferInfo src_transfer_info;
+ MakeTransferInfo(&src_transfer_info, mip_info, volume, locked_rect.Pitch,
+ slice_pitch);
+ TransferVolume(volume, mip_info, dst_transfer_info, data,
+ src_transfer_info, locked_rect.pBits);
+ HR(lock_texture->UnlockRect(level));
+ return true;
+}
+
+// Texture 3D functions
+
+// Destroys the 3D texture.
+Texture3DD3D9::~Texture3DD3D9() {
+ DCHECK(d3d_texture_);
+ d3d_texture_->Release();
+ d3d_texture_ = NULL;
+ if (d3d_shadow_) {
+ d3d_shadow_->Release();
+ d3d_shadow_ = NULL;
+ }
+}
+
+// Creates a 3D texture. For dynamic textures, create it in the default pool,
+// and a shadow version in the system memory pool (that we can lock). For
+// regular texture, simply create one in the managed pool.
+Texture3DD3D9 *Texture3DD3D9::Create(GAPID3D9 *gapi,
+ unsigned int width,
+ unsigned int height,
+ unsigned int depth,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags) {
+ DCHECK_GT(width, 0);
+ DCHECK_GT(height, 0);
+ DCHECK_GT(depth, 0);
+ DCHECK_GT(levels, 0);
+ D3DFORMAT d3d_format = D3DFormat(format);
+ IDirect3DDevice9 *device = gapi->d3d_device();
+ if (flags & texture::DYNAMIC) {
+ IDirect3DVolumeTexture9 *d3d_texture = NULL;
+ HRESULT result = device->CreateVolumeTexture(width, height, depth, levels,
+ D3DUSAGE_DYNAMIC, d3d_format,
+ D3DPOOL_DEFAULT, &d3d_texture,
+ NULL);
+ if (result != D3D_OK) {
+ LOG(ERROR) << "DirectX error when calling CreateTexture: "
+ << DXGetErrorStringA(result);
+ return NULL;
+ }
+ IDirect3DVolumeTexture9 *d3d_shadow = NULL;
+ result = device->CreateVolumeTexture(width, height, depth, levels,
+ D3DUSAGE_DYNAMIC, d3d_format,
+ D3DPOOL_SYSTEMMEM, &d3d_shadow, NULL);
+ if (result != D3D_OK) {
+ LOG(ERROR) << "DirectX error when calling CreateTexture: "
+ << DXGetErrorStringA(result);
+ d3d_texture->Release();
+ return NULL;
+ }
+ return new Texture3DD3D9(levels, format, flags, width, height, depth,
+ d3d_texture, d3d_shadow);
+ } else {
+ IDirect3DVolumeTexture9 *d3d_texture = NULL;
+ HRESULT result = device->CreateVolumeTexture(width, height, depth, levels,
+ D3DUSAGE_DYNAMIC, d3d_format,
+ D3DPOOL_MANAGED, &d3d_texture,
+ NULL);
+ if (result != D3D_OK) {
+ LOG(ERROR) << "DirectX error when calling CreateTexture: "
+ << DXGetErrorStringA(result);
+ return NULL;
+ }
+ return new Texture3DD3D9(levels, format, flags, width, height, depth,
+ d3d_texture, NULL);
+ }
+}
+
+// Sets the data in the texture, using LockBox()/UnlockBox(). For dynamic
+// textures, it copies the data to the shadow texture, then updates the actual
+// texture. For regular texture, it directly modifies the actual texture.
+bool Texture3DD3D9::SetData(GAPID3D9 *gapi,
+ const Volume &volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ const void *data) {
+ DCHECK(d3d_texture_);
+ IDirect3DVolumeTexture9 *lock_texture =
+ d3d_shadow_ ? d3d_shadow_ : d3d_texture_;
+
+ MipLevelInfo mip_info;
+ MakeMipLevelInfo(&mip_info, format(), width_, height_, depth_, level);
+ TransferInfo src_transfer_info;
+ MakeTransferInfo(&src_transfer_info, mip_info, volume, row_pitch,
+ slice_pitch);
+ if (!CheckVolume(mip_info, volume) || level >= levels() ||
+ size < src_transfer_info.total_size)
+ return false;
+
+ bool full_box = IsFullVolume(mip_info, volume);
+ D3DLOCKED_BOX locked_box;
+ D3DBOX box = {volume.x, volume.y, volume.z, volume.x+volume.width,
+ volume.y+volume.height, volume.z+volume.depth};
+ DWORD lock_flags =
+ full_box && (flags() & texture::DYNAMIC) ? D3DLOCK_DISCARD : 0;
+ HR(lock_texture->LockBox(level, &locked_box, full_box ? NULL : &box,
+ lock_flags));
+
+ TransferInfo dst_transfer_info;
+ MakeTransferInfo(&dst_transfer_info, mip_info, volume, locked_box.RowPitch,
+ locked_box.SlicePitch);
+ TransferVolume(volume, mip_info, dst_transfer_info, locked_box.pBits,
+ src_transfer_info, data);
+
+ HR(lock_texture->UnlockBox(level));
+ if (d3d_shadow_) {
+ HR(gapi->d3d_device()->UpdateTexture(d3d_shadow_, d3d_texture_));
+ }
+ return true;
+}
+
+// Gets the data from the texture, using LockBox()/UnlockBox(). For dynamic
+// textures, it gets the data from the shadow texture, For regular texture, it
+// gets it directly from the actual texture.
+bool Texture3DD3D9::GetData(GAPID3D9 *gapi,
+ const Volume &volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ void *data) {
+ DCHECK(d3d_texture_);
+ IDirect3DVolumeTexture9 *lock_texture =
+ d3d_shadow_ ? d3d_shadow_ : d3d_texture_;
+
+ MipLevelInfo mip_info;
+ MakeMipLevelInfo(&mip_info, format(), width_, height_, depth_, level);
+ TransferInfo dst_transfer_info;
+ MakeTransferInfo(&dst_transfer_info, mip_info, volume, row_pitch,
+ slice_pitch);
+ if (!CheckVolume(mip_info, volume) || level >= levels() ||
+ size < dst_transfer_info.total_size)
+ return false;
+
+ bool full_box = IsFullVolume(mip_info, volume);
+ D3DLOCKED_BOX locked_box;
+ D3DBOX box = {volume.x, volume.y, volume.z, volume.x+volume.width,
+ volume.y+volume.height, volume.z+volume.depth};
+ DWORD lock_flags = D3DLOCK_READONLY;
+ HR(lock_texture->LockBox(level, &locked_box, full_box ? NULL : &box,
+ lock_flags));
+ TransferInfo src_transfer_info;
+ MakeTransferInfo(&src_transfer_info, mip_info, volume, locked_box.RowPitch,
+ locked_box.SlicePitch);
+ TransferVolume(volume, mip_info, dst_transfer_info, data,
+ src_transfer_info, locked_box.pBits);
+ HR(lock_texture->UnlockBox(level));
+ return true;
+}
+
+// Texture Cube functions.
+
+// Destroys the cube map texture, releasing the D3D texture, and its shadow if
+// any.
+TextureCubeD3D9::~TextureCubeD3D9() {
+ DCHECK(d3d_texture_);
+ d3d_texture_->Release();
+ d3d_texture_ = NULL;
+ if (d3d_shadow_) {
+ d3d_shadow_->Release();
+ d3d_shadow_ = NULL;
+ }
+}
+
+// Creates a cube map texture. For dynamic textures, create it in the default
+// pool, and a shadow version in the system memory pool (that we can lock). For
+// regular texture, simply create one in the managed pool.
+TextureCubeD3D9 *TextureCubeD3D9::Create(GAPID3D9 *gapi,
+ unsigned int side,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags) {
+ DCHECK_GT(side, 0);
+ DCHECK_GT(levels, 0);
+ D3DFORMAT d3d_format = D3DFormat(format);
+ IDirect3DDevice9 *device = gapi->d3d_device();
+ if (flags & texture::DYNAMIC) {
+ IDirect3DCubeTexture9 *d3d_texture = NULL;
+ HRESULT result = device->CreateCubeTexture(side, levels, D3DUSAGE_DYNAMIC,
+ d3d_format, D3DPOOL_DEFAULT,
+ &d3d_texture, NULL);
+ if (result != D3D_OK) {
+ LOG(ERROR) << "DirectX error when calling CreateTexture: "
+ << DXGetErrorStringA(result);
+ return NULL;
+ }
+ IDirect3DCubeTexture9 *d3d_shadow = NULL;
+ result = device->CreateCubeTexture(side, levels, D3DUSAGE_DYNAMIC,
+ d3d_format, D3DPOOL_SYSTEMMEM,
+ &d3d_shadow, NULL);
+ if (result != D3D_OK) {
+ LOG(ERROR) << "DirectX error when calling CreateTexture: "
+ << DXGetErrorStringA(result);
+ d3d_texture->Release();
+ return NULL;
+ }
+ return new TextureCubeD3D9(levels, format, flags, side, d3d_texture,
+ d3d_shadow);
+ } else {
+ IDirect3DCubeTexture9 *d3d_texture = NULL;
+ HRESULT result = device->CreateCubeTexture(side, levels, 0, d3d_format,
+ D3DPOOL_MANAGED, &d3d_texture,
+ NULL);
+ if (result != D3D_OK) {
+ LOG(ERROR) << "DirectX error when calling CreateTexture: "
+ << DXGetErrorStringA(result);
+ return NULL;
+ }
+ return new TextureCubeD3D9(levels, format, flags, side, d3d_texture, NULL);
+ }
+}
+
+// Sets the data in the texture, using LockRect()/UnlockRect(). For dynamic
+// textures, it copies the data to the shadow texture, then updates the actual
+// texture. For regular texture, it directly modifies the actual texture.
+bool TextureCubeD3D9::SetData(GAPID3D9 *gapi,
+ const Volume &volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ const void *data) {
+ DCHECK(d3d_texture_);
+ IDirect3DCubeTexture9 *lock_texture =
+ d3d_shadow_ ? d3d_shadow_ : d3d_texture_;
+ MipLevelInfo mip_info;
+ MakeMipLevelInfo(&mip_info, format(), side_, side_, 1, level);
+ TransferInfo src_transfer_info;
+ MakeTransferInfo(&src_transfer_info, mip_info, volume, row_pitch,
+ slice_pitch);
+ if (!CheckVolume(mip_info, volume) || level >= levels() ||
+ size < src_transfer_info.total_size)
+ return false;
+
+ D3DCUBEMAP_FACES d3d_face = D3DFace(face);
+ bool full_rect = IsFullVolume(mip_info, volume);
+ D3DLOCKED_RECT locked_rect;
+ RECT rect = {volume.x, volume.y, volume.x+volume.width,
+ volume.y+volume.height};
+ DWORD lock_flags =
+ full_rect && (flags() & texture::DYNAMIC) ? D3DLOCK_DISCARD : 0;
+ HR(lock_texture->LockRect(d3d_face, level, &locked_rect,
+ full_rect ? NULL : &rect, lock_flags));
+
+ TransferInfo dst_transfer_info;
+ MakeTransferInfo(&dst_transfer_info, mip_info, volume, locked_rect.Pitch,
+ slice_pitch);
+ TransferVolume(volume, mip_info, dst_transfer_info, locked_rect.pBits,
+ src_transfer_info, data);
+
+ HR(lock_texture->UnlockRect(d3d_face, level));
+ if (d3d_shadow_) {
+ HR(gapi->d3d_device()->UpdateTexture(d3d_shadow_, d3d_texture_));
+ }
+ return true;
+}
+
+// Gets the data from the texture, using LockRect()/UnlockRect(). For dynamic
+// textures, it gets the data from the shadow texture, For regular texture, it
+// gets it directly from the actual texture.
+bool TextureCubeD3D9::GetData(GAPID3D9 *gapi,
+ const Volume &volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ void *data) {
+ DCHECK(d3d_texture_);
+ IDirect3DCubeTexture9 *lock_texture =
+ d3d_shadow_ ? d3d_shadow_ : d3d_texture_;
+
+ MipLevelInfo mip_info;
+ MakeMipLevelInfo(&mip_info, format(), side_, side_, 1, level);
+ TransferInfo dst_transfer_info;
+ MakeTransferInfo(&dst_transfer_info, mip_info, volume, row_pitch,
+ slice_pitch);
+ if (!CheckVolume(mip_info, volume) || level >= levels() ||
+ size < dst_transfer_info.total_size)
+ return false;
+
+ D3DCUBEMAP_FACES d3d_face = D3DFace(face);
+ bool full_rect = IsFullVolume(mip_info, volume);
+ D3DLOCKED_RECT locked_rect;
+ RECT rect = {volume.x, volume.y, volume.x+volume.width,
+ volume.y+volume.height};
+ DWORD lock_flags = D3DLOCK_READONLY;
+ HR(lock_texture->LockRect(d3d_face, level, &locked_rect,
+ full_rect ? NULL : &rect, lock_flags));
+ TransferInfo src_transfer_info;
+ MakeTransferInfo(&src_transfer_info, mip_info, volume, locked_rect.Pitch,
+ slice_pitch);
+ TransferVolume(volume, mip_info, dst_transfer_info, data,
+ src_transfer_info, locked_rect.pBits);
+ HR(lock_texture->UnlockRect(d3d_face, level));
+ return true;
+}
+// GAPID3D9 functions.
+
+// Destroys a texture resource.
+BufferSyncInterface::ParseError GAPID3D9::DestroyTexture(ResourceID id) {
+ // Dirty effect, because this texture id may be used
+ DirtyEffect();
+ return textures_.Destroy(id) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+// Creates a 2D texture resource.
+BufferSyncInterface::ParseError GAPID3D9::CreateTexture2D(
+ ResourceID id,
+ unsigned int width,
+ unsigned int height,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags) {
+ Texture2DD3D9 *texture = Texture2DD3D9::Create(this, width, height, levels,
+ format, flags);
+ if (!texture) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ // Dirty effect, because this texture id may be used
+ DirtyEffect();
+ textures_.Assign(id, texture);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+// Creates a 3D texture resource.
+BufferSyncInterface::ParseError GAPID3D9::CreateTexture3D(
+ ResourceID id,
+ unsigned int width,
+ unsigned int height,
+ unsigned int depth,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags) {
+ Texture3DD3D9 *texture = Texture3DD3D9::Create(this, width, height, depth,
+ levels, format, flags);
+ if (!texture) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ // Dirty effect, because this texture id may be used
+ DirtyEffect();
+ textures_.Assign(id, texture);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+// Creates a cube map texture resource.
+BufferSyncInterface::ParseError GAPID3D9::CreateTextureCube(
+ ResourceID id,
+ unsigned int side,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags) {
+ TextureCubeD3D9 *texture = TextureCubeD3D9::Create(this, side, levels,
+ format, flags);
+ if (!texture) return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ // Dirty effect, because this texture id may be used
+ DirtyEffect();
+ textures_.Assign(id, texture);
+ return BufferSyncInterface::PARSE_NO_ERROR;
+}
+
+// Copies the data into a texture resource.
+BufferSyncInterface::ParseError GAPID3D9::SetTextureData(
+ ResourceID id,
+ unsigned int x,
+ unsigned int y,
+ unsigned int z,
+ unsigned int width,
+ unsigned int height,
+ unsigned int depth,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ const void *data) {
+ TextureD3D9 *texture = textures_.Get(id);
+ if (!texture)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ Volume volume = {x, y, z, width, height, depth};
+ return texture->SetData(this, volume, level, face, row_pitch, slice_pitch,
+ size, data) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+// Copies the data from a texture resource.
+BufferSyncInterface::ParseError GAPID3D9::GetTextureData(
+ ResourceID id,
+ unsigned int x,
+ unsigned int y,
+ unsigned int z,
+ unsigned int width,
+ unsigned int height,
+ unsigned int depth,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ void *data) {
+ TextureD3D9 *texture = textures_.Get(id);
+ if (!texture)
+ return BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+ Volume volume = {x, y, z, width, height, depth};
+ return texture->GetData(this, volume, level, face, row_pitch, slice_pitch,
+ size, data) ?
+ BufferSyncInterface::PARSE_NO_ERROR :
+ BufferSyncInterface::PARSE_INVALID_ARGUMENTS;
+}
+
+} // namespace command_buffer
+} // namespace o3d
diff --git a/o3d/command_buffer/service/win/d3d9/texture_d3d9.h b/o3d/command_buffer/service/win/d3d9/texture_d3d9.h
new file mode 100644
index 0000000..c55ba4f
--- /dev/null
+++ b/o3d/command_buffer/service/win/d3d9/texture_d3d9.h
@@ -0,0 +1,245 @@
+/*
+ * Copyright 2009, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+
+#ifndef O3D_COMMAND_BUFFER_SERVICE_WIN_D3D9_TEXTURE_D3D9_H__
+#define O3D_COMMAND_BUFFER_SERVICE_WIN_D3D9_TEXTURE_D3D9_H__
+
+// This file contains the definition of the D3D9 versions of texture-related
+// resource classes.
+
+#include "command_buffer/common/cross/gapi_interface.h"
+#include "command_buffer/service/win/d3d9/d3d9_utils.h"
+#include "command_buffer/service/cross/resource.h"
+#include "command_buffer/service/cross/texture_utils.h"
+
+namespace o3d {
+namespace command_buffer {
+
+class GAPID3D9;
+
+// The base class for a D3D texture resource, providing access to the base D3D
+// texture that can be assigned to an effect parameter or a sampler unit.
+class TextureD3D9 : public Texture {
+ public:
+ TextureD3D9(texture::Type type,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags)
+ : Texture(type, levels, format, flags) {}
+ // Gets the D3D base texture.
+ virtual IDirect3DBaseTexture9 *d3d_base_texture() const = 0;
+ // Sets data into a texture resource.
+ virtual bool SetData(GAPID3D9 *gapi,
+ const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ const void *data) = 0;
+ // Gets data from a texture resource.
+ virtual bool GetData(GAPID3D9 *gapi,
+ const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ void *data) = 0;
+ private:
+ DISALLOW_COPY_AND_ASSIGN(TextureD3D9);
+};
+
+// A 2D texture resource for D3D.
+class Texture2DD3D9 : public TextureD3D9 {
+ public:
+ Texture2DD3D9(unsigned int levels,
+ texture::Format format,
+ unsigned int flags,
+ unsigned int width,
+ unsigned int height,
+ IDirect3DTexture9 *texture,
+ IDirect3DTexture9 *shadow)
+ : TextureD3D9(texture::TEXTURE_2D, levels, format, flags),
+ width_(width),
+ height_(height),
+ d3d_texture_(texture),
+ d3d_shadow_(shadow) {}
+ virtual ~Texture2DD3D9();
+
+ // Creates a 2D texture resource.
+ static Texture2DD3D9 *Create(GAPID3D9 *gapi,
+ unsigned int width,
+ unsigned int height,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags);
+ // Sets data into a 2D texture resource.
+ virtual bool SetData(GAPID3D9 *gapi,
+ const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ const void *data);
+ // Gets data from a 2D texture resource.
+ virtual bool GetData(GAPID3D9 *gapi,
+ const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ void *data);
+ // Gets the D3D base texture.
+ virtual IDirect3DBaseTexture9 *d3d_base_texture() const {
+ return d3d_texture_;
+ }
+ private:
+ unsigned int width_;
+ unsigned int height_;
+ IDirect3DTexture9 *d3d_texture_;
+ IDirect3DTexture9 *d3d_shadow_;
+ DISALLOW_COPY_AND_ASSIGN(Texture2DD3D9);
+};
+
+// A 3D texture resource for D3D.
+class Texture3DD3D9 : public TextureD3D9 {
+ public:
+ Texture3DD3D9(unsigned int levels,
+ texture::Format format,
+ unsigned int flags,
+ unsigned int width,
+ unsigned int height,
+ unsigned int depth,
+ IDirect3DVolumeTexture9 *texture,
+ IDirect3DVolumeTexture9 *shadow)
+ : TextureD3D9(texture::TEXTURE_2D, levels, format, flags),
+ width_(width),
+ height_(height),
+ depth_(depth),
+ d3d_texture_(texture),
+ d3d_shadow_(shadow) {}
+ virtual ~Texture3DD3D9();
+ // Creates a 3D texture resource.
+ static Texture3DD3D9 *Create(GAPID3D9 *gapi,
+ unsigned int width,
+ unsigned int height,
+ unsigned int depth,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags);
+ // Sets data into a 3D texture resource.
+ virtual bool SetData(GAPID3D9 *gapi,
+ const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ const void *data);
+ // Gets data from a 3D texture resource.
+ virtual bool GetData(GAPID3D9 *gapi,
+ const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ void *data);
+ // Gets the D3D base texture.
+ virtual IDirect3DBaseTexture9 *d3d_base_texture() const {
+ return d3d_texture_;
+ }
+ private:
+ unsigned int width_;
+ unsigned int height_;
+ unsigned int depth_;
+ IDirect3DVolumeTexture9 *d3d_texture_;
+ IDirect3DVolumeTexture9 *d3d_shadow_;
+ DISALLOW_COPY_AND_ASSIGN(Texture3DD3D9);
+};
+
+// A cube map texture resource for D3D.
+class TextureCubeD3D9 : public TextureD3D9 {
+ public:
+ TextureCubeD3D9(unsigned int levels,
+ texture::Format format,
+ unsigned int flags,
+ unsigned int side,
+ IDirect3DCubeTexture9 *texture,
+ IDirect3DCubeTexture9 *shadow)
+ : TextureD3D9(texture::TEXTURE_CUBE, levels, format, flags),
+ side_(side),
+ d3d_texture_(texture),
+ d3d_shadow_(shadow) {}
+ virtual ~TextureCubeD3D9();
+ // Creates a cube map texture resource.
+ static TextureCubeD3D9 *Create(GAPID3D9 *gapi,
+ unsigned int side,
+ unsigned int levels,
+ texture::Format format,
+ unsigned int flags);
+ // Sets data into a cube map texture resource.
+ virtual bool SetData(GAPID3D9 *gapi,
+ const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ const void *data);
+ // Gets data from a cube map texture resource.
+ virtual bool GetData(GAPID3D9 *gapi,
+ const Volume& volume,
+ unsigned int level,
+ texture::Face face,
+ unsigned int row_pitch,
+ unsigned int slice_pitch,
+ unsigned int size,
+ void *data);
+ // Gets the D3D base texture.
+ virtual IDirect3DBaseTexture9 *d3d_base_texture() const {
+ return d3d_texture_;
+ }
+ private:
+ unsigned int side_;
+ IDirect3DCubeTexture9 *d3d_texture_;
+ IDirect3DCubeTexture9 *d3d_shadow_;
+ DISALLOW_COPY_AND_ASSIGN(TextureCubeD3D9);
+};
+
+} // namespace command_buffer
+} // namespace o3d
+
+#endif // O3D_COMMAND_BUFFER_SERVICE_WIN_D3D9_TEXTURE_D3D9_H__
diff --git a/o3d/command_buffer/service/win/plugin.def b/o3d/command_buffer/service/win/plugin.def
new file mode 100644
index 0000000..292cf78
--- /dev/null
+++ b/o3d/command_buffer/service/win/plugin.def
@@ -0,0 +1,6 @@
+LIBRARY npo3d_cb_plugin
+
+EXPORTS
+ NP_GetEntryPoints @1
+ NP_Initialize @2
+ NP_Shutdown @3
diff --git a/o3d/command_buffer/service/win/plugin.rc b/o3d/command_buffer/service/win/plugin.rc
new file mode 100644
index 0000000..393668e
--- /dev/null
+++ b/o3d/command_buffer/service/win/plugin.rc
@@ -0,0 +1,100 @@
+// Microsoft Visual C++ generated resource script.
+//
+#define APSTUDIO_READONLY_SYMBOLS
+/////////////////////////////////////////////////////////////////////////////
+//
+// Generated from the TEXTINCLUDE 2 resource.
+//
+#include "afxres.h"
+
+/////////////////////////////////////////////////////////////////////////////
+#undef APSTUDIO_READONLY_SYMBOLS
+
+/////////////////////////////////////////////////////////////////////////////
+// English (U.S.) resources
+
+#if !defined(AFX_RESOURCE_DLL) || defined(AFX_TARG_ENU)
+#ifdef _WIN32
+LANGUAGE LANG_ENGLISH, SUBLANG_ENGLISH_US
+#pragma code_page(1252)
+#endif //_WIN32
+
+#ifdef APSTUDIO_INVOKED
+/////////////////////////////////////////////////////////////////////////////
+//
+// TEXTINCLUDE
+//
+
+1 TEXTINCLUDE
+BEGIN
+ "resource.h\0"
+END
+
+2 TEXTINCLUDE
+BEGIN
+ "#include ""afxres.h""\r\n"
+ "\0"
+END
+
+3 TEXTINCLUDE
+BEGIN
+ "\r\n"
+ "\0"
+END
+
+#endif // APSTUDIO_INVOKED
+
+
+/////////////////////////////////////////////////////////////////////////////
+//
+// Version
+//
+
+VS_VERSION_INFO VERSIONINFO
+ FILEVERSION 0,0,0,1
+ PRODUCTVERSION 0,0,0,1
+ FILEFLAGSMASK 0x17L
+#ifdef _DEBUG
+ FILEFLAGS 0x1L
+#else
+ FILEFLAGS 0x0L
+#endif
+ FILEOS 0x4L
+ FILETYPE 0x2L
+ FILESUBTYPE 0x0L
+BEGIN
+ BLOCK "StringFileInfo"
+ BEGIN
+ BLOCK "040904e4"
+ BEGIN
+ VALUE "FileDescription", "CommandBuffer SRV Plugin"
+ VALUE "FileVersion", "0,0,0,1"
+ VALUE "InternalName", ""
+ VALUE "LegalCopyright", "Copyright (C) 2008"
+ VALUE "MIMEType", "application/vnd.cmdbuf"
+ VALUE "OriginalFilename", "npo3d_cb_plugin.dll"
+ VALUE "ProductName", "CommandBuffer SRV Plugin"
+ VALUE "ProductVersion", "0,0,0,1"
+ END
+ END
+ BLOCK "VarFileInfo"
+ BEGIN
+ VALUE "Translation", 0x409, 1252
+ END
+END
+
+#endif // English (U.S.) resources
+/////////////////////////////////////////////////////////////////////////////
+
+
+
+#ifndef APSTUDIO_INVOKED
+/////////////////////////////////////////////////////////////////////////////
+//
+// Generated from the TEXTINCLUDE 3 resource.
+//
+
+
+/////////////////////////////////////////////////////////////////////////////
+#endif // not APSTUDIO_INVOKED
+