安卓opengles使用ndk方式读取png图片

2024-05-16 00:32

本文主要是介绍安卓opengles使用ndk方式读取png图片,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本博客主要是通过使用libpng来进行加载png图片

1libpng图片配置

(1)在官网下载最新的libpng压缩包和zlib压缩包,然后解压,分别放入如下图所示的png目录和zlib目录
最终存放目录
(2)将jni\png\scripts目录下的pnglibconf.h.prebuilt文件复制到jni\png目录下,并且重命名为pnglibconf.h。
(3)在jni\png目录下新建Android.mk文件,内容如下:

LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)LOCAL_MODULE    := libpng
LOCAL_SRC_FILES :=\png.c \pngerror.c \pngget.c \pngmem.c \pngpread.c \pngread.c \pngrio.c \pngrtran.c \pngrutil.c \pngset.c \pngtest.c \pngtrans.c \pngwio.c \pngwrite.c \pngwtran.c \pngwutil.cLOCAL_EXPORT_C_INCLUDES := $(LOCAL_PATH)/.include $(BUILD_STATIC_LIBRARY)

(4)在jni\zlib目录下新建Android.mk文件,内容如下:

include $(CLEAR_VARS)
LOCAL_MODULE:= zlib
LOCAL_SRC_FILES:= \
adler32.c \
compress.c \
crc32.c \
deflate.c \
gzclose.c \
gzlib.c \
gzread.c \
gzwrite.c \
infback.c \
inflate.c \
inftrees.c \
inffast.c \
trees.c \
uncompr.c \
zutil.c LOCAL_EXPORT_C_INCLUDES := $(LOCAL_PATH)
include $(BUILD_STATIC_LIBRARY)

(4)在jni\目录下新建Android.mk文件,文件目录内容如下:

