美文网首页
JNI学习-用JNI进行AES加密

JNI学习-用JNI进行AES加密

作者: 夏_Leon | 来源:发表于2019-05-10 10:36 被阅读0次

    只是一篇学习笔记,不系统。

    尝试用JNI进行AES加密

    要求使用cbc算法、Pkcs5Padding填充、可自定义key、初始向量,尝试一些库,记录下来。

    从openssl中抽出的AES相关代码 https://github.com/lcl101/AES_C
    很多教程里提到了这个库,应该是包含了比较完整的加密方式的,有时间再慢慢读代码。

    一个用NDK实现AES加密、签名校验防止二次打包 https://github.com/BruceWind/AESJniEncrypt
    这个库应该是比较完善的,start的人也很多,但是目前仅支持ecb加密,也不支持自定义初始向量,只能先了解,等后续完善了。https://github.com/kokke/tiny-AES-c 是这个库参考的算法,如果需要补全cbc加密等功能,应该可以在这个库里找到。

    https://github.com/panxw/android-aes-jni
    是github上直接搜出来的库,时间比较久远应该也不维护了,把其中需要的代码整理出来后,意外地很符合自己的需要,需要把256位密钥改为128位,密钥、初始向量也改为16位,另外C中的密钥和向量都是16进制数组,而原java中是用字符串的,转一下即可。
    第二天发现有巨坑,他没在C里写解密算法!解密算法是用java实现的!

    另一个实现比较臃肿没细看 https://www.jianshu.com/p/3f09a048a2cc

    另有 https://github.com/wtuadn/JNITool 因为Base64加密错误的原因,一直以为是该代码有问题,其实不然,应该也是能用的。

    最后是在 https://github.com/xiangdingquan/Aes 找到比较符合我要求的第三方代码。

    第一个坑

    java代码里的base64加密Base64.encode(str, Base64.URL_SAFE),必须选择URL_SAFE,不能选择DEFAULT,否则会有部分特殊符号不一致。

    第二个坑

    加解密测试下来大多没问题,但是在4.4系统中全部出错,加密的时候传入的字符串莫名短了很多。

    extern "C"
    JNIEXPORT jstring JNICALL
    Java_com_zhaodaoweizhi_trackcar_g4_1jni_AESUtil_encrypt(JNIEnv *env, jclass type, jstring src_) {
        const char *str = (char *) env->GetStringUTFChars(src_, 0);
        env->ReleaseStringUTFChars(src_, str);
        char *result = AES_CBC_PKCS7_Encrypt(str, AES_KEY, AES_IV);//AES CBC PKCS7Padding加密
        jstring out = env->NewStringUTF(result);
        return out;
    }
    

    这是加密的本地入口代码,错误信息并没有指向出错源头。为了找到源头,只能进入加密算法里面,一行一行去打印日志,比对8.0系统下的正常日志。
    最后发现是在char *result = AES_CBC_PKCS7_Encrypt(str, AES_KEY, AES_IV);这个方法里报错的,但奇怪的时候入口时str是正确的,在里面进一步计算的时候str就变短了。
    没有找到任何地方有修改过str,大胆猜测,是在前面那句env->ReleaseStringUTFChars(src_, str);把str修改了,调换了这句代码的位置再试试就通过了。

    extern "C"
    JNIEXPORT jstring JNICALL
    Java_com_zhaodaoweizhi_trackcar_g4_1jni_AESUtil_encrypt(JNIEnv *env, jclass type, jstring src_) {
        const char *str = (char *) env->GetStringUTFChars(src_, 0);
        char *result = AES_CBC_PKCS7_Encrypt(str, AES_KEY, AES_IV);//AES CBC PKCS7Padding加密
        env->ReleaseStringUTFChars(src_, str);
        jstring out = env->NewStringUTF(result);
        return out;
    }
    

    而这个问题只在4.4及以下系统下存在,其他系统并不报错,非常奇怪了。

    相关文章

      网友评论

          本文标题:JNI学习-用JNI进行AES加密

          本文链接:https://www.haomeiwen.com/subject/zvuooqtx.html