android : introduce starter project example (#4926)
* Introduce starter project for Android Based on examples/llama.swiftui. * Add github workflow * Set NDK version * Only build arm64-v8a in CI * Sync bench code * Rename CI prop to skip-armeabi-v7a * Remove unused tests
							
								
								
									
										30
									
								
								examples/llama.android/app/src/main/AndroidManifest.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,30 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <manifest xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:tools="http://schemas.android.com/tools"> | ||||
| 
 | ||||
|     <uses-permission android:name="android.permission.INTERNET" /> | ||||
| 
 | ||||
|     <application | ||||
|         android:allowBackup="true" | ||||
|         android:dataExtractionRules="@xml/data_extraction_rules" | ||||
|         android:fullBackupContent="@xml/backup_rules" | ||||
|         android:icon="@mipmap/ic_launcher" | ||||
|         android:label="@string/app_name" | ||||
|         android:roundIcon="@mipmap/ic_launcher_round" | ||||
|         android:supportsRtl="true" | ||||
|         android:theme="@style/Theme.LlamaAndroid" | ||||
|         > | ||||
| 
 | ||||
|         <activity | ||||
|             android:name=".MainActivity" | ||||
|             android:exported="true" | ||||
|             android:theme="@style/Theme.LlamaAndroid"> | ||||
|             <intent-filter> | ||||
|                 <action android:name="android.intent.action.MAIN" /> | ||||
| 
 | ||||
|                 <category android:name="android.intent.category.LAUNCHER" /> | ||||
|             </intent-filter> | ||||
|         </activity> | ||||
|     </application> | ||||
| 
 | ||||
| </manifest> | ||||
							
								
								
									
										50
									
								
								examples/llama.android/app/src/main/cpp/CMakeLists.txt
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,50 @@ | |||
| 
 | ||||
| # For more information about using CMake with Android Studio, read the | ||||
| # documentation: https://d.android.com/studio/projects/add-native-code.html. | ||||
| # For more examples on how to use CMake, see https://github.com/android/ndk-samples. | ||||
| 
 | ||||
| # Sets the minimum CMake version required for this project. | ||||
| cmake_minimum_required(VERSION 3.22.1) | ||||
| 
 | ||||
| # Declares the project name. The project name can be accessed via ${ PROJECT_NAME}, | ||||
| # Since this is the top level CMakeLists.txt, the project name is also accessible | ||||
| # with ${CMAKE_PROJECT_NAME} (both CMake variables are in-sync within the top level | ||||
| # build script scope). | ||||
| project("llama-android") | ||||
| 
 | ||||
| include(FetchContent) | ||||
| FetchContent_Declare( | ||||
|         llama | ||||
|         GIT_REPOSITORY https://github.com/ggerganov/llama.cpp | ||||
|         GIT_TAG        master | ||||
| ) | ||||
| 
 | ||||
| # Also provides "common" | ||||
| FetchContent_MakeAvailable(llama) | ||||
| 
 | ||||
| # Creates and names a library, sets it as either STATIC | ||||
| # or SHARED, and provides the relative paths to its source code. | ||||
| # You can define multiple libraries, and CMake builds them for you. | ||||
| # Gradle automatically packages shared libraries with your APK. | ||||
| # | ||||
| # In this top level CMakeLists.txt, ${CMAKE_PROJECT_NAME} is used to define | ||||
| # the target library name; in the sub-module's CMakeLists.txt, ${PROJECT_NAME} | ||||
| # is preferred for the same purpose. | ||||
| # | ||||
| # In order to load a library into your app from Java/Kotlin, you must call | ||||
| # System.loadLibrary() and pass the name of the library defined here; | ||||
| # for GameActivity/NativeActivity derived applications, the same library name must be | ||||
| # used in the AndroidManifest.xml file. | ||||
| add_library(${CMAKE_PROJECT_NAME} SHARED | ||||
|     # List C/C++ source files with relative paths to this CMakeLists.txt. | ||||
|     llama-android.cpp) | ||||
| 
 | ||||
| # Specifies libraries CMake should link to your target library. You | ||||
| # can link libraries from various origins, such as libraries defined in this | ||||
| # build script, prebuilt third-party libraries, or Android system libraries. | ||||
| target_link_libraries(${CMAKE_PROJECT_NAME} | ||||
|     # List libraries link to the target library | ||||
|     llama | ||||
|     common | ||||
|     android | ||||
|     log) | ||||
							
								
								
									
										394
									
								
								examples/llama.android/app/src/main/cpp/llama-android.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,394 @@ | |||
| #include <android/log.h> | ||||
| #include <jni.h> | ||||
| #include <iomanip> | ||||
| #include <math.h> | ||||
| #include <string> | ||||
| #include <unistd.h> | ||||
| #include "llama.h" | ||||
| #include "common/common.h" | ||||
| 
 | ||||
| // Write C++ code here.
 | ||||
| //
 | ||||
| // Do not forget to dynamically load the C++ library into your application.
 | ||||
| //
 | ||||
| // For instance,
 | ||||
| //
 | ||||
| // In MainActivity.java:
 | ||||
| //    static {
 | ||||
| //       System.loadLibrary("llama-android");
 | ||||
| //    }
 | ||||
| //
 | ||||
| // Or, in MainActivity.kt:
 | ||||
| //    companion object {
 | ||||
| //      init {
 | ||||
| //         System.loadLibrary("llama-android")
 | ||||
| //      }
 | ||||
| //    }
 | ||||
| 
 | ||||
| #define TAG "llama-android.cpp" | ||||
| #define LOGi(...) __android_log_print(ANDROID_LOG_INFO, TAG, __VA_ARGS__) | ||||
| #define LOGe(...) __android_log_print(ANDROID_LOG_ERROR, TAG, __VA_ARGS__) | ||||
| 
 | ||||
| jclass la_int_var; | ||||
| jmethodID la_int_var_value; | ||||
| jmethodID la_int_var_inc; | ||||
| 
 | ||||
| static void log_callback(ggml_log_level level, const char * fmt, void * data) { | ||||
|     if (level == GGML_LOG_LEVEL_ERROR)     __android_log_print(ANDROID_LOG_ERROR, TAG, fmt, data); | ||||
|     else if (level == GGML_LOG_LEVEL_INFO) __android_log_print(ANDROID_LOG_INFO, TAG, fmt, data); | ||||
|     else if (level == GGML_LOG_LEVEL_WARN) __android_log_print(ANDROID_LOG_WARN, TAG, fmt, data); | ||||
|     else __android_log_print(ANDROID_LOG_DEFAULT, TAG, fmt, data); | ||||
| } | ||||
| 
 | ||||
| extern "C" | ||||
| JNIEXPORT jlong JNICALL | ||||
| Java_com_example_llama_Llm_load_1model(JNIEnv *env, jobject, jstring filename) { | ||||
|     llama_model_params model_params = llama_model_default_params(); | ||||
| 
 | ||||
|     auto path_to_model = env->GetStringUTFChars(filename, 0); | ||||
|     LOGi("Loading model from %s", path_to_model); | ||||
| 
 | ||||
|     auto model = llama_load_model_from_file(path_to_model, model_params); | ||||
|     env->ReleaseStringUTFChars(filename, path_to_model); | ||||
| 
 | ||||
|     if (!model) { | ||||
|         LOGe("load_model() failed"); | ||||
|         env->ThrowNew(env->FindClass("java/lang/IllegalStateException"), "load_model() failed"); | ||||
|         return 0; | ||||
|     } | ||||
| 
 | ||||
|     return reinterpret_cast<jlong>(model); | ||||
| } | ||||
| 
 | ||||
| extern "C" | ||||
| JNIEXPORT void JNICALL | ||||
| Java_com_example_llama_Llm_free_1model(JNIEnv *, jobject, jlong model) { | ||||
|     llama_free_model(reinterpret_cast<llama_model *>(model)); | ||||
| } | ||||
| 
 | ||||
| extern "C" | ||||
| JNIEXPORT jlong JNICALL | ||||
| Java_com_example_llama_Llm_new_1context(JNIEnv *env, jobject, jlong jmodel) { | ||||
|     auto model = reinterpret_cast<llama_model *>(jmodel); | ||||
| 
 | ||||
|     if (!model) { | ||||
|         LOGe("new_context(): model cannot be null"); | ||||
|         env->ThrowNew(env->FindClass("java/lang/IllegalArgumentException"), "Model cannot be null"); | ||||
|         return 0; | ||||
|     } | ||||
| 
 | ||||
|     int n_threads = std::max(1, std::min(8, (int) sysconf(_SC_NPROCESSORS_ONLN) - 2)); | ||||
|     LOGi("Using %d threads", n_threads); | ||||
| 
 | ||||
|     llama_context_params ctx_params = llama_context_default_params(); | ||||
|     ctx_params.seed  = 1234; | ||||
|     ctx_params.n_ctx = 2048; | ||||
|     ctx_params.n_threads       = n_threads; | ||||
|     ctx_params.n_threads_batch = n_threads; | ||||
| 
 | ||||
|     llama_context * context = llama_new_context_with_model(model, ctx_params); | ||||
| 
 | ||||
|     if (!context) { | ||||
|         LOGe("llama_new_context_with_model() returned null)"); | ||||
|         env->ThrowNew(env->FindClass("java/lang/IllegalStateException"), | ||||
|                       "llama_new_context_with_model() returned null)"); | ||||
|         return 0; | ||||
|     } | ||||
| 
 | ||||
|     return reinterpret_cast<jlong>(context); | ||||
| } | ||||
| 
 | ||||
| extern "C" | ||||
| JNIEXPORT void JNICALL | ||||
| Java_com_example_llama_Llm_free_1context(JNIEnv *, jobject, jlong context) { | ||||
|     llama_free(reinterpret_cast<llama_context *>(context)); | ||||
| } | ||||
| 
 | ||||
| extern "C" | ||||
| JNIEXPORT void JNICALL | ||||
| Java_com_example_llama_Llm_backend_1free(JNIEnv *, jobject) { | ||||
|     llama_backend_free(); | ||||
| } | ||||
| 
 | ||||
| extern "C" | ||||
| JNIEXPORT void JNICALL | ||||
| Java_com_example_llama_Llm_log_1to_1android(JNIEnv *, jobject) { | ||||
|     llama_log_set(log_callback, NULL); | ||||
| } | ||||
| 
 | ||||
| extern "C" | ||||
| JNIEXPORT jstring JNICALL | ||||
| Java_com_example_llama_Llm_bench_1model( | ||||
|         JNIEnv *env, | ||||
|         jobject, | ||||
|         jlong context_pointer, | ||||
|         jlong model_pointer, | ||||
|         jlong batch_pointer, | ||||
|         jint pp, | ||||
|         jint tg, | ||||
|         jint pl, | ||||
|         jint nr | ||||
|         ) { | ||||
|     auto pp_avg = 0.0; | ||||
|     auto tg_avg = 0.0; | ||||
|     auto pp_std = 0.0; | ||||
|     auto tg_std = 0.0; | ||||
| 
 | ||||
|     const auto context = reinterpret_cast<llama_context *>(context_pointer); | ||||
|     const auto model = reinterpret_cast<llama_model *>(model_pointer); | ||||
|     const auto batch = reinterpret_cast<llama_batch *>(batch_pointer); | ||||
| 
 | ||||
|     const int n_ctx = llama_n_ctx(context); | ||||
| 
 | ||||
|     LOGi("n_ctx = %d", n_ctx); | ||||
| 
 | ||||
|     int i, j; | ||||
|     int nri; | ||||
|     for (nri = 0; nri < nr; nri++) { | ||||
|         LOGi("Benchmark prompt processing (pp)"); | ||||
| 
 | ||||
|         llama_batch_clear(*batch); | ||||
| 
 | ||||
|         const int n_tokens = pp; | ||||
|         for (i = 0; i < n_tokens; i++) { | ||||
|             llama_batch_add(*batch, 0, i, { 0 }, false); | ||||
|         } | ||||
| 
 | ||||
|         batch->logits[batch->n_tokens - 1] = true; | ||||
|         llama_kv_cache_clear(context); | ||||
| 
 | ||||
|         const auto t_pp_start = ggml_time_us(); | ||||
|         if (llama_decode(context, *batch) != 0) { | ||||
|             LOGi("llama_decode() failed during prompt processing"); | ||||
|         } | ||||
|         const auto t_pp_end = ggml_time_us(); | ||||
| 
 | ||||
|         // bench text generation
 | ||||
| 
 | ||||
|         LOGi("Benchmark text generation (tg)"); | ||||
| 
 | ||||
|         llama_kv_cache_clear(context); | ||||
|         const auto t_tg_start = ggml_time_us(); | ||||
|         for (i = 0; i < tg; i++) { | ||||
| 
 | ||||
|             llama_batch_clear(*batch); | ||||
|             for (j = 0; j < pl; j++) { | ||||
|                 llama_batch_add(*batch, 0, i, { j }, true); | ||||
|             } | ||||
| 
 | ||||
|             LOGi("llama_decode() text generation: %d", i); | ||||
|             if (llama_decode(context, *batch) != 0) { | ||||
|                 LOGi("llama_decode() failed during text generation"); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         const auto t_tg_end = ggml_time_us(); | ||||
| 
 | ||||
|         llama_kv_cache_clear(context); | ||||
| 
 | ||||
|         const auto t_pp = double(t_pp_end - t_pp_start) / 1000000.0; | ||||
|         const auto t_tg = double(t_tg_end - t_tg_start) / 1000000.0; | ||||
| 
 | ||||
|         const auto speed_pp = double(pp) / t_pp; | ||||
|         const auto speed_tg = double(pl * tg) / t_tg; | ||||
| 
 | ||||
|         pp_avg += speed_pp; | ||||
|         tg_avg += speed_tg; | ||||
| 
 | ||||
|         pp_std += speed_pp * speed_pp; | ||||
|         tg_std += speed_tg * speed_tg; | ||||
| 
 | ||||
|         LOGi("pp %f t/s, tg %f t/s", speed_pp, speed_tg); | ||||
|     } | ||||
| 
 | ||||
|     pp_avg /= double(nr); | ||||
|     tg_avg /= double(nr); | ||||
| 
 | ||||
|     if (nr > 1) { | ||||
|         pp_std = sqrt(pp_std / double(nr - 1) - pp_avg * pp_avg * double(nr) / double(nr - 1)); | ||||
|         tg_std = sqrt(tg_std / double(nr - 1) - tg_avg * tg_avg * double(nr) / double(nr - 1)); | ||||
|     } else { | ||||
|         pp_std = 0; | ||||
|         tg_std = 0; | ||||
|     } | ||||
| 
 | ||||
|     char model_desc[128]; | ||||
|     llama_model_desc(model, model_desc, sizeof(model_desc)); | ||||
| 
 | ||||
|     const auto model_size     = double(llama_model_size(model)) / 1024.0 / 1024.0 / 1024.0; | ||||
|     const auto model_n_params = double(llama_model_n_params(model)) / 1e9; | ||||
| 
 | ||||
|     const auto backend    = "(Android)"; // TODO: What should this be?
 | ||||
| 
 | ||||
|     std::stringstream result; | ||||
|     result << std::setprecision(2); | ||||
|     result << "| model | size | params | backend | test | t/s |\n"; | ||||
|     result << "| --- | --- | --- | --- | --- | --- |\n"; | ||||
|     result << "| " << model_desc << " | " << model_size << "GiB | " << model_n_params << "B | " << backend << " | pp " << pp << " | " << pp_avg << " ± " << pp_std << " |\n"; | ||||
|     result << "| " << model_desc << " | " << model_size << "GiB | " << model_n_params << "B | " << backend << " | tg " << tg << " | " << tg_avg << " ± " << tg_std << " |\n"; | ||||
| 
 | ||||
|     return env->NewStringUTF(result.str().c_str()); | ||||
| } | ||||
| 
 | ||||
| extern "C" | ||||
| JNIEXPORT void JNICALL | ||||
| Java_com_example_llama_Llm_free_1batch(JNIEnv *, jobject, jlong batch_pointer) { | ||||
|     llama_batch_free(*reinterpret_cast<llama_batch *>(batch_pointer)); | ||||
| } | ||||
| 
 | ||||
| extern "C" | ||||
| JNIEXPORT jlong JNICALL | ||||
| Java_com_example_llama_Llm_new_1batch(JNIEnv *, jobject, jint n_tokens, jint embd, jint n_seq_max) { | ||||
| 
 | ||||
|     // Source: Copy of llama.cpp:llama_batch_init but heap-allocated.
 | ||||
| 
 | ||||
|     llama_batch *batch = new llama_batch { | ||||
|         0, | ||||
|         nullptr, | ||||
|         nullptr, | ||||
|         nullptr, | ||||
|         nullptr, | ||||
|         nullptr, | ||||
|         nullptr, | ||||
|         0, | ||||
|         0, | ||||
|         0, | ||||
|     }; | ||||
| 
 | ||||
|     if (embd) { | ||||
|         batch->embd = (float *) malloc(sizeof(float) * n_tokens * embd); | ||||
|     } else { | ||||
|         batch->token = (llama_token *) malloc(sizeof(llama_token) * n_tokens); | ||||
|     } | ||||
| 
 | ||||
|     batch->pos      = (llama_pos *)     malloc(sizeof(llama_pos)      * n_tokens); | ||||
|     batch->n_seq_id = (int32_t *)       malloc(sizeof(int32_t)        * n_tokens); | ||||
|     batch->seq_id   = (llama_seq_id **) malloc(sizeof(llama_seq_id *) * n_tokens); | ||||
|     for (int i = 0; i < n_tokens; ++i) { | ||||
|         batch->seq_id[i] = (llama_seq_id *) malloc(sizeof(llama_seq_id) * n_seq_max); | ||||
|     } | ||||
|     batch->logits   = (int8_t *)        malloc(sizeof(int8_t)         * n_tokens); | ||||
| 
 | ||||
|     return reinterpret_cast<jlong>(batch); | ||||
| } | ||||
| 
 | ||||
| extern "C" | ||||
| JNIEXPORT void JNICALL | ||||
| Java_com_example_llama_Llm_backend_1init(JNIEnv *, jobject, jboolean numa) { | ||||
|     llama_backend_init(numa); | ||||
| } | ||||
| 
 | ||||
| extern "C" | ||||
| JNIEXPORT jstring JNICALL | ||||
| Java_com_example_llama_Llm_system_1info(JNIEnv *env, jobject) { | ||||
|     return env->NewStringUTF(llama_print_system_info()); | ||||
| } | ||||
| 
 | ||||
| extern "C" | ||||
| JNIEXPORT jint JNICALL | ||||
| Java_com_example_llama_Llm_completion_1init( | ||||
|         JNIEnv *env, | ||||
|         jobject, | ||||
|         jlong context_pointer, | ||||
|         jlong batch_pointer, | ||||
|         jstring jtext, | ||||
|         jint n_len | ||||
|     ) { | ||||
| 
 | ||||
|     const auto text = env->GetStringUTFChars(jtext, 0); | ||||
|     const auto context = reinterpret_cast<llama_context *>(context_pointer); | ||||
|     const auto batch = reinterpret_cast<llama_batch *>(batch_pointer); | ||||
| 
 | ||||
|     const auto tokens_list = llama_tokenize(context, text, 1); | ||||
| 
 | ||||
|     auto n_ctx = llama_n_ctx(context); | ||||
|     auto n_kv_req = tokens_list.size() + (n_len - tokens_list.size()); | ||||
| 
 | ||||
|     LOGi("n_len = %d, n_ctx = %d, n_kv_req = %d", n_len, n_ctx, n_kv_req); | ||||
| 
 | ||||
|     if (n_kv_req > n_ctx) { | ||||
|         LOGe("error: n_kv_req > n_ctx, the required KV cache size is not big enough"); | ||||
|     } | ||||
| 
 | ||||
|     for (auto id : tokens_list) { | ||||
|         LOGi("%s", llama_token_to_piece(context, id).c_str()); | ||||
|     } | ||||
| 
 | ||||
|     llama_batch_clear(*batch); | ||||
| 
 | ||||
|     // evaluate the initial prompt
 | ||||
|     for (auto i = 0; i < tokens_list.size(); i++) { | ||||
|         llama_batch_add(*batch, tokens_list[i], i, { 0 }, false); | ||||
|     } | ||||
| 
 | ||||
|     // llama_decode will output logits only for the last token of the prompt
 | ||||
|     batch->logits[batch->n_tokens - 1] = true; | ||||
| 
 | ||||
|     if (llama_decode(context, *batch) != 0) { | ||||
|         LOGe("llama_decode() failed"); | ||||
|     } | ||||
| 
 | ||||
|     env->ReleaseStringUTFChars(jtext, text); | ||||
| 
 | ||||
|     return batch->n_tokens; | ||||
| } | ||||
| 
 | ||||
| extern "C" | ||||
| JNIEXPORT jstring JNICALL | ||||
| Java_com_example_llama_Llm_completion_1loop( | ||||
|         JNIEnv * env, | ||||
|         jobject, | ||||
|         jlong context_pointer, | ||||
|         jlong batch_pointer, | ||||
|         jint n_len, | ||||
|         jobject intvar_ncur | ||||
| ) { | ||||
|     const auto context = reinterpret_cast<llama_context *>(context_pointer); | ||||
|     const auto batch = reinterpret_cast<llama_batch *>(batch_pointer); | ||||
|     const auto model = llama_get_model(context); | ||||
| 
 | ||||
|     if (!la_int_var) la_int_var = env->GetObjectClass(intvar_ncur); | ||||
|     if (!la_int_var_value) la_int_var_value = env->GetMethodID(la_int_var, "getValue", "()I"); | ||||
|     if (!la_int_var_inc) la_int_var_inc = env->GetMethodID(la_int_var, "inc", "()V"); | ||||
| 
 | ||||
|     auto n_vocab = llama_n_vocab(model); | ||||
|     auto logits = llama_get_logits_ith(context, batch->n_tokens - 1); | ||||
| 
 | ||||
|     std::vector<llama_token_data> candidates; | ||||
|     candidates.reserve(n_vocab); | ||||
| 
 | ||||
|     for (llama_token token_id = 0; token_id < n_vocab; token_id++) { | ||||
|         candidates.emplace_back(llama_token_data{ token_id, logits[token_id], 0.0f }); | ||||
|     } | ||||
| 
 | ||||
|     llama_token_data_array candidates_p = { candidates.data(), candidates.size(), false }; | ||||
| 
 | ||||
|     // sample the most likely token
 | ||||
|     const auto new_token_id = llama_sample_token_greedy(context, &candidates_p); | ||||
| 
 | ||||
|     const auto n_cur = env->CallIntMethod(intvar_ncur, la_int_var_value); | ||||
|     if (new_token_id == llama_token_eos(model) || n_cur == n_len) { | ||||
|         return env->NewStringUTF(""); | ||||
|     } | ||||
| 
 | ||||
|     auto new_token_chars = llama_token_to_piece(context, new_token_id); | ||||
|     LOGi("new_token_chars: `%s`", new_token_chars.c_str()); | ||||
|     auto new_token = env->NewStringUTF(new_token_chars.c_str()); | ||||
| 
 | ||||
|     llama_batch_clear(*batch); | ||||
|     llama_batch_add(*batch, new_token_id, n_cur, { 0 }, true); | ||||
| 
 | ||||
|     env->CallVoidMethod(intvar_ncur, la_int_var_inc); | ||||
| 
 | ||||
|     if (llama_decode(context, *batch) != 0) { | ||||
|         LOGe("llama_decode() returned null"); | ||||
|     } | ||||
| 
 | ||||
|     return new_token; | ||||
| } | ||||
| 
 | ||||
| extern "C" | ||||
| JNIEXPORT void JNICALL | ||||
| Java_com_example_llama_Llm_kv_1cache_1clear(JNIEnv *, jobject, jlong context) { | ||||
|     llama_kv_cache_clear(reinterpret_cast<llama_context *>(context)); | ||||
| } | ||||
|  | @ -0,0 +1,119 @@ | |||
| package com.example.llama | ||||
| 
 | ||||
| import android.app.DownloadManager | ||||
| import android.net.Uri | ||||
| import android.util.Log | ||||
| import androidx.compose.material3.Button | ||||
| import androidx.compose.material3.Text | ||||
| import androidx.compose.runtime.Composable | ||||
| import androidx.compose.runtime.getValue | ||||
| import androidx.compose.runtime.mutableDoubleStateOf | ||||
| import androidx.compose.runtime.mutableStateOf | ||||
| import androidx.compose.runtime.remember | ||||
| import androidx.compose.runtime.rememberCoroutineScope | ||||
| import androidx.compose.runtime.setValue | ||||
| import androidx.core.database.getLongOrNull | ||||
| import androidx.core.net.toUri | ||||
| import kotlinx.coroutines.delay | ||||
| import kotlinx.coroutines.launch | ||||
| import java.io.File | ||||
| 
 | ||||
| data class Downloadable(val name: String, val source: Uri, val destination: File) { | ||||
|     companion object { | ||||
|         @JvmStatic | ||||
|         private val tag: String? = this::class.qualifiedName | ||||
| 
 | ||||
|         sealed interface State | ||||
|         data object Ready: State | ||||
|         data class Downloading(val id: Long): State | ||||
|         data class Downloaded(val downloadable: Downloadable): State | ||||
|         data class Error(val message: String): State | ||||
| 
 | ||||
|         @JvmStatic | ||||
|         @Composable | ||||
|         fun Button(viewModel: MainViewModel, dm: DownloadManager, item: Downloadable) { | ||||
|             var status: State by remember { | ||||
|                 mutableStateOf( | ||||
|                     if (item.destination.exists()) Downloaded(item) | ||||
|                     else Ready | ||||
|                 ) | ||||
|             } | ||||
|             var progress by remember { mutableDoubleStateOf(0.0) } | ||||
| 
 | ||||
|             val coroutineScope = rememberCoroutineScope() | ||||
| 
 | ||||
|             suspend fun waitForDownload(result: Downloading, item: Downloadable): State { | ||||
|                 while (true) { | ||||
|                     val cursor = dm.query(DownloadManager.Query().setFilterById(result.id)) | ||||
| 
 | ||||
|                     if (cursor == null) { | ||||
|                         Log.e(tag, "dm.query() returned null") | ||||
|                         return Error("dm.query() returned null") | ||||
|                     } | ||||
| 
 | ||||
|                     if (!cursor.moveToFirst() || cursor.count < 1) { | ||||
|                         cursor.close() | ||||
|                         Log.i(tag, "cursor.moveToFirst() returned false or cursor.count < 1, download canceled?") | ||||
|                         return Ready | ||||
|                     } | ||||
| 
 | ||||
|                     val pix = cursor.getColumnIndex(DownloadManager.COLUMN_BYTES_DOWNLOADED_SO_FAR) | ||||
|                     val tix = cursor.getColumnIndex(DownloadManager.COLUMN_TOTAL_SIZE_BYTES) | ||||
|                     val sofar = cursor.getLongOrNull(pix) ?: 0 | ||||
|                     val total = cursor.getLongOrNull(tix) ?: 1 | ||||
|                     cursor.close() | ||||
| 
 | ||||
|                     if (sofar == total) { | ||||
|                         return Downloaded(item) | ||||
|                     } | ||||
| 
 | ||||
|                     progress = (sofar * 1.0) / total | ||||
| 
 | ||||
|                     delay(1000L) | ||||
|                 } | ||||
|             } | ||||
| 
 | ||||
|             fun onClick() { | ||||
|                 when (val s = status) { | ||||
|                     is Downloaded -> { | ||||
|                         viewModel.load(item.destination.path) | ||||
|                     } | ||||
| 
 | ||||
|                     is Downloading -> { | ||||
|                         coroutineScope.launch { | ||||
|                             status = waitForDownload(s, item) | ||||
|                         } | ||||
|                     } | ||||
| 
 | ||||
|                     else -> { | ||||
|                         item.destination.delete() | ||||
| 
 | ||||
|                         val request = DownloadManager.Request(item.source).apply { | ||||
|                             setTitle("Downloading model") | ||||
|                             setDescription("Downloading model: ${item.name}") | ||||
|                             setAllowedNetworkTypes(DownloadManager.Request.NETWORK_WIFI) | ||||
|                             setDestinationUri(item.destination.toUri()) | ||||
|                         } | ||||
| 
 | ||||
|                         viewModel.log("Saving ${item.name} to ${item.destination.path}") | ||||
|                         Log.i(tag, "Saving ${item.name} to ${item.destination.path}") | ||||
| 
 | ||||
|                         val id = dm.enqueue(request) | ||||
|                         status = Downloading(id) | ||||
|                         onClick() | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
| 
 | ||||
|             Button(onClick = { onClick() }, enabled = status !is Downloading) { | ||||
|                 when (status) { | ||||
|                     is Downloading -> Text(text = "Downloading ${(progress * 100).toInt()}%") | ||||
|                     is Downloaded -> Text("Load ${item.name}") | ||||
|                     is Ready -> Text("Download ${item.name}") | ||||
|                     is Error -> Text("Download ${item.name}") | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|     } | ||||
| } | ||||
|  | @ -0,0 +1,172 @@ | |||
| package com.example.llama | ||||
| 
 | ||||
| import android.util.Log | ||||
| import kotlinx.coroutines.CoroutineDispatcher | ||||
| import kotlinx.coroutines.asCoroutineDispatcher | ||||
| import kotlinx.coroutines.flow.Flow | ||||
| import kotlinx.coroutines.flow.flow | ||||
| import kotlinx.coroutines.flow.flowOn | ||||
| import kotlinx.coroutines.withContext | ||||
| import java.util.concurrent.Executors | ||||
| import kotlin.concurrent.thread | ||||
| 
 | ||||
| class Llm { | ||||
|     private val tag: String? = this::class.simpleName | ||||
| 
 | ||||
|     private val threadLocalState: ThreadLocal<State> = ThreadLocal.withInitial { State.Idle } | ||||
| 
 | ||||
|     private val runLoop: CoroutineDispatcher = Executors.newSingleThreadExecutor { | ||||
|         thread(start = false, name = "Llm-RunLoop") { | ||||
|             Log.d(tag, "Dedicated thread for native code: ${Thread.currentThread().name}") | ||||
| 
 | ||||
|             // No-op if called more than once. | ||||
|             System.loadLibrary("llama-android") | ||||
| 
 | ||||
|             // Set llama log handler to Android | ||||
|             log_to_android() | ||||
|             backend_init(false) | ||||
| 
 | ||||
|             Log.d(tag, system_info()) | ||||
| 
 | ||||
|             it.run() | ||||
|         }.apply { | ||||
|             uncaughtExceptionHandler = Thread.UncaughtExceptionHandler { _, exception: Throwable -> | ||||
|                 Log.e(tag, "Unhandled exception", exception) | ||||
|             } | ||||
|         } | ||||
|     }.asCoroutineDispatcher() | ||||
| 
 | ||||
|     private val nlen: Int = 64 | ||||
| 
 | ||||
|     private external fun log_to_android() | ||||
|     private external fun load_model(filename: String): Long | ||||
|     private external fun free_model(model: Long) | ||||
|     private external fun new_context(model: Long): Long | ||||
|     private external fun free_context(context: Long) | ||||
|     private external fun backend_init(numa: Boolean) | ||||
|     private external fun backend_free() | ||||
|     private external fun free_batch(batch: Long) | ||||
|     private external fun new_batch(nTokens: Int, embd: Int, nSeqMax: Int): Long | ||||
|     private external fun bench_model( | ||||
|         context: Long, | ||||
|         model: Long, | ||||
|         batch: Long, | ||||
|         pp: Int, | ||||
|         tg: Int, | ||||
|         pl: Int, | ||||
|         nr: Int | ||||
|     ): String | ||||
| 
 | ||||
|     private external fun system_info(): String | ||||
| 
 | ||||
|     private external fun completion_init( | ||||
|         context: Long, | ||||
|         batch: Long, | ||||
|         text: String, | ||||
|         nLen: Int | ||||
|     ): Int | ||||
| 
 | ||||
|     private external fun completion_loop( | ||||
|         context: Long, | ||||
|         batch: Long, | ||||
|         nLen: Int, | ||||
|         ncur: IntVar | ||||
|     ): String | ||||
| 
 | ||||
|     private external fun kv_cache_clear(context: Long) | ||||
| 
 | ||||
|     suspend fun bench(pp: Int, tg: Int, pl: Int, nr: Int = 1): String { | ||||
|         return withContext(runLoop) { | ||||
|             when (val state = threadLocalState.get()) { | ||||
|                 is State.Loaded -> { | ||||
|                     Log.d(tag, "bench(): $state") | ||||
|                     bench_model(state.context, state.model, state.batch, pp, tg, pl, nr) | ||||
|                 } | ||||
| 
 | ||||
|                 else -> throw IllegalStateException("No model loaded") | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     suspend fun load(pathToModel: String) { | ||||
|         withContext(runLoop) { | ||||
|             when (threadLocalState.get()) { | ||||
|                 is State.Idle -> { | ||||
|                     val model = load_model(pathToModel) | ||||
|                     if (model == 0L)  throw IllegalStateException("load_model() failed") | ||||
| 
 | ||||
|                     val context = new_context(model) | ||||
|                     if (context == 0L) throw IllegalStateException("new_context() failed") | ||||
| 
 | ||||
|                     val batch = new_batch(512, 0, 1) | ||||
|                     if (batch == 0L) throw IllegalStateException("new_batch() failed") | ||||
| 
 | ||||
|                     Log.i(tag, "Loaded model $pathToModel") | ||||
|                     threadLocalState.set(State.Loaded(model, context, batch)) | ||||
|                 } | ||||
|                 else -> throw IllegalStateException("Model already loaded") | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fun send(message: String): Flow<String> = flow { | ||||
|         when (val state = threadLocalState.get()) { | ||||
|             is State.Loaded -> { | ||||
|                 val ncur = IntVar(completion_init(state.context, state.batch, message, nlen)) | ||||
|                 while (ncur.value <= nlen) { | ||||
|                     val str = completion_loop(state.context, state.batch, nlen, ncur) | ||||
|                     if (str.isEmpty()) { | ||||
|                         break | ||||
|                     } | ||||
|                     emit(str) | ||||
|                 } | ||||
|                 kv_cache_clear(state.context) | ||||
|             } | ||||
|             else -> {} | ||||
|         } | ||||
|     }.flowOn(runLoop) | ||||
| 
 | ||||
|     /** | ||||
|      * Unloads the model and frees resources. | ||||
|      * | ||||
|      * This is a no-op if there's no model loaded. | ||||
|      */ | ||||
|     suspend fun unload() { | ||||
|         withContext(runLoop) { | ||||
|             when (val state = threadLocalState.get()) { | ||||
|                 is State.Loaded -> { | ||||
|                     free_context(state.context) | ||||
|                     free_model(state.model) | ||||
|                     free_batch(state.batch) | ||||
| 
 | ||||
|                     threadLocalState.set(State.Idle) | ||||
|                 } | ||||
|                 else -> {} | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     companion object { | ||||
|         private class IntVar(value: Int) { | ||||
|             @Volatile | ||||
|             var value: Int = value | ||||
|                 private set | ||||
| 
 | ||||
|             fun inc() { | ||||
|                 synchronized(this) { | ||||
|                     value += 1 | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         private sealed interface State { | ||||
|             data object Idle: State | ||||
|             data class Loaded(val model: Long, val context: Long, val batch: Long): State | ||||
|         } | ||||
| 
 | ||||
|         // Enforce only one instance of Llm. | ||||
|         private val _instance: Llm = Llm() | ||||
| 
 | ||||
|         fun instance(): Llm = _instance | ||||
|     } | ||||
| } | ||||
|  | @ -0,0 +1,154 @@ | |||
| package com.example.llama | ||||
| 
 | ||||
| import android.app.ActivityManager | ||||
| import android.app.DownloadManager | ||||
| import android.content.ClipData | ||||
| import android.content.ClipboardManager | ||||
| import android.net.Uri | ||||
| import android.os.Bundle | ||||
| import android.os.StrictMode | ||||
| import android.os.StrictMode.VmPolicy | ||||
| import android.text.format.Formatter | ||||
| import androidx.activity.ComponentActivity | ||||
| import androidx.activity.compose.setContent | ||||
| import androidx.activity.viewModels | ||||
| import androidx.compose.foundation.layout.Box | ||||
| import androidx.compose.foundation.layout.Column | ||||
| import androidx.compose.foundation.layout.Row | ||||
| import androidx.compose.foundation.layout.fillMaxSize | ||||
| import androidx.compose.foundation.layout.padding | ||||
| import androidx.compose.foundation.lazy.LazyColumn | ||||
| import androidx.compose.foundation.lazy.items | ||||
| import androidx.compose.foundation.lazy.rememberLazyListState | ||||
| import androidx.compose.material3.Button | ||||
| import androidx.compose.material3.LocalContentColor | ||||
| import androidx.compose.material3.MaterialTheme | ||||
| import androidx.compose.material3.OutlinedTextField | ||||
| import androidx.compose.material3.Surface | ||||
| import androidx.compose.material3.Text | ||||
| import androidx.compose.runtime.Composable | ||||
| import androidx.compose.ui.Modifier | ||||
| import androidx.compose.ui.unit.dp | ||||
| import androidx.core.content.getSystemService | ||||
| import com.example.llama.ui.theme.LlamaAndroidTheme | ||||
| import java.io.File | ||||
| 
 | ||||
| class MainActivity( | ||||
|     activityManager: ActivityManager? = null, | ||||
|     downloadManager: DownloadManager? = null, | ||||
|     clipboardManager: ClipboardManager? = null, | ||||
| ): ComponentActivity() { | ||||
|     private val tag: String? = this::class.simpleName | ||||
| 
 | ||||
|     private val activityManager by lazy { activityManager ?: getSystemService<ActivityManager>()!! } | ||||
|     private val downloadManager by lazy { downloadManager ?: getSystemService<DownloadManager>()!! } | ||||
|     private val clipboardManager by lazy { clipboardManager ?: getSystemService<ClipboardManager>()!! } | ||||
| 
 | ||||
|     private val viewModel: MainViewModel by viewModels() | ||||
| 
 | ||||
|     // Get a MemoryInfo object for the device's current memory status. | ||||
|     private fun availableMemory(): ActivityManager.MemoryInfo { | ||||
|         return ActivityManager.MemoryInfo().also { memoryInfo -> | ||||
|             activityManager.getMemoryInfo(memoryInfo) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         super.onCreate(savedInstanceState) | ||||
| 
 | ||||
|         StrictMode.setVmPolicy( | ||||
|             VmPolicy.Builder(StrictMode.getVmPolicy()) | ||||
|                 .detectLeakedClosableObjects() | ||||
|                 .build() | ||||
|         ) | ||||
| 
 | ||||
|         val free = Formatter.formatFileSize(this, availableMemory().availMem) | ||||
|         val total = Formatter.formatFileSize(this, availableMemory().totalMem) | ||||
| 
 | ||||
|         viewModel.log("Current memory: $free / $total") | ||||
|         viewModel.log("Downloads directory: ${getExternalFilesDir(null)}") | ||||
| 
 | ||||
|         val extFilesDir = getExternalFilesDir(null) | ||||
| 
 | ||||
|         val models = listOf( | ||||
|             Downloadable( | ||||
|                 "Phi-2 7B (Q4_0, 1.6 GiB)", | ||||
|                 Uri.parse("https://huggingface.co/ggml-org/models/resolve/main/phi-2/ggml-model-q4_0.gguf?download=true"), | ||||
|                 File(extFilesDir, "phi-2-q4_0.gguf"), | ||||
|             ), | ||||
|             Downloadable( | ||||
|                 "TinyLlama 1.1B (f16, 2.2 GiB)", | ||||
|                 Uri.parse("https://huggingface.co/ggml-org/models/resolve/main/tinyllama-1.1b/ggml-model-f16.gguf?download=true"), | ||||
|                 File(extFilesDir, "tinyllama-1.1-f16.gguf"), | ||||
|             ), | ||||
|             Downloadable( | ||||
|                 "Phi 2 DPO (Q3_K_M, 1.48 GiB)", | ||||
|                 Uri.parse("https://huggingface.co/TheBloke/phi-2-dpo-GGUF/resolve/main/phi-2-dpo.Q3_K_M.gguf?download=true"), | ||||
|                 File(extFilesDir, "phi-2-dpo.Q3_K_M.gguf") | ||||
|             ), | ||||
|         ) | ||||
| 
 | ||||
|         setContent { | ||||
|             LlamaAndroidTheme { | ||||
|                 // A surface container using the 'background' color from the theme | ||||
|                 Surface( | ||||
|                     modifier = Modifier.fillMaxSize(), | ||||
|                     color = MaterialTheme.colorScheme.background | ||||
|                 ) { | ||||
|                     MainCompose( | ||||
|                         viewModel, | ||||
|                         clipboardManager, | ||||
|                         downloadManager, | ||||
|                         models, | ||||
|                     ) | ||||
|                 } | ||||
| 
 | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @Composable | ||||
| fun MainCompose( | ||||
|     viewModel: MainViewModel, | ||||
|     clipboard: ClipboardManager, | ||||
|     dm: DownloadManager, | ||||
|     models: List<Downloadable> | ||||
| ) { | ||||
|     Column { | ||||
|         val scrollState = rememberLazyListState() | ||||
| 
 | ||||
|         Box(modifier = Modifier.weight(1f)) { | ||||
|             LazyColumn(state = scrollState) { | ||||
|                 items(viewModel.messages) { | ||||
|                     Text( | ||||
|                         it, | ||||
|                         style = MaterialTheme.typography.bodyLarge.copy(color = LocalContentColor.current), | ||||
|                         modifier = Modifier.padding(16.dp) | ||||
|                     ) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         OutlinedTextField( | ||||
|             value = viewModel.message, | ||||
|             onValueChange = { viewModel.updateMessage(it) }, | ||||
|             label = { Text("Message") }, | ||||
|         ) | ||||
|         Row { | ||||
|             Button({ viewModel.send() }) { Text("Send") } | ||||
|             Button({ viewModel.bench(8, 4, 1) }) { Text("Bench") } | ||||
|             Button({ viewModel.clear() }) { Text("Clear") } | ||||
|             Button({ | ||||
|                 viewModel.messages.joinToString("\n").let { | ||||
|                     clipboard.setPrimaryClip(ClipData.newPlainText("", it)) | ||||
|                 } | ||||
|             }) { Text("Copy") } | ||||
|         } | ||||
| 
 | ||||
|         Column { | ||||
|             for (model in models) { | ||||
|                 Downloadable.Button(viewModel, dm, model) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | @ -0,0 +1,104 @@ | |||
| package com.example.llama | ||||
| 
 | ||||
| import android.util.Log | ||||
| import androidx.compose.runtime.getValue | ||||
| import androidx.compose.runtime.mutableStateOf | ||||
| import androidx.compose.runtime.setValue | ||||
| import androidx.lifecycle.ViewModel | ||||
| import androidx.lifecycle.viewModelScope | ||||
| import kotlinx.coroutines.flow.catch | ||||
| import kotlinx.coroutines.launch | ||||
| 
 | ||||
| class MainViewModel(private val llm: Llm = Llm.instance()): ViewModel() { | ||||
|     companion object { | ||||
|         @JvmStatic | ||||
|         private val NanosPerSecond = 1_000_000_000.0 | ||||
|     } | ||||
| 
 | ||||
|     private val tag: String? = this::class.simpleName | ||||
| 
 | ||||
|     var messages by mutableStateOf(listOf("Initializing...")) | ||||
|         private set | ||||
| 
 | ||||
|     var message by mutableStateOf("") | ||||
|         private set | ||||
| 
 | ||||
|     override fun onCleared() { | ||||
|         super.onCleared() | ||||
| 
 | ||||
|         viewModelScope.launch { | ||||
|             try { | ||||
|                 llm.unload() | ||||
|             } catch (exc: IllegalStateException) { | ||||
|                 messages += exc.message!! | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fun send() { | ||||
|         val text = message | ||||
|         message = "" | ||||
| 
 | ||||
|         // Add to messages console. | ||||
|         messages += text | ||||
|         messages += "" | ||||
| 
 | ||||
|         viewModelScope.launch { | ||||
|             llm.send(text) | ||||
|                 .catch { | ||||
|                     Log.e(tag, "send() failed", it) | ||||
|                     messages += it.message!! | ||||
|                 } | ||||
|                 .collect { messages = messages.dropLast(1) + (messages.last() + it) } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fun bench(pp: Int, tg: Int, pl: Int, nr: Int = 1) { | ||||
|         viewModelScope.launch { | ||||
|             try { | ||||
|                 val start = System.nanoTime() | ||||
|                 val warmupResult = llm.bench(pp, tg, pl, nr) | ||||
|                 val end = System.nanoTime() | ||||
| 
 | ||||
|                 messages += warmupResult | ||||
| 
 | ||||
|                 val warmup = (end - start).toDouble() / NanosPerSecond | ||||
|                 messages += "Warm up time: $warmup seconds, please wait..." | ||||
| 
 | ||||
|                 if (warmup > 5.0) { | ||||
|                     messages += "Warm up took too long, aborting benchmark" | ||||
|                     return@launch | ||||
|                 } | ||||
| 
 | ||||
|                 messages += llm.bench(512, 128, 1, 3) | ||||
|             } catch (exc: IllegalStateException) { | ||||
|                 Log.e(tag, "bench() failed", exc) | ||||
|                 messages += exc.message!! | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fun load(pathToModel: String) { | ||||
|         viewModelScope.launch { | ||||
|             try { | ||||
|                 llm.load(pathToModel) | ||||
|                 messages += "Loaded $pathToModel" | ||||
|             } catch (exc: IllegalStateException) { | ||||
|                 Log.e(tag, "load() failed", exc) | ||||
|                 messages += exc.message!! | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fun updateMessage(newMessage: String) { | ||||
|         message = newMessage | ||||
|     } | ||||
| 
 | ||||
|     fun clear() { | ||||
|         messages = listOf() | ||||
|     } | ||||
| 
 | ||||
|     fun log(message: String) { | ||||
|         messages += message | ||||
|     } | ||||
| } | ||||
|  | @ -0,0 +1,11 @@ | |||
| package com.example.llama.ui.theme | ||||
| 
 | ||||
| import androidx.compose.ui.graphics.Color | ||||
| 
 | ||||
| val Purple80 = Color(0xFFD0BCFF) | ||||
| val PurpleGrey80 = Color(0xFFCCC2DC) | ||||
| val Pink80 = Color(0xFFEFB8C8) | ||||
| 
 | ||||
| val Purple40 = Color(0xFF6650a4) | ||||
| val PurpleGrey40 = Color(0xFF625b71) | ||||
| val Pink40 = Color(0xFF7D5260) | ||||
|  | @ -0,0 +1,70 @@ | |||
| package com.example.llama.ui.theme | ||||
| 
 | ||||
| import android.app.Activity | ||||
| import android.os.Build | ||||
| import androidx.compose.foundation.isSystemInDarkTheme | ||||
| import androidx.compose.material3.MaterialTheme | ||||
| import androidx.compose.material3.darkColorScheme | ||||
| import androidx.compose.material3.dynamicDarkColorScheme | ||||
| import androidx.compose.material3.dynamicLightColorScheme | ||||
| import androidx.compose.material3.lightColorScheme | ||||
| import androidx.compose.runtime.Composable | ||||
| import androidx.compose.runtime.SideEffect | ||||
| import androidx.compose.ui.graphics.toArgb | ||||
| import androidx.compose.ui.platform.LocalContext | ||||
| import androidx.compose.ui.platform.LocalView | ||||
| import androidx.core.view.WindowCompat | ||||
| 
 | ||||
| private val DarkColorScheme = darkColorScheme( | ||||
|     primary = Purple80, | ||||
|     secondary = PurpleGrey80, | ||||
|     tertiary = Pink80 | ||||
| ) | ||||
| 
 | ||||
| private val LightColorScheme = lightColorScheme( | ||||
|     primary = Purple40, | ||||
|     secondary = PurpleGrey40, | ||||
|     tertiary = Pink40 | ||||
| 
 | ||||
|     /* Other default colors to override | ||||
|     background = Color(0xFFFFFBFE), | ||||
|     surface = Color(0xFFFFFBFE), | ||||
|     onPrimary = Color.White, | ||||
|     onSecondary = Color.White, | ||||
|     onTertiary = Color.White, | ||||
|     onBackground = Color(0xFF1C1B1F), | ||||
|     onSurface = Color(0xFF1C1B1F), | ||||
|     */ | ||||
| ) | ||||
| 
 | ||||
| @Composable | ||||
| fun LlamaAndroidTheme( | ||||
|     darkTheme: Boolean = isSystemInDarkTheme(), | ||||
|     // Dynamic color is available on Android 12+ | ||||
|     dynamicColor: Boolean = true, | ||||
|     content: @Composable () -> Unit | ||||
| ) { | ||||
|     val colorScheme = when { | ||||
|         dynamicColor && Build.VERSION.SDK_INT >= Build.VERSION_CODES.S -> { | ||||
|             val context = LocalContext.current | ||||
|             if (darkTheme) dynamicDarkColorScheme(context) else dynamicLightColorScheme(context) | ||||
|         } | ||||
| 
 | ||||
|         darkTheme -> DarkColorScheme | ||||
|         else -> LightColorScheme | ||||
|     } | ||||
|     val view = LocalView.current | ||||
|     if (!view.isInEditMode) { | ||||
|         SideEffect { | ||||
|             val window = (view.context as Activity).window | ||||
|             window.statusBarColor = colorScheme.primary.toArgb() | ||||
|             WindowCompat.getInsetsController(window, view).isAppearanceLightStatusBars = darkTheme | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     MaterialTheme( | ||||
|         colorScheme = colorScheme, | ||||
|         typography = Typography, | ||||
|         content = content | ||||
|     ) | ||||
| } | ||||
|  | @ -0,0 +1,34 @@ | |||
| package com.example.llama.ui.theme | ||||
| 
 | ||||
| import androidx.compose.material3.Typography | ||||
| import androidx.compose.ui.text.TextStyle | ||||
| import androidx.compose.ui.text.font.FontFamily | ||||
| import androidx.compose.ui.text.font.FontWeight | ||||
| import androidx.compose.ui.unit.sp | ||||
| 
 | ||||
| // Set of Material typography styles to start with | ||||
| val Typography = Typography( | ||||
|     bodyLarge = TextStyle( | ||||
|         fontFamily = FontFamily.Default, | ||||
|         fontWeight = FontWeight.Normal, | ||||
|         fontSize = 16.sp, | ||||
|         lineHeight = 24.sp, | ||||
|         letterSpacing = 0.5.sp | ||||
|     ) | ||||
|     /* Other default text styles to override | ||||
|     titleLarge = TextStyle( | ||||
|         fontFamily = FontFamily.Default, | ||||
|         fontWeight = FontWeight.Normal, | ||||
|         fontSize = 22.sp, | ||||
|         lineHeight = 28.sp, | ||||
|         letterSpacing = 0.sp | ||||
|     ), | ||||
|     labelSmall = TextStyle( | ||||
|         fontFamily = FontFamily.Default, | ||||
|         fontWeight = FontWeight.Medium, | ||||
|         fontSize = 11.sp, | ||||
|         lineHeight = 16.sp, | ||||
|         letterSpacing = 0.5.sp | ||||
|     ) | ||||
|     */ | ||||
| ) | ||||
|  | @ -0,0 +1,170 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <vector xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     android:width="108dp" | ||||
|     android:height="108dp" | ||||
|     android:viewportWidth="108" | ||||
|     android:viewportHeight="108"> | ||||
|     <path | ||||
|         android:fillColor="#3DDC84" | ||||
|         android:pathData="M0,0h108v108h-108z" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M9,0L9,108" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M19,0L19,108" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M29,0L29,108" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M39,0L39,108" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M49,0L49,108" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M59,0L59,108" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M69,0L69,108" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M79,0L79,108" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M89,0L89,108" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M99,0L99,108" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M0,9L108,9" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M0,19L108,19" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M0,29L108,29" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M0,39L108,39" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M0,49L108,49" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M0,59L108,59" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M0,69L108,69" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M0,79L108,79" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M0,89L108,89" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M0,99L108,99" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M19,29L89,29" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M19,39L89,39" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M19,49L89,49" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M19,59L89,59" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M19,69L89,69" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M19,79L89,79" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M29,19L29,89" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M39,19L39,89" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M49,19L49,89" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M59,19L59,89" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M69,19L69,89" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
|     <path | ||||
|         android:fillColor="#00000000" | ||||
|         android:pathData="M79,19L79,89" | ||||
|         android:strokeWidth="0.8" | ||||
|         android:strokeColor="#33FFFFFF" /> | ||||
| </vector> | ||||
|  | @ -0,0 +1,30 @@ | |||
| <vector xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:aapt="http://schemas.android.com/aapt" | ||||
|     android:width="108dp" | ||||
|     android:height="108dp" | ||||
|     android:viewportWidth="108" | ||||
|     android:viewportHeight="108"> | ||||
|     <path android:pathData="M31,63.928c0,0 6.4,-11 12.1,-13.1c7.2,-2.6 26,-1.4 26,-1.4l38.1,38.1L107,108.928l-32,-1L31,63.928z"> | ||||
|         <aapt:attr name="android:fillColor"> | ||||
|             <gradient | ||||
|                 android:endX="85.84757" | ||||
|                 android:endY="92.4963" | ||||
|                 android:startX="42.9492" | ||||
|                 android:startY="49.59793" | ||||
|                 android:type="linear"> | ||||
|                 <item | ||||
|                     android:color="#44000000" | ||||
|                     android:offset="0.0" /> | ||||
|                 <item | ||||
|                     android:color="#00000000" | ||||
|                     android:offset="1.0" /> | ||||
|             </gradient> | ||||
|         </aapt:attr> | ||||
|     </path> | ||||
|     <path | ||||
|         android:fillColor="#FFFFFF" | ||||
|         android:fillType="nonZero" | ||||
|         android:pathData="M65.3,45.828l3.8,-6.6c0.2,-0.4 0.1,-0.9 -0.3,-1.1c-0.4,-0.2 -0.9,-0.1 -1.1,0.3l-3.9,6.7c-6.3,-2.8 -13.4,-2.8 -19.7,0l-3.9,-6.7c-0.2,-0.4 -0.7,-0.5 -1.1,-0.3C38.8,38.328 38.7,38.828 38.9,39.228l3.8,6.6C36.2,49.428 31.7,56.028 31,63.928h46C76.3,56.028 71.8,49.428 65.3,45.828zM43.4,57.328c-0.8,0 -1.5,-0.5 -1.8,-1.2c-0.3,-0.7 -0.1,-1.5 0.4,-2.1c0.5,-0.5 1.4,-0.7 2.1,-0.4c0.7,0.3 1.2,1 1.2,1.8C45.3,56.528 44.5,57.328 43.4,57.328L43.4,57.328zM64.6,57.328c-0.8,0 -1.5,-0.5 -1.8,-1.2s-0.1,-1.5 0.4,-2.1c0.5,-0.5 1.4,-0.7 2.1,-0.4c0.7,0.3 1.2,1 1.2,1.8C66.5,56.528 65.6,57.328 64.6,57.328L64.6,57.328z" | ||||
|         android:strokeWidth="1" | ||||
|         android:strokeColor="#00000000" /> | ||||
| </vector> | ||||
|  | @ -0,0 +1,6 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
|     <background android:drawable="@drawable/ic_launcher_background" /> | ||||
|     <foreground android:drawable="@drawable/ic_launcher_foreground" /> | ||||
|     <monochrome android:drawable="@drawable/ic_launcher_foreground" /> | ||||
| </adaptive-icon> | ||||
|  | @ -0,0 +1,6 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android"> | ||||
|     <background android:drawable="@drawable/ic_launcher_background" /> | ||||
|     <foreground android:drawable="@drawable/ic_launcher_foreground" /> | ||||
|     <monochrome android:drawable="@drawable/ic_launcher_foreground" /> | ||||
| </adaptive-icon> | ||||
| After Width: | Height: | Size: 1.4 KiB | 
| After Width: | Height: | Size: 2.8 KiB | 
| After Width: | Height: | Size: 982 B | 
| After Width: | Height: | Size: 1.7 KiB | 
| After Width: | Height: | Size: 1.9 KiB | 
| After Width: | Height: | Size: 3.8 KiB | 
| After Width: | Height: | Size: 2.8 KiB | 
| After Width: | Height: | Size: 5.8 KiB | 
| After Width: | Height: | Size: 3.8 KiB | 
| After Width: | Height: | Size: 7.6 KiB | 
							
								
								
									
										10
									
								
								examples/llama.android/app/src/main/res/values/colors.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,10 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <resources> | ||||
|     <color name="purple_200">#FFBB86FC</color> | ||||
|     <color name="purple_500">#FF6200EE</color> | ||||
|     <color name="purple_700">#FF3700B3</color> | ||||
|     <color name="teal_200">#FF03DAC5</color> | ||||
|     <color name="teal_700">#FF018786</color> | ||||
|     <color name="black">#FF000000</color> | ||||
|     <color name="white">#FFFFFFFF</color> | ||||
| </resources> | ||||
|  | @ -0,0 +1,3 @@ | |||
| <resources> | ||||
|     <string name="app_name">LlamaAndroid</string> | ||||
| </resources> | ||||
|  | @ -0,0 +1,5 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <resources> | ||||
| 
 | ||||
|     <style name="Theme.LlamaAndroid" parent="android:Theme.Material.Light.NoActionBar" /> | ||||
| </resources> | ||||
							
								
								
									
										13
									
								
								examples/llama.android/app/src/main/res/xml/backup_rules.xml
									
										
									
									
									
										Normal file
									
								
							
							
						
						|  | @ -0,0 +1,13 @@ | |||
| <?xml version="1.0" encoding="utf-8"?><!-- | ||||
|    Sample backup rules file; uncomment and customize as necessary. | ||||
|    See https://developer.android.com/guide/topics/data/autobackup | ||||
|    for details. | ||||
|    Note: This file is ignored for devices older that API 31 | ||||
|    See https://developer.android.com/about/versions/12/backup-restore | ||||
| --> | ||||
| <full-backup-content> | ||||
|     <!-- | ||||
|    <include domain="sharedpref" path="."/> | ||||
|    <exclude domain="sharedpref" path="device.xml"/> | ||||
| --> | ||||
| </full-backup-content> | ||||
|  | @ -0,0 +1,19 @@ | |||
| <?xml version="1.0" encoding="utf-8"?><!-- | ||||
|    Sample data extraction rules file; uncomment and customize as necessary. | ||||
|    See https://developer.android.com/about/versions/12/backup-restore#xml-changes | ||||
|    for details. | ||||
| --> | ||||
| <data-extraction-rules> | ||||
|     <cloud-backup> | ||||
|         <!-- TODO: Use <include> and <exclude> to control what is backed up. | ||||
|         <include .../> | ||||
|         <exclude .../> | ||||
|         --> | ||||
|     </cloud-backup> | ||||
|     <!-- | ||||
|     <device-transfer> | ||||
|         <include .../> | ||||
|         <exclude .../> | ||||
|     </device-transfer> | ||||
|     --> | ||||
| </data-extraction-rules> | ||||