blob: ec3665008d2ecf5a732092d0f084ddebd5dba262 [file] [log] [blame]
Brian Paulf02a5f62002-07-12 15:54:01 +00001/* $Id: readpix.c,v 1.8 2002/07/12 15:54:02 brianp Exp $ */
Brian Pauleca1bc92000-03-01 16:23:14 +00002
3/*
4 * glReadPixels and glCopyPixels test
5 *
6 * Brian Paul March 1, 2000 This file is in the public domain.
7 */
8
Brian Pauleca1bc92000-03-01 16:23:14 +00009#include <assert.h>
10#include <stdio.h>
11#include <stdlib.h>
12#include <math.h>
13#include <GL/glut.h>
14
15#include "../util/readtex.c" /* a hack, I know */
16
17#define IMAGE_FILE "../images/girl.rgb"
18
19static int ImgWidth, ImgHeight;
20static GLenum ImgFormat;
21static GLubyte *Image = NULL;
22
23static int APosX, APosY; /* simple drawpixels */
24static int BPosX, BPosY; /* read/draw pixels */
25static int CPosX, CPosY; /* copypixels */
26
27static GLboolean DrawFront = GL_FALSE;
28static GLboolean ScaleAndBias = GL_FALSE;
Brian Paul4f6d60e2000-03-23 19:47:25 +000029static GLboolean Benchmark = GL_FALSE;
Brian Pauleca1bc92000-03-01 16:23:14 +000030static GLubyte *TempImage = NULL;
31
Brian Paul1100b4d2000-10-16 21:24:39 +000032#if 0
Brian Pauld3d72802000-03-31 01:01:31 +000033#define ReadFormat ImgFormat
34#define ReadType GL_UNSIGNED_BYTE
35#endif
Brian Paul1100b4d2000-10-16 21:24:39 +000036#if 1
Alan Hourihanea5cdf992002-05-02 09:17:59 +000037static GLenum ReadFormat = GL_RGBA;
Brian Pauld3d72802000-03-31 01:01:31 +000038static GLenum ReadType = GL_UNSIGNED_BYTE;
39#endif
40#if 0
41static GLenum ReadFormat = GL_RGB;
42static GLenum ReadType = GL_UNSIGNED_SHORT_5_6_5;
43#endif
Brian Paul1100b4d2000-10-16 21:24:39 +000044#if 0
45static GLenum ReadFormat = GL_RGBA;
46static GLenum ReadType = GL_UNSIGNED_SHORT_1_5_5_5_REV;
47#endif
48#if 0
49static GLenum ReadFormat = GL_BGRA;
50static GLenum ReadType = GL_UNSIGNED_SHORT_5_5_5_1;
51#endif
52#if 0
53static GLenum ReadFormat = GL_BGRA;
54static GLenum ReadType = GL_UNSIGNED_SHORT_4_4_4_4_REV;
55#endif
Brian Pauld3d72802000-03-31 01:01:31 +000056
Brian Pauleca1bc92000-03-01 16:23:14 +000057
58static void
59Reset( void )
60{
61 APosX = 5; APosY = 20;
62 BPosX = APosX + ImgWidth + 5; BPosY = 20;
63 CPosX = BPosX + ImgWidth + 5; CPosY = 20;
64}
65
66
67static void
68PrintString(const char *s)
69{
70 while (*s) {
71 glutBitmapCharacter(GLUT_BITMAP_8_BY_13, (int) *s);
72 s++;
73 }
74}
75
76
77static void
78SetupPixelTransfer(GLboolean invert)
79{
80 if (invert) {
81 glPixelTransferf(GL_RED_SCALE, -1.0);
82 glPixelTransferf(GL_RED_BIAS, 1.0);
83 glPixelTransferf(GL_GREEN_SCALE, -1.0);
84 glPixelTransferf(GL_GREEN_BIAS, 1.0);
85 glPixelTransferf(GL_BLUE_SCALE, -1.0);
86 glPixelTransferf(GL_BLUE_BIAS, 1.0);
87 }
88 else {
89 glPixelTransferf(GL_RED_SCALE, 1.0);
90 glPixelTransferf(GL_RED_BIAS, 0.0);
91 glPixelTransferf(GL_GREEN_SCALE, 1.0);
92 glPixelTransferf(GL_GREEN_BIAS, 0.0);
93 glPixelTransferf(GL_BLUE_SCALE, 1.0);
94 glPixelTransferf(GL_BLUE_BIAS, 0.0);
95 }
96}
97
98
99static void
100Display( void )
101{
Brian Paul1100b4d2000-10-16 21:24:39 +0000102 glClearColor(.3, .3, .3, 1);
Brian Pauleca1bc92000-03-01 16:23:14 +0000103 glClear( GL_COLOR_BUFFER_BIT );
104
105 glRasterPos2i(5, ImgHeight+25);
Brian Paul4f6d60e2000-03-23 19:47:25 +0000106 PrintString("f = toggle front/back s = toggle scale/bias b = benchmark");
Brian Pauleca1bc92000-03-01 16:23:14 +0000107
108 /* draw original image */
109 glRasterPos2i(APosX, 5);
110 PrintString("Original");
111 glRasterPos2i(APosX, APosY);
112 glEnable(GL_DITHER);
113 SetupPixelTransfer(GL_FALSE);
114 glDrawPixels(ImgWidth, ImgHeight, ImgFormat, GL_UNSIGNED_BYTE, Image);
115
116 /* do readpixels, drawpixels */
117 glRasterPos2i(BPosX, 5);
118 PrintString("Read/DrawPixels");
119 SetupPixelTransfer(ScaleAndBias);
Brian Paul4f6d60e2000-03-23 19:47:25 +0000120 if (Benchmark) {
121 GLint reads = 0;
122 GLint endTime;
123 GLint startTime = glutGet(GLUT_ELAPSED_TIME);
124 GLdouble seconds, pixelsPerSecond;
125 printf("Benchmarking...\n");
126 do {
127 glReadPixels(APosX, APosY, ImgWidth, ImgHeight,
Brian Pauld3d72802000-03-31 01:01:31 +0000128 ReadFormat, ReadType, TempImage);
Brian Paul4f6d60e2000-03-23 19:47:25 +0000129 reads++;
130 endTime = glutGet(GLUT_ELAPSED_TIME);
131 } while (endTime - startTime < 4000); /* 4 seconds */
132 seconds = (double) (endTime - startTime) / 1000.0;
133 pixelsPerSecond = reads * ImgWidth * ImgHeight / seconds;
134 printf("Result: %d reads in %f seconds = %f pixels/sec\n",
135 reads, seconds, pixelsPerSecond);
136 Benchmark = GL_FALSE;
137 }
138 else {
Brian Paul1100b4d2000-10-16 21:24:39 +0000139 /* clear the temporary image to white (helpful for debugging */
140 memset(TempImage, 255, ImgWidth * ImgHeight * 4);
Brian Paul4f6d60e2000-03-23 19:47:25 +0000141 glReadPixels(APosX, APosY, ImgWidth, ImgHeight,
Brian Pauld3d72802000-03-31 01:01:31 +0000142 ReadFormat, ReadType, TempImage);
Brian Paul4f6d60e2000-03-23 19:47:25 +0000143 }
Brian Pauleca1bc92000-03-01 16:23:14 +0000144 glRasterPos2i(BPosX, BPosY);
145 glDisable(GL_DITHER);
146 SetupPixelTransfer(GL_FALSE);
Brian Pauld3d72802000-03-31 01:01:31 +0000147 glDrawPixels(ImgWidth, ImgHeight, ReadFormat, ReadType, TempImage);
Brian Pauleca1bc92000-03-01 16:23:14 +0000148
149 /* do copypixels */
150 glRasterPos2i(CPosX, 5);
151 PrintString("CopyPixels");
152 glRasterPos2i(CPosX, CPosY);
153 glDisable(GL_DITHER);
154 SetupPixelTransfer(ScaleAndBias);
Alan Hourihanea5cdf992002-05-02 09:17:59 +0000155 glCopyPixels(APosX, APosY, ImgWidth, ImgHeight, GL_COLOR);
Brian Pauleca1bc92000-03-01 16:23:14 +0000156
157 if (!DrawFront)
158 glutSwapBuffers();
Alan Hourihane056b3582002-05-02 09:15:22 +0000159 else
160 glFinish();
Brian Pauleca1bc92000-03-01 16:23:14 +0000161}
162
163
Brian Paul4f6d60e2000-03-23 19:47:25 +0000164static void
165Reshape( int width, int height )
Brian Pauleca1bc92000-03-01 16:23:14 +0000166{
167 glViewport( 0, 0, width, height );
168 glMatrixMode( GL_PROJECTION );
169 glLoadIdentity();
170 glOrtho( 0.0, width, 0.0, height, -1.0, 1.0 );
171 glMatrixMode( GL_MODELVIEW );
172 glLoadIdentity();
173}
174
175
Brian Paul4f6d60e2000-03-23 19:47:25 +0000176static void
177Key( unsigned char key, int x, int y )
Brian Pauleca1bc92000-03-01 16:23:14 +0000178{
179 (void) x;
180 (void) y;
181 switch (key) {
182 case 'b':
Brian Paul4f6d60e2000-03-23 19:47:25 +0000183 Benchmark = GL_TRUE;
184 break;
185 case 's':
Brian Pauleca1bc92000-03-01 16:23:14 +0000186 ScaleAndBias = !ScaleAndBias;
187 break;
188 case 'f':
189 DrawFront = !DrawFront;
Brian Paul4f6d60e2000-03-23 19:47:25 +0000190 if (DrawFront) {
Alan Hourihanea5cdf992002-05-02 09:17:59 +0000191 glDrawBuffer(GL_FRONT);
192 glReadBuffer(GL_FRONT);
Brian Paul4f6d60e2000-03-23 19:47:25 +0000193 }
194 else {
Alan Hourihanea5cdf992002-05-02 09:17:59 +0000195 glDrawBuffer(GL_BACK);
196 glReadBuffer(GL_BACK);
Brian Paul4f6d60e2000-03-23 19:47:25 +0000197 }
Brian Pauleca1bc92000-03-01 16:23:14 +0000198 printf("glDrawBuffer(%s)\n", DrawFront ? "GL_FRONT" : "GL_BACK");
199 break;
200 case 27:
201 exit(0);
202 break;
203 }
204 glutPostRedisplay();
205}
206
207
Brian Pauleca1bc92000-03-01 16:23:14 +0000208static void
209Init( GLboolean ciMode )
210{
211 printf("GL_VERSION = %s\n", (char *) glGetString(GL_VERSION));
212 printf("GL_RENDERER = %s\n", (char *) glGetString(GL_RENDERER));
213
214 Image = LoadRGBImage( IMAGE_FILE, &ImgWidth, &ImgHeight, &ImgFormat );
215 if (!Image) {
216 printf("Couldn't read %s\n", IMAGE_FILE);
217 exit(0);
218 }
219
220 if (ciMode) {
221 /* Convert RGB image to grayscale */
Brian Paulf02a5f62002-07-12 15:54:01 +0000222 GLubyte *indexImage = (GLubyte *) malloc( ImgWidth * ImgHeight );
Brian Pauleca1bc92000-03-01 16:23:14 +0000223 GLint i;
224 for (i=0; i<ImgWidth*ImgHeight; i++) {
225 int gray = Image[i*3] + Image[i*3+1] + Image[i*3+2];
226 indexImage[i] = gray / 3;
227 }
228 free(Image);
229 Image = indexImage;
230 ImgFormat = GL_COLOR_INDEX;
231
232 for (i=0;i<255;i++) {
233 float g = i / 255.0;
234 glutSetColor(i, g, g, g);
235 }
236 }
237
238 printf("Loaded %d by %d image\n", ImgWidth, ImgHeight );
239
240 glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
241 glPixelStorei(GL_UNPACK_ROW_LENGTH, ImgWidth);
242 glPixelStorei(GL_PACK_ALIGNMENT, 1);
243 glPixelStorei(GL_PACK_ROW_LENGTH, ImgWidth);
244
245 Reset();
246
247 TempImage = (GLubyte *) malloc(ImgWidth * ImgHeight * 4 * sizeof(GLubyte));
248 assert(TempImage);
249}
250
251
Brian Paul4f6d60e2000-03-23 19:47:25 +0000252int
253main( int argc, char *argv[] )
Brian Pauleca1bc92000-03-01 16:23:14 +0000254{
255 GLboolean ciMode = GL_FALSE;
Brian Pauleca1bc92000-03-01 16:23:14 +0000256 if (argc > 1 && strcmp(argv[1], "-ci")==0) {
257 ciMode = GL_TRUE;
258 }
Brian Pauleca1bc92000-03-01 16:23:14 +0000259 glutInit( &argc, argv );
260 glutInitWindowPosition( 0, 0 );
261 glutInitWindowSize( 750, 250 );
Brian Pauleca1bc92000-03-01 16:23:14 +0000262 if (ciMode)
263 glutInitDisplayMode( GLUT_INDEX | GLUT_DOUBLE );
264 else
265 glutInitDisplayMode( GLUT_RGB | GLUT_DOUBLE );
Brian Pauleca1bc92000-03-01 16:23:14 +0000266 glutCreateWindow(argv[0]);
Brian Pauleca1bc92000-03-01 16:23:14 +0000267 Init(ciMode);
Brian Pauleca1bc92000-03-01 16:23:14 +0000268 glutReshapeFunc( Reshape );
269 glutKeyboardFunc( Key );
Brian Pauleca1bc92000-03-01 16:23:14 +0000270 glutDisplayFunc( Display );
Brian Pauleca1bc92000-03-01 16:23:14 +0000271 glutMainLoop();
272 return 0;
273}