前两篇在这里:

Android上使用native IO

最近工作中的问题笔记

最近遇到的问题是,

java.io.IOException: FAT Full

StackOverflow的结果:

http://stackoverflow.com/questions/18906055/what-causes-jobb-tool-to-throw-fat-full-ioexception

提问者自己解释了原因, 原因是obb超过512M就出错了. 但是FAT16最大可以支持2G, 这个是jobb的bug.

同时作者提供了jobb修复的代码和bin:

https://github.com/monkey0506/jobbifier/tree/master/jObbifier/bin (由于不懂Java/eclipse,花了点时间才编译打包出来)

最后关于在native下 mount一直报错的问题(AOBB_STATE_ERROR_INTERNAL, AOBB_STATE_ERROR_COULD_NOT_MOUNT)
logcat没有任何输出, 同时网上也没有任何解决方法可以解决我这里遇到的问题.

最后改用在java端mount, 竟然毫无错误的成功了...表示很无语. What's wrong with the NDK team? why mounting obb in native fails but in Java end succeeds?

另外, 网上可以找到关于native API code里的问题 https://code.google.com/p/android/issues/detail?id=41983

AStorageManager::getMountedObbPath

https://github.com/android/platform_frameworks_base/blob/master/native/android/storage_manager.cpp

     const char* getMountedObbPath(const char* filename) {
String16 filename16(filename);
String16 path16;
if (mMountService->getMountedObbPath(filename16, path16)) {
return String8(path16).string(); //WTF? return a temp object's buffer?
} else {
return NULL;
}
}

由于没有看String8的实现, 但是单从表面上看, 返回一个local temp object的buffer, 应该是有问题的, 除非buffer是malloc的,但貌似文档又没有说要free之类的(或者是mountService内部的也可以). 而实际中我也遇到返回乱码的情况.

这个问题有人提出很久了, 但是一直没有人去改..

虽然obb的mount都是异步的, 但java的回调是同步的, 而且回调只有在开始了消息循环以后才会被调用. 而native的callback确定是在另外一个线程调用的,难道也要等到消息循环开始以后才可以? 即便是这样, 这种坑也应该在文档里面说清楚,或者给个native sample吧..现在只有java的obb sample.

感觉native API上对obb的支持有很多坑还没有发现. 这部分决定先用java了.


更新06/05/2014

native代码是这样的:

 Callback(const char* filename, const SYSTEM::int32_t state, void* callbackdata )
{
Android_App* app = (Android_App*)data;
if( state == AOBB_STATE_MOUNTED )
{
int isMounted = AStorageManager_isObbMounted(app->storage, filename);
assert( isMounted != ); const char* mntPath = AStorageManager_getMountedObbPath(app->storage, filename); //save persistent path data - current NDK returns tmp string that may even corrupted right after AStorageManager_getMountedObbPath() return
//https://code.google.com/p/android/issues/detail?id=41983
static char mountPath[PATH_MAX];
app->storageRoot = strcpy(mountPath, mntPath);
append(mountPath, "/data"); LOGI("OBB mounted: %s", filename);
pthread_cond_broadcast(&app->cond);
}
else if( state == AOBB_STATE_UNMOUNTED )
{
LOGI("OBB unmounted: %s", filename);
}
else if( state != AOBB_STATE_ERROR_NOT_MOUNTED)
{
LOGE("VCAndroid_ObbCallbackFunc: %d", state);
if( app != null )
pthread_cond_broadcast(&app->cond);
}
} mountOBB()
{
AStorageManager_mountObb( Callback );
pthread_cond_wait(&app->cond, &app->mutex); } //Main thread entry for activity create, called by NactiveActivity.java
NativeActivity_onCreate()
{
...
mountOBB();
createthread: android_main(); //this is pseudo code
return;
}

调用java的native代码是这样:

//pseudo codes:

mountOBB()
{
call Java code to mount
} bool isOBBMounted()
{
query Java code whether mount ready
} NativeActivity_onCreate(...)
{
...
mountOBB();
create thread: android_main();
} android_main()
{ while ((ident= ALooper_pollAll(...)) >= )
{
if( isOBBMounted() )
continue_app();
}
}

最后尝试把native代码也全放在android_main()线程里,而且等窗口创建好, 像用Java代码那样循环查询, 还是失败...

还有, 测试的时候还发现, 某些设备push OBB以后app读取不出来, 最后发现是没有使用新的标准路径,
有的设备上/sdcard/Android/obb/com.XXX.XXX/ 是无法访问的. 要使用sdk/tools/monnitor查看mount的真正路径,然后方进去

http://stackoverflow.com/questions/18064114/expansion-file-cant-load-obb-from-sdcard-android-obb-on-android-4-2

比如三星的Galaxy S4, 更新到4.4.2的系统, 直接adb push到/sdcard/Android/obb/XXX/main.1.XXX.obb就没有问题,

而Galaxy Note 10 2014新版虽然系统版本是4.3, 但是就有这个问题, push以后, 应用程序找不到该文件.

不过这个是调试的时候才有的问题. 真正发布的时候, 我们不需要知道真实路径, app会拿到系统给的路径, 然后下载obb放到该文件夹下. 调试的时候因为是手动上传的包, 可能路径不对.


目前在各种设备上测试确定Java的mount没有出现过问题.

