Библиотека ICU в Android NDK - PullRequest
       16

Библиотека ICU в Android NDK

2 голосов
/ 12 ноября 2010

Я пытаюсь создать оболочку JNI для библиотеки C, которая зависит от библиотек ICU (libicuuc.so и libicui18n.so).

Я пытался собрать ICU4C в моем NDK (как стандартной, так и версии CrystaXна машине с Mac OS X) и продолжал сталкиваться с такими проблемами:

/Users/kyip/KyVmShared/KyAndroid/myproject/obj/local/armeabi/objs/icuuc/udata.o: In function `openCommonData':
/Users/kyip/KyVmShared/KyAndroid/myproject/jni/icu4c/common/udata.c:836: undefined reference to `icudt42_dat'
/Users/kyip/KyVmShared/KyAndroid/myproject/obj/local/armeabi/objs/icuuc/ustr_wcs.o: In function `_strFromWCS':
/Users/kyip/KyVmShared/KyAndroid/myproject/jni/icu4c/common/ustr_wcs.c:365: undefined reference to `wcstombs'
/Users/kyip/KyVmShared/KyAndroid/myproject/jni/icu4c/common/ustr_wcs.c:415: undefined reference to `wcstombs'
/Users/kyip/KyVmShared/KyAndroid/myproject/jni/icu4c/common/ustr_wcs.c:314: undefined reference to `wcstombs'
/Users/kyip/KyVmShared/KyAndroid/myproject/obj/local/armeabi/objs/icuuc/ustr_wcs.o: In function `_strToWCS':
/Users/kyip/KyVmShared/KyAndroid/myproject/jni/icu4c/common/ustr_wcs.c:164: undefined reference to `mbstowcs'
collect2: ld returned 1 exit status

Я также попробовал предложение, приведенное в поддержка юникода в Android ndk , но не повезло.Я застрял на:

arm-eabi-g++ -I/ky/crystax/android-ndk-r4-crystax/build/platforms/android-8/arch-arm/usr/include/ -O3 -fno-short-wchar -DU_USING_ICU_NAMESPACE=0 -DU_GNUC_UTF16_STRING=0 -fno-short-enums -nostdlib -fPIC -DU_COMMON_IMPLEMENTATION  -D_REENTRANT -I../common -I../../icu/source/common -I../../icu/source/i18n   "-DDEFAULT_ICU_PLUGINS=\"/usr/local/lib/icu\" "  -DU_COMMON_IMPLEMENTATION -DHAVE_CONFIG_H  -I/ky/crystax/android-ndk-r4-crystax/build/platforms/android-8/arch-arm/usr/include/ -O3 -fno-short-wchar -DU_USING_ICU_NAMESPACE=0 -DU_GNUC_UTF16_STRING=0 -fno-short-enums -nostdlib -fPIC -DU_COMMON_IMPLEMENTATION  -std=c++0x  -fvisibility=hidden -c   -o errorcode.ao ../../icu/source/common/errorcode.cpp
In file included from ../../icu/source/common/unicode/ptypes.h:23,
                 from ../../icu/source/common/unicode/umachine.h:52,
                 from ../../icu/source/common/unicode/utypes.h:36,
                 from ../../icu/source/common/errorcode.cpp:17:
/ky/crystax/android-ndk-r4-crystax/build/platforms/android-8/arch-arm/usr/include/sys/types.h:122: error: 'uint64_t' does not name a type
make[1]: *** [errorcode.ao] Error 1
make: *** [all-recursive] Error 2

Любая помощь будет оценена.

Ответы [ 4 ]

0 голосов
/ 15 июля 2017

Предпринимались попытки предоставить оболочки NDK для библиотек ICU, являющихся частью системы: https://android -review.googlesource.com / c / 153001 / .

0 голосов
/ 23 мая 2013

