From b3e4995ccdde79aeb3acbb8ef2930e871c82b7c9 Mon Sep 17 00:00:00 2001 From: guoling Date: Fri, 15 Nov 2024 16:06:31 +0800 Subject: [PATCH 01/12] make copyFile() robust with rename() failture #1432 --- Core/MemoryFile.cpp | 7 +++++++ Core/MemoryFile_OSX.cpp | 6 +++++- 2 files changed, 12 insertions(+), 1 deletion(-) diff --git a/Core/MemoryFile.cpp b/Core/MemoryFile.cpp index 2e705e6a..808da7dc 100644 --- a/Core/MemoryFile.cpp +++ b/Core/MemoryFile.cpp @@ -520,6 +520,13 @@ bool copyFile(const MMKVPath_t &srcPath, const MMKVPath_t &dstPath) { if (copyFileContent(srcPath, tmpFD, false)) { MMKVInfo("copyfile [%s] to [%s]", srcPath.c_str(), tmpPath.c_str()); renamed = tryAtomicRename(tmpPath, dstPath); + if (!renamed) { + MMKVInfo("rename fail, try copy file content instead."); + if (copyFileContent(tmpPath, dstPath)) { + renamed = true; + ::unlink(tmpPath.c_str()); + } + } if (renamed) { MMKVInfo("copyfile [%s] to [%s] finish.", srcPath.c_str(), dstPath.c_str()); } diff --git a/Core/MemoryFile_OSX.cpp b/Core/MemoryFile_OSX.cpp index fa65b0e7..df9745a3 100644 --- a/Core/MemoryFile_OSX.cpp +++ b/Core/MemoryFile_OSX.cpp @@ -97,8 +97,12 @@ bool copyFile(const MMKVPath_t &srcPath, const MMKVPath_t &dstPath) { MMKVInfo("copyfile [%s] to [%s] finish.", srcPath.c_str(), dstPath.c_str()); return true; } + + MMKVInfo("rename fail, try copy file content instead."); + auto ret = copyFileContent(tmpFile.UTF8String, dstPath); + unlink(tmpFile.UTF8String); - return false; + return ret; } bool copyFileContent(const MMKVPath_t &srcPath, const MMKVPath_t &dstPath) { From dd8847ce3728f894f186f05ad00493e118e6102e Mon Sep 17 00:00:00 2001 From: guoling Date: Tue, 19 Nov 2024 11:20:42 +0800 Subject: [PATCH 02/12] Win32: only mmap() on ftruncate()/zeroFillFile() failture iff we have a valid file mapping before #1068 --- Core/MemoryFile_Win32.cpp | 12 ++++++++++-- 1 file changed, 10 insertions(+), 2 deletions(-) diff --git a/Core/MemoryFile_Win32.cpp b/Core/MemoryFile_Win32.cpp index 8bec794b..7ada2d87 100755 --- a/Core/MemoryFile_Win32.cpp +++ b/Core/MemoryFile_Win32.cpp @@ -122,7 +122,11 @@ bool MemoryFile::truncate(size_t size) { m_size = ((m_size / DEFAULT_MMAP_SIZE) + 1) * DEFAULT_MMAP_SIZE; } + // Win32 won't ftruncate a file if there's active file mmapping/handle, we have to unmmap/close ahead + bool needMMapOnFailure = false; if (m_ptr) { + // if we have a valid file mapping before, we should restore it regardless + needMMapOnFailure = true; if (!UnmapViewOfFile(m_ptr)) { MMKVError("fail to munmap [%ls], %d", m_diskFile.m_path.c_str(), GetLastError()); } @@ -136,14 +140,18 @@ bool MemoryFile::truncate(size_t size) { if (!ftruncate(m_diskFile.getFd(), m_size)) { MMKVError("fail to truncate [%ls] to size %zu", m_diskFile.m_path.c_str(), m_size); m_size = oldSize; - mmap(); + if (needMMapOnFailure) { + mmap(); + } return false; } if (m_size > oldSize) { if (!zeroFillFile(m_diskFile.getFd(), oldSize, m_size - oldSize)) { MMKVError("fail to zeroFile [%ls] to size %zu", m_diskFile.m_path.c_str(), m_size); m_size = oldSize; - mmap(); + if (needMMapOnFailure) { + mmap(); + } return false; } } From ba4f343de517daa6da8807eb953933f1d1cd5213 Mon Sep 17 00:00:00 2001 From: guoling Date: Thu, 28 Nov 2024 22:30:57 +0800 Subject: [PATCH 03/12] protect from invalid value size of auto-key-expire mmkv #1440 --- Core/MMKV_IO.cpp | 40 ++++++++++++++++++++++++++++++++++++++++ 1 file changed, 40 insertions(+) diff --git a/Core/MMKV_IO.cpp b/Core/MMKV_IO.cpp index bb1f302a..54fa9594 100755 --- a/Core/MMKV_IO.cpp +++ b/Core/MMKV_IO.cpp @@ -1731,6 +1731,14 @@ bool MMKV::disableAutoKeyExpire() { MMKVVector vec; auto packKeyValue = [&](const MMKVKey_t &key, const MMBuffer &value) { assert(value.length() >= Fixed32Size); + if (value.length() < Fixed32Size) { +#ifdef MMKV_APPLE + MMKVWarning("key [%@] has invalid value size %u", key, value.length()); +#else + MMKVWarning("key [%s] has invalid value size %u", key.data(), value.length()); +#endif + return; + } MMBuffer data(value.length() - Fixed32Size); auto ptr = (uint8_t *) data.getPtr(); memcpy(ptr, value.getPtr(), value.length() - Fixed32Size); @@ -1771,6 +1779,13 @@ uint32_t MMKV::getExpireTimeForKey(MMKVKey_t key) { auto raw = getRawDataForKey(key); assert(raw.length() == 0 || raw.length() >= Fixed32Size); if (raw.length() < Fixed32Size) { + if (raw.length() != 0) { +#ifdef MMKV_APPLE + MMKVWarning("key [%@] has invalid value size %u", key, raw.length()); +#else + MMKVWarning("key [%s] has invalid value size %u", key.data(), raw.length()); +#endif + } return 0; } auto ptr = (const uint8_t *) raw.getPtr() + raw.length() - Fixed32Size; @@ -1786,6 +1801,13 @@ mmkv::MMBuffer MMKV::getDataWithoutMTimeForKey(MMKVKey_t key) { auto raw = getRawDataForKey(key); assert(raw.length() == 0 || raw.length() >= Fixed32Size); if (raw.length() < Fixed32Size) { + if (raw.length() != 0) { +#ifdef MMKV_APPLE + MMKVWarning("key [%@] has invalid value size %u", key, raw.length()); +#else + MMKVWarning("key [%s] has invalid value size %u", key.data(), raw.length()); +#endif + } return raw; } auto newLength = raw.length() - Fixed32Size; @@ -1824,6 +1846,15 @@ size_t MMKV::filterExpiredKeys() { for (auto itr = m_dicCrypt->begin(); itr != m_dicCrypt->end(); NOOP) { auto &kvHolder = itr->second; assert(kvHolder.realValueSize() >= Fixed32Size); + if (kvHolder.realValueSize() < Fixed32Size) { +#ifdef MMKV_APPLE + MMKVWarning("key [%@] has invalid value size %u", itr->first, kvHolder.realValueSize()); +#else + MMKVWarning("key [%s] has invalid value size %u", itr->first.c_str(), kvHolder.realValueSize()); +#endif + itr++; + continue; + } auto buffer = kvHolder.toMMBuffer(basePtr, m_crypter); auto ptr = (uint8_t *) buffer.getPtr(); ptr += buffer.length() - Fixed32Size; @@ -1848,6 +1879,15 @@ size_t MMKV::filterExpiredKeys() { for (auto itr = m_dic->begin(); itr != m_dic->end(); NOOP) { auto &kvHolder = itr->second; assert(kvHolder.valueSize >= Fixed32Size); + if (kvHolder.valueSize < Fixed32Size) { +#ifdef MMKV_APPLE + MMKVWarning("key [%@] has invalid value size %u", itr->first, kvHolder.valueSize); +#else + MMKVWarning("key [%s] has invalid value size %u", itr->first.c_str(), kvHolder.valueSize); +#endif + itr++; + continue; + } auto ptr = basePtr + kvHolder.offset + kvHolder.computedKVSize; ptr += kvHolder.valueSize - Fixed32Size; auto time = *(const uint32_t *) ptr; From 4249a645aee6210778f218d8e0377db6610c6fdb Mon Sep 17 00:00:00 2001 From: guoling Date: Mon, 2 Dec 2024 17:32:20 +0800 Subject: [PATCH 04/12] obfuscation for OHOS looks like working #1385 --- OpenHarmony/MMKV/consumer-rules.txt | 46 +++++++++++++++++++++----- OpenHarmony/MMKV/obfuscation-rules.txt | 14 -------- 2 files changed, 37 insertions(+), 23 deletions(-) diff --git a/OpenHarmony/MMKV/consumer-rules.txt b/OpenHarmony/MMKV/consumer-rules.txt index e02e7bb9..7f036af1 100644 --- a/OpenHarmony/MMKV/consumer-rules.txt +++ b/OpenHarmony/MMKV/consumer-rules.txt @@ -17,16 +17,44 @@ # -keep-property-name: specifies property names that you want to keep # -keep-global-name: specifies names that you want to keep in the global scope --keep -./src/main/ets/utils/MMKV.ets -./src/main/ets/utils/MMKVHandler.ets -./src/main/ets/utils/MMKVLogLevel.ets -./src/main/ets/utils/NativeBuffer.ets -./src/main/ets/utils/Util.ts - --keep-file-name +-keep-global-name MMKV MMKVHandler +MMKVRecoverStrategic MMKVLogLevel NativeBuffer -Util +getObjKeys + +-keep-property-name +*_MODE +initializeWithPath +defaultMMKV +initData +wantLogRedirect +mmkvLog +onMMKV* +wantContentChangeNotification +onContentChangedByOuterProcess +Level* +pointer +size +backedUpMMKVWithID +decodeFloat32Array +decodeFloat64Array +decodeInt16Array +decodeInt8Array +decodeUint16Array +decodeUint32Array +decodeUint64Array +decodeUint8Array +decodeUint8ClampedArray +defaultMMKV +encodeBytesPart +encodeTypedArray +importFromPreferences +mmkvWithAshmemID +rootDir + +-keep-dts +/Users/lingol/Developer/mmkv/OpenHarmony/MMKV/src/main/cpp/types/libmmkv/ + diff --git a/OpenHarmony/MMKV/obfuscation-rules.txt b/OpenHarmony/MMKV/obfuscation-rules.txt index d4c3b4d2..51021121 100644 --- a/OpenHarmony/MMKV/obfuscation-rules.txt +++ b/OpenHarmony/MMKV/obfuscation-rules.txt @@ -16,17 +16,3 @@ # Keep options: # -keep-property-name: specifies property names that you want to keep # -keep-global-name: specifies names that you want to keep in the global scope - --keep -./src/main/ets/utils/MMKV.ets -./src/main/ets/utils/MMKVHandler.ets -./src/main/ets/utils/MMKVLogLevel.ets -./src/main/ets/utils/NativeBuffer.ets -./src/main/ets/utils/Util.ts - --keep-file-name -MMKV -MMKVHandler -MMKVLogLevel -NativeBuffer -Util \ No newline at end of file From 8d6377667c94782f2b61e7949a9f88d2085c7964 Mon Sep 17 00:00:00 2001 From: guoling Date: Mon, 2 Dec 2024 18:00:16 +0800 Subject: [PATCH 05/12] add testcase for backup/restore normal mmap file from custom path --- .../com/tencent/mmkvdemo/MainActivity.java | 24 +++++++++++++++++++ 1 file changed, 24 insertions(+) diff --git a/Android/MMKV/mmkvdemo/src/main/java/com/tencent/mmkvdemo/MainActivity.java b/Android/MMKV/mmkvdemo/src/main/java/com/tencent/mmkvdemo/MainActivity.java index 859ab889..118a4948 100644 --- a/Android/MMKV/mmkvdemo/src/main/java/com/tencent/mmkvdemo/MainActivity.java +++ b/Android/MMKV/mmkvdemo/src/main/java/com/tencent/mmkvdemo/MainActivity.java @@ -105,6 +105,10 @@ public void onClick(View v) { kv.checkContentChangedByOuterProcess(); kv.close(); + // prepare for backup customize root path + kv = testMMKV("test_backup", "MMKV Backup", false, otherDir); + kv.close(); + testAshmem(); testReKey(); @@ -564,6 +568,15 @@ private void testBackup() { Log.i("MMKV", "check on backup file[" + mmkv.mmapID() + "] allKeys: " + Arrays.toString(mmkv.allKeys())); } + // test backup a normal mmkv from custom root path + mmapID = "test_backup"; + ret = MMKV.backupOneToDirectory(mmapID, backupRootDir, otherDir); + Log.i("MMKV", "backup one [" + mmapID + "] ret = " + ret); + if (ret) { + MMKV mmkv = MMKV.backedUpMMKVWithID(mmapID, MMKV.SINGLE_PROCESS_MODE, "MMKV Backup", backupRootDir); + Log.i("MMKV", "check on backup file[" + mmkv.mmapID() + "] allKeys: " + Arrays.toString(mmkv.allKeys())); + } + /*{ MMKV mmkv = MMKV.mmkvWithID("imported"); mmkv.close(); @@ -603,6 +616,17 @@ private void testRestore() { Log.i("MMKV", "after restore [" + mmkv.mmapID() + "] allKeys: " + Arrays.toString(mmkv.allKeys())); } + // test backup a normal mmkv from custom root path + mmapID = "test_backup"; + mmkv = MMKV.mmkvWithID(mmapID, MMKV.SINGLE_PROCESS_MODE, "MMKV Backup", otherDir); + mmkv.encode("test_restore", 1024); + Log.i("MMKV", "before restore [" + mmkv.mmapID() + "] allKeys: " + Arrays.toString(mmkv.allKeys())); + ret = MMKV.restoreOneMMKVFromDirectory(mmapID, backupRootDir, otherDir); + Log.i("MMKV", "backup one [" + mmapID + "] ret = " + ret); + if (ret) { + Log.i("MMKV", "check on backup file[" + mmkv.mmapID() + "] allKeys: " + Arrays.toString(mmkv.allKeys())); + } + /*{ mmkv = MMKV.mmkvWithID("imported"); mmkv.close(); From 20b68dbc7134dbed39adbe854514fb9762b8ed96 Mon Sep 17 00:00:00 2001 From: guoling Date: Mon, 2 Dec 2024 15:15:18 +0800 Subject: [PATCH 06/12] fix tryAtomicRename() with src == dst bug --- Core/MMKV.h | 4 ++-- Core/MMKV_Android.cpp | 14 +++++++------- Core/MemoryFile_Linux.cpp | 5 ++++- Core/MemoryFile_OSX.cpp | 8 ++++++-- 4 files changed, 19 insertions(+), 12 deletions(-) diff --git a/Core/MMKV.h b/Core/MMKV.h index c031fec8..16e9d836 100644 --- a/Core/MMKV.h +++ b/Core/MMKV.h @@ -226,7 +226,7 @@ class MMKV { void notifyContentChanged(); #if defined(MMKV_ANDROID) && !defined(MMKV_DISABLE_CRYPT) - void checkReSetCryptKey(int fd, int metaFD, std::string *cryptKey); + void checkReSetCryptKey(int fd, int metaFD, const std::string *cryptKey); #endif static bool backupOneToDirectory(const std::string &mmapKey, const MMKVPath_t &dstPath, const MMKVPath_t &srcPath, bool compareFullPath); static size_t backupAllToDirectory(const MMKVPath_t &dstDir, const MMKVPath_t &srcDir, bool isInSpecialDir); @@ -591,5 +591,5 @@ bool MMKV::getVector(MMKVKey_t key, T &result) { MMKV_NAMESPACE_END -#endif +#endif // __cplusplus #endif // MMKV_MMKV_H diff --git a/Core/MMKV_Android.cpp b/Core/MMKV_Android.cpp index 344a6a5c..22028145 100644 --- a/Core/MMKV_Android.cpp +++ b/Core/MMKV_Android.cpp @@ -39,11 +39,11 @@ using namespace mmkv; extern unordered_map *g_instanceDic; extern ThreadLock *g_instanceLock; -MMKV::MMKV(const string &mmapID, int size, MMKVMode mode, string *cryptKey, string *rootPath, size_t expectedCapacity) +MMKV::MMKV(const string &mmapID, int size, MMKVMode mode, const string *cryptKey, const string *rootPath, size_t expectedCapacity) : m_mmapID((mode & MMKV_BACKUP) ? mmapID : mmapedKVKey(mmapID, rootPath)) // historically Android mistakenly use mmapKey as mmapID , m_mode(mode) , m_path(mappedKVPathWithID(m_mmapID, mode, rootPath)) - , m_crcPath(crcPathWithID(m_mmapID, mode, rootPath)) + , m_crcPath(crcPathWithPath(m_path)) , m_dic(nullptr) , m_dicCrypt(nullptr) , m_expectedCapacity(std::max(DEFAULT_MMAP_SIZE, roundUp(expectedCapacity, DEFAULT_MMAP_SIZE))) @@ -88,11 +88,11 @@ MMKV::MMKV(const string &mmapID, int size, MMKVMode mode, string *cryptKey, stri }*/ } -MMKV::MMKV(const string &mmapID, int ashmemFD, int ashmemMetaFD, string *cryptKey) +MMKV::MMKV(const string &mmapID, int ashmemFD, int ashmemMetaFD, const string *cryptKey) : m_mmapID(mmapID) , m_mode(MMKV_ASHMEM) , m_path(mappedKVPathWithID(m_mmapID, MMKV_ASHMEM, nullptr)) - , m_crcPath(crcPathWithID(m_mmapID, MMKV_ASHMEM, nullptr)) + , m_crcPath(crcPathWithPath(m_path)) , m_dic(nullptr) , m_dicCrypt(nullptr) , m_file(new MemoryFile(ashmemFD)) @@ -137,7 +137,7 @@ MMKV::MMKV(const string &mmapID, int ashmemFD, int ashmemMetaFD, string *cryptKe }*/ } -MMKV *MMKV::mmkvWithID(const string &mmapID, int size, MMKVMode mode, string *cryptKey, string *rootPath, size_t expectedCapacity) { +MMKV *MMKV::mmkvWithID(const string &mmapID, int size, MMKVMode mode, const string *cryptKey, const string *rootPath, size_t expectedCapacity) { if (mmapID.empty() || !g_instanceLock) { return nullptr; } @@ -162,7 +162,7 @@ MMKV *MMKV::mmkvWithID(const string &mmapID, int size, MMKVMode mode, string *cr return kv; } -MMKV *MMKV::mmkvWithAshmemFD(const string &mmapID, int fd, int metaFD, string *cryptKey) { +MMKV *MMKV::mmkvWithAshmemFD(const string &mmapID, int fd, int metaFD, const string *cryptKey) { if (fd < 0 || !g_instanceLock) { return nullptr; @@ -191,7 +191,7 @@ int MMKV::ashmemMetaFD() { } # ifndef MMKV_DISABLE_CRYPT -void MMKV::checkReSetCryptKey(int fd, int metaFD, string *cryptKey) { +void MMKV::checkReSetCryptKey(int fd, int metaFD, const string *cryptKey) { SCOPED_LOCK(m_lock); checkReSetCryptKey(cryptKey); diff --git a/Core/MemoryFile_Linux.cpp b/Core/MemoryFile_Linux.cpp index 843b41a2..bcc4300b 100644 --- a/Core/MemoryFile_Linux.cpp +++ b/Core/MemoryFile_Linux.cpp @@ -72,6 +72,10 @@ bool tryAtomicRename(const MMKVPath_t &srcPath, const MMKVPath_t &dstPath) { MMKVWarning("fail on syscall(SYS_renameat2) [%s] to [%s], %d(%s)", srcPath.c_str(), dstPath.c_str(), errno, strerror(errno)); } + + if (renamed && (srcPath != dstPath)) { + ::unlink(srcPath.c_str()); + } #endif // SYS_renameat2 && !MMKV_OHOS if (!renamed) { @@ -81,7 +85,6 @@ bool tryAtomicRename(const MMKVPath_t &srcPath, const MMKVPath_t &dstPath) { } } - ::unlink(srcPath.c_str()); return true; } diff --git a/Core/MemoryFile_OSX.cpp b/Core/MemoryFile_OSX.cpp index df9745a3..21d56e38 100644 --- a/Core/MemoryFile_OSX.cpp +++ b/Core/MemoryFile_OSX.cpp @@ -58,6 +58,9 @@ void tryResetFileProtection(const string &path) { namespace mmkv { bool tryAtomicRename(const char *src, const char *dst) { + if (!src || !dst) { + return false; + } bool renamed = false; // try atomic swap first @@ -65,6 +68,9 @@ bool tryAtomicRename(const char *src, const char *dst) { // renameat2() equivalent if (renamex_np(src, dst, RENAME_SWAP) == 0) { renamed = true; + if (strcmp(src, dst) != 0) { + ::unlink(src); + } } else if (errno != ENOENT) { MMKVError("fail to renamex_np %s to %s, %s", src, dst, strerror(errno)); } @@ -78,8 +84,6 @@ bool tryAtomicRename(const char *src, const char *dst) { } } - ::unlink(src); - return true; } From b097a0d013924456bdeb9ed3cb1d9c0acb20a4cb Mon Sep 17 00:00:00 2001 From: guoling Date: Tue, 10 Dec 2024 10:43:45 +0800 Subject: [PATCH 07/12] add renameat2() with RENAME_EXCHANGE for OHOS --- Core/MemoryFile_Linux.cpp | 4 ++-- OpenHarmony/MMKV/README.md | 5 +++++ 2 files changed, 7 insertions(+), 2 deletions(-) diff --git a/Core/MemoryFile_Linux.cpp b/Core/MemoryFile_Linux.cpp index bcc4300b..d9254216 100644 --- a/Core/MemoryFile_Linux.cpp +++ b/Core/MemoryFile_Linux.cpp @@ -54,7 +54,7 @@ bool tryAtomicRename(const MMKVPath_t &srcPath, const MMKVPath_t &dstPath) { bool renamed = false; // try renameat2() first -#if defined(SYS_renameat2) && !defined(MMKV_OHOS) +#if defined(SYS_renameat2) #ifdef MMKV_ANDROID static auto g_renameat2 = (renameat2_t) dlsym(RTLD_DEFAULT, "renameat2"); if (g_renameat2) { @@ -76,7 +76,7 @@ bool tryAtomicRename(const MMKVPath_t &srcPath, const MMKVPath_t &dstPath) { if (renamed && (srcPath != dstPath)) { ::unlink(srcPath.c_str()); } -#endif // SYS_renameat2 && !MMKV_OHOS +#endif // SYS_renameat2 if (!renamed) { if (::rename(srcPath.c_str(), dstPath.c_str()) != 0) { diff --git a/OpenHarmony/MMKV/README.md b/OpenHarmony/MMKV/README.md index e037e334..3d0f7b8b 100644 --- a/OpenHarmony/MMKV/README.md +++ b/OpenHarmony/MMKV/README.md @@ -89,6 +89,11 @@ export default class EntryAbility extends UIAbility { mmkv.encodeBytes('bytes', arrayBuffer); let bytes = mmkv.decodeBytes('bytes'); console.info('bytes = ', ArrayBufferToString(bytes)); + + let arr = new Uint8Array([0, 255, 1, 255]); + mmkv.encodeTypedArray('uint8-array', arr); + let newUI8Arr = kv.decodeUint8Array('uint8-array'); + console.info('uint8-array = ', newUI8Arr); ``` As you can see, MMKV is quite easy to use. From cf408a79626985cd96d26468989dc9112d5b934b Mon Sep 17 00:00:00 2001 From: guoling Date: Fri, 13 Dec 2024 12:07:14 +0800 Subject: [PATCH 08/12] check if current Android process is 64-bit --- .../src/main/java/com/tencent/mmkv/MMKV.java | 3 +++ .../UnsupportedArchitectureException.java | 27 +++++++++++++++++++ OpenHarmony/MMKV/BuildProfile.ets | 4 +-- 3 files changed, 32 insertions(+), 2 deletions(-) create mode 100644 Android/MMKV/mmkv/src/main/java/com/tencent/mmkv/UnsupportedArchitectureException.java diff --git a/Android/MMKV/mmkv/src/main/java/com/tencent/mmkv/MMKV.java b/Android/MMKV/mmkv/src/main/java/com/tencent/mmkv/MMKV.java index 14b85e93..45e44114 100644 --- a/Android/MMKV/mmkv/src/main/java/com/tencent/mmkv/MMKV.java +++ b/Android/MMKV/mmkv/src/main/java/com/tencent/mmkv/MMKV.java @@ -190,6 +190,9 @@ public static String initialize(Context context, String rootDir, LibLoader loade } public static String initialize(@NonNull Context context, String rootDir, LibLoader loader, MMKVLogLevel logLevel, MMKVHandler handler) { + if (!android.os.Process.is64Bit()) { + throw new UnsupportedArchitectureException("MMKV 2.0+ requires 64-bit App, use 1.3.x instead."); + } // disable process mode in release build // FIXME: Find a better way to getApplicationInfo() without using context. // If any one knows how, you're welcome to make a contribution. diff --git a/Android/MMKV/mmkv/src/main/java/com/tencent/mmkv/UnsupportedArchitectureException.java b/Android/MMKV/mmkv/src/main/java/com/tencent/mmkv/UnsupportedArchitectureException.java new file mode 100644 index 00000000..2ae95eac --- /dev/null +++ b/Android/MMKV/mmkv/src/main/java/com/tencent/mmkv/UnsupportedArchitectureException.java @@ -0,0 +1,27 @@ +/* + * Tencent is pleased to support the open source community by making + * MMKV available. + * + * Copyright (C) 2018 THL A29 Limited, a Tencent company. + * All rights reserved. + * + * Licensed under the BSD 3-Clause License (the "License"); you may not use + * this file except in compliance with the License. You may obtain a copy of + * the License at + * + * https://opensource.org/licenses/BSD-3-Clause + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package com.tencent.mmkv; + +public class UnsupportedArchitectureException extends RuntimeException { + public UnsupportedArchitectureException(String message) { + super(message); + } +} diff --git a/OpenHarmony/MMKV/BuildProfile.ets b/OpenHarmony/MMKV/BuildProfile.ets index 96577cf1..b82de3d4 100644 --- a/OpenHarmony/MMKV/BuildProfile.ets +++ b/OpenHarmony/MMKV/BuildProfile.ets @@ -2,8 +2,8 @@ * Use these variables when you tailor your ArkTS code. They must be of the const type. */ export const HAR_VERSION = '2.0.1'; -export const BUILD_MODE_NAME = 'release'; -export const DEBUG = false; +export const BUILD_MODE_NAME = 'debug'; +export const DEBUG = true; export const TARGET_NAME = 'default'; /** From c732fca22a4e5ac1a7d10df2569d99152b8e089a Mon Sep 17 00:00:00 2001 From: guoling Date: Fri, 13 Dec 2024 20:21:51 +0800 Subject: [PATCH 09/12] update requirement for 64-bit arch on home page --- README.md | 128 ++++++++++++++++++++++++++------------------------- README_CN.md | 77 ++++++++++++++++--------------- 2 files changed, 104 insertions(+), 101 deletions(-) diff --git a/README.md b/README.md index 4aa46183..b746ab91 100644 --- a/README.md +++ b/README.md @@ -33,6 +33,8 @@ dependencies { } ``` +Starting from v2.0.0, MMKV **no longer supports 32-bit** arch and API level 22 or 21, if you want 32-bit or API level 21~22, use v1.3.x LTS series. + For other installation options, see [Android Setup](https://github.com/Tencent/MMKV/wiki/android_setup). ### Quick Tutorial @@ -270,69 +272,69 @@ std::cout << "string = " << result << std::endl; ``` MMKV also supports **Multi-Process Access**. Full tutorials can be found here [POSIX Tutorial](https://github.com/Tencent/MMKV/wiki/posix_tutorial). - -# MMKV for HarmonyOS NEXT - -## Features - -* **Efficient**. MMKV uses mmap to keep memory synced with file, and protobuf to encode/decode values, making the most of native platform to achieve best performance. - * **Multi-Process concurrency**: MMKV supports concurrent read-read and read-write access between processes. - -* **Easy-to-use**. You can use MMKV as you go. All changes are saved immediately, no `sync`, no `flush` calls needed. - -* **Small**. - * **A handful of files**: MMKV contains process locks, encode/decode helpers and mmap logics and nothing more. It's really tidy. - * **About 600K in binary size**: MMKV adds about 600K per architecture on App size, and much less when zipped (HAR/HAP). - - -## Getting Started -### Installation via OHPM: - -```bash -ohpm install @tencent/mmkv -``` -### Quick Tutorial -You can use MMKV as you go. All changes are saved immediately, no `sync`, no `apply` calls needed. -Setup MMKV on App startup, say your `EntryAbility.onCreate()` function, add these lines: - -```js -import { MMKV } from '@tencent/mmkv'; - -export default class EntryAbility extends UIAbility { - onCreate(want: Want, launchParam: AbilityConstant.LaunchParam): void { - let appCtx = this.context.getApplicationContext(); - let mmkvRootDir = MMKV.initialize(appCtx); - console.info('mmkv rootDir: ', mmkvRootDir); - …… - } -``` - -MMKV has a global instance, that can be used directly: - -```js -import { MMKV } from '@tencent/mmkv'; - -let mmkv = MMKV.defaultMMKV(); -mmkv.encodeBool('bool', true); -console.info('bool = ', mmkv.decodeBool('bool')); - -mmkv.encodeInt32('int32', Math.pow(2, 31) - 1); -console.info('max int32 = ', mmkv.decodeInt32('int32')); - -mmkv.encodeInt64('int', BigInt(2**63) - BigInt(1)); -console.info('max int64 = ', mmkv.decodeInt64('int')); - -let str: string = 'Hello OpenHarmony from MMKV'; -mmkv.encodeString('string', str); -console.info('string = ', mmkv.decodeString('string')); - -let arrayBuffer: ArrayBuffer = StringToArrayBuffer('Hello OpenHarmony from MMKV with bytes'); -mmkv.encodeBytes('bytes', arrayBuffer); -let bytes = mmkv.decodeBytes('bytes'); -console.info('bytes = ', ArrayBufferToString(bytes)); -``` - -As you can see, MMKV is quite easy to use. + +# MMKV for HarmonyOS NEXT + +## Features + +* **Efficient**. MMKV uses mmap to keep memory synced with file, and protobuf to encode/decode values, making the most of native platform to achieve best performance. + * **Multi-Process concurrency**: MMKV supports concurrent read-read and read-write access between processes. + +* **Easy-to-use**. You can use MMKV as you go. All changes are saved immediately, no `sync`, no `flush` calls needed. + +* **Small**. + * **A handful of files**: MMKV contains process locks, encode/decode helpers and mmap logics and nothing more. It's really tidy. + * **About 600K in binary size**: MMKV adds about 600K per architecture on App size, and much less when zipped (HAR/HAP). + + +## Getting Started +### Installation via OHPM: + +```bash +ohpm install @tencent/mmkv +``` +### Quick Tutorial +You can use MMKV as you go. All changes are saved immediately, no `sync`, no `apply` calls needed. +Setup MMKV on App startup, say your `EntryAbility.onCreate()` function, add these lines: + +```js +import { MMKV } from '@tencent/mmkv'; + +export default class EntryAbility extends UIAbility { + onCreate(want: Want, launchParam: AbilityConstant.LaunchParam): void { + let appCtx = this.context.getApplicationContext(); + let mmkvRootDir = MMKV.initialize(appCtx); + console.info('mmkv rootDir: ', mmkvRootDir); + …… + } +``` + +MMKV has a global instance, that can be used directly: + +```js +import { MMKV } from '@tencent/mmkv'; + +let mmkv = MMKV.defaultMMKV(); +mmkv.encodeBool('bool', true); +console.info('bool = ', mmkv.decodeBool('bool')); + +mmkv.encodeInt32('int32', Math.pow(2, 31) - 1); +console.info('max int32 = ', mmkv.decodeInt32('int32')); + +mmkv.encodeInt64('int', BigInt(2**63) - BigInt(1)); +console.info('max int64 = ', mmkv.decodeInt64('int')); + +let str: string = 'Hello OpenHarmony from MMKV'; +mmkv.encodeString('string', str); +console.info('string = ', mmkv.decodeString('string')); + +let arrayBuffer: ArrayBuffer = StringToArrayBuffer('Hello OpenHarmony from MMKV with bytes'); +mmkv.encodeBytes('bytes', arrayBuffer); +let bytes = mmkv.decodeBytes('bytes'); +console.info('bytes = ', ArrayBufferToString(bytes)); +``` + +As you can see, MMKV is quite easy to use. For the full documentation, see [HarmonyOS NEXT Tutorial](https://github.com/Tencent/MMKV/wiki/ohos_setup). ## License diff --git a/README_CN.md b/README_CN.md index 665cc53b..0fc4a122 100644 --- a/README_CN.md +++ b/README_CN.md @@ -26,6 +26,7 @@ dependencies { // replace "2.0.1" with any available version } ``` +从 v2.0.0 起, MMKV **去掉了 32-bit 架构的支持**、API level 22 及以下的支持, 如有这类需求,请使用 v1.3.x LTS 版本。 更多安装指引参考 [Android Setup](https://github.com/Tencent/MMKV/wiki/android_setup_cn)。 ### 快速上手 @@ -226,14 +227,14 @@ std::cout << "string = " << result << std::endl; ``` MMKV 支持**多进程访问**,更详细的用法参考 [POSIX Tutorial](https://github.com/Tencent/MMKV/wiki/posix_tutorial_cn)。 - + ## HarmonyOS NEXT 指南 ### 安装引入 推荐使用 OHPM: - -```bash -ohpm install @tencent/mmkv -``` + +```bash +ohpm install @tencent/mmkv +``` 更多安装指引参考 [HarmonyOS NEXT Tutorial](https://github.com/Tencent/MMKV/wiki/ohos_setup)。 @@ -241,45 +242,45 @@ ohpm install @tencent/mmkv MMKV 的使用非常简单,所有变更立马生效,无需调用 `save`、`sync`。 在 App 启动时初始化 MMKV,设定 MMKV 的根目录,例如在 `EntryAbility.onCreate()` 里: -```js -import { MMKV } from '@tencent/mmkv'; - -export default class EntryAbility extends UIAbility { - onCreate(want: Want, launchParam: AbilityConstant.LaunchParam): void { - let appCtx = this.context.getApplicationContext(); - let mmkvRootDir = MMKV.initialize(appCtx); - console.info('mmkv rootDir: ', mmkvRootDir); - …… - } -``` +```js +import { MMKV } from '@tencent/mmkv'; + +export default class EntryAbility extends UIAbility { + onCreate(want: Want, launchParam: AbilityConstant.LaunchParam): void { + let appCtx = this.context.getApplicationContext(); + let mmkvRootDir = MMKV.initialize(appCtx); + console.info('mmkv rootDir: ', mmkvRootDir); + …… + } +``` MMKV 提供一个全局的实例,可以直接使用: -```js -import { MMKV } from '@tencent/mmkv'; - -let mmkv = MMKV.defaultMMKV(); -mmkv.encodeBool('bool', true); -console.info('bool = ', mmkv.decodeBool('bool')); - -mmkv.encodeInt32('int32', Math.pow(2, 31) - 1); -console.info('max int32 = ', mmkv.decodeInt32('int32')); - -mmkv.encodeInt64('int', BigInt(2**63) - BigInt(1)); -console.info('max int64 = ', mmkv.decodeInt64('int')); - -let str: string = 'Hello OpenHarmony from MMKV'; -mmkv.encodeString('string', str); -console.info('string = ', mmkv.decodeString('string')); - -let arrayBuffer: ArrayBuffer = StringToArrayBuffer('Hello OpenHarmony from MMKV with bytes'); -mmkv.encodeBytes('bytes', arrayBuffer); -let bytes = mmkv.decodeBytes('bytes'); -console.info('bytes = ', ArrayBufferToString(bytes)); +```js +import { MMKV } from '@tencent/mmkv'; + +let mmkv = MMKV.defaultMMKV(); +mmkv.encodeBool('bool', true); +console.info('bool = ', mmkv.decodeBool('bool')); + +mmkv.encodeInt32('int32', Math.pow(2, 31) - 1); +console.info('max int32 = ', mmkv.decodeInt32('int32')); + +mmkv.encodeInt64('int', BigInt(2**63) - BigInt(1)); +console.info('max int64 = ', mmkv.decodeInt64('int')); + +let str: string = 'Hello OpenHarmony from MMKV'; +mmkv.encodeString('string', str); +console.info('string = ', mmkv.decodeString('string')); + +let arrayBuffer: ArrayBuffer = StringToArrayBuffer('Hello OpenHarmony from MMKV with bytes'); +mmkv.encodeBytes('bytes', arrayBuffer); +let bytes = mmkv.decodeBytes('bytes'); +console.info('bytes = ', ArrayBufferToString(bytes)); ``` MMKV 更详细的用法参考 [HarmonyOS NEXT Tutorial](https://github.com/Tencent/MMKV/wiki/ohos_setup)。 - + ## License MMKV 以 BSD 3-Clause 证书开源,详情参见 [LICENSE.TXT](./LICENSE.TXT)。 From 3b8c11d944a7baea6b7a0934832b81c438eaf83d Mon Sep 17 00:00:00 2001 From: guoling Date: Tue, 17 Dec 2024 17:45:27 +0800 Subject: [PATCH 10/12] fix merge error --- Core/MMKV.h | 10 +++++----- Core/MMKV_Android.cpp | 4 ++-- 2 files changed, 7 insertions(+), 7 deletions(-) diff --git a/Core/MMKV.h b/Core/MMKV.h index 16e9d836..f5f2db3b 100644 --- a/Core/MMKV.h +++ b/Core/MMKV.h @@ -102,9 +102,9 @@ class MMKV { mmkv::InterProcessLock *m_sharedProcessModeLock; mmkv::InterProcessLock *m_exclusiveProcessModeLock; - MMKV(const std::string &mmapID, int size, MMKVMode mode, std::string *cryptKey, MMKVPath_t *rootPath, size_t expectedCapacity = 0); + MMKV(const std::string &mmapID, int size, MMKVMode mode, const std::string *cryptKey, const MMKVPath_t *rootPath, size_t expectedCapacity = 0); - MMKV(const std::string &mmapID, int ashmemFD, int ashmemMetaFd, std::string *cryptKey = nullptr); + MMKV(const std::string &mmapID, int ashmemFD, int ashmemMetaFd, const std::string *cryptKey = nullptr); #endif ~MMKV(); @@ -270,11 +270,11 @@ class MMKV { static MMKV *mmkvWithID(const std::string &mmapID, int size = mmkv::DEFAULT_MMAP_SIZE, MMKVMode mode = MMKV_SINGLE_PROCESS, - std::string *cryptKey = nullptr, - MMKVPath_t *rootPath = nullptr, + const std::string *cryptKey = nullptr, + const MMKVPath_t *rootPath = nullptr, size_t expectedCapacity = 0); - static MMKV *mmkvWithAshmemFD(const std::string &mmapID, int fd, int metaFD, std::string *cryptKey = nullptr); + static MMKV *mmkvWithAshmemFD(const std::string &mmapID, int fd, int metaFD, const std::string *cryptKey = nullptr); int ashmemFD(); diff --git a/Core/MMKV_Android.cpp b/Core/MMKV_Android.cpp index 22028145..5dd2601b 100644 --- a/Core/MMKV_Android.cpp +++ b/Core/MMKV_Android.cpp @@ -43,7 +43,7 @@ MMKV::MMKV(const string &mmapID, int size, MMKVMode mode, const string *cryptKey : m_mmapID((mode & MMKV_BACKUP) ? mmapID : mmapedKVKey(mmapID, rootPath)) // historically Android mistakenly use mmapKey as mmapID , m_mode(mode) , m_path(mappedKVPathWithID(m_mmapID, mode, rootPath)) - , m_crcPath(crcPathWithPath(m_path)) + , m_crcPath(crcPathWithID(m_mmapID, mode, rootPath)) , m_dic(nullptr) , m_dicCrypt(nullptr) , m_expectedCapacity(std::max(DEFAULT_MMAP_SIZE, roundUp(expectedCapacity, DEFAULT_MMAP_SIZE))) @@ -92,7 +92,7 @@ MMKV::MMKV(const string &mmapID, int ashmemFD, int ashmemMetaFD, const string *c : m_mmapID(mmapID) , m_mode(MMKV_ASHMEM) , m_path(mappedKVPathWithID(m_mmapID, MMKV_ASHMEM, nullptr)) - , m_crcPath(crcPathWithPath(m_path)) + , m_crcPath(crcPathWithID(m_mmapID, MMKV_ASHMEM, nullptr)) , m_dic(nullptr) , m_dicCrypt(nullptr) , m_file(new MemoryFile(ashmemFD)) From 0bd466efe34bf27e1c0aa1c882703a5163265c19 Mon Sep 17 00:00:00 2001 From: guoling Date: Tue, 24 Dec 2024 19:29:07 +0800 Subject: [PATCH 11/12] support correct filename on custom root path --- Core/MMKV.cpp | 24 ++++++++++++++++-------- Core/MMKV.h | 2 +- Core/MMKV_Android.cpp | 16 ++++++++++++++-- Core/MMKV_IO.cpp | 10 +++++++++- 4 files changed, 40 insertions(+), 12 deletions(-) diff --git a/Core/MMKV.cpp b/Core/MMKV.cpp index ccc730f4..62e0b525 100644 --- a/Core/MMKV.cpp +++ b/Core/MMKV.cpp @@ -343,11 +343,7 @@ void MMKV::close() { SCOPED_LOCK(g_instanceLock); m_lock->lock(); -#ifndef MMKV_ANDROID auto itr = g_instanceDic->find(m_mmapKey); -#else - auto itr = g_instanceDic->find(m_mmapID); -#endif if (itr != g_instanceDic->end()) { g_instanceDic->erase(itr); } @@ -1308,8 +1304,14 @@ bool MMKV::backupOneToDirectory(const string &mmapID, const MMKVPath_t &dstDir, auto dstPath = dstDir + MMKV_PATH_SLASH + encodePath; auto mmapKey = mmapedKVKey(mmapID, rootPath); #ifdef MMKV_ANDROID - // historically Android mistakenly use mmapKey as mmapID - auto srcPath = *rootPath + MMKV_PATH_SLASH + encodeFilePath(mmapKey, *rootPath); + string srcPath; + auto correctPath = *rootPath + MMKV_PATH_SLASH + encodePath; + if (srcDir && isFileExist(correctPath)) { + srcPath = correctPath; + } else { + // historically Android mistakenly use mmapKey as mmapID + srcPath = *rootPath + MMKV_PATH_SLASH + encodeFilePath(mmapKey, *rootPath); + } #else auto srcPath = *rootPath + MMKV_PATH_SLASH + encodePath; #endif @@ -1491,8 +1493,14 @@ bool MMKV::restoreOneFromDirectory(const string &mmapID, const MMKVPath_t &srcDi auto srcPath = srcDir + MMKV_PATH_SLASH + encodePath; auto mmapKey = mmapedKVKey(mmapID, rootPath); #ifdef MMKV_ANDROID - // historically Android mistakenly use mmapKey as mmapID - auto dstPath = *rootPath + MMKV_PATH_SLASH + encodeFilePath(mmapKey, *rootPath); + string dstPath; + auto correctPath = *rootPath + MMKV_PATH_SLASH + encodePath; + if (dstDir && isFileExist(correctPath)) { + dstPath = correctPath; + } else { + // historically Android mistakenly use mmapKey as mmapID + dstPath = *rootPath + MMKV_PATH_SLASH + encodeFilePath(mmapKey, *rootPath); + } #else auto dstPath = *rootPath + MMKV_PATH_SLASH + encodePath; #endif diff --git a/Core/MMKV.h b/Core/MMKV.h index f5f2db3b..68ac866e 100644 --- a/Core/MMKV.h +++ b/Core/MMKV.h @@ -95,7 +95,6 @@ concept MMKV_SUPPORTED_VALUE_TYPE = MMKV_SUPPORTED_PRIMITIVE_VALUE_TYPE || MM class MMKV { #ifndef MMKV_ANDROID - std::string m_mmapKey; MMKV(const std::string &mmapID, MMKVMode mode, std::string *cryptKey, MMKVPath_t *rootPath, size_t expectedCapacity = 0); #else // defined(MMKV_ANDROID) mmkv::FileLock *m_fileModeLock; @@ -109,6 +108,7 @@ class MMKV { ~MMKV(); + std::string m_mmapKey; std::string m_mmapID; const MMKVMode m_mode; MMKVPath_t m_path; diff --git a/Core/MMKV_Android.cpp b/Core/MMKV_Android.cpp index 5dd2601b..905d86ec 100644 --- a/Core/MMKV_Android.cpp +++ b/Core/MMKV_Android.cpp @@ -40,7 +40,7 @@ extern unordered_map *g_instanceDic; extern ThreadLock *g_instanceLock; MMKV::MMKV(const string &mmapID, int size, MMKVMode mode, const string *cryptKey, const string *rootPath, size_t expectedCapacity) - : m_mmapID((mode & MMKV_BACKUP) ? mmapID : mmapedKVKey(mmapID, rootPath)) // historically Android mistakenly use mmapKey as mmapID + : m_mmapID(mmapID) , m_mode(mode) , m_path(mappedKVPathWithID(m_mmapID, mode, rootPath)) , m_crcPath(crcPathWithID(m_mmapID, mode, rootPath)) @@ -157,7 +157,18 @@ MMKV *MMKV::mmkvWithID(const string &mmapID, int size, MMKVMode mode, const stri } MMKVInfo("prepare to load %s (id %s) from rootPath %s", mmapID.c_str(), mmapKey.c_str(), rootPath->c_str()); } - auto kv = new MMKV(mmapID, size, mode, cryptKey, rootPath, expectedCapacity); + + string realID; + auto correctPath = mappedKVPathWithID(mmapID, mode, rootPath); + if ((mode & MMKV_BACKUP) || (rootPath && isFileExist(correctPath))) { + // it's successfully migrated to the correct path by newer version of MMKV + realID = mmapID; + } else { + // historically Android mistakenly use mmapKey as mmapID + realID = mmapKey; + } + auto kv = new MMKV(realID, size, mode, cryptKey, rootPath, expectedCapacity); + kv->m_mmapKey = mmapKey; (*g_instanceDic)[mmapKey] = kv; return kv; } @@ -178,6 +189,7 @@ MMKV *MMKV::mmkvWithAshmemFD(const string &mmapID, int fd, int metaFD, const str return kv; } auto kv = new MMKV(mmapID, fd, metaFD, cryptKey); + kv->m_mmapKey = mmapID; (*g_instanceDic)[mmapID] = kv; return kv; } diff --git a/Core/MMKV_IO.cpp b/Core/MMKV_IO.cpp index 54fa9594..d347624a 100755 --- a/Core/MMKV_IO.cpp +++ b/Core/MMKV_IO.cpp @@ -1549,7 +1549,15 @@ bool MMKV::removeStorage(const std::string &mmapID, MMKVPath_t *relatePath) { } auto mmapKey = mmapedKVKey(mmapID, relatePath); #ifdef MMKV_ANDROID - auto &realID = mmapKey; // historically Android mistakenly use mmapKey as mmapID + std::string realID; + auto correctPath = mappedKVPathWithID(mmapID, MMKV_SINGLE_PROCESS, relatePath); + if (relatePath && isFileExist(correctPath)) { + // it's successfully migrated to the correct path by newer version of MMKV + realID = mmapID; + } else { + // historically Android mistakenly use mmapKey as mmapID + realID = mmapKey; + } #else auto &realID = mmapID; #endif From 2323ee49be29d96c31b1fde9409041daf2fe45d8 Mon Sep 17 00:00:00 2001 From: guoling Date: Fri, 27 Dec 2024 17:07:59 +0800 Subject: [PATCH 12/12] prepare for v2.0.2 --- Android/MMKV/gradle.properties | 2 +- Android/MMKV/mmkvdemo/build.gradle | 6 +++--- CHANGELOG.md | 18 ++++++++++++++++++ Core/MMKVPredef.h | 2 +- OpenHarmony/MMKV/CHANGELOG.md | 5 +++++ OpenHarmony/MMKV/README.md | 4 ++-- OpenHarmony/MMKV/oh-package.json5 | 2 +- OpenHarmony/entry/oh-package-lock.json5 | 16 +++++----------- OpenHarmony/entry/oh-package.json5 | 4 ++-- README.md | 6 +++--- README_CN.md | 4 ++-- flutter/mmkv/CHANGELOG.md | 3 +++ flutter/mmkv/README.md | 4 ++-- flutter/mmkv/pubspec.yaml | 10 +++++----- flutter/mmkv_ohos/CHANGELOG.md | 3 +++ flutter/mmkv_ohos/ohos/oh-package.json5 | 2 +- flutter/mmkv_ohos/pubspec.yaml | 2 +- iOS/MMKV.podspec | 4 ++-- iOS/MMKV/MMKV.xcodeproj/project.pbxproj | 6 ------ iOS/MMKVAppExtension.podspec | 4 ++-- iOS/MMKVCore.podspec | 2 +- iOS/MMKVWatchExtension.podspec | 4 ++-- 22 files changed, 65 insertions(+), 48 deletions(-) diff --git a/Android/MMKV/gradle.properties b/Android/MMKV/gradle.properties index 30da076e..3238487e 100644 --- a/Android/MMKV/gradle.properties +++ b/Android/MMKV/gradle.properties @@ -14,7 +14,7 @@ org.gradle.jvmargs=-Xmx1536m # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects # org.gradle.parallel=true -VERSION_NAME_PREFIX=2.0.1 +VERSION_NAME_PREFIX=2.0.2 #VERSION_NAME_SUFFIX=-SNAPSHOT VERSION_NAME_SUFFIX= diff --git a/Android/MMKV/mmkvdemo/build.gradle b/Android/MMKV/mmkvdemo/build.gradle index 4df3aa16..77c93af6 100644 --- a/Android/MMKV/mmkvdemo/build.gradle +++ b/Android/MMKV/mmkvdemo/build.gradle @@ -77,9 +77,9 @@ repositories { dependencies { implementation fileTree(include: ['*.jar'], dir: 'libs') implementation project(':mmkv') -// implementation 'com.tencent:mmkv:2.0.1' -// implementation 'com.tencent:mmkv-static:2.0.1' // this is identical to 'com.tencent:mmkv' -// implementation 'com.tencent:mmkv-shared:2.0.1' +// implementation 'com.tencent:mmkv:2.0.2' +// implementation 'com.tencent:mmkv-static:2.0.2' // this is identical to 'com.tencent:mmkv' +// implementation 'com.tencent:mmkv-shared:2.0.2' implementation 'androidx.appcompat:appcompat:1.6.1' implementation 'androidx.constraintlayout:constraintlayout:2.1.4' testImplementation 'junit:junit:4.13.2' diff --git a/CHANGELOG.md b/CHANGELOG.md index cd89fb83..2df3cee8 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,4 +1,22 @@ # MMKV Change Log +## v2.0.2 / 2024-12-27 +Mary holiday and a happy new year! +### Changes for All platforms +* Fix a bug that MMKV might fail to backup/restore across different filesystems. +* Add protection from invalid value size of auto-key-expire mmkv. + +### Android +* If the running App is 32-bit only, warn about it (by throwing `UnsupportedArchitectureException`) before trying to load native lib. +* Add forward support for the correct filename with a custom root path. + +### HarmonyOS NEXT +* Obfuscation fully supported. +* Use atomic file rename on OHOS. +* Add forward support for the correct filename with a custom root path. + +### Win32 +* Only `mmap()` on `ftruncate()`/`zeroFillFile()` failure iff we have a valid file mapping before. + ## v2.0.1 / 2024-11-12 **This is a hotfix release.** ### Changes for All platforms diff --git a/Core/MMKVPredef.h b/Core/MMKVPredef.h index e9b0acf1..ab9cd95e 100755 --- a/Core/MMKVPredef.h +++ b/Core/MMKVPredef.h @@ -34,7 +34,7 @@ #include #include -constexpr auto MMKV_VERSION = "v2.0.1"; +constexpr auto MMKV_VERSION = "v2.0.2"; #ifdef DEBUG # define MMKV_DEBUG diff --git a/OpenHarmony/MMKV/CHANGELOG.md b/OpenHarmony/MMKV/CHANGELOG.md index 3e6080e8..7415c634 100644 --- a/OpenHarmony/MMKV/CHANGELOG.md +++ b/OpenHarmony/MMKV/CHANGELOG.md @@ -1,4 +1,9 @@ # MMKV for HarmonyOS NEXT Change Log +## v2.0.2 / 2024-12-27 +* Obfuscation fully supported. +* Use atomic file rename on OHOS. +* Add forward support for the correct filename with a custom root path. + ## v2.0.1 / 2024-11-12 * Fix a bug that MMKV might become dead-locked for other threads after `decodeStringSet()` / `decodeNumberSet` / `decodeBoolSet` or decoding `TypedArray`. diff --git a/OpenHarmony/MMKV/README.md b/OpenHarmony/MMKV/README.md index 3d0f7b8b..a80202a0 100644 --- a/OpenHarmony/MMKV/README.md +++ b/OpenHarmony/MMKV/README.md @@ -1,6 +1,6 @@ [![license](https://img.shields.io/badge/license-BSD_3-brightgreen.svg?style=flat)](https://github.com/Tencent/MMKV/blob/master/LICENSE.TXT) [![PRs Welcome](https://img.shields.io/badge/PRs-welcome-brightgreen.svg)](https://github.com/Tencent/MMKV/pulls) -[![Release Version](https://img.shields.io/badge/release-2.0.1-brightgreen.svg)](https://github.com/Tencent/MMKV/releases) +[![Release Version](https://img.shields.io/badge/release-2.0.2-brightgreen.svg)](https://github.com/Tencent/MMKV/releases) [![Platform](https://img.shields.io/badge/Platform-%20HarmonyOS%20NEXT-brightgreen.svg)](https://github.com/Tencent/MMKV/wiki/home) MMKV is an **efficient**, **small**, **easy-to-use** mobile key-value storage framework used in the WeChat application. It's now available on **HarmonyOS NEXT**. @@ -37,7 +37,7 @@ Or, you can add it to your project manually. ```json "dependencies": { - "@tencent/mmkv": "^2.0.1", + "@tencent/mmkv": "~2.0.2", } ``` * Then run diff --git a/OpenHarmony/MMKV/oh-package.json5 b/OpenHarmony/MMKV/oh-package.json5 index 3e8fcb15..25dd6379 100644 --- a/OpenHarmony/MMKV/oh-package.json5 +++ b/OpenHarmony/MMKV/oh-package.json5 @@ -1,6 +1,6 @@ { "name": "@tencent/mmkv", - "version": "2.0.1", + "version": "2.0.2", "description": "The official OpenHarmony package of MMKV. An efficient, small mobile key-value storage framework developed by WeChat.", "main": "Index.ets", "author": "guoling", diff --git a/OpenHarmony/entry/oh-package-lock.json5 b/OpenHarmony/entry/oh-package-lock.json5 index 2f2ec6bb..4e3f869a 100644 --- a/OpenHarmony/entry/oh-package-lock.json5 +++ b/OpenHarmony/entry/oh-package-lock.json5 @@ -5,25 +5,19 @@ "lockfileVersion": 3, "ATTENTION": "THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.", "specifiers": { - "@tencent/mmkv@../MMKV": "@tencent/mmkv@../MMKV", - "libmmkv.so@../MMKV/src/main/cpp/types/libmmkv": "libmmkv.so@../MMKV/src/main/cpp/types/libmmkv" + "@tencent/mmkv@~2.0.0": "@tencent/mmkv@2.0.1" }, "packages": { - "@tencent/mmkv@../MMKV": { + "@tencent/mmkv@2.0.1": { "name": "@tencent/mmkv", "version": "2.0.1", - "resolved": "../MMKV", - "registryType": "local", + "integrity": "sha512-UlIzMArUoQaEFASFxLSIg0Q/LHY1K3mqv/XdCXdgzz9Pb1VS7AS2o5LwUrkY/O8iuT8Rr1SxGmp98xKRE4e8yg==", + "resolved": "https://ohpm.openharmony.cn/ohpm/@tencent/mmkv/-/mmkv-2.0.1.har", + "registryType": "ohpm", "dependencies": { "libmmkv.so": "file:./src/main/cpp/types/libmmkv" }, "packageType": "InterfaceHar" - }, - "libmmkv.so@../MMKV/src/main/cpp/types/libmmkv": { - "name": "libmmkv.so", - "version": "0.0.0", - "resolved": "../MMKV/src/main/cpp/types/libmmkv", - "registryType": "local" } } } \ No newline at end of file diff --git a/OpenHarmony/entry/oh-package.json5 b/OpenHarmony/entry/oh-package.json5 index b373db08..136d433f 100644 --- a/OpenHarmony/entry/oh-package.json5 +++ b/OpenHarmony/entry/oh-package.json5 @@ -6,8 +6,8 @@ "author": "guoling", "license": "BSD 3-Clause", "dependencies": { - "@tencent/mmkv": "file:../MMKV" -// "@tencent/mmkv": "2.0.0", + "@tencent/mmkv": "file:../MMKV", +// "@tencent/mmkv": "~2.0.2", } } diff --git a/README.md b/README.md index b746ab91..5e8401bf 100644 --- a/README.md +++ b/README.md @@ -1,6 +1,6 @@ [![license](https://img.shields.io/badge/license-BSD_3-brightgreen.svg?style=flat)](https://github.com/Tencent/MMKV/blob/master/LICENSE.TXT) [![PRs Welcome](https://img.shields.io/badge/PRs-welcome-brightgreen.svg)](https://github.com/Tencent/MMKV/pulls) -[![Release Version](https://img.shields.io/badge/release-2.0.1-brightgreen.svg)](https://github.com/Tencent/MMKV/releases) +[![Release Version](https://img.shields.io/badge/release-2.0.2-brightgreen.svg)](https://github.com/Tencent/MMKV/releases) [![Platform](https://img.shields.io/badge/Platform-%20Android%20%7C%20iOS%2FmacOS%20%7C%20Windows%20%7C%20POSIX%20%7C%20HarmonyOS%20NEXT-brightgreen.svg)](https://github.com/Tencent/MMKV/wiki/home) 中文版本请参看[这里](./README_CN.md) @@ -28,8 +28,8 @@ Add the following lines to `build.gradle` on your app module: ```gradle dependencies { - implementation 'com.tencent:mmkv:2.0.1' - // replace "2.0.1" with any available version + implementation 'com.tencent:mmkv:2.0.2' + // replace "2.0.2" with any available version } ``` diff --git a/README_CN.md b/README_CN.md index 0fc4a122..1ec6890e 100644 --- a/README_CN.md +++ b/README_CN.md @@ -22,8 +22,8 @@ MMKV 是基于 mmap 内存映射的 key-value 组件,底层序列化/反序列 ```gradle dependencies { - implementation 'com.tencent:mmkv:2.0.1' - // replace "2.0.1" with any available version + implementation 'com.tencent:mmkv:2.0.2' + // replace "2.0.2" with any available version } ``` 从 v2.0.0 起, MMKV **去掉了 32-bit 架构的支持**、API level 22 及以下的支持, 如有这类需求,请使用 v1.3.x LTS 版本。 diff --git a/flutter/mmkv/CHANGELOG.md b/flutter/mmkv/CHANGELOG.md index 78b8f5b3..ba919e32 100644 --- a/flutter/mmkv/CHANGELOG.md +++ b/flutter/mmkv/CHANGELOG.md @@ -1,4 +1,7 @@ # MMKV for Flutter Change Log +## v2.0.2 / 2024-12-27 +* Add version limitation of < v2.1.0 on MMKV platform plugins. + ## v2.0.1 / 2024-10-25 * Fix breaking changes on platform interface package. diff --git a/flutter/mmkv/README.md b/flutter/mmkv/README.md index 3d1414b3..5d93157c 100644 --- a/flutter/mmkv/README.md +++ b/flutter/mmkv/README.md @@ -1,6 +1,6 @@ [![license](https://img.shields.io/badge/license-BSD_3-brightgreen.svg?style=flat)](https://github.com/Tencent/MMKV/blob/master/LICENSE.TXT) [![PRs Welcome](https://img.shields.io/badge/PRs-welcome-brightgreen.svg)](https://github.com/Tencent/MMKV/pulls) -[![Release Version](https://img.shields.io/badge/release-2.0.1-brightgreen.svg)](https://github.com/Tencent/MMKV/releases) +[![Release Version](https://img.shields.io/badge/release-2.0.2-brightgreen.svg)](https://github.com/Tencent/MMKV/releases) [![Platform](https://img.shields.io/badge/Platform-%20Android%20%7C%20iOS-brightgreen.svg)](https://github.com/Tencent/MMKV/wiki/home) MMKV is an **efficient**, **small**, **easy-to-use** mobile key-value storage framework used in the WeChat application. It's currently available on **Android** and **iOS**. @@ -26,7 +26,7 @@ Add the following lines to `pubspec.yaml` on your app module. Then run `flutter ```yaml dependencies: - mmkv: "^2.0.1" + mmkv: "^2.0.2" ``` If you already include MMKV native lib in your App, you need to upgrade to version newer than v2.0.0. diff --git a/flutter/mmkv/pubspec.yaml b/flutter/mmkv/pubspec.yaml index 2575dd14..cd7e8642 100644 --- a/flutter/mmkv/pubspec.yaml +++ b/flutter/mmkv/pubspec.yaml @@ -1,6 +1,6 @@ name: mmkv description: An efficient, small mobile key-value storage framework developed by WeChat. Works on Android & iOS. -version: 2.0.1 +version: 2.0.2 homepage: https://github.com/Tencent/mmkv repository: https://github.com/Tencent/MMKV/tree/master/flutter/mmkv @@ -13,16 +13,16 @@ dependencies: sdk: flutter ffi: ^2.0.0 mmkv_ios: - ^2.0.0 + '>=2.0.0 <2.1.0' # path: ../mmkv_ios mmkv_android: - ^2.0.0 + '>=2.0.0 <2.1.0' # path: ../mmkv_android mmkv_ohos: - ^2.0.0 + '>=2.0.2 <2.1.0' # path: ../mmkv_ohos mmkv_platform_interface: - ^2.0.0 + '>=2.0.0 <2.1.0' # path: ../mmkv_platform_interface dev_dependencies: diff --git a/flutter/mmkv_ohos/CHANGELOG.md b/flutter/mmkv_ohos/CHANGELOG.md index c7db7dbb..8ba69ac8 100644 --- a/flutter/mmkv_ohos/CHANGELOG.md +++ b/flutter/mmkv_ohos/CHANGELOG.md @@ -1,4 +1,7 @@ # MMKV Platform OHOS Change Log +## v2.0.2 / 2024-12-27 +Keep up with native lib v2.0.2. And add limitation of < v2.1. + ## v2.0.1 / 2024-10-25 Keep up with native lib v2.0.1. diff --git a/flutter/mmkv_ohos/ohos/oh-package.json5 b/flutter/mmkv_ohos/ohos/oh-package.json5 index 601aa441..343b4606 100644 --- a/flutter/mmkv_ohos/ohos/oh-package.json5 +++ b/flutter/mmkv_ohos/ohos/oh-package.json5 @@ -7,7 +7,7 @@ "license": "Apache-2.0", "dependencies": { "@ohos/flutter_ohos": "file:./har/flutter.har", - "@tencent/mmkv": "^2.0.1" + "@tencent/mmkv": ">=2.0.2 <2.1" // "@tencent/mmkv": "file:/Users/lingol/Developer/mmkv/OpenHarmony/MMKV/build/default/outputs/default/MMKV.har" }, "devDependencies": {}, diff --git a/flutter/mmkv_ohos/pubspec.yaml b/flutter/mmkv_ohos/pubspec.yaml index 8a2ac82f..479cdbbf 100644 --- a/flutter/mmkv_ohos/pubspec.yaml +++ b/flutter/mmkv_ohos/pubspec.yaml @@ -1,7 +1,7 @@ name: mmkv_ohos description: OHOS platform implementation of MMKV. repository: https://github.com/Tencent/MMKV/tree/master/flutter/mmkv_ohos -version: 2.0.1 +version: 2.0.2 homepage: https://github.com/Tencent/mmkv environment: diff --git a/iOS/MMKV.podspec b/iOS/MMKV.podspec index b8807307..c70e44d6 100644 --- a/iOS/MMKV.podspec +++ b/iOS/MMKV.podspec @@ -1,7 +1,7 @@ Pod::Spec.new do |s| s.name = "MMKV" - s.version = "2.0.0" + s.version = "2.0.2" s.summary = "MMKV is a cross-platform key-value storage framework developed by WeChat." s.description = <<-DESC @@ -33,7 +33,7 @@ Pod::Spec.new do |s| "CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF" => "NO", } - s.dependency 'MMKVCore', '~> 2.0.0' + s.dependency 'MMKVCore', '~> 2.0.2' end diff --git a/iOS/MMKV/MMKV.xcodeproj/project.pbxproj b/iOS/MMKV/MMKV.xcodeproj/project.pbxproj index c9596622..6a54d113 100644 --- a/iOS/MMKV/MMKV.xcodeproj/project.pbxproj +++ b/iOS/MMKV/MMKV.xcodeproj/project.pbxproj @@ -493,7 +493,6 @@ "@loader_path/Frameworks", ); MACOSX_DEPLOYMENT_TARGET = "$(RECOMMENDED_MACOSX_DEPLOYMENT_TARGET)"; - MARKETING_VERSION = 2.0.0; MODULE_VERIFIER_SUPPORTED_LANGUAGE_STANDARDS = "gnu11 gnu++17"; "OTHER_LDFLAGS[sdk=iphoneos*]" = ( "-framework", @@ -539,7 +538,6 @@ "@loader_path/Frameworks", ); MACOSX_DEPLOYMENT_TARGET = "$(RECOMMENDED_MACOSX_DEPLOYMENT_TARGET)"; - MARKETING_VERSION = 2.0.0; MODULE_VERIFIER_SUPPORTED_LANGUAGE_STANDARDS = "gnu11 gnu++17"; "OTHER_LDFLAGS[sdk=iphoneos*]" = ( "-framework", @@ -749,7 +747,6 @@ "@loader_path/Frameworks", ); MACOSX_DEPLOYMENT_TARGET = "$(RECOMMENDED_MACOSX_DEPLOYMENT_TARGET)"; - MARKETING_VERSION = 2.0.0; MODULE_VERIFIER_SUPPORTED_LANGUAGE_STANDARDS = "gnu11 gnu++17"; "OTHER_LDFLAGS[sdk=iphoneos*]" = ( "-framework", @@ -794,7 +791,6 @@ "@loader_path/Frameworks", ); MACOSX_DEPLOYMENT_TARGET = "$(RECOMMENDED_MACOSX_DEPLOYMENT_TARGET)"; - MARKETING_VERSION = 2.0.0; MODULE_VERIFIER_SUPPORTED_LANGUAGE_STANDARDS = "gnu11 gnu++17"; "OTHER_LDFLAGS[sdk=iphoneos*]" = ( "-framework", @@ -843,7 +839,6 @@ "@loader_path/Frameworks", ); MACOSX_DEPLOYMENT_TARGET = "$(RECOMMENDED_MACOSX_DEPLOYMENT_TARGET)"; - MARKETING_VERSION = 2.0.0; MODULE_VERIFIER_SUPPORTED_LANGUAGE_STANDARDS = "gnu11 gnu++17"; "OTHER_LDFLAGS[sdk=iphoneos*]" = ( "-framework", @@ -891,7 +886,6 @@ "@loader_path/Frameworks", ); MACOSX_DEPLOYMENT_TARGET = "$(RECOMMENDED_MACOSX_DEPLOYMENT_TARGET)"; - MARKETING_VERSION = 2.0.0; MODULE_VERIFIER_SUPPORTED_LANGUAGE_STANDARDS = "gnu11 gnu++17"; "OTHER_LDFLAGS[sdk=iphoneos*]" = ( "-framework", diff --git a/iOS/MMKVAppExtension.podspec b/iOS/MMKVAppExtension.podspec index 33040d09..bacd23ce 100644 --- a/iOS/MMKVAppExtension.podspec +++ b/iOS/MMKVAppExtension.podspec @@ -1,7 +1,7 @@ Pod::Spec.new do |s| s.name = "MMKVAppExtension" - s.version = "2.0.0" + s.version = "2.0.2" s.summary = "MMKV is a cross-platform key-value storage framework developed by WeChat." s.module_name = "MMKVAppExtension" @@ -31,7 +31,7 @@ Pod::Spec.new do |s| "GCC_PREPROCESSOR_DEFINITIONS" => "MMKV_IOS_EXTENSION", } - s.dependency 'MMKVCore', '~> 2.0.0' + s.dependency 'MMKVCore', '~> 2.0.2' end diff --git a/iOS/MMKVCore.podspec b/iOS/MMKVCore.podspec index ac778004..4b7910c0 100644 --- a/iOS/MMKVCore.podspec +++ b/iOS/MMKVCore.podspec @@ -1,7 +1,7 @@ Pod::Spec.new do |s| s.name = "MMKVCore" - s.version = "2.0.0" + s.version = "2.0.2" s.summary = "MMKVCore for MMKV. MMKV is a cross-platform key-value storage framework developed by WeChat." s.description = <<-DESC diff --git a/iOS/MMKVWatchExtension.podspec b/iOS/MMKVWatchExtension.podspec index e8b02c2a..b2e177e1 100644 --- a/iOS/MMKVWatchExtension.podspec +++ b/iOS/MMKVWatchExtension.podspec @@ -1,7 +1,7 @@ Pod::Spec.new do |s| s.name = "MMKVWatchExtension" - s.version = "2.0.0" + s.version = "2.0.2" s.summary = "MMKV is a cross-platform key-value storage framework developed by WeChat." s.module_name = "MMKVWatchExtension" @@ -31,7 +31,7 @@ Pod::Spec.new do |s| "GCC_PREPROCESSOR_DEFINITIONS" => "MMKV_IOS_EXTENSION", } - s.dependency 'MMKVCore', '~> 2.0.0' + s.dependency 'MMKVCore', '~> 2.0.2' end