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