1 /* Display a cleared blue window. This demo has no dependencies on
2 * any utility code, just the graw interface and gallium.
3 */
4
5 #include "state_tracker/graw.h"
6 #include "pipe/p_screen.h"
7 #include "pipe/p_context.h"
8 #include "pipe/p_shader_tokens.h"
9 #include "pipe/p_state.h"
10 #include "pipe/p_defines.h"
11 #include <stdio.h> /* for fread(), etc */
12
13 #include "util/u_inlines.h"
14 #include "util/u_memory.h" /* Offset() */
15 #include "util/u_draw_quad.h"
16 #include "util/u_box.h"
17
18 static const char *filename = NULL;
19 unsigned show_fps = 0;
20
21
usage(char * name)22 static void usage(char *name)
23 {
24 fprintf(stderr, "usage: %s [ options ] shader_filename\n", name);
25 #ifndef WIN32
26 fprintf(stderr, "\n" );
27 fprintf(stderr, "options:\n");
28 fprintf(stderr, " -fps show frames per second\n");
29 #endif
30 }
31
32
33 enum pipe_format formats[] = {
34 PIPE_FORMAT_R8G8B8A8_UNORM,
35 PIPE_FORMAT_B8G8R8A8_UNORM,
36 PIPE_FORMAT_NONE
37 };
38
39 static const int WIDTH = 250;
40 static const int HEIGHT = 250;
41
42 static struct pipe_screen *screen = NULL;
43 static struct pipe_context *ctx = NULL;
44 static struct pipe_resource *rttex = NULL;
45 static struct pipe_resource *constbuf1 = NULL;
46 static struct pipe_resource *constbuf2 = NULL;
47 static struct pipe_surface *surf = NULL;
48 static struct pipe_sampler_view *sv = NULL;
49 static void *sampler = NULL;
50 static void *window = NULL;
51 static struct pipe_resource *samptex = NULL;
52
53 struct vertex {
54 float position[4];
55 float color[4];
56 float texcoord[4];
57 };
58
59 /* Vertex data matches progs/fp/fp-tri.c, but flipped in Y dimension
60 * so that the final images are the same.
61 */
62 static struct vertex vertices[] =
63 {
64 { { 0.9, 0.9, 0.0, 1.0 },
65 { 0, 0, 1, 1 },
66 { 1, 1, 0, 1 } },
67
68 { { 0.9, -0.9, 0.0, 1.0 },
69 { 1, 0, 0, 1 },
70 { 1, -1, 0, 1 } },
71
72 { {-0.9, 0.0, 0.0, 1.0 },
73 { 0, 1, 0, 1 },
74 { -1, 0, 0, 1 } },
75 };
76
77 static float constants1[] =
78 { 0.4, 0, 0, 1,
79 1, 1, 1, 1,
80 2, 2, 2, 2,
81 4, 8, 16, 32,
82
83 3, 0, 0, 0,
84 0, .5, 0, 0,
85 1, 0, 0, 1,
86 0, 0, 0, 1,
87
88 1, 0, 0, 0.5,
89 0, 1, 0, 0.5,
90 0, 0, 1, 0,
91 0, 0, 0, 1,
92 };
93
94
95 static float constants2[] =
96 { 1, 0, 0, 1,
97 0, 1, 0, 1,
98 0, 0, 1, 1,
99 0, 0, 0, 0,
100
101 1, 1, 0, 1,
102 1, .5, 0, 1,
103 1, 0, 0, 1,
104 0, 0, 0, 1,
105
106 1, 0, 0, 0.5,
107 0, 1, 0, 0.5,
108 0, 0, 1, 0,
109 0, 0, 0, 1,
110 };
111
init_fs_constbuf(void)112 static void init_fs_constbuf( void )
113 {
114 struct pipe_resource templat;
115 struct pipe_box box;
116
117 templat.target = PIPE_BUFFER;
118 templat.format = PIPE_FORMAT_R8_UNORM;
119 templat.width0 = sizeof(constants1);
120 templat.height0 = 1;
121 templat.depth0 = 1;
122 templat.array_size = 1;
123 templat.last_level = 0;
124 templat.nr_samples = 1;
125 templat.bind = PIPE_BIND_CONSTANT_BUFFER;
126
127 constbuf1 = screen->resource_create(screen,
128 &templat);
129 if (constbuf1 == NULL)
130 exit(4);
131
132 constbuf2 = screen->resource_create(screen,
133 &templat);
134 if (constbuf2 == NULL)
135 exit(4);
136
137
138 {
139 u_box_2d(0,0,sizeof(constants1),1, &box);
140
141 ctx->transfer_inline_write(ctx,
142 constbuf1,
143 0,
144 PIPE_TRANSFER_WRITE,
145 &box,
146 constants1,
147 sizeof constants1,
148 sizeof constants1);
149
150
151 pipe_set_constant_buffer(ctx,
152 PIPE_SHADER_FRAGMENT, 0,
153 constbuf1);
154 }
155 {
156 u_box_2d(0,0,sizeof(constants2),1, &box);
157
158 ctx->transfer_inline_write(ctx,
159 constbuf2,
160 0,
161 PIPE_TRANSFER_WRITE,
162 &box,
163 constants2,
164 sizeof constants2,
165 sizeof constants2);
166
167
168 pipe_set_constant_buffer(ctx,
169 PIPE_SHADER_FRAGMENT, 1,
170 constbuf2);
171 }
172 }
173
174
set_viewport(float x,float y,float width,float height,float near,float far)175 static void set_viewport( float x, float y,
176 float width, float height,
177 float near, float far)
178 {
179 float z = far;
180 float half_width = (float)width / 2.0f;
181 float half_height = (float)height / 2.0f;
182 float half_depth = ((float)far - (float)near) / 2.0f;
183 struct pipe_viewport_state vp;
184
185 vp.scale[0] = half_width;
186 vp.scale[1] = half_height;
187 vp.scale[2] = half_depth;
188 vp.scale[3] = 1.0f;
189
190 vp.translate[0] = half_width + x;
191 vp.translate[1] = half_height + y;
192 vp.translate[2] = half_depth + z;
193 vp.translate[3] = 0.0f;
194
195 ctx->set_viewport_state( ctx, &vp );
196 }
197
set_vertices(void)198 static void set_vertices( void )
199 {
200 struct pipe_vertex_element ve[3];
201 struct pipe_vertex_buffer vbuf;
202 void *handle;
203
204 memset(ve, 0, sizeof ve);
205
206 ve[0].src_offset = Offset(struct vertex, position);
207 ve[0].src_format = PIPE_FORMAT_R32G32B32A32_FLOAT;
208 ve[1].src_offset = Offset(struct vertex, color);
209 ve[1].src_format = PIPE_FORMAT_R32G32B32A32_FLOAT;
210 ve[2].src_offset = Offset(struct vertex, texcoord);
211 ve[2].src_format = PIPE_FORMAT_R32G32B32A32_FLOAT;
212
213 handle = ctx->create_vertex_elements_state(ctx, 3, ve);
214 ctx->bind_vertex_elements_state(ctx, handle);
215
216
217 vbuf.stride = sizeof( struct vertex );
218 vbuf.buffer_offset = 0;
219 vbuf.buffer = pipe_buffer_create_with_data(ctx,
220 PIPE_BIND_VERTEX_BUFFER,
221 PIPE_USAGE_STATIC,
222 sizeof(vertices),
223 vertices);
224
225 ctx->set_vertex_buffers(ctx, 1, &vbuf);
226 }
227
set_vertex_shader(void)228 static void set_vertex_shader( void )
229 {
230 void *handle;
231 const char *text =
232 "VERT\n"
233 "DCL IN[0]\n"
234 "DCL IN[1]\n"
235 "DCL IN[2]\n"
236 "DCL OUT[0], POSITION\n"
237 "DCL OUT[1], COLOR[0]\n"
238 "DCL OUT[2], GENERIC[0]\n"
239 " MOV OUT[0], IN[0]\n"
240 " MOV OUT[1], IN[1]\n"
241 " MOV OUT[2], IN[2]\n"
242 " END\n";
243
244 handle = graw_parse_vertex_shader(ctx, text);
245 ctx->bind_vs_state(ctx, handle);
246 }
247
set_fragment_shader(const char * filename)248 static void set_fragment_shader( const char *filename )
249 {
250 FILE *f;
251 char buf[50000];
252 void *handle;
253 int sz;
254
255 if ((f = fopen(filename, "r")) == NULL) {
256 fprintf(stderr, "Couldn't open %s\n", filename);
257 exit(1);
258 }
259
260 sz = fread(buf, 1, sizeof(buf), f);
261 if (!feof(f)) {
262 printf("file too long\n");
263 exit(1);
264 }
265 printf("%.*s\n", sz, buf);
266 buf[sz] = 0;
267
268 handle = graw_parse_fragment_shader(ctx, buf);
269 ctx->bind_fs_state(ctx, handle);
270 fclose(f);
271 }
272
273
draw(void)274 static void draw( void )
275 {
276 union pipe_color_union clear_color = { {.1,.3,.5,0} };
277
278 ctx->clear(ctx, PIPE_CLEAR_COLOR, &clear_color, 0, 0);
279 util_draw_arrays(ctx, PIPE_PRIM_TRIANGLES, 0, 3);
280 ctx->flush(ctx, NULL);
281
282 graw_save_surface_to_file(ctx, surf, NULL);
283
284 screen->flush_frontbuffer(screen, rttex, 0, 0, window);
285 }
286
287 #define SIZE 16
288
init_tex(void)289 static void init_tex( void )
290 {
291 struct pipe_sampler_view sv_template;
292 struct pipe_sampler_state sampler_desc;
293 struct pipe_resource templat;
294 struct pipe_box box;
295 ubyte tex2d[SIZE][SIZE][4];
296 int s, t;
297
298 #if (SIZE != 2)
299 for (s = 0; s < SIZE; s++) {
300 for (t = 0; t < SIZE; t++) {
301 if (0) {
302 int x = (s ^ t) & 1;
303 tex2d[t][s][0] = (x) ? 0 : 63;
304 tex2d[t][s][1] = (x) ? 0 : 128;
305 tex2d[t][s][2] = 0;
306 tex2d[t][s][3] = 0xff;
307 }
308 else {
309 int x = ((s ^ t) >> 2) & 1;
310 tex2d[t][s][0] = s*255/(SIZE-1);
311 tex2d[t][s][1] = t*255/(SIZE-1);
312 tex2d[t][s][2] = (x) ? 0 : 128;
313 tex2d[t][s][3] = 0xff;
314 }
315 }
316 }
317 #else
318 tex2d[0][0][0] = 0;
319 tex2d[0][0][1] = 255;
320 tex2d[0][0][2] = 255;
321 tex2d[0][0][3] = 0;
322
323 tex2d[0][1][0] = 0;
324 tex2d[0][1][1] = 0;
325 tex2d[0][1][2] = 255;
326 tex2d[0][1][3] = 255;
327
328 tex2d[1][0][0] = 255;
329 tex2d[1][0][1] = 255;
330 tex2d[1][0][2] = 0;
331 tex2d[1][0][3] = 255;
332
333 tex2d[1][1][0] = 255;
334 tex2d[1][1][1] = 0;
335 tex2d[1][1][2] = 0;
336 tex2d[1][1][3] = 255;
337 #endif
338
339 templat.target = PIPE_TEXTURE_2D;
340 templat.format = PIPE_FORMAT_B8G8R8A8_UNORM;
341 templat.width0 = SIZE;
342 templat.height0 = SIZE;
343 templat.depth0 = 1;
344 templat.array_size = 1;
345 templat.last_level = 0;
346 templat.nr_samples = 1;
347 templat.bind = PIPE_BIND_SAMPLER_VIEW;
348
349
350 samptex = screen->resource_create(screen,
351 &templat);
352 if (samptex == NULL)
353 exit(4);
354
355 u_box_2d(0,0,SIZE,SIZE, &box);
356
357 ctx->transfer_inline_write(ctx,
358 samptex,
359 0,
360 PIPE_TRANSFER_WRITE,
361 &box,
362 tex2d,
363 sizeof tex2d[0],
364 sizeof tex2d);
365
366 /* Possibly read back & compare against original data:
367 */
368 if (0)
369 {
370 struct pipe_transfer *t;
371 uint32_t *ptr;
372 t = pipe_get_transfer(ctx, samptex,
373 0, 0, /* level, layer */
374 PIPE_TRANSFER_READ,
375 0, 0, SIZE, SIZE); /* x, y, width, height */
376
377 ptr = ctx->transfer_map(ctx, t);
378
379 if (memcmp(ptr, tex2d, sizeof tex2d) != 0) {
380 assert(0);
381 exit(9);
382 }
383
384 ctx->transfer_unmap(ctx, t);
385
386 ctx->transfer_destroy(ctx, t);
387 }
388
389 memset(&sv_template, 0, sizeof sv_template);
390 sv_template.format = samptex->format;
391 sv_template.texture = samptex;
392 sv_template.swizzle_r = 0;
393 sv_template.swizzle_g = 1;
394 sv_template.swizzle_b = 2;
395 sv_template.swizzle_a = 3;
396 sv = ctx->create_sampler_view(ctx, samptex, &sv_template);
397 if (sv == NULL)
398 exit(5);
399
400 ctx->set_fragment_sampler_views(ctx, 1, &sv);
401
402
403 memset(&sampler_desc, 0, sizeof sampler_desc);
404 sampler_desc.wrap_s = PIPE_TEX_WRAP_REPEAT;
405 sampler_desc.wrap_t = PIPE_TEX_WRAP_REPEAT;
406 sampler_desc.wrap_r = PIPE_TEX_WRAP_REPEAT;
407 sampler_desc.min_img_filter = PIPE_TEX_FILTER_NEAREST;
408 sampler_desc.min_mip_filter = PIPE_TEX_MIPFILTER_NONE;
409 sampler_desc.mag_img_filter = PIPE_TEX_FILTER_NEAREST;
410 sampler_desc.compare_mode = PIPE_TEX_COMPARE_NONE;
411 sampler_desc.compare_func = 0;
412 sampler_desc.normalized_coords = 1;
413 sampler_desc.max_anisotropy = 0;
414
415 sampler = ctx->create_sampler_state(ctx, &sampler_desc);
416 if (sampler == NULL)
417 exit(6);
418
419 ctx->bind_fragment_sampler_states(ctx, 1, &sampler);
420
421 }
422
init(void)423 static void init( void )
424 {
425 struct pipe_framebuffer_state fb;
426 struct pipe_resource templat;
427 struct pipe_surface surf_tmpl;
428 int i;
429
430 /* It's hard to say whether window or screen should be created
431 * first. Different environments would prefer one or the other.
432 *
433 * Also, no easy way of querying supported formats if the screen
434 * cannot be created first.
435 */
436 for (i = 0; formats[i] != PIPE_FORMAT_NONE; i++) {
437 screen = graw_create_window_and_screen(0, 0, 300, 300,
438 formats[i],
439 &window);
440 if (window && screen)
441 break;
442 }
443 if (!screen || !window) {
444 fprintf(stderr, "Unable to create window\n");
445 exit(1);
446 }
447
448 ctx = screen->context_create(screen, NULL);
449 if (ctx == NULL)
450 exit(3);
451
452 templat.target = PIPE_TEXTURE_2D;
453 templat.format = formats[i];
454 templat.width0 = WIDTH;
455 templat.height0 = HEIGHT;
456 templat.depth0 = 1;
457 templat.array_size = 1;
458 templat.last_level = 0;
459 templat.nr_samples = 1;
460 templat.bind = (PIPE_BIND_RENDER_TARGET |
461 PIPE_BIND_DISPLAY_TARGET);
462
463 rttex = screen->resource_create(screen,
464 &templat);
465 if (rttex == NULL)
466 exit(4);
467
468 surf_tmpl.format = templat.format;
469 surf_tmpl.usage = PIPE_BIND_RENDER_TARGET;
470 surf_tmpl.u.tex.level = 0;
471 surf_tmpl.u.tex.first_layer = 0;
472 surf_tmpl.u.tex.last_layer = 0;
473 surf = ctx->create_surface(ctx, rttex, &surf_tmpl);
474 if (surf == NULL)
475 exit(5);
476
477 memset(&fb, 0, sizeof fb);
478 fb.nr_cbufs = 1;
479 fb.width = WIDTH;
480 fb.height = HEIGHT;
481 fb.cbufs[0] = surf;
482
483 ctx->set_framebuffer_state(ctx, &fb);
484
485 {
486 struct pipe_blend_state blend;
487 void *handle;
488 memset(&blend, 0, sizeof blend);
489 blend.rt[0].colormask = PIPE_MASK_RGBA;
490 handle = ctx->create_blend_state(ctx, &blend);
491 ctx->bind_blend_state(ctx, handle);
492 }
493
494 {
495 struct pipe_depth_stencil_alpha_state depthstencil;
496 void *handle;
497 memset(&depthstencil, 0, sizeof depthstencil);
498 handle = ctx->create_depth_stencil_alpha_state(ctx, &depthstencil);
499 ctx->bind_depth_stencil_alpha_state(ctx, handle);
500 }
501
502 {
503 struct pipe_rasterizer_state rasterizer;
504 void *handle;
505 memset(&rasterizer, 0, sizeof rasterizer);
506 rasterizer.cull_face = PIPE_FACE_NONE;
507 rasterizer.gl_rasterization_rules = 1;
508 rasterizer.depth_clip = 1;
509 handle = ctx->create_rasterizer_state(ctx, &rasterizer);
510 ctx->bind_rasterizer_state(ctx, handle);
511 }
512
513 set_viewport(0, 0, WIDTH, HEIGHT, 30, 1000);
514
515 init_tex();
516 init_fs_constbuf();
517
518 set_vertices();
519 set_vertex_shader();
520 set_fragment_shader(filename);
521 }
522
args(int argc,char * argv[])523 static void args(int argc, char *argv[])
524 {
525 int i;
526
527 for (i = 1; i < argc;) {
528 if (graw_parse_args(&i, argc, argv)) {
529 continue;
530 }
531 if (strcmp(argv[i], "-fps") == 0) {
532 show_fps = 1;
533 i++;
534 }
535 else if (i == argc - 1) {
536 filename = argv[i];
537 i++;
538 }
539 else {
540 usage(argv[0]);
541 exit(1);
542 }
543 }
544
545 if (!filename) {
546 usage(argv[0]);
547 exit(1);
548 }
549 }
550
main(int argc,char * argv[])551 int main( int argc, char *argv[] )
552 {
553 args(argc,argv);
554 init();
555
556 graw_set_display_func( draw );
557 graw_main_loop();
558 return 0;
559 }
560