/*
* 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 "GLUtils.h"
#include <stdlib.h>
#include <sys/time.h>
#include <android/asset_manager_jni.h>
#define LOG_TAG "CTS_OPENGL"
#define LOG_NDEBUG 0
#include <android/log.h>
static JNIEnv* sEnv = NULL;
static jobject sAssetManager = NULL;
void GLUtils::setEnvAndAssetManager(JNIEnv* env, jobject assetManager) {
sEnv = env;
sAssetManager = assetManager;
}
static AAsset* loadAsset(const char* path) {
AAssetManager* nativeManager = AAssetManager_fromJava(sEnv, sAssetManager);
if (nativeManager == NULL) {
return NULL;
}
return AAssetManager_open(nativeManager, path, AASSET_MODE_UNKNOWN);;
}
char* GLUtils::openTextFile(const char* path) {
AAsset* asset = loadAsset(path);
if (asset == NULL) {
__android_log_print(ANDROID_LOG_ERROR, LOG_TAG, "Couldn't load %s", path);
return NULL;
}
off_t length = AAsset_getLength(asset);
char* buffer = new char[length + 1];
int num = AAsset_read(asset, buffer, length);
AAsset_close(asset);
if (num != length) {
__android_log_print(ANDROID_LOG_ERROR, LOG_TAG, "Couldn't read %s", path);
delete[] buffer;
return NULL;
}
buffer[length] = '\0';
return buffer;
}
GLuint GLUtils::loadTexture(const char* path) {
GLuint textureId = 0;
jclass activityClass = sEnv->FindClass("android/opengl2/cts/reference/GLGameActivity");
if (activityClass == NULL) {
__android_log_print(ANDROID_LOG_ERROR, LOG_TAG, "Couldn't find activity class");
return -1;
}
jmethodID loadTexture = sEnv->GetStaticMethodID(activityClass, "loadTexture",
"(Landroid/content/res/AssetManager;Ljava/lang/String;)I");
if (loadTexture == NULL) {
__android_log_print(ANDROID_LOG_ERROR, LOG_TAG, "Couldn't find loadTexture method");
return -1;
}
jstring pathStr = sEnv->NewStringUTF(path);
textureId = sEnv->CallStaticIntMethod(activityClass, loadTexture, sAssetManager, pathStr);
sEnv->DeleteLocalRef(pathStr);
return textureId;
}
static int readInt(char* b) {
unsigned char* ub = (unsigned char*) b;
return (((int) ub[0]) << 24) | (((int) ub[1]) << 16) | (((int) ub[2]) << 8) | ((int) ub[3]);
}
static float readFloat(char* b) {
union {
int input;
float output;
} data;
data.input = readInt(b);
return data.output;
}
Mesh* GLUtils::loadMesh(const char* path) {
char* buffer = openTextFile(path);
if (buffer == NULL) {
return NULL;
}
int index = 0;
int numVertices = readInt(buffer + index);
index += 4;
float* vertices = new float[numVertices * 3];
float* normals = new float[numVertices * 3];
float* texCoords = new float[numVertices * 2];
for (int i = 0; i < numVertices; i++) {
// Vertices
int vIndex = i * 3;
vertices[vIndex + 0] = readFloat(buffer + index);
index += 4;
vertices[vIndex + 1] = readFloat(buffer + index);
index += 4;
vertices[vIndex + 2] = readFloat(buffer + index);
index += 4;
// Normals
normals[vIndex + 0] = readFloat(buffer + index);
index += 4;
normals[vIndex + 1] = readFloat(buffer + index);
index += 4;
normals[vIndex + 2] = readFloat(buffer + index);
index += 4;
// Texture Coordinates
int tIndex = i * 2;
texCoords[tIndex + 0] = readFloat(buffer + index);
index += 4;
texCoords[tIndex + 1] = readFloat(buffer + index);
index += 4;
}
return new Mesh(vertices, normals, texCoords, numVertices);
}
// Loads the given source code as a shader of the given type.
static GLuint loadShader(GLenum shaderType, const char** source) {
GLuint shader = glCreateShader(shaderType);
if (shader) {
glShaderSource(shader, 1, source, NULL);
glCompileShader(shader);
GLint compiled = 0;
glGetShaderiv(shader, GL_COMPILE_STATUS, &compiled);
if (!compiled) {
GLint infoLen = 0;
glGetShaderiv(shader, GL_INFO_LOG_LENGTH, &infoLen);
if (infoLen > 0) {
char* infoLog = (char*) malloc(sizeof(char) * infoLen);
glGetShaderInfoLog(shader, infoLen, NULL, infoLog);
__android_log_print(ANDROID_LOG_ERROR, LOG_TAG,
"Error compiling shader:\n%s\n", infoLog);
free(infoLog);
}
glDeleteShader(shader);
shader = 0;
}
}
return shader;
}
GLuint GLUtils::createProgram(const char** vertexSource, const char** fragmentSource) {
GLuint vertexShader = loadShader(GL_VERTEX_SHADER, vertexSource);
if (!vertexShader) {
return 0;
}
GLuint fragmentShader = loadShader(GL_FRAGMENT_SHADER, fragmentSource);
if (!fragmentShader) {
return 0;
}
GLuint program = glCreateProgram();
if (program) {
glAttachShader(program, vertexShader);
glAttachShader(program, fragmentShader);
GLint linkStatus;
glLinkProgram(program);
glGetProgramiv(program, GL_LINK_STATUS, &linkStatus);
if (!linkStatus) {
GLint infoLen = 0;
glGetProgramiv(program, GL_INFO_LOG_LENGTH, &infoLen);
if (infoLen > 0) {
char* infoLog = (char*) malloc(sizeof(char) * infoLen);
glGetProgramInfoLog(program, infoLen, NULL, infoLog);
__android_log_print(ANDROID_LOG_ERROR, LOG_TAG,
"Error linking program:\n%s\n", infoLog);
free(infoLog);
}
glDeleteProgram(program);
program = 0;
}
}
return program;
}
double GLUtils::currentTimeMillis() {
struct timeval tv;
gettimeofday(&tv, (struct timezone *) NULL);
return tv.tv_sec * 1000.0 + tv.tv_usec / 1000.0;
}
// Rounds a number up to the smallest power of 2 that is greater than or equal to x.
int GLUtils::roundUpToSmallestPowerOf2(int x) {
if (x < 0) {
return 0;
}
--x;
x |= x >> 1;
x |= x >> 2;
x |= x >> 4;
x |= x >> 8;
x |= x >> 16;
return x + 1;
}
GLuint GLUtils::genTexture(int texWidth, int texHeight, int fill) {
GLuint textureId = 0;
int w = roundUpToSmallestPowerOf2(texWidth);
int h = roundUpToSmallestPowerOf2(texHeight);
uint32_t* m = new uint32_t[w * h];
if (m != NULL) {
uint32_t* d = m;
for (int y = 0; y < h; y++) {
for (int x = 0; x < w; x++) {
if (fill == RANDOM_FILL) {
*d = 0xff000000 | ((y & 0xff) << 16) | ((x & 0xff) << 8) | ((x + y) & 0xff);
} else {
*d = 0xff000000 | fill;
}
d++;
}
}
glGenTextures(1, &textureId);
glBindTexture(GL_TEXTURE_2D, textureId);
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, w, h, 0, GL_RGBA, GL_UNSIGNED_BYTE, m);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
}
delete[] m;
return textureId;
}