Welcome to mirror list, hosted at ThFree Co, Russian Federation.

git.blender.org/blender.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
Diffstat (limited to 'intern/cycles/device/device_cpu.cpp')
-rw-r--r--intern/cycles/device/device_cpu.cpp216
1 files changed, 216 insertions, 0 deletions
diff --git a/intern/cycles/device/device_cpu.cpp b/intern/cycles/device/device_cpu.cpp
new file mode 100644
index 00000000000..bb47a127f20
--- /dev/null
+++ b/intern/cycles/device/device_cpu.cpp
@@ -0,0 +1,216 @@
+/*
+ * Copyright 2011, Blender Foundation.
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+
+#include <stdlib.h>
+#include <string.h>
+
+#include "device.h"
+#include "device_intern.h"
+
+#include "kernel.h"
+#include "kernel_types.h"
+
+#include "osl_shader.h"
+
+#include "util_debug.h"
+#include "util_foreach.h"
+#include "util_function.h"
+#include "util_opengl.h"
+#include "util_progress.h"
+#include "util_system.h"
+#include "util_thread.h"
+
+CCL_NAMESPACE_BEGIN
+
+class CPUDevice : public Device
+{
+public:
+ vector<thread*> threads;
+ ThreadQueue<DeviceTask> tasks;
+ KernelGlobals *kg;
+
+ CPUDevice()
+ {
+ kg = kernel_globals_create();
+ threads.resize(system_cpu_thread_count());
+
+ for(size_t i = 0; i < threads.size(); i++)
+ threads[i] = new thread(function_bind(&CPUDevice::thread_run, this, i));
+ }
+
+ ~CPUDevice()
+ {
+ tasks.stop();
+
+ foreach(thread *t, threads) {
+ t->join();
+ delete t;
+ }
+
+ kernel_globals_free(kg);
+ }
+
+ string description()
+ {
+ return system_cpu_brand_string();
+ }
+
+ void mem_alloc(device_memory& mem, MemoryType type)
+ {
+ mem.device_pointer = mem.data_pointer;
+ }
+
+ void mem_copy_to(device_memory& mem)
+ {
+ /* no-op */
+ }
+
+ void mem_copy_from(device_memory& mem, size_t offset, size_t size)
+ {
+ /* no-op */
+ }
+
+ void mem_zero(device_memory& mem)
+ {
+ memset((void*)mem.device_pointer, 0, mem.memory_size());
+ }
+
+ void mem_free(device_memory& mem)
+ {
+ mem.device_pointer = 0;
+ }
+
+ void const_copy_to(const char *name, void *host, size_t size)
+ {
+ kernel_const_copy(kg, name, host, size);
+ }
+
+ void tex_alloc(const char *name, device_memory& mem, bool interpolation, bool periodic)
+ {
+ kernel_tex_copy(kg, name, mem.data_pointer, mem.data_width, mem.data_height);
+ mem.device_pointer = mem.data_pointer;
+ }
+
+ void tex_free(device_memory& mem)
+ {
+ mem.device_pointer = 0;
+ }
+
+ void *osl_memory()
+ {
+#ifdef WITH_OSL
+ return kernel_osl_memory(kg);
+#else
+ return NULL;
+#endif
+ }
+
+ void thread_run(int t)
+ {
+ DeviceTask task;
+
+ while(tasks.worker_wait_pop(task)) {
+ if(task.type == DeviceTask::PATH_TRACE)
+ thread_path_trace(task);
+ else if(task.type == DeviceTask::TONEMAP)
+ thread_tonemap(task);
+ else if(task.type == DeviceTask::DISPLACE)
+ thread_displace(task);
+
+ tasks.worker_done();
+ }
+ }
+
+ void thread_path_trace(DeviceTask& task)
+ {
+ if(tasks.worker_cancel())
+ return;
+
+#ifdef WITH_OSL
+ if(kernel_osl_use(kg))
+ OSLShader::thread_init(kg);
+#endif
+
+ for(int y = task.y; y < task.y + task.h; y++) {
+ for(int x = task.x; x < task.x + task.w; x++)
+ kernel_cpu_path_trace(kg, (float4*)task.buffer, (unsigned int*)task.rng_state, task.pass, x, y);
+
+ if(tasks.worker_cancel())
+ break;
+ }
+
+#ifdef WITH_OSL
+ if(kernel_osl_use(kg))
+ OSLShader::thread_free(kg);
+#endif
+ }
+
+ void thread_tonemap(DeviceTask& task)
+ {
+ for(int y = task.y; y < task.y + task.h; y++) {
+ for(int x = task.x; x < task.x + task.w; x++)
+ kernel_cpu_tonemap(kg, (uchar4*)task.rgba, (float4*)task.buffer, task.pass, task.resolution, x, y);
+ }
+ }
+
+ void thread_displace(DeviceTask& task)
+ {
+#ifdef WITH_OSL
+ if(kernel_osl_use(kg))
+ OSLShader::thread_init(kg);
+#endif
+
+ for(int x = task.displace_x; x < task.displace_x + task.displace_w; x++) {
+ kernel_cpu_displace(kg, (uint4*)task.displace_input, (float3*)task.displace_offset, x);
+
+ if(tasks.worker_cancel())
+ break;
+ }
+
+#ifdef WITH_OSL
+ if(kernel_osl_use(kg))
+ OSLShader::thread_free(kg);
+#endif
+ }
+
+ void task_add(DeviceTask& task)
+ {
+ if(task.type == DeviceTask::TONEMAP)
+ tasks.push(task);
+ else
+ task.split(tasks, threads.size());
+ }
+
+ void task_wait()
+ {
+ tasks.wait_done();
+ }
+
+ void task_cancel()
+ {
+ tasks.cancel();
+ }
+};
+
+Device *device_cpu_create()
+{
+ return new CPUDevice();
+}
+
+CCL_NAMESPACE_END
+