LOCAL_PATH:=$(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE:=libmygles
LOCAL_SRC_FILES:=opengl.cpp
LOCAL_STATIC_LIBRARIES := libpng
LOCAL_STATIC_LIBRARIES += zlib
LOCAL_C_INCLUDES := $(LOCAL_PATH)/png
LOCAL_C_INCLUDES += $(LOCAL_PATH)/zlibLOCAL_LDLIBS:=-llog -lGLESv2 -landroidinclude $(BUILD_SHARED_LIBRARY)include $(CLEAR_VARS)include $(LOCAL_PATH)/png/Android.mk $(LOCAL_PATH)/zlib/Android.mk

(5)修改jni/png/pngpriv.h文件中


#  ifdef __ARM_NEON__
#     define PNG_ARM_NEON_OPT 2
#  else
#     define PNG_ARM_NEON_OPT 0
#  en

#define PNG_ARM_NEON_OPT 0

不然会报错 png_init_filter_fuction_neon
(6)使用ndk编译主文件,我的文件是jni目录下的opengl.cpp文件,该文件中需要包括#include “png.h”,然后编译成功。
(7)使用如下函数读取并创建纹理,试了网上好多函数都不能用,使用该函数编译后,将png图片拷贝到assets函数下,即可调用
“`

GLuint CreateTextureFromPng(const char *filename)
{
info(“enter into CreateTextureFromPng”);

int bitDepth;
int colorType;
int interlaceype;png_uint_32 width;
png_uint_32 height;
void *pixelData = nullptr;
AAsset *asset = nullptr;
unsigned char head[8];
// off_t size;
asset = AAssetManager_open(sAssetManager, filename, AASSET_MODE_UNKNOWN);
if (asset == nullptr)
{info("load file %s fail", filename);return 0;
}
AAsset_read(asset, head, 8);
if (png_sig_cmp(head, 0, 8))
{info("File %s, is not PNG", filename);return 0;
}info("enter into CreateTextureFromPng1");png_structp pngPtr = png_create_read_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
if (!pngPtr)
{info("Unable to create PNG structure: %s", filename);return 0;
}// Allocate/initialize the memory for image information.  REQUIRED
png_infop infoPtr = png_create_info_struct(pngPtr);if (!infoPtr)
{png_destroy_read_struct(&pngPtr, NULL, NULL);info("Unable to create PNG info : %s", filename);return 0;
}
info("enter into CreateTextureFromPng2");
png_infop endInfo = png_create_info_struct(pngPtr);
if (!endInfo)
{png_destroy_read_struct(&pngPtr, &infoPtr, NULL);info("Unable to create PNG end info : %s", filename);return 0;
}
// Set error handling if you are using the setjmp/longjmp method (this is
// the normal method of doing things with libpng).  REQUIRED unless you
// set up your own error handlers in the png_create_read_struct() earlier.
if (setjmp(png_jmpbuf(pngPtr)))
{// Free all of the memory associated with the png_ptr and info_ptrpng_destroy_read_struct(&pngPtr, &infoPtr, &endInfo);info("Error during setjmp : %s", filename);return 0;
}
png_set_read_fn(pngPtr, (void *)asset, readPngData);
info("enter into CreateTextureFromPng3");// If we have already read some of the signature
png_set_sig_bytes(pngPtr, 8);png_read_info(pngPtr, infoPtr);png_get_IHDR(pngPtr, infoPtr, &width, &height, &bitDepth, &colorType, &interlaceype, NULL, NULL);info("PNG width = %d, height = %d", width, height);// Update the png info struct.
png_read_update_info(pngPtr, infoPtr);// Allocate the memory to hold the image using the fields of info_ptrunsigned int rowBytes = png_get_rowbytes(pngPtr, infoPtr);
info("Row size: %d bytes", rowBytes);// Allocate the pixel data as a big block, to be given to openGL
pixelData = png_malloc(pngPtr, rowBytes * height);
if (!pixelData)
{png_destroy_read_struct(&pngPtr, &infoPtr, &endInfo);info("Unable to allocate PNG pixel data while loading %s", filename);return 0;
}
int numberPasses = png_set_interlace_handling(pngPtr);
info("interlacing passes = %d", numberPasses);
for (int pass = 0; pass < numberPasses; pass++)
{for (int row = 0; row < height; row++){png_read_row(pngPtr, ((unsigned char *)pixelData + (row * rowBytes)), NULL);}
}png_read_end(pngPtr, infoPtr);png_destroy_read_struct(&pngPtr, &infoPtr, &endInfo);info("enter into CreateTextureFromPng4");AAsset_close(asset);
//此时  pixelData 已经赋值结束     png_uint_32 width;       png_uint_32 height;GLuint texture;
glGenTextures(1, &texture);
glBindTexture(GL_TEXTURE_2D, texture);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, width, height, 0, GL_RGBA, GL_UNSIGNED_BYTE, pixelData);
glBindTexture(GL_TEXTURE_2D, 0);info("enter into CreateTextureFromPng5");
return texture;

}

2参考链接


(1)http://www.xuebuyuan.com/zh-hant/1894472.html
(2)http://blog.csdn.net/talkxin/article/details/50968313
(3)http://blog.sina.com.cn/s/blog_643634b80101hzi8.html
(4)https://stackoverflow.com/questions/19089014/iphone-device-linker-error

这篇关于安卓opengles使用ndk方式读取png图片的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/993363

相关文章

golang程序打包成脚本部署到Linux系统方式

《golang程序打包成脚本部署到Linux系统方式》Golang程序通过本地编译(设置GOOS为linux生成无后缀二进制文件),上传至Linux服务器后赋权执行,使用nohup命令实现后台运行,完... 目录本地编译golang程序上传Golang二进制文件到linux服务器总结本地编译Golang程序

深入理解Go语言中二维切片的使用

《深入理解Go语言中二维切片的使用》本文深入讲解了Go语言中二维切片的概念与应用,用于表示矩阵、表格等二维数据结构,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧... 目录引言二维切片的基本概念定义创建二维切片二维切片的操作访问元素修改元素遍历二维切片二维切片的动态调整追加行动态

Linux下删除乱码文件和目录的实现方式

《Linux下删除乱码文件和目录的实现方式》:本文主要介绍Linux下删除乱码文件和目录的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux下删除乱码文件和目录方法1方法2总结Linux下删除乱码文件和目录方法1使用ls -i命令找到文件或目录

prometheus如何使用pushgateway监控网路丢包

《prometheus如何使用pushgateway监控网路丢包》:本文主要介绍prometheus如何使用pushgateway监控网路丢包问题,具有很好的参考价值,希望对大家有所帮助,如有错误... 目录监控网路丢包脚本数据图表总结监控网路丢包脚本[root@gtcq-gt-monitor-prome

Python通用唯一标识符模块uuid使用案例详解

《Python通用唯一标识符模块uuid使用案例详解》Pythonuuid模块用于生成128位全局唯一标识符,支持UUID1-5版本,适用于分布式系统、数据库主键等场景,需注意隐私、碰撞概率及存储优... 目录简介核心功能1. UUID版本2. UUID属性3. 命名空间使用场景1. 生成唯一标识符2. 数

Linux在线解压jar包的实现方式

《Linux在线解压jar包的实现方式》:本文主要介绍Linux在线解压jar包的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux在线解压jar包解压 jar包的步骤总结Linux在线解压jar包在 Centos 中解压 jar 包可以使用 u

Java中读取YAML文件配置信息常见问题及解决方法

《Java中读取YAML文件配置信息常见问题及解决方法》:本文主要介绍Java中读取YAML文件配置信息常见问题及解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要... 目录1 使用Spring Boot的@ConfigurationProperties2. 使用@Valu

SpringBoot中如何使用Assert进行断言校验

《SpringBoot中如何使用Assert进行断言校验》Java提供了内置的assert机制,而Spring框架也提供了更强大的Assert工具类来帮助开发者进行参数校验和状态检查,下... 目录前言一、Java 原生assert简介1.1 使用方式1.2 示例代码1.3 优缺点分析二、Spring Fr

Android kotlin中 Channel 和 Flow 的区别和选择使用场景分析

《Androidkotlin中Channel和Flow的区别和选择使用场景分析》Kotlin协程中,Flow是冷数据流,按需触发,适合响应式数据处理;Channel是热数据流,持续发送,支持... 目录一、基本概念界定FlowChannel二、核心特性对比数据生产触发条件生产与消费的关系背压处理机制生命周期

java使用protobuf-maven-plugin的插件编译proto文件详解

《java使用protobuf-maven-plugin的插件编译proto文件详解》:本文主要介绍java使用protobuf-maven-plugin的插件编译proto文件,具有很好的参考价... 目录protobuf文件作为数据传输和存储的协议主要介绍在Java使用maven编译proto文件的插件