现在为了方便测试, 会优先读取/sdcard/main.1.XXX.obb 这样测试的时候obb直接放到sd卡里面就好了,一般测试的时候都是这么做吧.

但是还是不确定新的安卓系统对于非app私有的数据是否有读取权限限制, 应该没有, 不过最好先看文档确认下.

因为Android4.4 KitKat已经没有外部写权限了(我觉得这样更安全,本来是个好事,但是应该一开始就这么搞,现在突然这么搞会出现软件兼容性问题), 不知道外部读会不会有问题, 现在测过一两个4.4的设备还没有发现问题.

工作记录[续] android OBB的更多相关文章

  1. 【工作记录】android手势事件操作记录

    /* 用户按下触摸屏.快速移动后松开 public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float vel ...

  2. 工作记录 - OBB的解决方案

    之前关于OBB的内容: Android上使用native IO 最近工作中的问题笔记 工作记录[续] android OBB 自从用了Java来mount OBB, 再也没有遇到挂载的问题. 但最近在 ...

  3. [工作记录] Android OpenGL ES: non-square texture - continue

    previous: [工作记录] Android OpenGL ES 2.0: square texture not supported on some device recently I found ...

  4. [转]倍数提高工作效率的 Android Studio 奇技

    转自:http://android.jobbole.com/81687/ 倍数提高工作效率的 Android Studio 奇技 2015/10/08 · 技术分享 · 4 评论· Android S ...

  5. MySQL跨表更新字段 工作记录

    工作中遇到两表查询,从user表中获取用户唯一id字段 写入到另外一张qiuzu表中的uid字段中; 二者可以关联起来的只有用户的手机号码tel字段; 了解需求后数据量稍多,不可能一个一个的手动修改 ...

  6. 最近的linux工作记录

    最近的linux工作记录 最近公司走了一些同事,部分服务器交到了我的手里,总结一些常用的操作 注:大写的字符串一般是用来占位,需要替换 创建账户和使用密钥对登陆 1,账户系列 useradd 选项 用 ...

  7. 『记录』Android参考资料

    欢迎留言推荐好的教程.资料.博客及作者等. 『记录』Android参考资料 1.前期环境 Android Studio使用Git Android Studio快捷键总结 Android Studio及 ...

  8. 工作记录--WPF自定义控件,实现一个可设置编辑模式的TextBox

    原文:工作记录--WPF自定义控件,实现一个可设置编辑模式的TextBox 1. 背景 因为最近在使用wpf开发桌面端应用,在查看页面需要把TextBox和Combox等控件设置为只读的.原本是个很简 ...

  9. 图书馆管理系统程序+全套开发文档(系统计划书,系统使用说明,测试报告,UML分析与设计,工作记录)

    图书馆管理系统程序+全套开发文档(系统计划书,系统使用说明,测试报告,UML分析与设计,工作记录): https://download.csdn.net/download/qq_39932172/11 ...

随机推荐

  1. JavaScript -基础- 函数与对象(三)Date对象

    一.Date对象 1.创建方法 var date_obj=new Date(); alert(date_obj.toLocaleString()) var date_obj=new Date(&quo ...

  2. C++ Templates 关于程序库的概念和通用工具

    using namespace std所谓的命名空间,就是一种将程序库名称封装起来的方法,它就像在程序库中竖立了一道围墙 标准程序库中有一部分,比如string classes,支持具体的错误处理,它 ...

  3. IIS设置上传文件大小限制

    单位为字节. 500*1024*1024=524288000

  4. C++关于运算符的注意事项

    1.函数调用也是一种特殊的运算符,对运算对象的个数不作限制. 2.几元运算符,是基于作用的对象的数量. 3.不同类型的运算对象进行运算,可能会出现类型转换,一般情况下小整数类型会被转换成较大的整数类型 ...

  5. Android : 输入设备键值从底层到应用层的映射流程

    一.Android输入子系统简介: Android输入事件的源头是位于/dev/input/下的设备节点,而输入系统的终点是由WMS管理的某个窗口.最初的输入事件为内核生成的原始事件,而最终交付给窗口 ...

  6. java中coroutine使用

    链接1:http://jm.taobao.org/2010/09/17/326/ 链接2:https://www.jianshu.com/p/0f1a6943eab5

  7. Java语法基础DayFive

    一.继承 1.格式:class 子类 extends 父类 2.好处:提高代码的复用性:让类与类之间产生了关系,是多态的前提. 3.弊端: (1)类的耦合性增强了,而开发的原则是高内聚,低耦合.内聚是 ...

  8. 20165326 java第九周学习笔记

    第九周学习笔记 URL类 属于java.net包 最基本三部分:协议(对象所在的Java虚拟机支持).地址(能连接的有效IP地址或域名).资源(主机上的任何一个文件) 常用构造方法 public UR ...

  9. 通过调整浏览器UA设置欺骗限制上网

    先上图片, 通过调整浏览器UA,欺骗识别,原来这个WIFI是只能手机端使用的,打开IE   F12,进行如上图所示,进行修改,正常输入手机号,获取验证码,登陆后,即可上网了.虽然显示的是400,但实际 ...

  10. shell脚本实例-内存磁盘使用警告

    1,磁盘使用警告并发送邮件 #!usr/bin/bash #df -Th|grep '/$' 这个是获取内存使用的那一条记录 #后面两句是获取内存的使用率 disk=`df -Th|grep '/$' ...