OpenGL ES: частота кадров и текстуры
Я заметил, что размер текстурного атласа действительно влияет на приложение OpenGL ES 2. Размер атласа составляет 2048 X 2048, а при использовании на Samsung Galaxy S7 производительность практически соответствует 60 кадрам в секунду. Даже 4096 X 4096 подойдет, хотя, возможно, некоторая потеря производительности, но не сильно.
Если я сейчас скомпилирую и запуском того же приложения на Samsung Galaxy S3, частота кадров будет очень плохой, если я использую 2048 X 2048 из Textureatlas. Для рисования рамки требуется около 25 мс. Но если я уменьшу текстурный атлас до 1024 х 1024, производительность значительно улучшится - 60 кадров в секунду.
Итак, можно ли сделать вывод, что я должен использовать разные размеры текстуры в зависимости от устройства? Но как именно размер соотносится с устройством? Для размера экрана может быть? Например - размер экрана 720 X 1280 Я использую текстурный атлас 1024 X 1024, а текстурный атлас 1440 X 2560 и 2048 X 2048?
Или есть проблема с моим источником? К настоящему времени я не знаю, как отслеживать, скажем, 100 спрайтов, которые совместно используют один и тот же frameBuffer. Вместо этого каждый спрайт имеет свой собственный буфер, и в onDRawFrame я делаю отдельный вызов для каждого буфера. Но я думаю, что bindBuffer быстрее, чем bindTexture?
ИЛИ Есть ли проблема, как я загружаю растровое изображение в OpenGL? Я использую флаг
Bitmap.Config.RGB_565
но без увеличения производительности.
Источник рендерера
public MyGLRenderer(Context context) {
this.context = context;
nSprites = 75;
alienUvs = new float[16][12 * nSprites];
bgUvs = new float[12];
vertices = new float[18 *nSprites];
verticesBG = new float[18];
vertexBuffer= new FloatBuffer[nSprites];
vertexBufferBG = new FloatBuffer[1];
// The indices for all textured quads
indices = new short[nSprites*6];
uvBufferAlien = new FloatBuffer[16]; // antalet frames
uvBufferBG = new FloatBuffer[1];
bufferId = new int[16 + nSprites + 2]; // 16 stycken för aliens, nsprites antal för punkterna
modulo = 2;
start = System.currentTimeMillis();
//slumpa x o y
xPos = new float[nSprites];
yPos = new float[nSprites];
for (int i = 0; i < nSprites; i++) {
Random r = new Random();
xPos[i] = (float)(r.nextDouble()-0.5) * 6;
r = new Random();
yPos[i] = (float)(r.nextDouble()-0.5) * 6;
}
}
@Override
public void onSurfaceCreated(GL10 unused, EGLConfig config) {
GLES20.glClear(GLES20.GL_COLOR_BUFFER_BIT | GLES20.GL_DEPTH_BUFFER_BIT);
vertexBuffer = GLData.createVertices(nSprites, vertices);
vertexBufferBG[0] = GLData.createVerticesBG(verticesBG);
GLData.createUvsData(nSprites, alienUvs, uvBufferAlien);
uvBufferBG[0] = GLData.createUvsDataBG(bgUvs);
GLData.createVertexBufferObject(vertexBuffer, vertexBufferBG, nSprites, uvBufferAlien, uvBufferBG, bufferId.length, bufferId);
createCamera();
GLES20.glEnable(GLES20.GL_CULL_FACE);
GLES20.glEnable(GLES20.GL_BLEND);
GLES20.glBlendFunc(GLES20.GL_ONE, GLES20.GL_ONE_MINUS_SRC_ALPHA);
// inlästa shaderprogram (källkod i strängform)
final String vertexShader = getVertexShader();
final String fragmentShader = getFragmentShader();
final int vertexShaderHandle = ShaderHelper.compileShader(GLES20.GL_VERTEX_SHADER, vertexShader);
final int fragmentShaderHandle = ShaderHelper.compileShader(GLES20.GL_FRAGMENT_SHADER, fragmentShader);
mProgramHandle = ShaderHelper.createAndLinkProgram(vertexShaderHandle, fragmentShaderHandle,
new String[] {"a_Position", "a_Color", "a_Normal", "a_TexCoordinate"});
//skapa texture
textureHandle = TextureHelper.loadTexture(context);
GLES20.glUseProgram(mProgramHandle);
mMVPMatrixHandle = GLES20.glGetUniformLocation(mProgramHandle, "u_MVPMatrix");
mMVMatrixHandle = GLES20.glGetUniformLocation(mProgramHandle, "u_MVMatrix");
mTextureCoordinateHandle = GLES20.glGetAttribLocation(mProgramHandle, "a_TexCoordinate");
mPositionHandle = GLES20.glGetAttribLocation(mProgramHandle, "a_Position");
}
@Override
public void onSurfaceChanged(GL10 unused, int width, int height) {
game_width = width;
game_height = height;
GLES20.glViewport(0, 0, width, height);
final float ratio = (float) width / height;
final float left = -ratio;
final float right = ratio;
final float bottom = -1.0f;
final float top = 1.0f;
final float near = 1f;
final float far = 20.0f;
Matrix.frustumM(mProjectionMatrix, 0, left, right, bottom, top, near, far);
}
@Override
public void onDrawFrame(GL10 unused) {
int rest = 0;
if (modulo > 0) {
rest = frameCntr % modulo;
}
if (rest == 0) {
bitmapIndex++;
}
if (bitmapIndex == 16) {
bitmapIndex = 0;
}
frameCntr++;
//uvs, 16 bilder
GLES20.glBindBuffer(GLES20.GL_ARRAY_BUFFER, bufferId[bitmapIndex]);
GLES20.glEnableVertexAttribArray(mTextureCoordinateHandle);
GLES20.glVertexAttribPointer(mTextureCoordinateHandle, mTextureCoordinateDataSize, GLES20.GL_FLOAT, false, 0, 0);
for (int i = 0; i < nSprites; i++) {
Matrix.setIdentityM(mModelMatrix, 0);
Matrix.translateM(mModelMatrix, 0, xPos[i], yPos[i], -8f);
GLES20.glBindBuffer(GLES20.GL_ARRAY_BUFFER, bufferId[16 + i]);
GLES20.glEnableVertexAttribArray(mPositionHandle);
GLES20.glVertexAttribPointer(mPositionHandle, mPositionDataSize, GLES20.GL_FLOAT, false, 0, 0);
draw();
}
//mease framerate
long stop = System.currentTimeMillis();
nbFrames++;
if (stop - start >= 1000) {
System.out.println("" + (float)1000 / nbFrames);
nbFrames = 0;
start += 1000;
}
}
private void draw() {
GLES20.glBindBuffer(GLES20.GL_ARRAY_BUFFER, 0);
Matrix.multiplyMM(mMVPMatrix, 0, mViewMatrix, 0, mModelMatrix, 0);
GLES20.glUniformMatrix4fv(mMVMatrixHandle, 1, false, mMVPMatrix, 0);
Matrix.multiplyMM(mMVPMatrix, 0, mProjectionMatrix, 0, mMVPMatrix, 0);
GLES20.glUniformMatrix4fv(mMVPMatrixHandle, 1, false, mMVPMatrix, 0);
GLES20.glDrawArrays(GLES20.GL_TRIANGLES, 0, 6);
}
TextureHelper
public class TextureHelper {
public static int[] loadTexture(Context context) {
int[] texturenames = new int[1];
GLES20.glGenTextures(1, texturenames, 0);
int resId = context.getResources().getIdentifier("aliens", "drawable", context.getPackageName());
//Bitmap bmp = BitmapDecoder.decodeSampledBitmapFromResource(context.getResources(), resId, 960, 928);
//Bitmap bmp = BitmapDecoder.decodeSampledBitmapFromResource(context.getResources(), resId, 780, 1280);
Bitmap bmp = BitmapDecoder.decodeSampledBitmapFromResource(context.getResources(), resId, 2048, 2048);
GLES20.glActiveTexture(GLES20.GL_TEXTURE0);
GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, texturenames[0]);
GLES20.glTexParameteri(GLES20.GL_TEXTURE_2D, GLES20.GL_TEXTURE_MIN_FILTER, GLES20.GL_LINEAR);
GLES20.glTexParameteri(GLES20.GL_TEXTURE_2D, GLES20.GL_TEXTURE_MAG_FILTER, GLES20.GL_LINEAR);
GLUtils.texImage2D(GLES20.GL_TEXTURE_2D, 0, bmp, 0);
bmp.recycle();
return texturenames;
}
}
BitmapDecoder (фрагмент)
protected static Bitmap decodeSampledBitmapFromResource(Resources res, int resId, int reqWidth, int reqHeight) {
// First decode with inJustDecodeBounds=true to check dimensions
final BitmapFactory.Options options = new BitmapFactory.Options();
options.inJustDecodeBounds = true;
BitmapFactory.decodeResource(res, resId, options);
// Calculate inSampleSize
options.inSampleSize = calculateInSampleSize(options, reqWidth, reqHeight);
options.inPreferredConfig = Bitmap.Config.RGB_565;
options.inDither = true;
//options.inScaled = false;
// Decode bitmap with inSampleSize set
options.inJustDecodeBounds = false;
return Bitmap.createScaledBitmap( BitmapFactory.decodeResource(res, resId, options), reqWidth, reqHeight, true);
}