| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright 2016, 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. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Contributor(s): Blender Institute | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \file blender/draw/intern/draw_manager_shader.c
 | 
					
						
							|  |  |  |  *  \ingroup draw | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "draw_manager.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | #include "DNA_world_types.h"
 | 
					
						
							|  |  |  | #include "DNA_material_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_listbase.h"
 | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | #include "BLI_string.h"
 | 
					
						
							|  |  |  | #include "BLI_string_utils.h"
 | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | #include "BLI_threads.h"
 | 
					
						
							|  |  |  | #include "BLI_task.h"
 | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | #include "BKE_global.h"
 | 
					
						
							|  |  |  | #include "BKE_main.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | #include "GPU_shader.h"
 | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | #include "GPU_material.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "WM_api.h"
 | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | #include "WM_types.h"
 | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | extern char datatoc_gpu_shader_2D_vert_glsl[]; | 
					
						
							|  |  |  | extern char datatoc_gpu_shader_3D_vert_glsl[]; | 
					
						
							| 
									
										
										
										
											2018-03-22 16:10:54 +01:00
										 |  |  | extern char datatoc_common_fullscreen_vert_glsl[]; | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-29 15:58:46 +02:00
										 |  |  | #define USE_DEFERRED_COMPILATION 1
 | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \name Deferred Compilation (DRW_deferred)
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Since compiling shader can take a long time, we do it in a non blocking | 
					
						
							|  |  |  |  * manner in another thread. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct DRWDeferredShader { | 
					
						
							|  |  |  | 	struct DRWDeferredShader *prev, *next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	GPUMaterial *mat; | 
					
						
							|  |  |  | } DRWDeferredShader; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct DRWShaderCompiler { | 
					
						
							|  |  |  | 	ListBase queue; /* DRWDeferredShader */ | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 	SpinLock list_lock; | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	DRWDeferredShader *mat_compiling; | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 	ThreadMutex compilation_lock; | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-11 11:38:26 +02:00
										 |  |  | 	void *gl_context; | 
					
						
							| 
									
										
										
										
											2018-07-27 13:57:46 +02:00
										 |  |  | 	bool own_context; | 
					
						
							| 
									
										
										
										
											2018-03-07 19:40:08 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 	int shaders_done; /* To compute progress. */ | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | } DRWShaderCompiler; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void drw_deferred_shader_free(DRWDeferredShader *dsh) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* Make sure it is not queued before freeing. */ | 
					
						
							|  |  |  | 	MEM_freeN(dsh); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | static void drw_deferred_shader_queue_free(ListBase *queue) | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 	DRWDeferredShader *dsh; | 
					
						
							| 
									
										
										
										
											2018-04-19 07:47:03 +02:00
										 |  |  | 	while ((dsh = BLI_pophead(queue))) { | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 		drw_deferred_shader_free(dsh); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void drw_deferred_shader_compilation_exec(void *custom_data, short *stop, short *do_update, float *progress) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	DRWShaderCompiler *comp = (DRWShaderCompiler *)custom_data; | 
					
						
							| 
									
										
										
										
											2018-06-11 11:38:26 +02:00
										 |  |  | 	void *gl_context = comp->gl_context; | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-11 11:38:26 +02:00
										 |  |  | 	WM_opengl_context_activate(gl_context); | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	while (true) { | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 		BLI_spin_lock(&comp->list_lock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (*stop != 0) { | 
					
						
							|  |  |  | 			/* We don't want user to be able to cancel the compilation
 | 
					
						
							|  |  |  | 			 * but wm can kill the task if we are closing blender. */ | 
					
						
							|  |  |  | 			BLI_spin_unlock(&comp->list_lock); | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* Pop tail because it will be less likely to lock the main thread
 | 
					
						
							|  |  |  | 		 * if all GPUMaterials are to be freed (see DRW_deferred_shader_remove()). */ | 
					
						
							|  |  |  | 		comp->mat_compiling = BLI_poptail(&comp->queue); | 
					
						
							|  |  |  | 		if (comp->mat_compiling == NULL) { | 
					
						
							|  |  |  | 			/* No more Shader to compile. */ | 
					
						
							|  |  |  | 			BLI_spin_unlock(&comp->list_lock); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		comp->shaders_done++; | 
					
						
							|  |  |  | 		int total = BLI_listbase_count(&comp->queue) + comp->shaders_done; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		BLI_mutex_lock(&comp->compilation_lock); | 
					
						
							|  |  |  | 		BLI_spin_unlock(&comp->list_lock); | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* Do the compilation. */ | 
					
						
							| 
									
										
										
										
											2018-06-07 11:58:15 +02:00
										 |  |  | 		GPU_material_compile(comp->mat_compiling->mat); | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		*progress = (float)comp->shaders_done / (float)total; | 
					
						
							|  |  |  | 		*do_update = true; | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-13 10:38:13 -03:00
										 |  |  | 		glFlush(); | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 		BLI_mutex_unlock(&comp->compilation_lock); | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 		drw_deferred_shader_free(comp->mat_compiling); | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-11 11:38:26 +02:00
										 |  |  | 	WM_opengl_context_release(gl_context); | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void drw_deferred_shader_compilation_free(void *custom_data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	DRWShaderCompiler *comp = (DRWShaderCompiler *)custom_data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	drw_deferred_shader_queue_free(&comp->queue); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BLI_spin_end(&comp->list_lock); | 
					
						
							|  |  |  | 	BLI_mutex_end(&comp->compilation_lock); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-27 13:57:46 +02:00
										 |  |  | 	if (comp->own_context) { | 
					
						
							| 
									
										
										
										
											2018-04-10 19:30:13 +02:00
										 |  |  | 		/* Only destroy if the job owns the context. */ | 
					
						
							| 
									
										
										
										
											2018-06-11 11:38:26 +02:00
										 |  |  | 		WM_opengl_context_dispose(comp->gl_context); | 
					
						
							| 
									
										
										
										
											2018-04-10 19:30:13 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-03-07 19:40:08 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 	MEM_freeN(comp); | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-10 14:40:28 +02:00
										 |  |  | static void drw_deferred_shader_add(GPUMaterial *mat, bool deferred) | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-13 03:58:00 +01:00
										 |  |  | 	/* Do not deferre the compilation if we are rendering for image. */ | 
					
						
							| 
									
										
										
										
											2018-07-10 14:40:28 +02:00
										 |  |  | 	if (DRW_state_is_image_render() || !USE_DEFERRED_COMPILATION || !deferred) { | 
					
						
							| 
									
										
										
										
											2018-03-13 03:58:00 +01:00
										 |  |  | 		/* Double checking that this GPUMaterial is not going to be
 | 
					
						
							|  |  |  | 		 * compiled by another thread. */ | 
					
						
							|  |  |  | 		DRW_deferred_shader_remove(mat); | 
					
						
							| 
									
										
										
										
											2018-06-07 11:58:15 +02:00
										 |  |  | 		GPU_material_compile(mat); | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	DRWDeferredShader *dsh = MEM_callocN(sizeof(DRWDeferredShader), "Deferred Shader"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	dsh->mat = mat; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 	BLI_assert(DST.draw_ctx.evil_C); | 
					
						
							|  |  |  | 	wmWindowManager *wm = CTX_wm_manager(DST.draw_ctx.evil_C); | 
					
						
							|  |  |  | 	wmWindow *win = CTX_wm_window(DST.draw_ctx.evil_C); | 
					
						
							|  |  |  | 	Scene *scene = DST.draw_ctx.scene; | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 	/* Get the running job or a new one if none is running. Can only have one job per type & owner.  */ | 
					
						
							|  |  |  | 	wmJob *wm_job = WM_jobs_get(wm, win, scene, "Shaders Compilation", | 
					
						
							|  |  |  | 	                            WM_JOB_PROGRESS | WM_JOB_SUSPEND, WM_JOB_TYPE_SHADER_COMPILATION); | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 	DRWShaderCompiler *old_comp = (DRWShaderCompiler *)WM_jobs_customdata_get(wm_job); | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 	DRWShaderCompiler *comp = MEM_callocN(sizeof(DRWShaderCompiler), "DRWShaderCompiler"); | 
					
						
							|  |  |  | 	BLI_spin_init(&comp->list_lock); | 
					
						
							|  |  |  | 	BLI_mutex_init(&comp->compilation_lock); | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 	if (old_comp) { | 
					
						
							|  |  |  | 		BLI_spin_lock(&old_comp->list_lock); | 
					
						
							|  |  |  | 		BLI_movelisttolist(&comp->queue, &old_comp->queue); | 
					
						
							|  |  |  | 		BLI_spin_unlock(&old_comp->list_lock); | 
					
						
							| 
									
										
										
										
											2018-04-10 19:30:13 +02:00
										 |  |  | 		/* Do not recreate context, just pass ownership. */ | 
					
						
							| 
									
										
										
										
											2018-07-27 13:57:46 +02:00
										 |  |  | 		if (old_comp->gl_context) { | 
					
						
							|  |  |  | 			comp->gl_context = old_comp->gl_context; | 
					
						
							|  |  |  | 			old_comp->own_context = false; | 
					
						
							|  |  |  | 			comp->own_context = true; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 	BLI_addtail(&comp->queue, dsh); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-10 19:30:13 +02:00
										 |  |  | 	/* Create only one context. */ | 
					
						
							| 
									
										
										
										
											2018-06-11 11:38:26 +02:00
										 |  |  | 	if (comp->gl_context == NULL) { | 
					
						
							|  |  |  | 		comp->gl_context = WM_opengl_context_create(); | 
					
						
							|  |  |  | 		WM_opengl_context_activate(DST.gl_context); | 
					
						
							| 
									
										
										
										
											2018-07-27 13:57:46 +02:00
										 |  |  | 		comp->own_context = true; | 
					
						
							| 
									
										
										
										
											2018-04-10 19:30:13 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-03-07 19:40:08 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 	WM_jobs_customdata_set(wm_job, comp, drw_deferred_shader_compilation_free); | 
					
						
							|  |  |  | 	WM_jobs_timer(wm_job, 0.1, NC_MATERIAL | ND_SHADING_DRAW, 0); | 
					
						
							|  |  |  | 	WM_jobs_callbacks(wm_job, drw_deferred_shader_compilation_exec, NULL, NULL, NULL); | 
					
						
							|  |  |  | 	WM_jobs_start(wm, wm_job); | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | void DRW_deferred_shader_remove(GPUMaterial *mat) | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 	Scene *scene = GPU_material_scene(mat); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-19 19:30:47 +02:00
										 |  |  | 	for (wmWindowManager *wm = G_MAIN->wm.first; wm; wm = wm->id.next) { | 
					
						
							| 
									
										
										
										
											2018-03-07 01:39:09 +01:00
										 |  |  | 		if (WM_jobs_test(wm, scene, WM_JOB_TYPE_SHADER_COMPILATION) == false) { | 
					
						
							|  |  |  | 			/* No job running, do not create a new one by calling WM_jobs_get. */ | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-03-05 21:50:56 +01:00
										 |  |  | 		for (wmWindow *win = wm->windows.first; win; win = win->next) { | 
					
						
							|  |  |  | 			wmJob *wm_job = WM_jobs_get(wm, win, scene, "Shaders Compilation", | 
					
						
							|  |  |  | 			                            WM_JOB_PROGRESS | WM_JOB_SUSPEND, WM_JOB_TYPE_SHADER_COMPILATION); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			DRWShaderCompiler *comp = (DRWShaderCompiler *)WM_jobs_customdata_get(wm_job); | 
					
						
							|  |  |  | 			if (comp != NULL) { | 
					
						
							|  |  |  | 				BLI_spin_lock(&comp->list_lock); | 
					
						
							|  |  |  | 				DRWDeferredShader *dsh; | 
					
						
							|  |  |  | 				dsh = (DRWDeferredShader *)BLI_findptr(&comp->queue, mat, offsetof(DRWDeferredShader, mat)); | 
					
						
							|  |  |  | 				if (dsh) { | 
					
						
							|  |  |  | 					BLI_remlink(&comp->queue, dsh); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				/* Wait for compilation to finish */ | 
					
						
							|  |  |  | 				if (comp->mat_compiling != NULL) { | 
					
						
							|  |  |  | 					if (comp->mat_compiling->mat == mat) { | 
					
						
							|  |  |  | 						BLI_mutex_lock(&comp->compilation_lock); | 
					
						
							|  |  |  | 						BLI_mutex_unlock(&comp->compilation_lock); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				BLI_spin_unlock(&comp->list_lock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (dsh) { | 
					
						
							|  |  |  | 					drw_deferred_shader_free(dsh); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | GPUShader *DRW_shader_create(const char *vert, const char *geom, const char *frag, const char *defines) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-08-02 18:31:38 +02:00
										 |  |  | 	return GPU_shader_create(vert, frag, geom, NULL, defines, __func__); | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GPUShader *DRW_shader_create_with_lib( | 
					
						
							|  |  |  |         const char *vert, const char *geom, const char *frag, const char *lib, const char *defines) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	GPUShader *sh; | 
					
						
							|  |  |  | 	char *vert_with_lib = NULL; | 
					
						
							|  |  |  | 	char *frag_with_lib = NULL; | 
					
						
							|  |  |  | 	char *geom_with_lib = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	vert_with_lib = BLI_string_joinN(lib, vert); | 
					
						
							|  |  |  | 	frag_with_lib = BLI_string_joinN(lib, frag); | 
					
						
							|  |  |  | 	if (geom) { | 
					
						
							|  |  |  | 		geom_with_lib = BLI_string_joinN(lib, geom); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-02 18:31:38 +02:00
										 |  |  | 	sh = GPU_shader_create(vert_with_lib, frag_with_lib, geom_with_lib, NULL, defines, __func__); | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	MEM_freeN(vert_with_lib); | 
					
						
							|  |  |  | 	MEM_freeN(frag_with_lib); | 
					
						
							|  |  |  | 	if (geom) { | 
					
						
							|  |  |  | 		MEM_freeN(geom_with_lib); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return sh; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-16 12:47:15 +02:00
										 |  |  | GPUShader *DRW_shader_create_with_transform_feedback( | 
					
						
							|  |  |  |         const char *vert, const char *geom, const char *defines, | 
					
						
							|  |  |  |         const GPUShaderTFBType prim_type, const char **varying_names, const int varying_count) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return GPU_shader_create_ex(vert, NULL, geom, NULL, defines, GPU_SHADER_FLAGS_NONE, | 
					
						
							| 
									
										
										
										
											2018-08-02 18:31:38 +02:00
										 |  |  | 	                            prim_type, varying_names, varying_count, __func__); | 
					
						
							| 
									
										
										
										
											2018-05-16 12:47:15 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | GPUShader *DRW_shader_create_2D(const char *frag, const char *defines) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-08-02 18:31:38 +02:00
										 |  |  | 	return GPU_shader_create(datatoc_gpu_shader_2D_vert_glsl, frag, NULL, NULL, defines, __func__); | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GPUShader *DRW_shader_create_3D(const char *frag, const char *defines) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-08-02 18:31:38 +02:00
										 |  |  | 	return GPU_shader_create(datatoc_gpu_shader_3D_vert_glsl, frag, NULL, NULL, defines, __func__); | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GPUShader *DRW_shader_create_fullscreen(const char *frag, const char *defines) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-08-02 18:31:38 +02:00
										 |  |  | 	return GPU_shader_create(datatoc_common_fullscreen_vert_glsl, frag, NULL, NULL, defines, __func__); | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GPUShader *DRW_shader_create_3D_depth_only(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return GPU_shader_get_builtin_shader(GPU_SHADER_3D_DEPTH_ONLY); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-10 14:40:28 +02:00
										 |  |  | GPUMaterial *DRW_shader_find_from_world(World *wo, const void *engine_type, int options, bool deferred) | 
					
						
							| 
									
										
										
										
											2018-03-13 03:58:00 +01:00
										 |  |  | { | 
					
						
							|  |  |  | 	GPUMaterial *mat = GPU_material_from_nodetree_find(&wo->gpumaterial, engine_type, options); | 
					
						
							| 
									
										
										
										
											2018-07-10 14:40:28 +02:00
										 |  |  | 	if (DRW_state_is_image_render() || !deferred) { | 
					
						
							| 
									
										
										
										
											2018-03-13 03:58:00 +01:00
										 |  |  | 		if (mat != NULL && GPU_material_status(mat) == GPU_MAT_QUEUED) { | 
					
						
							|  |  |  | 			/* XXX Hack : we return NULL so that the engine will call DRW_shader_create_from_XXX
 | 
					
						
							|  |  |  | 			 * with the shader code and we will resume the compilation from there. */ | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return mat; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-10 14:40:28 +02:00
										 |  |  | GPUMaterial *DRW_shader_find_from_material(Material *ma, const void *engine_type, int options, bool deferred) | 
					
						
							| 
									
										
										
										
											2018-03-13 03:58:00 +01:00
										 |  |  | { | 
					
						
							|  |  |  | 	GPUMaterial *mat = GPU_material_from_nodetree_find(&ma->gpumaterial, engine_type, options); | 
					
						
							| 
									
										
										
										
											2018-07-10 14:40:28 +02:00
										 |  |  | 	if (DRW_state_is_image_render() || !deferred) { | 
					
						
							| 
									
										
										
										
											2018-03-13 03:58:00 +01:00
										 |  |  | 		if (mat != NULL && GPU_material_status(mat) == GPU_MAT_QUEUED) { | 
					
						
							|  |  |  | 			/* XXX Hack : we return NULL so that the engine will call DRW_shader_create_from_XXX
 | 
					
						
							|  |  |  | 			 * with the shader code and we will resume the compilation from there. */ | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return mat; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | GPUMaterial *DRW_shader_create_from_world( | 
					
						
							|  |  |  |         struct Scene *scene, World *wo, const void *engine_type, int options, | 
					
						
							| 
									
										
										
										
											2018-07-10 14:40:28 +02:00
										 |  |  |         const char *vert, const char *geom, const char *frag_lib, const char *defines, bool deferred) | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-13 03:58:00 +01:00
										 |  |  | 	GPUMaterial *mat = NULL; | 
					
						
							|  |  |  | 	if (DRW_state_is_image_render()) { | 
					
						
							|  |  |  | 		mat = GPU_material_from_nodetree_find(&wo->gpumaterial, engine_type, options); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (mat == NULL) { | 
					
						
							|  |  |  | 		mat = GPU_material_from_nodetree( | 
					
						
							| 
									
										
										
										
											2018-06-07 11:58:15 +02:00
										 |  |  | 		        scene, wo->nodetree, &wo->gpumaterial, engine_type, options, | 
					
						
							| 
									
										
										
										
											2018-08-02 18:31:38 +02:00
										 |  |  | 		        vert, geom, frag_lib, defines, wo->id.name); | 
					
						
							| 
									
										
										
										
											2018-03-13 03:58:00 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-07 11:58:15 +02:00
										 |  |  | 	if (GPU_material_status(mat) == GPU_MAT_QUEUED) { | 
					
						
							| 
									
										
										
										
											2018-07-10 14:40:28 +02:00
										 |  |  | 		drw_deferred_shader_add(mat, deferred); | 
					
						
							| 
									
										
										
										
											2018-06-07 11:58:15 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return mat; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GPUMaterial *DRW_shader_create_from_material( | 
					
						
							|  |  |  |         struct Scene *scene, Material *ma, const void *engine_type, int options, | 
					
						
							| 
									
										
										
										
											2018-07-10 14:40:28 +02:00
										 |  |  |         const char *vert, const char *geom, const char *frag_lib, const char *defines, bool deferred) | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-13 03:58:00 +01:00
										 |  |  | 	GPUMaterial *mat = NULL; | 
					
						
							|  |  |  | 	if (DRW_state_is_image_render()) { | 
					
						
							|  |  |  | 		mat = GPU_material_from_nodetree_find(&ma->gpumaterial, engine_type, options); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (mat == NULL) { | 
					
						
							|  |  |  | 		mat = GPU_material_from_nodetree( | 
					
						
							| 
									
										
										
										
											2018-06-07 11:58:15 +02:00
										 |  |  | 		        scene, ma->nodetree, &ma->gpumaterial, engine_type, options, | 
					
						
							| 
									
										
										
										
											2018-08-02 18:31:38 +02:00
										 |  |  | 		        vert, geom, frag_lib, defines, ma->id.name); | 
					
						
							| 
									
										
										
										
											2018-03-13 03:58:00 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-07 11:58:15 +02:00
										 |  |  | 	if (GPU_material_status(mat) == GPU_MAT_QUEUED) { | 
					
						
							| 
									
										
										
										
											2018-07-10 14:40:28 +02:00
										 |  |  | 		drw_deferred_shader_add(mat, deferred); | 
					
						
							| 
									
										
										
										
											2018-06-07 11:58:15 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-03-05 00:54:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return mat; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | void DRW_shader_free(GPUShader *shader) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	GPU_shader_free(shader); | 
					
						
							|  |  |  | } |