У меня также была эта проблема: неопределенная ссылка на `mbstowcs '

Вы должны собрать и связать с более высокой версией Android API.

Примечание. Я пытался связать его с библиотеками из android-ndk / platform / android-4 ... Я думал, что 4 - это версия Android, но 4 - это версия Android API.А Android API 4 соответствует Android 1.6, который очень и очень стар, в libc

действительно нет функции mbstowcs.
0 голосов
/ 24 июня 2014

Вот как я решил проблему. Это грязно, но это работает. Библиотека была скомпилирована:

1. файл: /icu4c/common/cwchar.h

закомментируйте #if U_HAVE_WCHAR_H и соответствующие #endif, поэтому <wchar.h> всегда включается.

заменить предыдущее uprv_wcstombs определение на:

#define uprv_wcstombs(mbstr, wcstr, count) U_STANDARD_CPP_NAMESPACE wcs2mbs(mbstr, wcstr, count)

заменить предыдущее uprv_mbstowcs определение на:

#define uprv_mbstowcs(wcstr, mbstr, count) U_STANDARD_CPP_NAMESPACE mbs2wcs(wcstr, mbstr, count)

2. файл: /icu4c/common/ustr_wcs.cpp

где-то вверху, под уже существующими включениями добавьте строку:

#include "../wcsmbs.h"

3. создать новый файл "icu4c / wcsmbs.h"

size_t mbs2wcs(wchar_t * __ restrict pwcs, const char * __ restrict s, size_t n)
{

   mbstate_t mbs;
   const char *sp;
   memset(&mbs, 0, sizeof(mbs));
   sp=s;
   return (mbsrtowcs(pwcs,&sp,n,&mbs));
}


size_t wcs2mbs(char * __restrict s, const wchar_t * __restrict pwcs, size_t n)
{
   mbstate_t mbs;
   const wchar_t *pwcsp;
   memset(&mbs,0,sizeof(mbs));
   pwcsp = pwcs;
   return (wcsrtombs(s,&pwcsp,n,&mbs));
}

Надеюсь, это поможет.

0 голосов
/ 10 февраля 2012

Похоже, что в этой проблеме участвуют два файла. icu / source / common / unicode / ptypes.h, который вызывает sys / types.h, включает

#if ! U_HAVE_UINT64_T
    typedef unsigned long long uint64_t;
/* else we may not have a 64-bit type */
#endif

Включая sys / types.h из Android, мы задействуем (около строки 122/124)

#ifdef __BSD_VISIBLE
typedef unsigned char   u_char;
typedef unsigned short  u_short;
typedef unsigned int    u_int;
typedef unsigned long   u_long;

typedef uint32_t       u_int32_t;
typedef uint16_t       u_int16_t;
typedef uint8_t        u_int8_t;
typedef uint64_t       u_int64_t;
#endif

Кажется, что uint64_t не был объявлен, когда он назначен на u_int64_t. Действительно, sys / types.h включает в себя stdint.h, который имеет следующее:

#if !defined __STRICT_ANSI__ || __STDC_VERSION__ >= 199901L
#  define __STDC_INT64__
#endif

typedef __int8_t      int8_t;
typedef __uint8_t     uint8_t;
typedef __int16_t     int16_t;
typedef __uint16_t    uint16_t;
typedef __int32_t     int32_t;
typedef __uint32_t    uint32_t;
#if defined(__STDC_INT64__)
typedef __int64_t     int64_t;
typedef __uint64_t    uint64_t;
#endif

Вероятно STRICT_ANSI не определено. Похоже, это ошибка в коде Android в sys / types.h. Если STDC_INT64 не определен, он не будет определять uint64_t, поэтому он не может определить u_int64_t. Возможно, реальное решение состоит в том, чтобы изменить sys / types.h так, чтобы он имел

#ifdef __BSD_VISIBLE
typedef unsigned char   u_char;
typedef unsigned short  u_short;
typedef unsigned int    u_int;
typedef unsigned long   u_long;

typedef uint32_t       u_int32_t;
typedef uint16_t       u_int16_t;
typedef uint8_t        u_int8_t;
$if defined(__STDC_INT64__)
typedef uint64_t       u_int64_t;
#endif
#endif

Если вы исправите это, следующая ошибка будет в cstring.h: 109

icu/source/common/cstring.h:109: error: 'int64_t' has not been declared

Если вместо этого вы будете использовать #define STDC_INT64 в файле common / unicode / ptypes.h, он будет значительно дальше, но закончится на

icu/source/common/ustrenum.cpp:118: error: must #include <typeinfo> before using typeid

с дополнительной информацией здесь: http://groups.google.com/group/android-ndk/browse_thread/thread/2ec9dc289d815ba3?pli=1, но без реальных решений

...