/*
* Copyright (C) 2013 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
* in compliance with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under the License
* is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
* or implied. See the License for the specific language governing permissions and limitations under
* the License.
*/
#include "ShaderPerfRenderer.h"
#include <graphics/GLUtils.h>
#include <math.h>
#include <Trace.h>
static const float GOLDEN_RATIO = (1.0f + sqrt(5.0f)) / 2.0f;
static const int SP_NUM_VERTICES = 6;
static const float SP_VERTICES[SP_NUM_VERTICES * 3] = {
1.0f, 1.0f, -1.0f,
-1.0f, 1.0f, -1.0f,
-1.0f, -1.0f, -1.0f,
-1.0f, -1.0f, -1.0f,
1.0f, -1.0f, -1.0f,
1.0f, 1.0f, -1.0f };
static const float SP_TEX_COORDS[SP_NUM_VERTICES * 2] = {
1.0f, 1.0f,
0.0f, 1.0f,
0.0f, 0.0f,
0.0f, 0.0f,
1.0f, 0.0f,
1.0f, 1.0f };
static const char* SP_VERTEX =
"attribute vec4 a_Position;"
"attribute vec2 a_TexCoord;"
"varying vec2 v_TexCoord;"
"void main() {"
" v_TexCoord = a_TexCoord;"
" gl_Position = a_Position;"
"}";
static const char* SP_FRAGMENT_1 =
"precision mediump float;"
"uniform vec2 u_Seed;"
"uniform sampler2D u_Texture;"
"varying vec2 v_TexCoord;"
"void main() {"
" int count = ";
//Add workload here
static const char* SP_FRAGMENT_2 =
" * 4;"//workload * 4 (4 is a tweaking number, bigger = more work)
" vec2 z;"
" z.x = 3.0 * (v_TexCoord.x - 0.5);"
" z.y = 2.0 * (v_TexCoord.y - 0.5);"
" float u = 0.0;"
" for (int i = 0; i < count; i++) {"
" float x = (z.x * z.x - z.y * z.y) + u_Seed.x;"
" float y = (z.y * z.x + z.x * z.y) + u_Seed.y;"
" if (((x * x + y * y) > 4.0) && (u == 0.0)) {"
" u = float(i) / float(count);"
" }"
" z.x = x;"
" z.y = y;"
" }"
" gl_FragColor = texture2D(u_Texture, vec2(u, 0.0));"
"}";
// Copies the source array from 0 up to and including the '\0' character to the
// destination array starting from the given start position. Unlike strcpy, this
// returns the number of characters which were copied.
static int charCopy(const char* source, char* dest, int destStart) {
int srcAddr = 0;
int destAddr = destStart;
char current;
do {
current = source[srcAddr];
dest[destAddr] = current;
srcAddr++;
destAddr++;
} while (current != '\0');
return destAddr - destStart;
}
ShaderPerfRenderer::ShaderPerfRenderer(ANativeWindow* window, bool offscreen) :
Renderer(window, offscreen) {
}
bool ShaderPerfRenderer::setUp(int workload) {
SCOPED_TRACE();
if (!Renderer::setUp(workload)) {
return false;
}
const int MAX_FRAGMENT_SHADER_SIZE = 1000;
char* spFragment = new char[MAX_FRAGMENT_SHADER_SIZE];
// Add the first part.
int index = charCopy(SP_FRAGMENT_1, spFragment, 0);
// Add the count, overwriting the '\0' added by charCopy.
spFragment[index - 1] = (char) (((int) '0') + workload);
// Add the second part.
index += charCopy(SP_FRAGMENT_2, spFragment, index);
// Create program.
mProgramId = GLUtils::createProgram(&SP_VERTEX, const_cast<const char**>(&spFragment));
delete[] spFragment;
if (mProgramId == 0) {
return false;
}
// Bind attributes.
mTextureUniformHandle = glGetUniformLocation(mProgramId, "u_Texture");
mSeedUniformHandle = glGetUniformLocation(mProgramId, "u_Seed");
mPositionHandle = glGetAttribLocation(mProgramId, "a_Position");
mTexCoordHandle = glGetAttribLocation(mProgramId, "a_TexCoord");
const int SIZE = 256;
uint32_t* m = new uint32_t[SIZE];
if (m != NULL) {
uint32_t* d = m;
for (int i = 0; i < SIZE; i++) {
*d = 0xff000000 | ((i & 0xff) << 16);
d++;
}
glGenTextures(1, &mTextureId);
glBindTexture(GL_TEXTURE_2D, mTextureId);
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, SIZE, 1, 0, GL_RGBA, GL_UNSIGNED_BYTE, m);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
}
delete[] m;
return true;
}
void ShaderPerfRenderer::drawWorkload() {
SCOPED_TRACE();
glUseProgram(mProgramId);
// Set the background clear color.
glClearColor(0.0f, 0.0f, 0.0f, 0.0f);
glClear(GL_DEPTH_BUFFER_BIT | GL_COLOR_BUFFER_BIT);
// No culling of back faces
glDisable(GL_CULL_FACE);
// Bind the texture.
glActiveTexture(GL_TEXTURE0);
glBindTexture(GL_TEXTURE_2D, mTextureId);
glUniform1i(mTextureUniformHandle, 0);
// Bind the seed.
glUniform2f(mSeedUniformHandle, GOLDEN_RATIO - 2.0f, GOLDEN_RATIO - 1.0f);
// Bind the vertices.
glEnableVertexAttribArray(mPositionHandle);
glEnableVertexAttribArray(mTexCoordHandle);
glVertexAttribPointer(mPositionHandle, 3, GL_FLOAT, false, 0, SP_VERTICES);
glVertexAttribPointer(mTexCoordHandle, 2, GL_FLOAT, false, 0, SP_TEX_COORDS);
glDrawArrays(GL_TRIANGLES, 0, SP_NUM_VERTICES);
}