forked from Shirakumo/trial
-
Notifications
You must be signed in to change notification settings - Fork 0
/
shader-pass.lisp
618 lines (509 loc) · 24.9 KB
/
shader-pass.lisp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
(in-package #:org.shirakumo.fraf.trial)
(eval-when (:compile-toplevel :load-toplevel :execute)
(defclass shader-pass-class (shader-entity-class flow:static-node-class)
()))
(defmethod c2mop:validate-superclass ((class shader-pass-class) (superclass T))
NIL)
(defmethod c2mop:validate-superclass ((class T) (superclass shader-pass-class))
NIL)
(defmethod c2mop:validate-superclass ((class shader-pass-class) (superclass standard-class))
T)
;; FIXME: change texspec on per-instance basis to allow customising stuff
;; like texture size.
(defclass texture-port (flow:port)
((texture :initform NIL :accessor texture)
(texspec :initarg :texspec :accessor texspec)
(unit-id :initform NIL :accessor unit-id))
(:default-initargs
:texspec ()))
(defmethod stage ((port texture-port) (area staging-area))
(stage (texture port) area))
(defmethod (setf texture) :after ((texture texture) (port texture-port))
(dolist (connection (flow:connections port))
(let ((other (if (eq port (flow:left connection)) (flow:right connection) (flow:left connection))))
(setf (slot-value other 'texture) texture)
;; KLUDGE: this is ugly
(when (typep other 'output) (update-output-fb other texture)))))
(flow:define-port-value-slot texture-port texture texture)
;; FIXME: What about binding multiple levels and layers of the same texture?
(defclass image-port (texture-port)
((binding :initarg :binding :initform 0 :accessor binding)
(access :initarg :access :initform (error "ACCESS required.") :accessor access)))
(defmethod check-consistent ((port image-port)))
(defclass image-in (image-port flow:in-port flow:1-port)
((access :initform :read-only)))
(defclass image-out (image-port flow:out-port flow:n-port)
((access :initform :write-only)))
;; FIXME: check for duplicate inputs/outputs.
(defclass uniform-port (flow:port)
((uniform-name :initarg :uniform :initform NIL :accessor uniform-name)))
(defmethod initialize-instance :after ((port uniform-port) &key)
(unless (uniform-name port)
(setf (uniform-name port) (symbol->c-name (flow:name port)))))
(defclass input (flow:in-port flow:1-port texture-port uniform-port)
())
(defmethod check-consistent ((input input))
(unless (flow:connections input)
(error "Pipeline is not consistent.~%~
Pass ~s is missing a connection to its input ~s."
(flow:node input) input))
(let ((other (flow:left (first (flow:connections input)))))
(unless (or (not (texspec input))
(join-texspec (normalized-texspec (texspec input))
(normalized-texspec (texspec other))))
(error "Pipeline is not consistent.~%~
Pass ~s' input ~s~% ~s~%is not texture compatible with output ~s'~% ~s."
(flow:node input) input (normalized-texspec (texspec input))
other (normalized-texspec (texspec other))))))
(defclass output (flow:out-port flow:n-port texture-port)
((attachment :initarg :attachment :accessor attachment))
(:default-initargs :attachment :color-attachment0))
(defmethod check-consistent ((output output))
())
(defun update-output-fb (port new-texture)
(let ((fb (framebuffer (flow:node port))))
(when (and fb (not (eq new-texture (texture port))))
(setf (attachments fb)
(loop for (attachment texture . args) in (attachments fb)
collect (list* attachment
(if (eql attachment (attachment port))
new-texture
texture)
args))))))
(defmethod (setf texture) :before ((new-texture texture) (port output))
(update-output-fb port new-texture))
(defclass fixed-input (input)
())
(defmethod shared-initialize :after ((input fixed-input) slots &key texture)
(when texture
(setf (texture input) (eval texture))))
(defmethod check-consistent ((input fixed-input))
(unless (texture input)
(error "Pass ~s is missing an input texture ~s."
(flow:node input) input)))
(defclass static-input (flow:out-port flow:n-port texture-port uniform-port)
())
(defmethod check-consistent ((input static-input)))
(define-shader-entity shader-pass (flow:static-node)
((framebuffer :initform NIL :accessor framebuffer)
(active-p :initform T :accessor active-p)
(clear-color :initform (vec4) :initarg :clear-color :accessor clear-color))
(:metaclass shader-pass-class)
(:inhibit-shaders (shader-entity :fragment-shader)))
(defmethod update-instance-for-different-class :after ((prev shader-pass) (pass shader-pass) &rest args)
(declare (ignore args))
(when (and +main+ (slot-boundp +main+ 'scene) (scene +main+))
(handle (make-event 'instance-class-changed :instance pass) +main+)))
(defmethod shared-initialize :after ((pass shader-pass) slots &key)
(loop with texture-index = (max 16 (if *context* (gl:get-integer :max-texture-image-units) 0))
for port in (flow:ports pass)
do (typecase port
(texture-port
(setf (unit-id port) (decf texture-index))))))
(defgeneric bind-textures (object))
(defgeneric object-renderable-p (object pass))
(defgeneric compute-shader (shader-type pass object))
(defmethod object-renderable-p (object (pass shader-pass)) NIL)
(defmethod object-renderable-p ((renderable renderable) (pass shader-pass)) T)
(defmethod apply-transforms progn ((object renderable)))
(defmethod bind-textures ((object renderable)))
(defmethod bind-textures ((object standalone-shader-entity)))
(defmethod stage ((pass shader-pass) (area staging-area))
(dolist (port (flow:ports pass))
(stage port area))
(stage (framebuffer pass) area))
(defmethod check-consistent ((pass shader-pass))
(dolist (port (flow:ports pass))
(check-consistent port)))
(defmethod compute-shader (type pass object)
())
(defmethod compute-shader (type (pass shader-pass) object)
(append (call-next-method) (enlist (effective-shader type pass))))
(defmethod compute-shader (type pass (object shader-entity))
(append (call-next-method) (enlist (effective-shader type object))))
(defmethod compute-shader (type pass (object shader-entity-class))
(append (call-next-method) (enlist (effective-shader type object))))
(defmethod make-pass-shader-program ((pass shader-pass) object)
;; TODO: alias program against identical programs
(let* ((shaders ())
(directives (append (compute-preprocessor-directives pass)
(when (typep object 'shader-entity)
(compute-preprocessor-directives object))
(buffer-sources object)
(buffer-sources pass))))
(loop for type in *shader-type-list*
for inputs = (compute-shader type pass object)
do (when inputs
(let ((input (glsl-toolkit:merge-shader-sources
(list `(glsl-toolkit:shader ,@directives)
(glsl-toolkit:combine-methods inputs))
:min-version (glsl-target-version T))))
(push (make-instance 'shader :source input :type type) shaders))))
(make-instance 'shader-program :shaders shaders
:buffers (delete-duplicates (append (buffers object) (buffers pass))))))
(defmethod make-pass-shader-program ((pass shader-pass) (entity standalone-shader-entity))
(shader-program entity))
(defmethod make-pass-framebuffer ((pass shader-pass))
(let ((width 0) (height 0))
(loop for port in (flow:ports pass)
do (when (typep port 'output)
(setf width (max width (width (texture port))))
(setf height (max height (height (texture port))))))
(when (= 0 width)
(error "~a has no outputs!" pass))
(make-instance 'framebuffer
:width width
:height height
:clear-color (clear-color pass)
:attachments (loop for port in (flow:ports pass)
when (typep port 'output)
collect (list (attachment port) (texture port))))))
(defmethod finalize :after ((pass shader-pass))
(when (framebuffer pass)
(finalize (framebuffer pass))))
(defmethod enter ((container container) (pass shader-pass))
(prog1 (when (next-method-p)
(call-next-method))
(sequence:dosequence (object container)
(enter object pass))))
(defmethod leave ((container container) (pass shader-pass))
(prog1 (when (next-method-p)
(call-next-method))
(for:for ((object over container))
(leave object pass))))
(defmethod render (object (pass shader-pass))
(render object (shader-program-for-pass pass object)))
(defmethod width ((pass shader-pass))
(width (framebuffer pass)))
(defmethod height ((pass shader-pass))
(height (framebuffer pass)))
(defmacro define-shader-pass (name direct-superclasses direct-slots &rest options)
(setf direct-superclasses (append direct-superclasses (list 'shader-pass)))
(unless (find :metaclass options :key #'car)
(push '(:metaclass shader-pass-class) options))
`(defclass ,name ,direct-superclasses
,direct-slots
,@options))
(defmethod prepare-pass-program ((pass shader-pass) program)
(loop for slot in (c2mop:class-slots (class-of pass))
when (and (typep slot 'flow::effective-port-definition) (flow:port-type slot))
do (let ((port (flow::port-slot-value pass slot)))
(typecase port
(uniform-port
(if (texture port)
(setf (uniform program (uniform-name port)) (unit-id port))
(setf (uniform program (uniform-name port)) (slot-value pass (c2mop:slot-definition-name slot)))))))))
(defmethod bind-textures ((pass shader-pass))
;; FIXME: I kinda hate this and we could definitely optimise the iteration away.
(loop for slot in (c2mop:class-slots (class-of pass))
when (and (typep slot 'flow::effective-port-definition) (flow:port-type slot))
do (let ((port (flow::port-slot-value pass slot)))
(typecase port
(uniform-port
(when (texture port)
(bind (texture port) (unit-id port))))
(image-port
(when (texture port)
(%gl:bind-image-texture (binding port) (gl-name (texture port)) 0 T 0 (access port)
(internal-format (texture port)))))))))
(defmethod prepare-pass-program :around ((pass shader-pass) (program shader-program))
(unless (eq +current-shader-program+ program)
(activate program)
(call-next-method))
(update-uniforms pass program))
(defmethod blit-to-screen ((pass shader-pass))
(blit-to-screen (framebuffer pass)))
(defmethod capture ((pass shader-pass) &rest args)
(apply #'capture (framebuffer pass) args))
(defmethod render (object (pass shader-pass))
(let ((program (shader-program-for-pass pass object))
(framebuffer (framebuffer pass)))
(activate framebuffer)
(bind-textures pass)
(prepare-pass-program pass program)
(render object program)
(deactivate framebuffer)))
(defmethod save-image ((pass shader-pass) target type &rest args)
(apply #'save-image (framebuffer pass) target type args))
(defmacro define-pass-shader ((class pass-class shader-type) &body body)
`(progn (defmethod compute-shader ((type (eql ,shader-type)) (pass ,pass-class) (class (eql (find-class ',class))))
(load-time-value (list (glsl-toolkit:parse (progn ,@body)))))
(when (and +main+ (slot-boundp +main+ 'scene) (scene +main+))
(handle (make-event 'class-changed :changed-class (find-class ',class)) +main+))))
(define-shader-pass per-object-pass (listener)
((program-table :initform (make-hash-table :test 'eq) :accessor program-table)
(renderable-table :initform (make-hash-table :test 'eq) :accessor renderable-table)
(frame :initform (map-into (make-array 128 :adjustable T :fill-pointer 0) (lambda () (cons NIL NIL))) :accessor frame)))
(defgeneric construct-frame (pass))
(defgeneric render-frame (pass frame))
(defgeneric sort-frame (pass frame))
(defgeneric render-with (pass object program))
(defmethod camera ((pass shader-pass))
(camera (scene +main+)))
(defmethod scene ((pass shader-pass))
(scene +main+))
(defmethod sort-frame ((pass per-object-pass) frame)
frame)
(defmethod map-visible (function (camera camera) (pass per-object-pass))
(loop for object being the hash-keys of (renderable-table pass) using (hash-value program)
do (when (and (not (typep object 'class))
(in-view-p object camera))
(funcall function object))))
(defmethod map-visible (function (camera null) (pass per-object-pass))
(loop for object being the hash-keys of (renderable-table pass) using (hash-value program)
do (when (not (typep object 'class))
(funcall function object))))
(defmethod shader-program-for-pass ((pass per-object-pass) object)
(gethash object (renderable-table pass)))
(defmethod enter (object (pass per-object-pass))
(when (next-method-p)
(call-next-method))
(when (or (object-renderable-p object pass)
(typep object 'shader-entity-class))
(let ((renderable-table (renderable-table pass)))
(or (gethash object renderable-table)
(let* ((program-table (program-table pass))
(target (if (typep object '(or standalone-shader-entity dynamic-renderable))
object
(effective-shader-class object)))
(program (gethash target renderable-table)))
(unless program
(setf program (make-pass-shader-program pass target))
(unless (gethash program program-table)
(setf (gethash program program-table) (cons 0 NIL))
(setf (gethash target renderable-table) program)))
(incf (car (gethash program program-table)))
(setf (gethash object renderable-table) program))))))
(defmethod leave (object (pass per-object-pass))
(when (next-method-p)
(call-next-method))
(let* ((renderable-table (renderable-table pass))
(program (gethash object renderable-table)))
(when program
(decf (car (gethash program (program-table pass))))
(remhash object renderable-table))))
(defmethod stage ((pass per-object-pass) (area staging-area))
(call-next-method)
(loop for program being the hash-keys of (program-table pass) using (hash-value (count . cached))
do (cond ((<= count 0)
(remhash program (program-table pass))
(loop for other being the hash-values of (renderable-table pass) using (hash-key key)
do (when (eq other program) (remhash key (renderable-table pass)))))
(T
(stage program area)))))
(defmethod stage ((object shader-entity) (pass per-object-pass))
(unless (typep object 'dynamic-renderable)
(stage (effective-shader-class object) pass)))
(defmethod stage ((object shader-entity-class) (pass per-object-pass))
(enter object pass))
;; FIXME: Maybe consider determining effective class for each
;; individual shader stage as they might each change
;; at different levels and could thus be cached more
;; effectively.
;; FIXME: Share SHADER assets between shader programs by caching
;; them... somewhere somehow?
(defmethod handle ((ev class-changed) (pass per-object-pass))
(call-next-method)
(let ((class (changed-class ev))
(program-table (program-table pass))
(renderable-table (renderable-table pass)))
;; FIXME: What happens if the effective shader class changes?
;; We might be leaking shader programs for stale classes then.
(flet ((refresh (class)
(let ((prev (gethash class renderable-table)))
(v:info :trial.shader-pass "Refreshing shader program for ~a" class)
(let ((new (make-pass-shader-program pass class)))
(setf (buffers prev) (buffers new))
(setf (shaders prev) (shaders new))
(setf (cdr (gethash prev program-table)) NIL)))))
(cond ((eql class (class-of pass))
;; Pass changed, recompile everything
(loop for object being the hash-keys of renderable-table
do (when (typep object 'standard-class)
(refresh object))))
((and (gethash class renderable-table) (eql class (effective-shader-class class)))
;; Object changed, recompile it
(refresh class))))))
(defmethod handle ((ev instance-class-changed) (pass per-object-pass))
(call-next-method)
(let ((object (instance ev))
(renderable-table (renderable-table pass)))
(cond ((eq object pass)
;; Pass changed, recompile everything
(handle (make-event 'class-changed :changed-class (class-of object)) pass))
((gethash object renderable-table)
;; Just re-enter the object to facilitate the update.
(leave object pass)
(let ((program (enter object pass)))
(unless (allocated-p program)
(commit program (loader +main+) :unload NIL)))))))
(defmethod render ((pass per-object-pass) (_ null))
(render-frame pass (construct-frame pass)))
(defmethod prepare-pass-program ((pass per-object-pass) program)
(let ((entry (gethash program (program-table pass))))
(unless (cdr entry)
;; Only update if we haven't cached yet.
(call-next-method)
(setf (cdr entry) T))))
(defmethod construct-frame ((pass per-object-pass))
(let* ((frame (frame pass))
(index 0)
(total (array-total-size frame))
(renderable-table (renderable-table pass)))
(flet ((store (object program)
(when (<= total (incf index))
(adjust-array frame (* 2 total))
(loop for i from total below (* 2 total)
do (setf (aref frame i) (cons NIL NIL)))
(setf total (* 2 total)))
(let ((entry (aref frame (1- index))))
(setf (car entry) object)
(setf (cdr entry) program))))
(do-visible (object (camera pass) (scene pass))
(let ((program (gethash object renderable-table)))
(when program
(store object program)))))
(setf (fill-pointer frame) index)
(sort-frame pass frame)))
(defmethod render-frame ((pass per-object-pass) frame)
(declare (type (and vector (not simple-vector)) frame))
(let ((framebuffer (framebuffer pass)))
(activate framebuffer)
(bind-textures pass)
(loop for (object . program) across frame
do (render-with pass object program))
(deactivate framebuffer)))
(defmethod render-with :around ((pass per-object-pass) (object renderable) (program shader-program))
(restart-case
(progn
(prepare-pass-program pass program)
(with-pushed-matrix ()
(apply-transforms object)
(bind-textures object)
(update-uniforms object program)
(call-next-method)))
#-kandria-release
(leave ()
:report "Leave the object"
(leave object T))))
(defmethod render-with ((pass shader-pass) object program)
(render object program))
;;; This is pretty hacky.
(defmethod stage :after ((renderable renderable) (area staging-area))
(loop for observer in (gethash 'renderable (observers area))
do (observe-load-state observer renderable :staged area)))
(defmethod stage :after ((pass per-object-pass) (area staging-area))
(register-load-observer area pass 'renderable))
(defmethod observe-load-state ((pass per-object-pass) (renderable renderable) (state (eql :staged)) (area staging-area))
(let ((program (enter renderable pass)))
(when program
(stage program area))))
(define-shader-pass single-shader-pass ()
((shader-program :accessor shader-program)))
(defmethod update-instance-for-different-class :after ((previous single-shader-pass) (pass single-shader-pass) &key)
(when (allocated-p (shader-program pass))
(setf (shaders (shader-program pass))
(shaders (make-shader-program pass)))))
(defmethod (setf c2mop:slot-value-using-class) :after (value (class shader-pass-class) (entity single-shader-pass) (slotd buffer-slot-definition))
(when (and (slot-boundp entity 'shader-program) (allocated-p (shader-program entity)))
(bind value (shader-program entity))))
(defmethod stage ((pass single-shader-pass) (area staging-area))
(unless (slot-boundp pass 'shader-program)
(setf (shader-program pass) (make-shader-program pass)))
(call-next-method)
(stage (shader-program pass) area))
(defmethod handle ((ev class-changed) (pass single-shader-pass))
(when (eql (changed-class ev) (class-of pass))
(let ((prev (shader-program pass))
(new (make-shader-program pass)))
(v:info :trial.shader-pass "Refreshing shader program for ~a" (class-of pass))
(setf (buffers prev) (buffers new))
(setf (shaders prev) (shaders new)))))
(defmethod shader-program-for-pass ((pass single-shader-pass) o)
(shader-program pass))
(defmethod make-pass-shader-program ((pass single-shader-pass) o)
(shader-program pass))
(defmethod render ((pass single-shader-pass) (_ null))
(bind-textures pass)
(render pass (shader-program pass)))
(defmethod render :around ((pass single-shader-pass) (program shader-program))
(let ((framebuffer (framebuffer pass)))
(when framebuffer (activate framebuffer))
(prepare-pass-program pass program)
(call-next-method)
(when framebuffer (deactivate framebuffer))))
(define-shader-pass post-effect-pass (single-shader-pass)
((vertex-array :initform (// 'trial 'fullscreen-square) :accessor vertex-array)))
(defmethod stage :after ((pass post-effect-pass) (area staging-area))
(stage (vertex-array pass) area))
(defmethod object-renderable-p ((renderable renderable) (pass post-effect-pass)) NIL)
(defmethod enter (thing (pass post-effect-pass)))
(defmethod leave (thing (pass post-effect-pass)))
(defmethod handle ((event event) (pass post-effect-pass)))
(defmethod render ((pass post-effect-pass) (program shader-program))
(render (vertex-array pass) program))
(define-class-shader (post-effect-pass :vertex-shader)
"
layout (location = TRIAL_V_LOCATION) in vec3 position;
layout (location = TRIAL_V_UV) in vec2 in_uv;
out vec2 uv;
void main(){
gl_Position = vec4(position, 1.0f);
uv = in_uv;
}")
(define-class-shader (post-effect-pass :fragment-shader)
"
in vec2 uv;")
(define-shader-pass sample-reduction-pass (post-effect-pass)
((previous-pass :port-type input :texspec (:target :texture-2d-multisample))
(color :port-type output :texspec (:target :texture-2d) :reader color)))
(define-class-shader (sample-reduction-pass :fragment-shader)
"uniform sampler2DMS previous_pass;
in vec2 uv;
out vec4 color;
void main(){
color = texture(previous_pass, uv);
}")
(define-shader-pass compute-pass (single-shader-pass)
((work-groups :initform (vec 1 1 1) :initarg :work-groups :accessor work-groups)
(barrier :initform 4294967295)))
(defmethod initialize-instance :after ((pass compute-pass) &key)
(unless (integerp (slot-value pass 'barrier))
(setf (barrier pass) (slot-value pass 'barrier))))
(defmethod shared-initialize :after ((pass compute-pass) slots &key (barrier NIL barrier-p))
(when barrier-p (setf (barrier pass) barrier)))
(defmethod handle ((event event) (pass compute-pass)))
(defmethod barrier ((pass compute-pass))
(cffi:foreign-bitfield-symbols '%gl::MemoryBarrierMask (slot-value pass 'barrier)))
(defmethod (setf barrier) ((bits list) (pass compute-pass))
(setf (slot-value pass 'barrier) (cffi:foreign-bitfield-value '%gl::MemoryBarrierMask bits)))
(defmethod (setf barrier) ((bits symbol) (pass compute-pass))
(setf (slot-value pass 'barrier) (cffi:foreign-bitfield-value '%gl::MemoryBarrierMask (list bits))))
(defmethod render ((pass compute-pass) (_ null))
(bind-textures pass)
(render pass (or (shader-program pass)
(error "Shader program was never allocated!!"))))
(defmethod render ((pass compute-pass) (program shader-program))
(let ((work-groups (work-groups pass))
(barrier (slot-value pass 'barrier)))
(etypecase work-groups
(vec3
(%gl:dispatch-compute
(truncate (vx work-groups))
(truncate (vy work-groups))
(truncate (vz work-groups))))
(integer
(%gl:dispatch-compute-indirect work-groups))
(buffer-object
(%gl:bind-buffer :dispatch-indirect-buffer (gl-name work-groups))
(%gl:dispatch-compute-indirect 0)))
(when (/= 0 barrier)
(%gl:memory-barrier barrier))))
(defmethod render ((pass compute-pass) (work-groups vec3))
(setf (work-groups pass) work-groups)
(render pass NIL))
;; KLUDGE: this sucks as we override more than we need to.
(defmethod (setf class-shader) :before (shader type (class shader-pass-class))
(when (and (c2mop:subclassp class (find-class 'compute-pass))
(not (eql type :compute-shader)))
(error "May only attach a compute shader to compute-passes.")))