xenocara/dist/Mesa/progs/perf/readpixels.c

170 lines
4.3 KiB
C

/*
* Copyright (C) 2009 VMware, Inc. All Rights Reserved.
*
* Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the "Software"),
* to deal in the Software without restriction, including without limitation
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
* and/or sell copies of the Software, and to permit persons to whom the
* Software is furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included
* in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
* VMWARE BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
* AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
/**
* Measure glReadPixels speed.
* XXX also read into a PBO?
* XXX also read from FBOs?
*
* Brian Paul
* 23 Sep 2009
*/
#include <string.h>
#include <assert.h>
#include "glmain.h"
#include "common.h"
int WinWidth = 1000, WinHeight = 1000;
static GLuint VBO;
static const GLboolean DrawPoint = GL_TRUE;
static const GLboolean BufferSubDataInHalves = GL_TRUE;
static const GLfloat Vertex0[2] = { 0.0, 0.0 };
static GLenum HaveDepthStencil;
static GLenum ReadFormat, ReadType;
static GLint ReadWidth, ReadHeight;
static GLvoid *ReadBuffer;
/** Called from test harness/main */
void
PerfInit(void)
{
/* setup VBO */
glGenBuffersARB(1, &VBO);
glBindBufferARB(GL_ARRAY_BUFFER_ARB, VBO);
glBufferDataARB(GL_ARRAY_BUFFER_ARB, sizeof(Vertex0), Vertex0, GL_STATIC_DRAW_ARB);
glVertexPointer(2, GL_FLOAT, sizeof(Vertex0), (void *) 0);
glEnableClientState(GL_VERTEX_ARRAY);
glPixelStorei(GL_PACK_ALIGNMENT, 1);
HaveDepthStencil = PerfExtensionSupported("GL_EXT_packed_depth_stencil");
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT | GL_STENCIL_BUFFER_BIT);
glEnable(GL_DEPTH_TEST);
glEnable(GL_STENCIL_TEST);
}
static void
ReadPixels(unsigned count)
{
unsigned i;
for (i = 0; i < count; i++) {
/* read from random pos */
GLint x, y;
x = WinWidth - ReadWidth;
y = WinHeight - ReadHeight;
if (x > 0)
x = rand() % x;
if (y > 0)
y = rand() % y;
if (DrawPoint)
glDrawArrays(GL_POINTS, 0, 1);
glReadPixels(x, y, ReadWidth, ReadHeight,
ReadFormat, ReadType, ReadBuffer);
}
glFinish();
}
static const GLsizei Sizes[] = {
10,
100,
500,
1000,
0
};
static const struct {
GLenum format;
GLenum type;
const char *name;
GLuint pixel_size;
} DstFormats[] = {
{ GL_RGBA, GL_UNSIGNED_BYTE, "RGBA/ubyte", 4 },
{ GL_BGRA, GL_UNSIGNED_BYTE, "BGRA/ubyte", 4 },
{ GL_RGB, GL_UNSIGNED_SHORT_5_6_5, "RGB/565", 2 },
{ GL_LUMINANCE, GL_UNSIGNED_BYTE, "L/ubyte", 1 },
{ GL_DEPTH_COMPONENT, GL_UNSIGNED_INT, "Z/uint", 4 },
{ GL_DEPTH_STENCIL_EXT, GL_UNSIGNED_INT_24_8_EXT, "Z+S/uint", 4 },
{ 0, 0, NULL, 0 }
};
/** Called from test harness/main */
void
PerfNextRound(void)
{
}
/** Called from test harness/main */
void
PerfDraw(void)
{
double rate, mbPerSec;
int fmt, sz;
/* loop over formats */
for (fmt = 0; DstFormats[fmt].format; fmt++) {
ReadFormat = DstFormats[fmt].format;
ReadType = DstFormats[fmt].type;
/* loop over sizes */
for (sz = 0; Sizes[sz]; sz++) {
int imgSize;
ReadWidth = ReadHeight = Sizes[sz];
imgSize = ReadWidth * ReadHeight * DstFormats[fmt].pixel_size;
ReadBuffer = malloc(imgSize);
if (ReadFormat == GL_DEPTH_STENCIL_EXT && !HaveDepthStencil) {
rate = 0.0;
mbPerSec = 0.0;
}
else {
rate = PerfMeasureRate(ReadPixels);
mbPerSec = rate * imgSize / (1024.0 * 1024.0);
}
perf_printf("glReadPixels(%d x %d, %s): %.1f images/sec, %.1f Mpixels/sec\n",
ReadWidth, ReadHeight,
DstFormats[fmt].name, rate, mbPerSec);
free(ReadBuffer);
}
}
exit(0);
}