diff --git a/README.md b/README.md
index ed956bb02e..e59612f7ae 100644
--- a/README.md
+++ b/README.md
@@ -482,21 +482,6 @@ To learn more about model quantization, [read this documentation](tools/quantize
-## [`llama-run`](tools/run)
-
-#### A comprehensive example for running `llama.cpp` models. Useful for inferencing. Used with RamaLama [^3].
-
--
- Run a model with a specific prompt (by default it's pulled from Ollama registry)
-
- ```bash
- llama-run granite-code
- ```
-
-
-
-[^3]: [RamaLama](https://github.com/containers/ramalama)
-
## [`llama-simple`](examples/simple)
#### A minimal example for implementing apps with `llama.cpp`. Useful for developers.
@@ -600,7 +585,6 @@ $ echo "source ~/.llama-completion.bash" >> ~/.bashrc
- [stb-image](https://github.com/nothings/stb) - Single-header image format decoder, used by multimodal subsystem - Public domain
- [nlohmann/json](https://github.com/nlohmann/json) - Single-header JSON library, used by various tools/examples - MIT License
- [minja](https://github.com/google/minja) - Minimal Jinja parser in C++, used by various tools/examples - MIT License
-- [linenoise.cpp](./tools/run/linenoise.cpp/linenoise.cpp) - C++ library that provides readline-like line editing capabilities, used by `llama-run` - BSD 2-Clause License
- [curl](https://curl.se/) - Client-side URL transfer library, used by various tools/examples - [CURL License](https://curl.se/docs/copyright.html)
- [miniaudio.h](https://github.com/mackron/miniaudio) - Single-header audio format decoder, used by multimodal subsystem - Public domain
- [subprocess.h](https://github.com/sheredom/subprocess.h) - Single-header process launching solution for C and C++ - Public domain
diff --git a/common/arg.cpp b/common/arg.cpp
index a67a26e2dc..e7966d9d5c 100644
--- a/common/arg.cpp
+++ b/common/arg.cpp
@@ -679,7 +679,6 @@ static void common_params_print_completion(common_params_context & ctx_arg) {
"llama-quantize",
"llama-qwen2vl-cli",
"llama-retrieval",
- "llama-run",
"llama-save-load-state",
"llama-server",
"llama-simple",
diff --git a/licenses/LICENSE-linenoise b/licenses/LICENSE-linenoise
deleted file mode 100644
index b006b3b24d..0000000000
--- a/licenses/LICENSE-linenoise
+++ /dev/null
@@ -1,26 +0,0 @@
-Copyright (c) 2010-2014, Salvatore Sanfilippo
-Copyright (c) 2010-2013, Pieter Noordhuis
-Copyright (c) 2025, Eric Curtin
-
-All rights reserved.
-
-Redistribution and use in source and binary forms, with or without
-modification, are permitted provided that the following conditions are met:
-
-* Redistributions of source code must retain the above copyright notice,
- this list of conditions and the following disclaimer.
-
-* Redistributions in binary form must reproduce the above copyright notice,
- this list of conditions and the following disclaimer in the documentation
- and/or other materials provided with the distribution.
-
-THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
-ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
-WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
-DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
-ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
-(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
-LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
-ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
-(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
-SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
diff --git a/tools/CMakeLists.txt b/tools/CMakeLists.txt
index 8df3f41003..48959fefb5 100644
--- a/tools/CMakeLists.txt
+++ b/tools/CMakeLists.txt
@@ -25,7 +25,6 @@ else()
if (LLAMA_BUILD_SERVER)
add_subdirectory(server)
endif()
- add_subdirectory(run)
add_subdirectory(tokenize)
add_subdirectory(tts)
add_subdirectory(mtmd)
diff --git a/tools/run/CMakeLists.txt b/tools/run/CMakeLists.txt
deleted file mode 100644
index 6ad7534e29..0000000000
--- a/tools/run/CMakeLists.txt
+++ /dev/null
@@ -1,23 +0,0 @@
-set(TARGET llama-run)
-add_executable(${TARGET} run.cpp linenoise.cpp/linenoise.cpp)
-
-# TODO: avoid copying this code block from common/CMakeLists.txt
-set(LLAMA_RUN_EXTRA_LIBS "")
-if (LLAMA_CURL)
- find_package(CURL REQUIRED)
- target_compile_definitions(${TARGET} PUBLIC LLAMA_USE_CURL)
- include_directories(${CURL_INCLUDE_DIRS})
- set(LLAMA_RUN_EXTRA_LIBS ${LLAMA_RUN_EXTRA_LIBS} ${CURL_LIBRARIES})
-endif ()
-
-if(LLAMA_TOOLS_INSTALL)
- install(TARGETS ${TARGET} RUNTIME)
-endif()
-
-if (CMAKE_SYSTEM_NAME MATCHES "AIX")
- # AIX's flock() function comes from libbsd.a
- target_link_libraries(${TARGET} PRIVATE -lbsd)
-endif()
-
-target_link_libraries(${TARGET} PRIVATE common llama ${CMAKE_THREAD_LIBS_INIT} ${LLAMA_RUN_EXTRA_LIBS})
-target_compile_features(${TARGET} PRIVATE cxx_std_17)
diff --git a/tools/run/README.md b/tools/run/README.md
deleted file mode 100644
index 5fd769b44c..0000000000
--- a/tools/run/README.md
+++ /dev/null
@@ -1,52 +0,0 @@
-# llama.cpp/example/run
-
-The purpose of this example is to demonstrate a minimal usage of llama.cpp for running models.
-
-```bash
-llama-run granite3-moe
-```
-
-```bash
-Description:
- Runs a llm
-
-Usage:
- llama-run [options] model [prompt]
-
-Options:
- -c, --context-size
- Context size (default: 2048)
- -n, -ngl, --ngl
- Number of GPU layers (default: 0)
- --temp
- Temperature (default: 0.8)
- -v, --verbose, --log-verbose
- Set verbosity level to infinity (i.e. log all messages, useful for debugging)
- -h, --help
- Show help message
-
-Commands:
- model
- Model is a string with an optional prefix of
- huggingface:// (hf://), ollama://, https:// or file://.
- If no protocol is specified and a file exists in the specified
- path, file:// is assumed, otherwise if a file does not exist in
- the specified path, ollama:// is assumed. Models that are being
- pulled are downloaded with .partial extension while being
- downloaded and then renamed as the file without the .partial
- extension when complete.
-
-Examples:
- llama-run llama3
- llama-run ollama://granite-code
- llama-run ollama://smollm:135m
- llama-run hf://QuantFactory/SmolLM-135M-GGUF/SmolLM-135M.Q2_K.gguf
- llama-run huggingface://bartowski/SmolLM-1.7B-Instruct-v0.2-GGUF/SmolLM-1.7B-Instruct-v0.2-IQ3_M.gguf
- llama-run ms://QuantFactory/SmolLM-135M-GGUF/SmolLM-135M.Q2_K.gguf
- llama-run modelscope://bartowski/SmolLM-1.7B-Instruct-v0.2-GGUF/SmolLM-1.7B-Instruct-v0.2-IQ3_M.gguf
- llama-run https://example.com/some-file1.gguf
- llama-run some-file2.gguf
- llama-run file://some-file3.gguf
- llama-run --ngl 999 some-file4.gguf
- llama-run --ngl 999 some-file5.gguf Hello World
-```
diff --git a/tools/run/linenoise.cpp/linenoise.cpp b/tools/run/linenoise.cpp/linenoise.cpp
deleted file mode 100644
index 9cb9399003..0000000000
--- a/tools/run/linenoise.cpp/linenoise.cpp
+++ /dev/null
@@ -1,1995 +0,0 @@
-#ifndef _WIN32
-/*
- * You can find the latest source code at:
- *
- * http://github.com/ericcurtin/linenoise.cpp
- *
- * Does a number of crazy assumptions that happen to be true in 99.9999% of
- * the 2010 UNIX computers around.
- *
- * ------------------------------------------------------------------------
- *
- * Copyright (c) 2010-2023, Salvatore Sanfilippo
- * Copyright (c) 2010-2013, Pieter Noordhuis
- * Copyright (c) 2025, Eric Curtin
- *
- * All rights reserved.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions are
- * met:
- *
- * * Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- *
- * * Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- *
- * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
- * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
- * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
- * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
- * HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
- * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
- * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
- * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
- * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
- * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
- * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- *
- * ------------------------------------------------------------------------
- *
- * References:
- * - http://invisible-island.net/xterm/ctlseqs/ctlseqs.html
- * - http://www.3waylabs.com/nw/WWW/products/wizcon/vt220.html
- *
- * Todo list:
- * - Filter bogus Ctrl+ combinations.
- * - Win32 support
- *
- * Bloat:
- * - History search like Ctrl+r in readline?
- *
- * List of escape sequences used by this program, we do everything just
- * with three sequences. In order to be so cheap we may have some
- * flickering effect with some slow terminal, but the lesser sequences
- * the more compatible.
- *
- * EL (Erase Line)
- * Sequence: ESC [ n K
- * Effect: if n is 0 or missing, clear from cursor to end of line
- * Effect: if n is 1, clear from beginning of line to cursor
- * Effect: if n is 2, clear entire line
- *
- * CUF (CUrsor Forward)
- * Sequence: ESC [ n C
- * Effect: moves cursor forward n chars
- *
- * CUB (CUrsor Backward)
- * Sequence: ESC [ n D
- * Effect: moves cursor backward n chars
- *
- * The following is used to get the terminal width if getting
- * the width with the TIOCGWINSZ ioctl fails
- *
- * DSR (Device Status Report)
- * Sequence: ESC [ 6 n
- * Effect: reports the current cursor position as ESC [ n ; m R
- * where n is the row and m is the column
- *
- * When multi line mode is enabled, we also use an additional escape
- * sequence. However multi line editing is disabled by default.
- *
- * CUU (Cursor Up)
- * Sequence: ESC [ n A
- * Effect: moves cursor up of n chars.
- *
- * CUD (Cursor Down)
- * Sequence: ESC [ n B
- * Effect: moves cursor down of n chars.
- *
- * When linenoiseClearScreen() is called, two additional escape sequences
- * are used in order to clear the screen and position the cursor at home
- * position.
- *
- * CUP (Cursor position)
- * Sequence: ESC [ H
- * Effect: moves the cursor to upper left corner
- *
- * ED (Erase display)
- * Sequence: ESC [ 2 J
- * Effect: clear the whole screen
- *
- */
-
-# include "linenoise.h"
-
-# include
-# include
-# include
-# include
-# include
-# include
-# include
-# include
-# include
-# include
-# include
-
-# include
-# include
-# include
-
-# define LINENOISE_DEFAULT_HISTORY_MAX_LEN 100
-# define LINENOISE_MAX_LINE 4096
-static std::vector unsupported_term = { "dumb", "cons25", "emacs" };
-static linenoiseCompletionCallback *completionCallback = NULL;
-static linenoiseHintsCallback *hintsCallback = NULL;
-static linenoiseFreeHintsCallback *freeHintsCallback = NULL;
-static char *linenoiseNoTTY(void);
-static void refreshLineWithCompletion(struct linenoiseState *ls, linenoiseCompletions *lc, int flags);
-static void refreshLineWithFlags(struct linenoiseState *l, int flags);
-
-static struct termios orig_termios; /* In order to restore at exit.*/
-static int maskmode = 0; /* Show "***" instead of input. For passwords. */
-static int rawmode = 0; /* For atexit() function to check if restore is needed*/
-static int mlmode = 0; /* Multi line mode. Default is single line. */
-static int atexit_registered = 0; /* Register atexit just 1 time. */
-static int history_max_len = LINENOISE_DEFAULT_HISTORY_MAX_LEN;
-static int history_len = 0;
-static char **history = NULL;
-
-enum KEY_ACTION{
- KEY_NULL = 0, /* NULL */
- CTRL_A = 1, /* Ctrl+a */
- CTRL_B = 2, /* Ctrl-b */
- CTRL_C = 3, /* Ctrl-c */
- CTRL_D = 4, /* Ctrl-d */
- CTRL_E = 5, /* Ctrl-e */
- CTRL_F = 6, /* Ctrl-f */
- CTRL_H = 8, /* Ctrl-h */
- TAB = 9, /* Tab */
- CTRL_K = 11, /* Ctrl+k */
- CTRL_L = 12, /* Ctrl+l */
- ENTER = 13, /* Enter */
- CTRL_N = 14, /* Ctrl-n */
- CTRL_P = 16, /* Ctrl-p */
- CTRL_T = 20, /* Ctrl-t */
- CTRL_U = 21, /* Ctrl+u */
- CTRL_W = 23, /* Ctrl+w */
- ESC = 27, /* Escape */
- BACKSPACE = 127 /* Backspace */
-};
-
-static void linenoiseAtExit(void);
-int linenoiseHistoryAdd(const char *line);
-#define REFRESH_CLEAN (1<<0) // Clean the old prompt from the screen
-#define REFRESH_WRITE (1<<1) // Rewrite the prompt on the screen.
-#define REFRESH_ALL (REFRESH_CLEAN|REFRESH_WRITE) // Do both.
-static void refreshLine(struct linenoiseState *l);
-
-class File {
- public:
- FILE * file = nullptr;
-
- FILE * open(const std::string & filename, const char * mode) {
- file = fopen(filename.c_str(), mode);
-
- return file;
- }
-
- int lock() {
- if (file) {
- fd = fileno(file);
- if (flock(fd, LOCK_EX | LOCK_NB) != 0) {
- fd = -1;
-
- return 1;
- }
- }
-
- return 0;
- }
-
- ~File() {
- if (fd >= 0) {
- flock(fd, LOCK_UN);
- }
-
- if (file) {
- fclose(file);
- }
- }
-
- private:
- int fd = -1;
-};
-
-#if 0
-/* Debugging function. */
-__attribute__((format(printf, 1, 2)))
-static void lndebug(const char *fmt, ...) {
- static File file;
- if (file.file == nullptr) {
- file.open("/tmp/lndebug.txt", "a");
- }
-
- if (file.file != nullptr) {
- va_list args;
- va_start(args, fmt);
- vfprintf(file.file, fmt, args);
- va_end(args);
- fflush(file.file);
- }
-}
-#endif
-
-/* ========================== Encoding functions ============================= */
-
-/* Get length of previous UTF8 codepoint */
-static size_t prevUtf8CodePointLen(const char * buf, int pos) {
- int end = pos--;
- while (pos >= 0 && ((unsigned char) buf[pos] & 0xC0) == 0x80) {
- pos--;
- }
- return end - pos;
-}
-
-/* Convert UTF8 to Unicode code point */
-static size_t utf8BytesToCodePoint(const char * buf, size_t len, int * cp) {
- if (len) {
- unsigned char byte = buf[0];
- if ((byte & 0x80) == 0) {
- *cp = byte;
- return 1;
- } else if ((byte & 0xE0) == 0xC0) {
- if (len >= 2) {
- *cp = (((unsigned long) (buf[0] & 0x1F)) << 6) | ((unsigned long) (buf[1] & 0x3F));
- return 2;
- }
- } else if ((byte & 0xF0) == 0xE0) {
- if (len >= 3) {
- *cp = (((unsigned long) (buf[0] & 0x0F)) << 12) | (((unsigned long) (buf[1] & 0x3F)) << 6) |
- ((unsigned long) (buf[2] & 0x3F));
- return 3;
- }
- } else if ((byte & 0xF8) == 0xF0) {
- if (len >= 4) {
- *cp = (((unsigned long) (buf[0] & 0x07)) << 18) | (((unsigned long) (buf[1] & 0x3F)) << 12) |
- (((unsigned long) (buf[2] & 0x3F)) << 6) | ((unsigned long) (buf[3] & 0x3F));
- return 4;
- }
- }
- }
- return 0;
-}
-
-/* Check if the code is a wide character */
-static const unsigned long wideCharTable[][2] = {
- /* BEGIN: WIDE CHAR TABLE */
- { 0x1100, 0x115F },
- { 0x231A, 0x231B },
- { 0x2329, 0x232A },
- { 0x23E9, 0x23EC },
- { 0x23F0, 0x23F0 },
- { 0x23F3, 0x23F3 },
- { 0x25FD, 0x25FE },
- { 0x2614, 0x2615 },
- { 0x2630, 0x2637 },
- { 0x2648, 0x2653 },
- { 0x267F, 0x267F },
- { 0x268A, 0x268F },
- { 0x2693, 0x2693 },
- { 0x26A1, 0x26A1 },
- { 0x26AA, 0x26AB },
- { 0x26BD, 0x26BE },
- { 0x26C4, 0x26C5 },
- { 0x26CE, 0x26CE },
- { 0x26D4, 0x26D4 },
- { 0x26EA, 0x26EA },
- { 0x26F2, 0x26F3 },
- { 0x26F5, 0x26F5 },
- { 0x26FA, 0x26FA },
- { 0x26FD, 0x26FD },
- { 0x2705, 0x2705 },
- { 0x270A, 0x270B },
- { 0x2728, 0x2728 },
- { 0x274C, 0x274C },
- { 0x274E, 0x274E },
- { 0x2753, 0x2755 },
- { 0x2757, 0x2757 },
- { 0x2795, 0x2797 },
- { 0x27B0, 0x27B0 },
- { 0x27BF, 0x27BF },
- { 0x2B1B, 0x2B1C },
- { 0x2B50, 0x2B50 },
- { 0x2B55, 0x2B55 },
- { 0x2E80, 0x2E99 },
- { 0x2E9B, 0x2EF3 },
- { 0x2F00, 0x2FD5 },
- { 0x2FF0, 0x303E },
- { 0x3041, 0x3096 },
- { 0x3099, 0x30FF },
- { 0x3105, 0x312F },
- { 0x3131, 0x318E },
- { 0x3190, 0x31E5 },
- { 0x31EF, 0x321E },
- { 0x3220, 0x3247 },
- { 0x3250, 0xA48C },
- { 0xA490, 0xA4C6 },
- { 0xA960, 0xA97C },
- { 0xAC00, 0xD7A3 },
- { 0xF900, 0xFAFF },
- { 0xFE10, 0xFE19 },
- { 0xFE30, 0xFE52 },
- { 0xFE54, 0xFE66 },
- { 0xFE68, 0xFE6B },
- { 0xFF01, 0xFF60 },
- { 0xFFE0, 0xFFE6 },
- { 0x16FE0, 0x16FE4 },
- { 0x16FF0, 0x16FF1 },
- { 0x17000, 0x187F7 },
- { 0x18800, 0x18CD5 },
- { 0x18CFF, 0x18D08 },
- { 0x1AFF0, 0x1AFF3 },
- { 0x1AFF5, 0x1AFFB },
- { 0x1AFFD, 0x1AFFE },
- { 0x1B000, 0x1B122 },
- { 0x1B132, 0x1B132 },
- { 0x1B150, 0x1B152 },
- { 0x1B155, 0x1B155 },
- { 0x1B164, 0x1B167 },
- { 0x1B170, 0x1B2FB },
- { 0x1D300, 0x1D356 },
- { 0x1D360, 0x1D376 },
- { 0x1F004, 0x1F004 },
- { 0x1F0CF, 0x1F0CF },
- { 0x1F18E, 0x1F18E },
- { 0x1F191, 0x1F19A },
- { 0x1F200, 0x1F202 },
- { 0x1F210, 0x1F23B },
- { 0x1F240, 0x1F248 },
- { 0x1F250, 0x1F251 },
- { 0x1F260, 0x1F265 },
- { 0x1F300, 0x1F320 },
- { 0x1F32D, 0x1F335 },
- { 0x1F337, 0x1F37C },
- { 0x1F37E, 0x1F393 },
- { 0x1F3A0, 0x1F3CA },
- { 0x1F3CF, 0x1F3D3 },
- { 0x1F3E0, 0x1F3F0 },
- { 0x1F3F4, 0x1F3F4 },
- { 0x1F3F8, 0x1F43E },
- { 0x1F440, 0x1F440 },
- { 0x1F442, 0x1F4FC },
- { 0x1F4FF, 0x1F53D },
- { 0x1F54B, 0x1F54E },
- { 0x1F550, 0x1F567 },
- { 0x1F57A, 0x1F57A },
- { 0x1F595, 0x1F596 },
- { 0x1F5A4, 0x1F5A4 },
- { 0x1F5FB, 0x1F64F },
- { 0x1F680, 0x1F6C5 },
- { 0x1F6CC, 0x1F6CC },
- { 0x1F6D0, 0x1F6D2 },
- { 0x1F6D5, 0x1F6D7 },
- { 0x1F6DC, 0x1F6DF },
- { 0x1F6EB, 0x1F6EC },
- { 0x1F6F4, 0x1F6FC },
- { 0x1F7E0, 0x1F7EB },
- { 0x1F7F0, 0x1F7F0 },
- { 0x1F90C, 0x1F93A },
- { 0x1F93C, 0x1F945 },
- { 0x1F947, 0x1F9FF },
- { 0x1FA70, 0x1FA7C },
- { 0x1FA80, 0x1FA89 },
- { 0x1FA8F, 0x1FAC6 },
- { 0x1FACE, 0x1FADC },
- { 0x1FADF, 0x1FAE9 },
- { 0x1FAF0, 0x1FAF8 },
- { 0x20000, 0x2FFFD },
- { 0x30000, 0x3FFFD }
- /* END: WIDE CHAR TABLE */
-};
-
-static const size_t wideCharTableSize = sizeof(wideCharTable) / sizeof(wideCharTable[0]);
-
-static bool isWideChar(unsigned long cp) {
- for (size_t i = 0; i < wideCharTableSize; i++) {
- auto first_code = wideCharTable[i][0];
- auto last_code = wideCharTable[i][1];
- if (first_code > cp) {
- return false;
- }
- if (first_code <= cp && cp <= last_code) {
- return true;
- }
- }
- return false;
-}
-
-/* Check if the code is a combining character */
-static const unsigned long combiningCharTable[] = {
- /* BEGIN: COMBINING CHAR TABLE */
- 0x0300, 0x0301, 0x0302, 0x0303, 0x0304, 0x0305, 0x0306, 0x0307, 0x0308, 0x0309, 0x030A, 0x030B, 0x030C,
- 0x030D, 0x030E, 0x030F, 0x0310, 0x0311, 0x0312, 0x0313, 0x0314, 0x0315, 0x0316, 0x0317, 0x0318, 0x0319,
- 0x031A, 0x031B, 0x031C, 0x031D, 0x031E, 0x031F, 0x0320, 0x0321, 0x0322, 0x0323, 0x0324, 0x0325, 0x0326,
- 0x0327, 0x0328, 0x0329, 0x032A, 0x032B, 0x032C, 0x032D, 0x032E, 0x032F, 0x0330, 0x0331, 0x0332, 0x0333,
- 0x0334, 0x0335, 0x0336, 0x0337, 0x0338, 0x0339, 0x033A, 0x033B, 0x033C, 0x033D, 0x033E, 0x033F, 0x0340,
- 0x0341, 0x0342, 0x0343, 0x0344, 0x0345, 0x0346, 0x0347, 0x0348, 0x0349, 0x034A, 0x034B, 0x034C, 0x034D,
- 0x034E, 0x034F, 0x0350, 0x0351, 0x0352, 0x0353, 0x0354, 0x0355, 0x0356, 0x0357, 0x0358, 0x0359, 0x035A,
- 0x035B, 0x035C, 0x035D, 0x035E, 0x035F, 0x0360, 0x0361, 0x0362, 0x0363, 0x0364, 0x0365, 0x0366, 0x0367,
- 0x0368, 0x0369, 0x036A, 0x036B, 0x036C, 0x036D, 0x036E, 0x036F, 0x0483, 0x0484, 0x0485, 0x0486, 0x0487,
- 0x0591, 0x0592, 0x0593, 0x0594, 0x0595, 0x0596, 0x0597, 0x0598, 0x0599, 0x059A, 0x059B, 0x059C, 0x059D,
- 0x059E, 0x059F, 0x05A0, 0x05A1, 0x05A2, 0x05A3, 0x05A4, 0x05A5, 0x05A6, 0x05A7, 0x05A8, 0x05A9, 0x05AA,
- 0x05AB, 0x05AC, 0x05AD, 0x05AE, 0x05AF, 0x05B0, 0x05B1, 0x05B2, 0x05B3, 0x05B4, 0x05B5, 0x05B6, 0x05B7,
- 0x05B8, 0x05B9, 0x05BA, 0x05BB, 0x05BC, 0x05BD, 0x05BF, 0x05C1, 0x05C2, 0x05C4, 0x05C5, 0x05C7, 0x0610,
- 0x0611, 0x0612, 0x0613, 0x0614, 0x0615, 0x0616, 0x0617, 0x0618, 0x0619, 0x061A, 0x064B, 0x064C, 0x064D,
- 0x064E, 0x064F, 0x0650, 0x0651, 0x0652, 0x0653, 0x0654, 0x0655, 0x0656, 0x0657, 0x0658, 0x0659, 0x065A,
- 0x065B, 0x065C, 0x065D, 0x065E, 0x065F, 0x0670, 0x06D6, 0x06D7, 0x06D8, 0x06D9, 0x06DA, 0x06DB, 0x06DC,
- 0x06DF, 0x06E0, 0x06E1, 0x06E2, 0x06E3, 0x06E4, 0x06E7, 0x06E8, 0x06EA, 0x06EB, 0x06EC, 0x06ED, 0x0711,
- 0x0730, 0x0731, 0x0732, 0x0733, 0x0734, 0x0735, 0x0736, 0x0737, 0x0738, 0x0739, 0x073A, 0x073B, 0x073C,
- 0x073D, 0x073E, 0x073F, 0x0740, 0x0741, 0x0742, 0x0743, 0x0744, 0x0745, 0x0746, 0x0747, 0x0748, 0x0749,
- 0x074A, 0x07A6, 0x07A7, 0x07A8, 0x07A9, 0x07AA, 0x07AB, 0x07AC, 0x07AD, 0x07AE, 0x07AF, 0x07B0, 0x07EB,
- 0x07EC, 0x07ED, 0x07EE, 0x07EF, 0x07F0, 0x07F1, 0x07F2, 0x07F3, 0x07FD, 0x0816, 0x0817, 0x0818, 0x0819,
- 0x081B, 0x081C, 0x081D, 0x081E, 0x081F, 0x0820, 0x0821, 0x0822, 0x0823, 0x0825, 0x0826, 0x0827, 0x0829,
- 0x082A, 0x082B, 0x082C, 0x082D, 0x0859, 0x085A, 0x085B, 0x0897, 0x0898, 0x0899, 0x089A, 0x089B, 0x089C,
- 0x089D, 0x089E, 0x089F, 0x08CA, 0x08CB, 0x08CC, 0x08CD, 0x08CE, 0x08CF, 0x08D0, 0x08D1, 0x08D2, 0x08D3,
- 0x08D4, 0x08D5, 0x08D6, 0x08D7, 0x08D8, 0x08D9, 0x08DA, 0x08DB, 0x08DC, 0x08DD, 0x08DE, 0x08DF, 0x08E0,
- 0x08E1, 0x08E3, 0x08E4, 0x08E5, 0x08E6, 0x08E7, 0x08E8, 0x08E9, 0x08EA, 0x08EB, 0x08EC, 0x08ED, 0x08EE,
- 0x08EF, 0x08F0, 0x08F1, 0x08F2, 0x08F3, 0x08F4, 0x08F5, 0x08F6, 0x08F7, 0x08F8, 0x08F9, 0x08FA, 0x08FB,
- 0x08FC, 0x08FD, 0x08FE, 0x08FF, 0x0900, 0x0901, 0x0902, 0x093A, 0x093C, 0x0941, 0x0942, 0x0943, 0x0944,
- 0x0945, 0x0946, 0x0947, 0x0948, 0x094D, 0x0951, 0x0952, 0x0953, 0x0954, 0x0955, 0x0956, 0x0957, 0x0962,
- 0x0963, 0x0981, 0x09BC, 0x09C1, 0x09C2, 0x09C3, 0x09C4, 0x09CD, 0x09E2, 0x09E3, 0x09FE, 0x0A01, 0x0A02,
- 0x0A3C, 0x0A41, 0x0A42, 0x0A47, 0x0A48, 0x0A4B, 0x0A4C, 0x0A4D, 0x0A51, 0x0A70, 0x0A71, 0x0A75, 0x0A81,
- 0x0A82, 0x0ABC, 0x0AC1, 0x0AC2, 0x0AC3, 0x0AC4, 0x0AC5, 0x0AC7, 0x0AC8, 0x0ACD, 0x0AE2, 0x0AE3, 0x0AFA,
- 0x0AFB, 0x0AFC, 0x0AFD, 0x0AFE, 0x0AFF, 0x0B01, 0x0B3C, 0x0B3F, 0x0B41, 0x0B42, 0x0B43, 0x0B44, 0x0B4D,
- 0x0B55, 0x0B56, 0x0B62, 0x0B63, 0x0B82, 0x0BC0, 0x0BCD, 0x0C00, 0x0C04, 0x0C3C, 0x0C3E, 0x0C3F, 0x0C40,
- 0x0C46, 0x0C47, 0x0C48, 0x0C4A, 0x0C4B, 0x0C4C, 0x0C4D, 0x0C55, 0x0C56, 0x0C62, 0x0C63, 0x0C81, 0x0CBC,
- 0x0CBF, 0x0CC6, 0x0CCC, 0x0CCD, 0x0CE2, 0x0CE3, 0x0D00, 0x0D01, 0x0D3B, 0x0D3C, 0x0D41, 0x0D42, 0x0D43,
- 0x0D44, 0x0D4D, 0x0D62, 0x0D63, 0x0D81, 0x0DCA, 0x0DD2, 0x0DD3, 0x0DD4, 0x0DD6, 0x0E31, 0x0E34, 0x0E35,
- 0x0E36, 0x0E37, 0x0E38, 0x0E39, 0x0E3A, 0x0E47, 0x0E48, 0x0E49, 0x0E4A, 0x0E4B, 0x0E4C, 0x0E4D, 0x0E4E,
- 0x0EB1, 0x0EB4, 0x0EB5, 0x0EB6, 0x0EB7, 0x0EB8, 0x0EB9, 0x0EBA, 0x0EBB, 0x0EBC, 0x0EC8, 0x0EC9, 0x0ECA,
- 0x0ECB, 0x0ECC, 0x0ECD, 0x0ECE, 0x0F18, 0x0F19, 0x0F35, 0x0F37, 0x0F39, 0x0F71, 0x0F72, 0x0F73, 0x0F74,
- 0x0F75, 0x0F76, 0x0F77, 0x0F78, 0x0F79, 0x0F7A, 0x0F7B, 0x0F7C, 0x0F7D, 0x0F7E, 0x0F80, 0x0F81, 0x0F82,
- 0x0F83, 0x0F84, 0x0F86, 0x0F87, 0x0F8D, 0x0F8E, 0x0F8F, 0x0F90, 0x0F91, 0x0F92, 0x0F93, 0x0F94, 0x0F95,
- 0x0F96, 0x0F97, 0x0F99, 0x0F9A, 0x0F9B, 0x0F9C, 0x0F9D, 0x0F9E, 0x0F9F, 0x0FA0, 0x0FA1, 0x0FA2, 0x0FA3,
- 0x0FA4, 0x0FA5, 0x0FA6, 0x0FA7, 0x0FA8, 0x0FA9, 0x0FAA, 0x0FAB, 0x0FAC, 0x0FAD, 0x0FAE, 0x0FAF, 0x0FB0,
- 0x0FB1, 0x0FB2, 0x0FB3, 0x0FB4, 0x0FB5, 0x0FB6, 0x0FB7, 0x0FB8, 0x0FB9, 0x0FBA, 0x0FBB, 0x0FBC, 0x0FC6,
- 0x102D, 0x102E, 0x102F, 0x1030, 0x1032, 0x1033, 0x1034, 0x1035, 0x1036, 0x1037, 0x1039, 0x103A, 0x103D,
- 0x103E, 0x1058, 0x1059, 0x105E, 0x105F, 0x1060, 0x1071, 0x1072, 0x1073, 0x1074, 0x1082, 0x1085, 0x1086,
- 0x108D, 0x109D, 0x135D, 0x135E, 0x135F, 0x1712, 0x1713, 0x1714, 0x1732, 0x1733, 0x1752, 0x1753, 0x1772,
- 0x1773, 0x17B4, 0x17B5, 0x17B7, 0x17B8, 0x17B9, 0x17BA, 0x17BB, 0x17BC, 0x17BD, 0x17C6, 0x17C9, 0x17CA,
- 0x17CB, 0x17CC, 0x17CD, 0x17CE, 0x17CF, 0x17D0, 0x17D1, 0x17D2, 0x17D3, 0x17DD, 0x180B, 0x180C, 0x180D,
- 0x180F, 0x1885, 0x1886, 0x18A9, 0x1920, 0x1921, 0x1922, 0x1927, 0x1928, 0x1932, 0x1939, 0x193A, 0x193B,
- 0x1A17, 0x1A18, 0x1A1B, 0x1A56, 0x1A58, 0x1A59, 0x1A5A, 0x1A5B, 0x1A5C, 0x1A5D, 0x1A5E, 0x1A60, 0x1A62,
- 0x1A65, 0x1A66, 0x1A67, 0x1A68, 0x1A69, 0x1A6A, 0x1A6B, 0x1A6C, 0x1A73, 0x1A74, 0x1A75, 0x1A76, 0x1A77,
- 0x1A78, 0x1A79, 0x1A7A, 0x1A7B, 0x1A7C, 0x1A7F, 0x1AB0, 0x1AB1, 0x1AB2, 0x1AB3, 0x1AB4, 0x1AB5, 0x1AB6,
- 0x1AB7, 0x1AB8, 0x1AB9, 0x1ABA, 0x1ABB, 0x1ABC, 0x1ABD, 0x1ABF, 0x1AC0, 0x1AC1, 0x1AC2, 0x1AC3, 0x1AC4,
- 0x1AC5, 0x1AC6, 0x1AC7, 0x1AC8, 0x1AC9, 0x1ACA, 0x1ACB, 0x1ACC, 0x1ACD, 0x1ACE, 0x1B00, 0x1B01, 0x1B02,
- 0x1B03, 0x1B34, 0x1B36, 0x1B37, 0x1B38, 0x1B39, 0x1B3A, 0x1B3C, 0x1B42, 0x1B6B, 0x1B6C, 0x1B6D, 0x1B6E,
- 0x1B6F, 0x1B70, 0x1B71, 0x1B72, 0x1B73, 0x1B80, 0x1B81, 0x1BA2, 0x1BA3, 0x1BA4, 0x1BA5, 0x1BA8, 0x1BA9,
- 0x1BAB, 0x1BAC, 0x1BAD, 0x1BE6, 0x1BE8, 0x1BE9, 0x1BED, 0x1BEF, 0x1BF0, 0x1BF1, 0x1C2C, 0x1C2D, 0x1C2E,
- 0x1C2F, 0x1C30, 0x1C31, 0x1C32, 0x1C33, 0x1C36, 0x1C37, 0x1CD0, 0x1CD1, 0x1CD2, 0x1CD4, 0x1CD5, 0x1CD6,
- 0x1CD7, 0x1CD8, 0x1CD9, 0x1CDA, 0x1CDB, 0x1CDC, 0x1CDD, 0x1CDE, 0x1CDF, 0x1CE0, 0x1CE2, 0x1CE3, 0x1CE4,
- 0x1CE5, 0x1CE6, 0x1CE7, 0x1CE8, 0x1CED, 0x1CF4, 0x1CF8, 0x1CF9, 0x1DC0, 0x1DC1, 0x1DC2, 0x1DC3, 0x1DC4,
- 0x1DC5, 0x1DC6, 0x1DC7, 0x1DC8, 0x1DC9, 0x1DCA, 0x1DCB, 0x1DCC, 0x1DCD, 0x1DCE, 0x1DCF, 0x1DD0, 0x1DD1,
- 0x1DD2, 0x1DD3, 0x1DD4, 0x1DD5, 0x1DD6, 0x1DD7, 0x1DD8, 0x1DD9, 0x1DDA, 0x1DDB, 0x1DDC, 0x1DDD, 0x1DDE,
- 0x1DDF, 0x1DE0, 0x1DE1, 0x1DE2, 0x1DE3, 0x1DE4, 0x1DE5, 0x1DE6, 0x1DE7, 0x1DE8, 0x1DE9, 0x1DEA, 0x1DEB,
- 0x1DEC, 0x1DED, 0x1DEE, 0x1DEF, 0x1DF0, 0x1DF1, 0x1DF2, 0x1DF3, 0x1DF4, 0x1DF5, 0x1DF6, 0x1DF7, 0x1DF8,
- 0x1DF9, 0x1DFA, 0x1DFB, 0x1DFC, 0x1DFD, 0x1DFE, 0x1DFF, 0x20D0, 0x20D1, 0x20D2, 0x20D3, 0x20D4, 0x20D5,
- 0x20D6, 0x20D7, 0x20D8, 0x20D9, 0x20DA, 0x20DB, 0x20DC, 0x20E1, 0x20E5, 0x20E6, 0x20E7, 0x20E8, 0x20E9,
- 0x20EA, 0x20EB, 0x20EC, 0x20ED, 0x20EE, 0x20EF, 0x20F0, 0x2CEF, 0x2CF0, 0x2CF1, 0x2D7F, 0x2DE0, 0x2DE1,
- 0x2DE2, 0x2DE3, 0x2DE4, 0x2DE5, 0x2DE6, 0x2DE7, 0x2DE8, 0x2DE9, 0x2DEA, 0x2DEB, 0x2DEC, 0x2DED, 0x2DEE,
- 0x2DEF, 0x2DF0, 0x2DF1, 0x2DF2, 0x2DF3, 0x2DF4, 0x2DF5, 0x2DF6, 0x2DF7, 0x2DF8, 0x2DF9, 0x2DFA, 0x2DFB,
- 0x2DFC, 0x2DFD, 0x2DFE, 0x2DFF, 0x302A, 0x302B, 0x302C, 0x302D, 0x3099, 0x309A, 0xA66F, 0xA674, 0xA675,
- 0xA676, 0xA677, 0xA678, 0xA679, 0xA67A, 0xA67B, 0xA67C, 0xA67D, 0xA69E, 0xA69F, 0xA6F0, 0xA6F1, 0xA802,
- 0xA806, 0xA80B, 0xA825, 0xA826, 0xA82C, 0xA8C4, 0xA8C5, 0xA8E0, 0xA8E1, 0xA8E2, 0xA8E3, 0xA8E4, 0xA8E5,
- 0xA8E6, 0xA8E7, 0xA8E8, 0xA8E9, 0xA8EA, 0xA8EB, 0xA8EC, 0xA8ED, 0xA8EE, 0xA8EF, 0xA8F0, 0xA8F1, 0xA8FF,
- 0xA926, 0xA927, 0xA928, 0xA929, 0xA92A, 0xA92B, 0xA92C, 0xA92D, 0xA947, 0xA948, 0xA949, 0xA94A, 0xA94B,
- 0xA94C, 0xA94D, 0xA94E, 0xA94F, 0xA950, 0xA951, 0xA980, 0xA981, 0xA982, 0xA9B3, 0xA9B6, 0xA9B7, 0xA9B8,
- 0xA9B9, 0xA9BC, 0xA9BD, 0xA9E5, 0xAA29, 0xAA2A, 0xAA2B, 0xAA2C, 0xAA2D, 0xAA2E, 0xAA31, 0xAA32, 0xAA35,
- 0xAA36, 0xAA43, 0xAA4C, 0xAA7C, 0xAAB0, 0xAAB2, 0xAAB3, 0xAAB4, 0xAAB7, 0xAAB8, 0xAABE, 0xAABF, 0xAAC1,
- 0xAAEC, 0xAAED, 0xAAF6, 0xABE5, 0xABE8, 0xABED, 0xFB1E, 0xFE00, 0xFE01, 0xFE02, 0xFE03, 0xFE04, 0xFE05,
- 0xFE06, 0xFE07, 0xFE08, 0xFE09, 0xFE0A, 0xFE0B, 0xFE0C, 0xFE0D, 0xFE0E, 0xFE0F, 0xFE20, 0xFE21, 0xFE22,
- 0xFE23, 0xFE24, 0xFE25, 0xFE26, 0xFE27, 0xFE28, 0xFE29, 0xFE2A, 0xFE2B, 0xFE2C, 0xFE2D, 0xFE2E, 0xFE2F,
- 0x101FD, 0x102E0, 0x10376, 0x10377, 0x10378, 0x10379, 0x1037A, 0x10A01, 0x10A02, 0x10A03, 0x10A05, 0x10A06, 0x10A0C,
- 0x10A0D, 0x10A0E, 0x10A0F, 0x10A38, 0x10A39, 0x10A3A, 0x10A3F, 0x10AE5, 0x10AE6, 0x10D24, 0x10D25, 0x10D26, 0x10D27,
- 0x10D69, 0x10D6A, 0x10D6B, 0x10D6C, 0x10D6D, 0x10EAB, 0x10EAC, 0x10EFC, 0x10EFD, 0x10EFE, 0x10EFF, 0x10F46, 0x10F47,
- 0x10F48, 0x10F49, 0x10F4A, 0x10F4B, 0x10F4C, 0x10F4D, 0x10F4E, 0x10F4F, 0x10F50, 0x10F82, 0x10F83, 0x10F84, 0x10F85,
- 0x11001, 0x11038, 0x11039, 0x1103A, 0x1103B, 0x1103C, 0x1103D, 0x1103E, 0x1103F, 0x11040, 0x11041, 0x11042, 0x11043,
- 0x11044, 0x11045, 0x11046, 0x11070, 0x11073, 0x11074, 0x1107F, 0x11080, 0x11081, 0x110B3, 0x110B4, 0x110B5, 0x110B6,
- 0x110B9, 0x110BA, 0x110C2, 0x11100, 0x11101, 0x11102, 0x11127, 0x11128, 0x11129, 0x1112A, 0x1112B, 0x1112D, 0x1112E,
- 0x1112F, 0x11130, 0x11131, 0x11132, 0x11133, 0x11134, 0x11173, 0x11180, 0x11181, 0x111B6, 0x111B7, 0x111B8, 0x111B9,
- 0x111BA, 0x111BB, 0x111BC, 0x111BD, 0x111BE, 0x111C9, 0x111CA, 0x111CB, 0x111CC, 0x111CF, 0x1122F, 0x11230, 0x11231,
- 0x11234, 0x11236, 0x11237, 0x1123E, 0x11241, 0x112DF, 0x112E3, 0x112E4, 0x112E5, 0x112E6, 0x112E7, 0x112E8, 0x112E9,
- 0x112EA, 0x11300, 0x11301, 0x1133B, 0x1133C, 0x11340, 0x11366, 0x11367, 0x11368, 0x11369, 0x1136A, 0x1136B, 0x1136C,
- 0x11370, 0x11371, 0x11372, 0x11373, 0x11374, 0x113BB, 0x113BC, 0x113BD, 0x113BE, 0x113BF, 0x113C0, 0x113CE, 0x113D0,
- 0x113D2, 0x113E1, 0x113E2, 0x11438, 0x11439, 0x1143A, 0x1143B, 0x1143C, 0x1143D, 0x1143E, 0x1143F, 0x11442, 0x11443,
- 0x11444, 0x11446, 0x1145E, 0x114B3, 0x114B4, 0x114B5, 0x114B6, 0x114B7, 0x114B8, 0x114BA, 0x114BF, 0x114C0, 0x114C2,
- 0x114C3, 0x115B2, 0x115B3, 0x115B4, 0x115B5, 0x115BC, 0x115BD, 0x115BF, 0x115C0, 0x115DC, 0x115DD, 0x11633, 0x11634,
- 0x11635, 0x11636, 0x11637, 0x11638, 0x11639, 0x1163A, 0x1163D, 0x1163F, 0x11640, 0x116AB, 0x116AD, 0x116B0, 0x116B1,
- 0x116B2, 0x116B3, 0x116B4, 0x116B5, 0x116B7, 0x1171D, 0x1171F, 0x11722, 0x11723, 0x11724, 0x11725, 0x11727, 0x11728,
- 0x11729, 0x1172A, 0x1172B, 0x1182F, 0x11830, 0x11831, 0x11832, 0x11833, 0x11834, 0x11835, 0x11836, 0x11837, 0x11839,
- 0x1183A, 0x1193B, 0x1193C, 0x1193E, 0x11943, 0x119D4, 0x119D5, 0x119D6, 0x119D7, 0x119DA, 0x119DB, 0x119E0, 0x11A01,
- 0x11A02, 0x11A03, 0x11A04, 0x11A05, 0x11A06, 0x11A07, 0x11A08, 0x11A09, 0x11A0A, 0x11A33, 0x11A34, 0x11A35, 0x11A36,
- 0x11A37, 0x11A38, 0x11A3B, 0x11A3C, 0x11A3D, 0x11A3E, 0x11A47, 0x11A51, 0x11A52, 0x11A53, 0x11A54, 0x11A55, 0x11A56,
- 0x11A59, 0x11A5A, 0x11A5B, 0x11A8A, 0x11A8B, 0x11A8C, 0x11A8D, 0x11A8E, 0x11A8F, 0x11A90, 0x11A91, 0x11A92, 0x11A93,
- 0x11A94, 0x11A95, 0x11A96, 0x11A98, 0x11A99, 0x11C30, 0x11C31, 0x11C32, 0x11C33, 0x11C34, 0x11C35, 0x11C36, 0x11C38,
- 0x11C39, 0x11C3A, 0x11C3B, 0x11C3C, 0x11C3D, 0x11C3F, 0x11C92, 0x11C93, 0x11C94, 0x11C95, 0x11C96, 0x11C97, 0x11C98,
- 0x11C99, 0x11C9A, 0x11C9B, 0x11C9C, 0x11C9D, 0x11C9E, 0x11C9F, 0x11CA0, 0x11CA1, 0x11CA2, 0x11CA3, 0x11CA4, 0x11CA5,
- 0x11CA6, 0x11CA7, 0x11CAA, 0x11CAB, 0x11CAC, 0x11CAD, 0x11CAE, 0x11CAF, 0x11CB0, 0x11CB2, 0x11CB3, 0x11CB5, 0x11CB6,
- 0x11D31, 0x11D32, 0x11D33, 0x11D34, 0x11D35, 0x11D36, 0x11D3A, 0x11D3C, 0x11D3D, 0x11D3F, 0x11D40, 0x11D41, 0x11D42,
- 0x11D43, 0x11D44, 0x11D45, 0x11D47, 0x11D90, 0x11D91, 0x11D95, 0x11D97, 0x11EF3, 0x11EF4, 0x11F00, 0x11F01, 0x11F36,
- 0x11F37, 0x11F38, 0x11F39, 0x11F3A, 0x11F40, 0x11F42, 0x11F5A, 0x13440, 0x13447, 0x13448, 0x13449, 0x1344A, 0x1344B,
- 0x1344C, 0x1344D, 0x1344E, 0x1344F, 0x13450, 0x13451, 0x13452, 0x13453, 0x13454, 0x13455, 0x1611E, 0x1611F, 0x16120,
- 0x16121, 0x16122, 0x16123, 0x16124, 0x16125, 0x16126, 0x16127, 0x16128, 0x16129, 0x1612D, 0x1612E, 0x1612F, 0x16AF0,
- 0x16AF1, 0x16AF2, 0x16AF3, 0x16AF4, 0x16B30, 0x16B31, 0x16B32, 0x16B33, 0x16B34, 0x16B35, 0x16B36, 0x16F4F, 0x16F8F,
- 0x16F90, 0x16F91, 0x16F92, 0x16FE4, 0x1BC9D, 0x1BC9E, 0x1CF00, 0x1CF01, 0x1CF02, 0x1CF03, 0x1CF04, 0x1CF05, 0x1CF06,
- 0x1CF07, 0x1CF08, 0x1CF09, 0x1CF0A, 0x1CF0B, 0x1CF0C, 0x1CF0D, 0x1CF0E, 0x1CF0F, 0x1CF10, 0x1CF11, 0x1CF12, 0x1CF13,
- 0x1CF14, 0x1CF15, 0x1CF16, 0x1CF17, 0x1CF18, 0x1CF19, 0x1CF1A, 0x1CF1B, 0x1CF1C, 0x1CF1D, 0x1CF1E, 0x1CF1F, 0x1CF20,
- 0x1CF21, 0x1CF22, 0x1CF23, 0x1CF24, 0x1CF25, 0x1CF26, 0x1CF27, 0x1CF28, 0x1CF29, 0x1CF2A, 0x1CF2B, 0x1CF2C, 0x1CF2D,
- 0x1CF30, 0x1CF31, 0x1CF32, 0x1CF33, 0x1CF34, 0x1CF35, 0x1CF36, 0x1CF37, 0x1CF38, 0x1CF39, 0x1CF3A, 0x1CF3B, 0x1CF3C,
- 0x1CF3D, 0x1CF3E, 0x1CF3F, 0x1CF40, 0x1CF41, 0x1CF42, 0x1CF43, 0x1CF44, 0x1CF45, 0x1CF46, 0x1D167, 0x1D168, 0x1D169,
- 0x1D17B, 0x1D17C, 0x1D17D, 0x1D17E, 0x1D17F, 0x1D180, 0x1D181, 0x1D182, 0x1D185, 0x1D186, 0x1D187, 0x1D188, 0x1D189,
- 0x1D18A, 0x1D18B, 0x1D1AA, 0x1D1AB, 0x1D1AC, 0x1D1AD, 0x1D242, 0x1D243, 0x1D244, 0x1DA00, 0x1DA01, 0x1DA02, 0x1DA03,
- 0x1DA04, 0x1DA05, 0x1DA06, 0x1DA07, 0x1DA08, 0x1DA09, 0x1DA0A, 0x1DA0B, 0x1DA0C, 0x1DA0D, 0x1DA0E, 0x1DA0F, 0x1DA10,
- 0x1DA11, 0x1DA12, 0x1DA13, 0x1DA14, 0x1DA15, 0x1DA16, 0x1DA17, 0x1DA18, 0x1DA19, 0x1DA1A, 0x1DA1B, 0x1DA1C, 0x1DA1D,
- 0x1DA1E, 0x1DA1F, 0x1DA20, 0x1DA21, 0x1DA22, 0x1DA23, 0x1DA24, 0x1DA25, 0x1DA26, 0x1DA27, 0x1DA28, 0x1DA29, 0x1DA2A,
- 0x1DA2B, 0x1DA2C, 0x1DA2D, 0x1DA2E, 0x1DA2F, 0x1DA30, 0x1DA31, 0x1DA32, 0x1DA33, 0x1DA34, 0x1DA35, 0x1DA36, 0x1DA3B,
- 0x1DA3C, 0x1DA3D, 0x1DA3E, 0x1DA3F, 0x1DA40, 0x1DA41, 0x1DA42, 0x1DA43, 0x1DA44, 0x1DA45, 0x1DA46, 0x1DA47, 0x1DA48,
- 0x1DA49, 0x1DA4A, 0x1DA4B, 0x1DA4C, 0x1DA4D, 0x1DA4E, 0x1DA4F, 0x1DA50, 0x1DA51, 0x1DA52, 0x1DA53, 0x1DA54, 0x1DA55,
- 0x1DA56, 0x1DA57, 0x1DA58, 0x1DA59, 0x1DA5A, 0x1DA5B, 0x1DA5C, 0x1DA5D, 0x1DA5E, 0x1DA5F, 0x1DA60, 0x1DA61, 0x1DA62,
- 0x1DA63, 0x1DA64, 0x1DA65, 0x1DA66, 0x1DA67, 0x1DA68, 0x1DA69, 0x1DA6A, 0x1DA6B, 0x1DA6C, 0x1DA75, 0x1DA84, 0x1DA9B,
- 0x1DA9C, 0x1DA9D, 0x1DA9E, 0x1DA9F, 0x1DAA1, 0x1DAA2, 0x1DAA3, 0x1DAA4, 0x1DAA5, 0x1DAA6, 0x1DAA7, 0x1DAA8, 0x1DAA9,
- 0x1DAAA, 0x1DAAB, 0x1DAAC, 0x1DAAD, 0x1DAAE, 0x1DAAF, 0x1E000, 0x1E001, 0x1E002, 0x1E003, 0x1E004, 0x1E005, 0x1E006,
- 0x1E008, 0x1E009, 0x1E00A, 0x1E00B, 0x1E00C, 0x1E00D, 0x1E00E, 0x1E00F, 0x1E010, 0x1E011, 0x1E012, 0x1E013, 0x1E014,
- 0x1E015, 0x1E016, 0x1E017, 0x1E018, 0x1E01B, 0x1E01C, 0x1E01D, 0x1E01E, 0x1E01F, 0x1E020, 0x1E021, 0x1E023, 0x1E024,
- 0x1E026, 0x1E027, 0x1E028, 0x1E029, 0x1E02A, 0x1E08F, 0x1E130, 0x1E131, 0x1E132, 0x1E133, 0x1E134, 0x1E135, 0x1E136,
- 0x1E2AE, 0x1E2EC, 0x1E2ED, 0x1E2EE, 0x1E2EF, 0x1E4EC, 0x1E4ED, 0x1E4EE, 0x1E4EF, 0x1E5EE, 0x1E5EF, 0x1E8D0, 0x1E8D1,
- 0x1E8D2, 0x1E8D3, 0x1E8D4, 0x1E8D5, 0x1E8D6, 0x1E944, 0x1E945, 0x1E946, 0x1E947, 0x1E948, 0x1E949, 0x1E94A, 0xE0100,
- 0xE0101, 0xE0102, 0xE0103, 0xE0104, 0xE0105, 0xE0106, 0xE0107, 0xE0108, 0xE0109, 0xE010A, 0xE010B, 0xE010C, 0xE010D,
- 0xE010E, 0xE010F, 0xE0110, 0xE0111, 0xE0112, 0xE0113, 0xE0114, 0xE0115, 0xE0116, 0xE0117, 0xE0118, 0xE0119, 0xE011A,
- 0xE011B, 0xE011C, 0xE011D, 0xE011E, 0xE011F, 0xE0120, 0xE0121, 0xE0122, 0xE0123, 0xE0124, 0xE0125, 0xE0126, 0xE0127,
- 0xE0128, 0xE0129, 0xE012A, 0xE012B, 0xE012C, 0xE012D, 0xE012E, 0xE012F, 0xE0130, 0xE0131, 0xE0132, 0xE0133, 0xE0134,
- 0xE0135, 0xE0136, 0xE0137, 0xE0138, 0xE0139, 0xE013A, 0xE013B, 0xE013C, 0xE013D, 0xE013E, 0xE013F, 0xE0140, 0xE0141,
- 0xE0142, 0xE0143, 0xE0144, 0xE0145, 0xE0146, 0xE0147, 0xE0148, 0xE0149, 0xE014A, 0xE014B, 0xE014C, 0xE014D, 0xE014E,
- 0xE014F, 0xE0150, 0xE0151, 0xE0152, 0xE0153, 0xE0154, 0xE0155, 0xE0156, 0xE0157, 0xE0158, 0xE0159, 0xE015A, 0xE015B,
- 0xE015C, 0xE015D, 0xE015E, 0xE015F, 0xE0160, 0xE0161, 0xE0162, 0xE0163, 0xE0164, 0xE0165, 0xE0166, 0xE0167, 0xE0168,
- 0xE0169, 0xE016A, 0xE016B, 0xE016C, 0xE016D, 0xE016E, 0xE016F, 0xE0170, 0xE0171, 0xE0172, 0xE0173, 0xE0174, 0xE0175,
- 0xE0176, 0xE0177, 0xE0178, 0xE0179, 0xE017A, 0xE017B, 0xE017C, 0xE017D, 0xE017E, 0xE017F, 0xE0180, 0xE0181, 0xE0182,
- 0xE0183, 0xE0184, 0xE0185, 0xE0186, 0xE0187, 0xE0188, 0xE0189, 0xE018A, 0xE018B, 0xE018C, 0xE018D, 0xE018E, 0xE018F,
- 0xE0190, 0xE0191, 0xE0192, 0xE0193, 0xE0194, 0xE0195, 0xE0196, 0xE0197, 0xE0198, 0xE0199, 0xE019A, 0xE019B, 0xE019C,
- 0xE019D, 0xE019E, 0xE019F, 0xE01A0, 0xE01A1, 0xE01A2, 0xE01A3, 0xE01A4, 0xE01A5, 0xE01A6, 0xE01A7, 0xE01A8, 0xE01A9,
- 0xE01AA, 0xE01AB, 0xE01AC, 0xE01AD, 0xE01AE, 0xE01AF, 0xE01B0, 0xE01B1, 0xE01B2, 0xE01B3, 0xE01B4, 0xE01B5, 0xE01B6,
- 0xE01B7, 0xE01B8, 0xE01B9, 0xE01BA, 0xE01BB, 0xE01BC, 0xE01BD, 0xE01BE, 0xE01BF, 0xE01C0, 0xE01C1, 0xE01C2, 0xE01C3,
- 0xE01C4, 0xE01C5, 0xE01C6, 0xE01C7, 0xE01C8, 0xE01C9, 0xE01CA, 0xE01CB, 0xE01CC, 0xE01CD, 0xE01CE, 0xE01CF, 0xE01D0,
- 0xE01D1, 0xE01D2, 0xE01D3, 0xE01D4, 0xE01D5, 0xE01D6, 0xE01D7, 0xE01D8, 0xE01D9, 0xE01DA, 0xE01DB, 0xE01DC, 0xE01DD,
- 0xE01DE, 0xE01DF, 0xE01E0, 0xE01E1, 0xE01E2, 0xE01E3, 0xE01E4, 0xE01E5, 0xE01E6, 0xE01E7, 0xE01E8, 0xE01E9, 0xE01EA,
- 0xE01EB, 0xE01EC, 0xE01ED, 0xE01EE, 0xE01EF
- /* END: COMBINING CHAR TABLE */
-};
-
-static const unsigned long combiningCharTableSize = sizeof(combiningCharTable) / sizeof(combiningCharTable[0]);
-
-static bool isCombiningChar(unsigned long cp) {
- for (size_t i = 0; i < combiningCharTableSize; i++) {
- auto code = combiningCharTable[i];
- if (code > cp) {
- return false;
- }
- if (code == cp) {
- return true;
- }
- }
- return false;
-}
-
-/* Get length of previous grapheme */
-static size_t defaultPrevCharLen(const char * buf, size_t /*buf_len*/, size_t pos, size_t * col_len) {
- size_t end = pos;
- while (pos > 0) {
- size_t len = prevUtf8CodePointLen(buf, pos);
- pos -= len;
- int cp;
- utf8BytesToCodePoint(buf + pos, len, &cp);
- if (!isCombiningChar(cp)) {
- if (col_len != NULL) {
- *col_len = isWideChar(cp) ? 2 : 1;
- }
- return end - pos;
- }
- }
- /* NOTREACHED */
- return 0;
-}
-
-/* Get length of next grapheme */
-static size_t defaultNextCharLen(const char * buf, size_t buf_len, size_t pos, size_t * col_len) {
- size_t beg = pos;
- int cp;
- size_t len = utf8BytesToCodePoint(buf + pos, buf_len - pos, &cp);
- if (isCombiningChar(cp)) {
- /* NOTREACHED */
- return 0;
- }
- if (col_len != NULL) {
- *col_len = isWideChar(cp) ? 2 : 1;
- }
- pos += len;
- while (pos < buf_len) {
- int cp;
- len = utf8BytesToCodePoint(buf + pos, buf_len - pos, &cp);
- if (!isCombiningChar(cp)) {
- return pos - beg;
- }
- pos += len;
- }
- return pos - beg;
-}
-
-/* Read a Unicode from file. */
-static size_t defaultReadCode(int fd, char * buf, size_t buf_len, int * cp) {
- if (buf_len < 1) {
- return -1;
- }
- size_t nread = read(fd, &buf[0], 1);
- if (nread <= 0) {
- return nread;
- }
-
- unsigned char byte = buf[0];
- if ((byte & 0x80) == 0) {
- ;
- } else if ((byte & 0xE0) == 0xC0) {
- if (buf_len < 2) {
- return -1;
- }
- nread = read(fd, &buf[1], 1);
- if (nread <= 0) {
- return nread;
- }
- } else if ((byte & 0xF0) == 0xE0) {
- if (buf_len < 3) {
- return -1;
- }
- nread = read(fd, &buf[1], 2);
- if (nread <= 0) {
- return nread;
- }
- } else if ((byte & 0xF8) == 0xF0) {
- if (buf_len < 3) {
- return -1;
- }
- nread = read(fd, &buf[1], 3);
- if (nread <= 0) {
- return nread;
- }
- } else {
- return -1;
- }
-
- return utf8BytesToCodePoint(buf, buf_len, cp);
-}
-
-/* Set default encoding functions */
-static linenoisePrevCharLen * prevCharLen = defaultPrevCharLen;
-static linenoiseNextCharLen * nextCharLen = defaultNextCharLen;
-static linenoiseReadCode * readCode = defaultReadCode;
-
-/* Set used defined encoding functions */
-void linenoiseSetEncodingFunctions(linenoisePrevCharLen * prevCharLenFunc, linenoiseNextCharLen * nextCharLenFunc,
- linenoiseReadCode * readCodeFunc) {
- prevCharLen = prevCharLenFunc;
- nextCharLen = nextCharLenFunc;
- readCode = readCodeFunc;
-}
-
-/* ======================= Low level terminal handling ====================== */
-
-/* Enable "mask mode". When it is enabled, instead of the input that
- * the user is typing, the terminal will just display a corresponding
- * number of asterisks, like "****". This is useful for passwords and other
- * secrets that should not be displayed. */
-void linenoiseMaskModeEnable(void) {
- maskmode = 1;
-}
-
-/* Disable mask mode. */
-void linenoiseMaskModeDisable(void) {
- maskmode = 0;
-}
-
-/* Set if to use or not the multi line mode. */
-void linenoiseSetMultiLine(int ml) {
- mlmode = ml;
-}
-
-/* Return true if the terminal name is in the list of terminals we know are
- * not able to understand basic escape sequences. */
-static int isUnsupportedTerm(void) {
- char *term = getenv("TERM");
- if (term == NULL) return 0;
- for (size_t j = 0; j < unsupported_term.size(); ++j) {
- if (!strcasecmp(term, unsupported_term[j])) {
- return 1;
- }
- }
- return 0;
-}
-
-/* Raw mode: 1960 magic shit. */
-static int enableRawMode(int fd) {
- struct termios raw;
-
- if (!isatty(STDIN_FILENO)) goto fatal;
- if (!atexit_registered) {
- atexit(linenoiseAtExit);
- atexit_registered = 1;
- }
- if (tcgetattr(fd,&orig_termios) == -1) goto fatal;
-
- raw = orig_termios; /* modify the original mode */
- /* input modes: no break, no CR to NL, no parity check, no strip char,
- * no start/stop output control. */
- raw.c_iflag &= ~(BRKINT | ICRNL | INPCK | ISTRIP | IXON);
- /* output modes - disable post processing */
- raw.c_oflag &= ~(OPOST);
- /* control modes - set 8 bit chars */
- raw.c_cflag |= (CS8);
- /* local modes - choing off, canonical off, no extended functions,
- * no signal chars (^Z,^C) */
- raw.c_lflag &= ~(ECHO | ICANON | IEXTEN | ISIG);
- /* control chars - set return condition: min number of bytes and timer.
- * We want read to return every single byte, without timeout. */
- raw.c_cc[VMIN] = 1; raw.c_cc[VTIME] = 0; /* 1 byte, no timer */
-
- /* put terminal in raw mode after flushing */
- if (tcsetattr(fd,TCSAFLUSH,&raw) < 0) goto fatal;
- rawmode = 1;
- return 0;
-
-fatal:
- errno = ENOTTY;
- return -1;
-}
-
-static void disableRawMode(int fd) {
- /* Don't even check the return value as it's too late. */
- if (rawmode && tcsetattr(fd,TCSAFLUSH,&orig_termios) != -1)
- rawmode = 0;
-}
-
-/* Use the ESC [6n escape sequence to query the horizontal cursor position
- * and return it. On error -1 is returned, on success the position of the
- * cursor. */
-static int getCursorPosition(int ifd, int ofd) {
- char buf[32];
- int cols, rows;
- unsigned int i = 0;
-
- /* Report cursor location */
- if (write(ofd, "\x1b[6n", 4) != 4) return -1;
-
- /* Read the response: ESC [ rows ; cols R */
- while (i < sizeof(buf)-1) {
- if (read(ifd,buf+i,1) != 1) break;
- if (buf[i] == 'R') break;
- i++;
- }
- buf[i] = '\0';
-
- /* Parse it. */
- if (buf[0] != ESC || buf[1] != '[') return -1;
- if (sscanf(buf+2,"%d;%d",&rows,&cols) != 2) return -1;
- return cols;
-}
-
-/* Try to get the number of columns in the current terminal, or assume 80
- * if it fails. */
-static int getColumns(int ifd, int ofd) {
- struct winsize ws;
-
- if (ioctl(1, TIOCGWINSZ, &ws) == -1 || ws.ws_col == 0) {
- /* ioctl() failed. Try to query the terminal itself. */
- int start, cols;
-
- /* Get the initial position so we can restore it later. */
- start = getCursorPosition(ifd,ofd);
- if (start == -1) goto failed;
-
- /* Go to right margin and get position. */
- if (write(ofd,"\x1b[999C",6) != 6) goto failed;
- cols = getCursorPosition(ifd,ofd);
- if (cols == -1) goto failed;
-
- /* Restore position. */
- if (cols > start) {
- char seq[32];
- snprintf(seq,32,"\x1b[%dD",cols-start);
- if (write(ofd,seq,strlen(seq)) == -1) {
- /* Can't recover... */
- }
- }
- return cols;
- } else {
- return ws.ws_col;
- }
-
-failed:
- return 80;
-}
-
-/* Clear the screen. Used to handle ctrl+l */
-void linenoiseClearScreen(void) {
- if (write(STDOUT_FILENO,"\x1b[H\x1b[2J",7) <= 0) {
- /* nothing to do, just to avoid warning. */
- }
-}
-
-/* Beep, used for completion when there is nothing to complete or when all
- * the choices were already shown. */
-static void linenoiseBeep(void) {
- fprintf(stderr, "\x7");
- fflush(stderr);
-}
-
-/* Called by completeLine() and linenoiseShow() to render the current
- * edited line with the proposed completion. If the current completion table
- * is already available, it is passed as second argument, otherwise the
- * function will use the callback to obtain it.
- *
- * Flags are the same as refreshLine*(), that is REFRESH_* macros. */
-static void refreshLineWithCompletion(struct linenoiseState *ls, linenoiseCompletions *lc, int flags) {
- /* Obtain the table of completions if the caller didn't provide one. */
- linenoiseCompletions ctable;
- if (lc == NULL) {
- completionCallback(ls->buf, &ctable);
- lc = &ctable;
- }
-
- /* Show the edited line with completion if possible, or just refresh. */
- if (ls->completion_idx < lc->len) {
- struct linenoiseState saved = *ls;
- ls->len = ls->pos = strlen(lc->cvec[ls->completion_idx]);
- ls->buf = lc->cvec[ls->completion_idx];
- refreshLineWithFlags(ls, flags);
- ls->len = saved.len;
- ls->pos = saved.pos;
- ls->buf = saved.buf;
- } else {
- refreshLineWithFlags(ls, flags);
- }
-
- if (lc == &ctable) {
- ctable.to_free = false;
- }
-}
-
-enum ESC_TYPE { ESC_NULL = 0, ESC_DELETE, ESC_UP, ESC_DOWN, ESC_RIGHT, ESC_LEFT, ESC_HOME, ESC_END };
-
-static ESC_TYPE readEscapeSequence(struct linenoiseState * l) {
- /* Check if the file input has additional data. */
- struct pollfd pfd;
- pfd.fd = l->ifd;
- pfd.events = POLLIN;
-
- auto ret = poll(&pfd, 1, 1); // 1 millisecond timeout
- if (ret <= 0) { // -1: error, 0: timeout
- return ESC_NULL;
- }
-
- /* Read the next two bytes representing the escape sequence.
- * Use two calls to handle slow terminals returning the two
- * chars at different times. */
- char seq[3];
- if (read(l->ifd, seq, 1) == -1) {
- return ESC_NULL;
- }
- if (read(l->ifd, seq + 1, 1) == -1) {
- return ESC_NULL;
- }
-
- /* ESC [ sequences. */
- if (seq[0] == '[') {
- if (seq[1] >= '0' && seq[1] <= '9') {
- /* Extended escape, read additional byte. */
- if (read(l->ifd, seq + 2, 1) == -1) {
- return ESC_NULL;
- }
- if (seq[2] == '~') {
- switch (seq[1]) {
- case '3':
- return ESC_DELETE;
- }
- }
- } else {
- switch (seq[1]) {
- case 'A':
- return ESC_UP;
- case 'B':
- return ESC_DOWN;
- case 'C':
- return ESC_RIGHT;
- case 'D':
- return ESC_LEFT;
- case 'H':
- return ESC_HOME;
- case 'F':
- return ESC_END;
- }
- }
- }
-
- /* ESC O sequences. */
- else if (seq[0] == 'O') {
- switch (seq[1]) {
- case 'H':
- return ESC_HOME;
- case 'F':
- return ESC_END;
- }
- }
- return ESC_NULL;
-}
-
-/* This is an helper function for linenoiseEdit*() and is called when the
- * user types the key in order to complete the string currently in the
- * input.
- *
- * The state of the editing is encapsulated into the pointed linenoiseState
- * structure as described in the structure definition.
- *
- * If the function returns non-zero, the caller should handle the
- * returned value as a byte read from the standard input, and process
- * it as usually: this basically means that the function may return a byte
- * read from the terminal but not processed. Otherwise, if zero is returned,
- * the input was consumed by the completeLine() function to navigate the
- * possible completions, and the caller should read for the next characters
- * from stdin. */
-static int completeLine(struct linenoiseState * ls, int keypressed, ESC_TYPE esc_type) {
- linenoiseCompletions lc;
- int nwritten;
- char c = keypressed;
-
- completionCallback(ls->buf, &lc);
- if (lc.len == 0) {
- linenoiseBeep();
- ls->in_completion = 0;
- } else {
- if (c == TAB) {
- if (ls->in_completion == 0) {
- ls->in_completion = 1;
- ls->completion_idx = 0;
- } else {
- ls->completion_idx = (ls->completion_idx + 1) % (lc.len + 1);
- if (ls->completion_idx == lc.len) {
- linenoiseBeep();
- }
- }
- c = 0;
- } else if (c == ESC && esc_type == ESC_NULL) {
- /* Re-show original buffer */
- if (ls->completion_idx < lc.len) {
- refreshLine(ls);
- }
- ls->in_completion = 0;
- c = 0;
- } else {
- /* Update buffer and return */
- if (ls->completion_idx < lc.len) {
- nwritten = snprintf(ls->buf, ls->buflen, "%s", lc.cvec[ls->completion_idx]);
- ls->len = ls->pos = nwritten;
- }
- ls->in_completion = 0;
- }
-
- /* Show completion or original buffer */
- if (ls->in_completion && ls->completion_idx < lc.len) {
- refreshLineWithCompletion(ls, &lc, REFRESH_ALL);
- } else {
- refreshLine(ls);
- }
- }
-
- return c; /* Return last read character */
-}
-
-/* Register a callback function to be called for tab-completion. */
-void linenoiseSetCompletionCallback(linenoiseCompletionCallback *fn) {
- completionCallback = fn;
-}
-
-/* Register a hits function to be called to show hits to the user at the
- * right of the prompt. */
-void linenoiseSetHintsCallback(linenoiseHintsCallback *fn) {
- hintsCallback = fn;
-}
-
-/* Register a function to free the hints returned by the hints callback
- * registered with linenoiseSetHintsCallback(). */
-void linenoiseSetFreeHintsCallback(linenoiseFreeHintsCallback *fn) {
- freeHintsCallback = fn;
-}
-
-/* This function is used by the callback function registered by the user
- * in order to add completion options given the input string when the
- * user typed . See the example.c source code for a very easy to
- * understand example. */
-void linenoiseAddCompletion(linenoiseCompletions *lc, const char *str) {
- const size_t len = strlen(str);
- auto copy = std::make_unique(len + 1);
- if (!copy) {
- return;
- }
-
- memcpy(copy.get(), str, len + 1);
- char ** cvec = static_cast(std::realloc(lc->cvec, sizeof(char *) * (lc->len + 1)));
- if (cvec == nullptr) {
- return;
- }
-
- lc->cvec = cvec;
- lc->cvec[lc->len++] = copy.release();
-}
-
-/* Get column length from begining of buffer to current byte position */
-static size_t columnPos(const char * buf, size_t buf_len, size_t pos) {
- size_t ret = 0;
- size_t off = 0;
- while (off < pos) {
- size_t col_len;
- size_t len = nextCharLen(buf, buf_len, off, &col_len);
- off += len;
- ret += col_len;
- }
- return ret;
-}
-
-/* Helper of refreshSingleLine() and refreshMultiLine() to show hints
- * to the right of the prompt. */
-static void refreshShowHints(std::string & ab, struct linenoiseState * l, int pcollen) {
- char seq[64];
- size_t collen = pcollen + columnPos(l->buf, l->len, l->len);
- if (hintsCallback && collen < l->cols) {
- int color = -1, bold = 0;
- const char *hint = hintsCallback(l->buf,&color,&bold);
- if (hint) {
- int hintlen = strlen(hint);
- int hintmaxlen = l->cols - collen;
- if (hintlen > hintmaxlen) hintlen = hintmaxlen;
- if (bold == 1 && color == -1) color = 37;
- if (color != -1 || bold != 0)
- snprintf(seq,64,"\033[%d;%d;49m",bold,color);
- else
- seq[0] = '\0';
- ab.append(seq);
- ab.append(hint, hintlen);
- if (color != -1 || bold != 0)
- ab.append("\033[0m");
-
- /* Call the function to free the hint returned. */
- if (freeHintsCallback) freeHintsCallback(hint);
- }
- }
-}
-
-/* Check if text is an ANSI escape sequence */
-static int isAnsiEscape(const char * buf, size_t buf_len, size_t * len) {
- if (buf_len > 2 && !memcmp("\033[", buf, 2)) {
- size_t off = 2;
- while (off < buf_len) {
- switch (buf[off++]) {
- case 'A':
- case 'B':
- case 'C':
- case 'D':
- case 'E':
- case 'F':
- case 'G':
- case 'H':
- case 'J':
- case 'K':
- case 'S':
- case 'T':
- case 'f':
- case 'm':
- *len = off;
- return 1;
- }
- }
- }
- return 0;
-}
-
-/* Get column length of prompt text */
-static size_t promptTextColumnLen(const char * prompt, size_t plen) {
- char buf[LINENOISE_MAX_LINE];
- size_t buf_len = 0;
- size_t off = 0;
- while (off < plen) {
- size_t len;
- if (isAnsiEscape(prompt + off, plen - off, &len)) {
- off += len;
- continue;
- }
- buf[buf_len++] = prompt[off++];
- }
- return columnPos(buf, buf_len, buf_len);
-}
-
-/* Single line low level line refresh.
- *
- * Rewrite the currently edited line accordingly to the buffer content,
- * cursor position, and number of columns of the terminal.
- *
- * Flags is REFRESH_* macros. The function can just remove the old
- * prompt, just write it, or both. */
-static void refreshSingleLine(struct linenoiseState *l, int flags) {
- char seq[64];
- size_t pcollen = promptTextColumnLen(l->prompt, strlen(l->prompt));
- int fd = l->ofd;
- char *buf = l->buf;
- size_t len = l->len;
- size_t pos = l->pos;
- std::string ab;
-
- while ((pcollen + columnPos(buf, len, pos)) >= l->cols) {
- int chlen = nextCharLen(buf, len, 0, NULL);
- buf += chlen;
- len -= chlen;
- pos -= chlen;
- }
- while (pcollen + columnPos(buf, len, len) > l->cols) {
- len -= prevCharLen(buf, len, len, NULL);
- }
-
- /* Cursor to left edge */
- snprintf(seq,sizeof(seq),"\r");
- ab.append(seq);
-
- if (flags & REFRESH_WRITE) {
- /* Write the prompt and the current buffer content */
- ab.append(l->prompt);
- if (maskmode == 1) {
- while (len--) {
- ab.append("*");
- }
- } else {
- ab.append(buf, len);
- }
- /* Show hits if any. */
- refreshShowHints(ab, l, pcollen);
- }
-
- /* Erase to right */
- snprintf(seq,sizeof(seq),"\x1b[0K");
- ab.append(seq);
- if (flags & REFRESH_WRITE) {
- /* Move cursor to original position. */
- snprintf(seq, sizeof(seq), "\r\x1b[%dC", (int) (columnPos(buf, len, pos) + pcollen));
- ab.append(seq);
- }
-
- (void) !write(fd, ab.c_str(), ab.size()); /* Can't recover from write error. */
-}
-
-/* Get column length from begining of buffer to current byte position for multiline mode*/
-static size_t columnPosForMultiLine(const char * buf, size_t buf_len, size_t pos, size_t cols, size_t ini_pos) {
- size_t ret = 0;
- size_t colwid = ini_pos;
-
- size_t off = 0;
- while (off < buf_len) {
- size_t col_len;
- size_t len = nextCharLen(buf, buf_len, off, &col_len);
-
- int dif = (int) (colwid + col_len) - (int) cols;
- if (dif > 0) {
- ret += dif;
- colwid = col_len;
- } else if (dif == 0) {
- colwid = 0;
- } else {
- colwid += col_len;
- }
-
- if (off >= pos) {
- break;
- }
- off += len;
- ret += col_len;
- }
-
- return ret;
-}
-
-/* Multi line low level line refresh.
- *
- * Rewrite the currently edited line accordingly to the buffer content,
- * cursor position, and number of columns of the terminal.
- *
- * Flags is REFRESH_* macros. The function can just remove the old
- * prompt, just write it, or both. */
-static void refreshMultiLine(struct linenoiseState *l, int flags) {
- char seq[64];
- size_t pcollen = promptTextColumnLen(l->prompt, strlen(l->prompt));
- int colpos = columnPosForMultiLine(l->buf, l->len, l->len, l->cols, pcollen);
- int colpos2; /* cursor column position. */
- int rows = (pcollen + colpos + l->cols - 1) / l->cols; /* rows used by current buf. */
- int rpos = (pcollen + l->oldcolpos + l->cols) / l->cols; /* cursor relative row. */
- int rpos2; /* rpos after refresh. */
- int col; /* column position, zero-based. */
- int old_rows = l->oldrows;
- int fd = l->ofd, j;
- std::string ab;
- l->oldrows = rows;
-
- /* First step: clear all the lines used before. To do so start by
- * going to the last row. */
- if (flags & REFRESH_CLEAN) {
- if (old_rows - rpos > 0) {
- snprintf(seq,64,"\x1b[%dB", old_rows-rpos);
- ab.append(seq);
- }
-
- /* Now for every row clear it, go up. */
- for (j = 0; j < old_rows - 1; j++) {
- snprintf(seq,64,"\r\x1b[0K\x1b[1A");
- ab.append(seq);
- }
- }
-
- if (flags & REFRESH_ALL) {
- /* Clean the top line. */
- snprintf(seq,64,"\r\x1b[0K");
- ab.append(seq);
- }
-
- /* Get column length to cursor position */
- colpos2 = columnPosForMultiLine(l->buf, l->len, l->pos, l->cols, pcollen);
-
- if (flags & REFRESH_WRITE) {
- /* Write the prompt and the current buffer content */
- ab.append(l->prompt);
- if (maskmode == 1) {
- for (unsigned int i = 0; i < l->len; ++i) {
- ab.append("*");
- }
- } else {
- ab.append(l->buf, l->len);
- }
-
- /* Show hits if any. */
- refreshShowHints(ab, l, pcollen);
-
- /* If we are at the very end of the screen with our prompt, we need to
- * emit a newline and move the prompt to the first column. */
- if (l->pos && l->pos == l->len && (colpos2 + pcollen) % l->cols == 0) {
- ab.append("\n");
- snprintf(seq,64,"\r");
- ab.append(seq);
- rows++;
- if (rows > (int)l->oldrows) l->oldrows = rows;
- }
-
- /* Move cursor to right position. */
- rpos2 = (pcollen + colpos2 + l->cols) / l->cols; /* Current cursor relative row */
-
- /* Go up till we reach the expected position. */
- if (rows - rpos2 > 0) {
- snprintf(seq,64,"\x1b[%dA", rows-rpos2);
- ab.append(seq);
- }
-
- /* Set column. */
- col = (pcollen + colpos2) % l->cols;
- if (col)
- snprintf(seq,64,"\r\x1b[%dC", col);
- else
- snprintf(seq,64,"\r");
- ab.append(seq);
- }
-
- l->oldcolpos = colpos2;
-
- (void) !write(fd, ab.c_str(), ab.size()); /* Can't recover from write error. */
-}
-
-/* Calls the two low level functions refreshSingleLine() or
- * refreshMultiLine() according to the selected mode. */
-static void refreshLineWithFlags(struct linenoiseState *l, int flags) {
- if (mlmode)
- refreshMultiLine(l,flags);
- else
- refreshSingleLine(l,flags);
-}
-
-/* Utility function to avoid specifying REFRESH_ALL all the times. */
-static void refreshLine(struct linenoiseState *l) {
- refreshLineWithFlags(l,REFRESH_ALL);
-}
-
-/* Hide the current line, when using the multiplexing API. */
-void linenoiseHide(struct linenoiseState *l) {
- if (mlmode)
- refreshMultiLine(l,REFRESH_CLEAN);
- else
- refreshSingleLine(l,REFRESH_CLEAN);
-}
-
-/* Show the current line, when using the multiplexing API. */
-void linenoiseShow(struct linenoiseState *l) {
- if (l->in_completion) {
- refreshLineWithCompletion(l,NULL,REFRESH_WRITE);
- } else {
- refreshLineWithFlags(l,REFRESH_WRITE);
- }
-}
-
-/* Insert the character 'c' at cursor current position.
- *
- * On error writing to the terminal -1 is returned, otherwise 0. */
-static int linenoiseEditInsert(struct linenoiseState * l, const char * cbuf, int clen) {
- if (l->len + clen <= l->buflen) {
- if (l->len == l->pos) {
- memcpy(&l->buf[l->pos], cbuf, clen);
- l->pos += clen;
- l->len += clen;
- ;
- l->buf[l->len] = '\0';
- if ((!mlmode && promptTextColumnLen(l->prompt, l->plen) + columnPos(l->buf, l->len, l->len) < l->cols &&
- !hintsCallback)) {
- /* Avoid a full update of the line in the
- * trivial case. */
- if (maskmode == 1) {
- static const char d = '*';
- if (write(l->ofd, &d, 1) == -1) {
- return -1;
- }
- } else {
- if (write(l->ofd, cbuf, clen) == -1) {
- return -1;
- }
- }
- } else {
- refreshLine(l);
- }
- } else {
- memmove(l->buf + l->pos + clen, l->buf + l->pos, l->len - l->pos);
- memcpy(&l->buf[l->pos], cbuf, clen);
- l->pos += clen;
- l->len += clen;
- l->buf[l->len] = '\0';
- refreshLine(l);
- }
- }
- return 0;
-}
-
-/* Move cursor on the left. */
-static void linenoiseEditMoveLeft(struct linenoiseState * l) {
- if (l->pos > 0) {
- l->pos -= prevCharLen(l->buf, l->len, l->pos, NULL);
- refreshLine(l);
- }
-}
-
-/* Move cursor on the right. */
-static void linenoiseEditMoveRight(struct linenoiseState * l) {
- if (l->pos != l->len) {
- l->pos += nextCharLen(l->buf, l->len, l->pos, NULL);
- refreshLine(l);
- }
-}
-
-/* Move cursor to the start of the line. */
-static void linenoiseEditMoveHome(struct linenoiseState * l) {
- if (l->pos != 0) {
- l->pos = 0;
- refreshLine(l);
- }
-}
-
-/* Move cursor to the end of the line. */
-static void linenoiseEditMoveEnd(struct linenoiseState * l) {
- if (l->pos != l->len) {
- l->pos = l->len;
- refreshLine(l);
- }
-}
-
-/* Substitute the currently edited line with the next or previous history
- * entry as specified by 'dir'. */
-#define LINENOISE_HISTORY_NEXT 0
-#define LINENOISE_HISTORY_PREV 1
-
-static void linenoiseEditHistoryNext(struct linenoiseState * l, int dir) {
- if (history_len > 1) {
- /* Update the current history entry before to
- * overwrite it with the next one. */
- free(history[history_len - 1 - l->history_index]);
- history[history_len - 1 - l->history_index] = strdup(l->buf);
- /* Show the new entry */
- l->history_index += (dir == LINENOISE_HISTORY_PREV) ? 1 : -1;
- if (l->history_index < 0) {
- l->history_index = 0;
- return;
- } else if (l->history_index >= history_len) {
- l->history_index = history_len-1;
- return;
- }
- strncpy(l->buf,history[history_len - 1 - l->history_index],l->buflen);
- l->buf[l->buflen-1] = '\0';
- l->len = l->pos = strlen(l->buf);
- refreshLine(l);
- }
-}
-
-/* Delete the character at the right of the cursor without altering the cursor
- * position. Basically this is what happens with the "Delete" keyboard key. */
-static void linenoiseEditDelete(struct linenoiseState * l) {
- if (l->len > 0 && l->pos < l->len) {
- int chlen = nextCharLen(l->buf, l->len, l->pos, NULL);
- memmove(l->buf + l->pos, l->buf + l->pos + chlen, l->len - l->pos - chlen);
- l->len -= chlen;
- l->buf[l->len] = '\0';
- refreshLine(l);
- }
-}
-
-/* Backspace implementation. */
-static void linenoiseEditBackspace(struct linenoiseState * l) {
- if (l->pos > 0 && l->len > 0) {
- int chlen = prevCharLen(l->buf, l->len, l->pos, NULL);
- memmove(l->buf + l->pos - chlen, l->buf + l->pos, l->len - l->pos);
- l->pos -= chlen;
- l->len -= chlen;
- l->buf[l->len] = '\0';
- refreshLine(l);
- }
-}
-
-/* Delete the previous word, maintaining the cursor at the start of the
- * current word. */
-static void linenoiseEditDeletePrevWord(struct linenoiseState * l) {
- size_t old_pos = l->pos;
- size_t diff;
-
- while (l->pos > 0 && l->buf[l->pos-1] == ' ')
- l->pos--;
- while (l->pos > 0 && l->buf[l->pos-1] != ' ')
- l->pos--;
- diff = old_pos - l->pos;
- memmove(l->buf+l->pos,l->buf+old_pos,l->len-old_pos+1);
- l->len -= diff;
- refreshLine(l);
-}
-
-/* This function is part of the multiplexed API of Linenoise, that is used
- * in order to implement the blocking variant of the API but can also be
- * called by the user directly in an event driven program. It will:
- *
- * 1. Initialize the linenoise state passed by the user.
- * 2. Put the terminal in RAW mode.
- * 3. Show the prompt.
- * 4. Return control to the user, that will have to call linenoiseEditFeed()
- * each time there is some data arriving in the standard input.
- *
- * The user can also call linenoiseEditHide() and linenoiseEditShow() if it
- * is required to show some input arriving asynchronously, without mixing
- * it with the currently edited line.
- *
- * When linenoiseEditFeed() returns non-NULL, the user finished with the
- * line editing session (pressed enter CTRL-D/C): in this case the caller
- * needs to call linenoiseEditStop() to put back the terminal in normal
- * mode. This will not destroy the buffer, as long as the linenoiseState
- * is still valid in the context of the caller.
- *
- * The function returns 0 on success, or -1 if writing to standard output
- * fails. If stdin_fd or stdout_fd are set to -1, the default is to use
- * STDIN_FILENO and STDOUT_FILENO.
- */
-int linenoiseEditStart(struct linenoiseState *l, int stdin_fd, int stdout_fd, char *buf, size_t buflen, const char *prompt) {
- /* Populate the linenoise state that we pass to functions implementing
- * specific editing functionalities. */
- l->in_completion = 0;
- l->ifd = stdin_fd != -1 ? stdin_fd : STDIN_FILENO;
- l->ofd = stdout_fd != -1 ? stdout_fd : STDOUT_FILENO;
- l->buf = buf;
- l->buflen = buflen;
- l->prompt = prompt;
- l->plen = strlen(prompt);
- l->oldcolpos = l->pos = 0;
- l->len = 0;
-
- /* Enter raw mode. */
- if (enableRawMode(l->ifd) == -1) return -1;
-
- l->cols = getColumns(stdin_fd, stdout_fd);
- l->oldrows = 0;
- l->history_index = 0;
-
- /* Buffer starts empty. */
- l->buf[0] = '\0';
- l->buflen--; /* Make sure there is always space for the nullterm */
-
- /* If stdin is not a tty, stop here with the initialization. We
- * will actually just read a line from standard input in blocking
- * mode later, in linenoiseEditFeed(). */
- if (!isatty(l->ifd)) return 0;
-
- /* The latest history entry is always our current buffer, that
- * initially is just an empty string. */
- linenoiseHistoryAdd("");
-
- if (write(l->ofd,prompt,l->plen) == -1) return -1;
- return 0;
-}
-
-const char* linenoiseEditMore = "If you see this, you are misusing the API: when linenoiseEditFeed() is called, if it returns linenoiseEditMore the user is yet editing the line. See the README file for more information.";
-
-static const char * handleEnterKey(struct linenoiseState * l) {
- --history_len;
- free(history[history_len]);
- if (mlmode) {
- linenoiseEditMoveEnd(l);
- }
- if (hintsCallback) {
- /* Force a refresh without hints to leave the previous
- * line as the user typed it after a newline. */
- linenoiseHintsCallback * hc = hintsCallback;
- hintsCallback = NULL;
- refreshLine(l);
- hintsCallback = hc;
- }
-
- return strdup(l->buf);
-}
-
-static const char * handleCtrlCKey() {
- errno = EAGAIN;
- return NULL;
-}
-
-static const char * handleCtrlDKey(struct linenoiseState * l) {
- if (l->len > 0) {
- linenoiseEditDelete(l);
- return linenoiseEditMore;
- }
-
- --history_len;
- free(history[history_len]);
- errno = ENOENT;
- return NULL;
-}
-
-static void handleCtrlTKey(struct linenoiseState * l) {
- if (l->pos > 0 && l->pos < l->len) {
- auto prev_chlen = prevCharLen(l->buf, l->len, l->pos, NULL);
- auto curr_chlen = nextCharLen(l->buf, l->len, l->pos, NULL);
-
- std::string prev_char(prev_chlen, 0);
- memcpy(prev_char.data(), l->buf + l->pos - prev_chlen, prev_chlen);
- memmove(l->buf + l->pos - prev_chlen, l->buf + l->pos, curr_chlen);
- memmove(l->buf + l->pos - prev_chlen + curr_chlen, prev_char.data(), prev_chlen);
-
- l->pos = l->pos - prev_chlen + curr_chlen;
- if (l->pos + prev_chlen != l->len) {
- l->pos += prev_chlen;
- }
-
- refreshLine(l);
- }
-}
-
-static void handleEscapeSequence(struct linenoiseState * l, int esc_type) {
- switch (esc_type) {
- case ESC_NULL:
- break;
- case ESC_DELETE:
- linenoiseEditDelete(l);
- break;
- case ESC_UP:
- linenoiseEditHistoryNext(l, LINENOISE_HISTORY_PREV);
- break;
- case ESC_DOWN:
- linenoiseEditHistoryNext(l, LINENOISE_HISTORY_NEXT);
- break;
- case ESC_RIGHT:
- linenoiseEditMoveRight(l);
- break;
- case ESC_LEFT:
- linenoiseEditMoveLeft(l);
- break;
- case ESC_HOME:
- linenoiseEditMoveHome(l);
- break;
- case ESC_END:
- linenoiseEditMoveEnd(l);
- break;
- }
-}
-
-static void handleCtrlUKey(struct linenoiseState * l) {
- l->buf[0] = '\0';
- l->pos = l->len = 0;
- refreshLine(l);
-}
-
-static void handleCtrlKKey(struct linenoiseState * l) {
- l->buf[l->pos] = '\0';
- l->len = l->pos;
- refreshLine(l);
-}
-
-static const char * processInputCharacter(struct linenoiseState * l, int c, char * cbuf, int nread, int esc_type) {
- switch (c) {
- case ENTER:
- return handleEnterKey(l);
- case CTRL_C:
- return handleCtrlCKey();
- case BACKSPACE:
- case CTRL_H:
- linenoiseEditBackspace(l);
- break;
- case CTRL_D: /* ctrl-d, remove char at right of cursor, or if the
- line is empty, act as end-of-file. */
- return handleCtrlDKey(l);
- case CTRL_T:
- handleCtrlTKey(l);
- break;
- case CTRL_B:
- linenoiseEditMoveLeft(l);
- break;
- case CTRL_F:
- linenoiseEditMoveRight(l);
- break;
- case CTRL_P:
- linenoiseEditHistoryNext(l, LINENOISE_HISTORY_PREV);
- break;
- case CTRL_N:
- linenoiseEditHistoryNext(l, LINENOISE_HISTORY_NEXT);
- break;
- case ESC:
- handleEscapeSequence(l, esc_type);
- break;
- default:
- if (linenoiseEditInsert(l, cbuf, nread)) {
- return NULL;
- }
- break;
- case CTRL_U: /* Ctrl+u, delete the whole line. */
- handleCtrlUKey(l);
- break;
- case CTRL_K: /* Ctrl+k, delete from current to end of line. */
- handleCtrlKKey(l);
- break;
- case CTRL_A: /* Ctrl+a, go to the start of the line */
- linenoiseEditMoveHome(l);
- break;
- case CTRL_E: /* ctrl+e, go to the end of the line */
- linenoiseEditMoveEnd(l);
- break;
- case CTRL_L: /* ctrl+l, clear screen */
- linenoiseClearScreen();
- refreshLine(l);
- break;
- case CTRL_W: /* ctrl+w, delete previous word */
- linenoiseEditDeletePrevWord(l);
- break;
- }
- return linenoiseEditMore;
-}
-
-/* This function is part of the multiplexed API of linenoise, see the top
- * comment on linenoiseEditStart() for more information. Call this function
- * each time there is some data to read from the standard input file
- * descriptor. In the case of blocking operations, this function can just be
- * called in a loop, and block.
- *
- * The function returns linenoiseEditMore to signal that line editing is still
- * in progress, that is, the user didn't yet pressed enter / CTRL-D. Otherwise
- * the function returns the pointer to the heap-allocated buffer with the
- * edited line, that the user should free with linenoiseFree().
- *
- * On special conditions, NULL is returned and errno is populated:
- *
- * EAGAIN if the user pressed Ctrl-C
- * ENOENT if the user pressed Ctrl-D
- *
- * Some other errno: I/O error.
- */
-const char * linenoiseEditFeed(struct linenoiseState * l) {
- /* Not a TTY, pass control to line reading without character count
- * limits. */
- if (!isatty(l->ifd)) return linenoiseNoTTY();
-
- int c;
- int nread;
- char cbuf[32];
-
- nread = readCode(l->ifd, cbuf, sizeof(cbuf), &c);
- if (nread <= 0) return NULL;
-
- auto esc_type = ESC_NULL;
- if (c == ESC) {
- esc_type = readEscapeSequence(l);
- }
-
- /* Only autocomplete when the callback is set. It returns < 0 when
- * there was an error reading from fd. Otherwise it will return the
- * character that should be handled next. */
- if ((l->in_completion || c == 9) && completionCallback != NULL) {
- c = completeLine(l, c, esc_type);
- /* Read next character when 0 */
- if (c == 0) return linenoiseEditMore;
- }
-
- return processInputCharacter(l, c, cbuf, nread, esc_type);
-}
-
-/* This is part of the multiplexed linenoise API. See linenoiseEditStart()
- * for more information. This function is called when linenoiseEditFeed()
- * returns something different than NULL. At this point the user input
- * is in the buffer, and we can restore the terminal in normal mode. */
-void linenoiseEditStop(struct linenoiseState *l) {
- if (!isatty(l->ifd)) return;
- disableRawMode(l->ifd);
- printf("\n");
-}
-
-/* This just implements a blocking loop for the multiplexed API.
- * In many applications that are not event-driven, we can just call
- * the blocking linenoise API, wait for the user to complete the editing
- * and return the buffer. */
-static const char *linenoiseBlockingEdit(int stdin_fd, int stdout_fd, char *buf, size_t buflen, const char *prompt)
-{
- struct linenoiseState l;
-
- /* Editing without a buffer is invalid. */
- if (buflen == 0) {
- errno = EINVAL;
- return NULL;
- }
-
- linenoiseEditStart(&l,stdin_fd,stdout_fd,buf,buflen,prompt);
- const char *res;
- while((res = linenoiseEditFeed(&l)) == linenoiseEditMore);
- linenoiseEditStop(&l);
- return res;
-}
-
-/* This special mode is used by linenoise in order to print scan codes
- * on screen for debugging / development purposes. It is implemented
- * by the linenoise_example program using the --keycodes option. */
-void linenoisePrintKeyCodes(void) {
- char quit[4];
-
- printf("Linenoise key codes debugging mode.\n"
- "Press keys to see scan codes. Type 'quit' at any time to exit.\n");
- if (enableRawMode(STDIN_FILENO) == -1) return;
- memset(quit,' ',4);
- while(1) {
- char c;
- int nread;
-
- nread = read(STDIN_FILENO,&c,1);
- if (nread <= 0) continue;
- memmove(quit,quit+1,sizeof(quit)-1); /* shift string to left. */
- quit[sizeof(quit)-1] = c; /* Insert current char on the right. */
- if (memcmp(quit,"quit",sizeof(quit)) == 0) break;
-
- printf("'%c' %02x (%d) (type quit to exit)\n", isprint((int) c) ? c : '?', (int) c, (int) c);
- printf("\r"); /* Go left edge manually, we are in raw mode. */
- fflush(stdout);
- }
- disableRawMode(STDIN_FILENO);
-}
-
-/* This function is called when linenoise() is called with the standard
- * input file descriptor not attached to a TTY. So for example when the
- * program using linenoise is called in pipe or with a file redirected
- * to its standard input. In this case, we want to be able to return the
- * line regardless of its length (by default we are limited to 4k). */
-static char *linenoiseNoTTY(void) {
- char *line = NULL;
- size_t len = 0, maxlen = 0;
-
- while(1) {
- if (len == maxlen) {
- if (maxlen == 0) maxlen = 16;
- maxlen *= 2;
- char *oldval = line;
- line = (char*) realloc(line,maxlen);
- if (line == NULL) {
- if (oldval) free(oldval);
- return NULL;
- }
- }
- int c = fgetc(stdin);
- if (c == EOF || c == '\n') {
- if (c == EOF && len == 0) {
- free(line);
- return NULL;
- } else {
- line[len] = '\0';
- return line;
- }
- } else {
- line[len] = c;
- len++;
- }
- }
-}
-
-/* The high level function that is the main API of the linenoise library.
- * This function checks if the terminal has basic capabilities, just checking
- * for a blacklist of stupid terminals, and later either calls the line
- * editing function or uses dummy fgets() so that you will be able to type
- * something even in the most desperate of the conditions. */
-const char *linenoise(const char *prompt) {
- char buf[LINENOISE_MAX_LINE];
-
- if (!isatty(STDIN_FILENO)) {
- /* Not a tty: read from file / pipe. In this mode we don't want any
- * limit to the line size, so we call a function to handle that. */
- return linenoiseNoTTY();
- } else if (isUnsupportedTerm()) {
- size_t len;
-
- printf("%s",prompt);
- fflush(stdout);
- if (fgets(buf,LINENOISE_MAX_LINE,stdin) == NULL) return NULL;
- len = strlen(buf);
- while(len && (buf[len-1] == '\n' || buf[len-1] == '\r')) {
- len--;
- buf[len] = '\0';
- }
- return strdup(buf);
- } else {
- const char *retval = linenoiseBlockingEdit(STDIN_FILENO,STDOUT_FILENO,buf,LINENOISE_MAX_LINE,prompt);
- return retval;
- }
-}
-
-/* This is just a wrapper the user may want to call in order to make sure
- * the linenoise returned buffer is freed with the same allocator it was
- * created with. Useful when the main program is using an alternative
- * allocator. */
-void linenoiseFree(void *ptr) {
- if (ptr == linenoiseEditMore) return; // Protect from API misuse.
- free(ptr);
-}
-
-/* ================================ History ================================= */
-
-/* Free the history, but does not reset it. Only used when we have to
- * exit() to avoid memory leaks are reported by valgrind & co. */
-static void freeHistory(void) {
- if (history) {
- int j;
-
- for (j = 0; j < history_len; j++)
- free(history[j]);
- free(history);
- }
-}
-
-/* At exit we'll try to fix the terminal to the initial conditions. */
-static void linenoiseAtExit(void) {
- disableRawMode(STDIN_FILENO);
- freeHistory();
-}
-
-/* This is the API call to add a new entry in the linenoise history.
- * It uses a fixed array of char pointers that are shifted (memmoved)
- * when the history max length is reached in order to remove the older
- * entry and make room for the new one, so it is not exactly suitable for huge
- * histories, but will work well for a few hundred of entries.
- *
- * Using a circular buffer is smarter, but a bit more complex to handle. */
-int linenoiseHistoryAdd(const char *line) {
- char *linecopy;
-
- if (history_max_len == 0) return 0;
-
- /* Initialization on first call. */
- if (history == NULL) {
- history = (char**) malloc(sizeof(char*)*history_max_len);
- if (history == NULL) return 0;
- memset(history,0,(sizeof(char*)*history_max_len));
- }
-
- /* Don't add duplicated lines. */
- if (history_len && !strcmp(history[history_len-1], line)) return 0;
-
- /* Add an heap allocated copy of the line in the history.
- * If we reached the max length, remove the older line. */
- linecopy = strdup(line);
- if (!linecopy) return 0;
- if (history_len == history_max_len) {
- free(history[0]);
- memmove(history,history+1,sizeof(char*)*(history_max_len-1));
- history_len--;
- }
- history[history_len] = linecopy;
- history_len++;
- return 1;
-}
-
-/* Set the maximum length for the history. This function can be called even
- * if there is already some history, the function will make sure to retain
- * just the latest 'len' elements if the new history length value is smaller
- * than the amount of items already inside the history. */
-int linenoiseHistorySetMaxLen(int len) {
- char **new_ptr;
-
- if (len < 1) return 0;
- if (history) {
- int tocopy = history_len;
-
- new_ptr = (char**) malloc(sizeof(char*)*len);
- if (new_ptr == NULL) return 0;
-
- /* If we can't copy everything, free the elements we'll not use. */
- if (len < tocopy) {
- int j;
-
- for (j = 0; j < tocopy-len; j++) free(history[j]);
- tocopy = len;
- }
- memset(new_ptr,0,sizeof(char*)*len);
- memcpy(new_ptr,history+(history_len-tocopy), sizeof(char*)*tocopy);
- free(history);
- history = new_ptr;
- }
- history_max_len = len;
- if (history_len > history_max_len)
- history_len = history_max_len;
- return 1;
-}
-
-/* Save the history in the specified file. On success 0 is returned
- * otherwise -1 is returned. */
-int linenoiseHistorySave(const char *filename) {
- mode_t old_umask = umask(S_IXUSR|S_IRWXG|S_IRWXO);
- File file;
- file.open(filename, "w");
- umask(old_umask);
- if (file.file == NULL) {
- return -1;
- }
- chmod(filename,S_IRUSR|S_IWUSR);
- for (int j = 0; j < history_len; ++j) {
- fprintf(file.file, "%s\n", history[j]);
- }
-
- return 0;
-}
-
-/* Load the history from the specified file. If the file does not exist
- * zero is returned and no operation is performed.
- *
- * If the file exists and the operation succeeded 0 is returned, otherwise
- * on error -1 is returned. */
-int linenoiseHistoryLoad(const char *filename) {
- File file;
- file.open(filename, "r");
- char buf[LINENOISE_MAX_LINE];
- if (file.file == NULL) {
- return -1;
- }
-
- while (fgets(buf, LINENOISE_MAX_LINE, file.file) != NULL) {
- char *p;
-
- p = strchr(buf,'\r');
- if (!p) p = strchr(buf,'\n');
- if (p) *p = '\0';
- linenoiseHistoryAdd(buf);
- }
- return 0;
-}
-#endif
diff --git a/tools/run/linenoise.cpp/linenoise.h b/tools/run/linenoise.cpp/linenoise.h
deleted file mode 100644
index 9823ca36d0..0000000000
--- a/tools/run/linenoise.cpp/linenoise.h
+++ /dev/null
@@ -1,137 +0,0 @@
-/* linenoise.h -- VERSION 1.0
- *
- * Guerrilla line editing library against the idea that a line editing lib
- * needs to be 20,000 lines of C++ code.
- *
- * See linenoise.cpp for more information.
- *
- * ------------------------------------------------------------------------
- *
- * Copyright (c) 2010-2023, Salvatore Sanfilippo
- * Copyright (c) 2010-2013, Pieter Noordhuis
- * Copyright (c) 2025, Eric Curtin
- *
- * All rights reserved.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions are
- * met:
- *
- * * Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- *
- * * Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- *
- * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
- * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
- * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
- * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
- * HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
- * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
- * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
- * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
- * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
- * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
- * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- */
-
-#ifndef __LINENOISE_H
-#define __LINENOISE_H
-
-#ifdef __cplusplus
-extern "C" {
-#endif
-
-#include /* For size_t. */
-#include
-
-extern const char * linenoiseEditMore;
-
-/* The linenoiseState structure represents the state during line editing.
- * We pass this state to functions implementing specific editing
- * functionalities. */
-struct linenoiseState {
- int in_completion; /* The user pressed TAB and we are now in completion
- * mode, so input is handled by completeLine(). */
- size_t completion_idx; /* Index of next completion to propose. */
- int ifd; /* Terminal stdin file descriptor. */
- int ofd; /* Terminal stdout file descriptor. */
- char * buf; /* Edited line buffer. */
- size_t buflen; /* Edited line buffer size. */
- const char * prompt; /* Prompt to display. */
- size_t plen; /* Prompt length. */
- size_t pos; /* Current cursor position. */
- size_t oldcolpos; /* Previous refresh cursor column position. */
- size_t len; /* Current edited line length. */
- size_t cols; /* Number of columns in terminal. */
- size_t oldrows; /* Rows used by last refreshed line (multiline mode) */
- int history_index; /* The history index we are currently editing. */
-};
-
-struct linenoiseCompletions {
- size_t len = 0;
- char ** cvec = nullptr;
- bool to_free = true;
-
- ~linenoiseCompletions() {
- if (!to_free) {
- return;
- }
-
- for (size_t i = 0; i < len; ++i) {
- free(cvec[i]);
- }
-
- free(cvec);
- }
-};
-
-/* Non blocking API. */
-int linenoiseEditStart(struct linenoiseState * l, int stdin_fd, int stdout_fd, char * buf, size_t buflen,
- const char * prompt);
-const char * linenoiseEditFeed(struct linenoiseState * l);
-void linenoiseEditStop(struct linenoiseState * l);
-void linenoiseHide(struct linenoiseState * l);
-void linenoiseShow(struct linenoiseState * l);
-
-/* Blocking API. */
-const char * linenoise(const char * prompt);
-void linenoiseFree(void * ptr);
-
-/* Completion API. */
-typedef void(linenoiseCompletionCallback)(const char *, linenoiseCompletions *);
-typedef const char *(linenoiseHintsCallback) (const char *, int * color, int * bold);
-typedef void(linenoiseFreeHintsCallback)(const char *);
-void linenoiseSetCompletionCallback(linenoiseCompletionCallback *);
-void linenoiseSetHintsCallback(linenoiseHintsCallback *);
-void linenoiseSetFreeHintsCallback(linenoiseFreeHintsCallback *);
-void linenoiseAddCompletion(linenoiseCompletions *, const char *);
-
-/* History API. */
-int linenoiseHistoryAdd(const char * line);
-int linenoiseHistorySetMaxLen(int len);
-int linenoiseHistorySave(const char * filename);
-int linenoiseHistoryLoad(const char * filename);
-
-/* Other utilities. */
-void linenoiseClearScreen(void);
-void linenoiseSetMultiLine(int ml);
-void linenoisePrintKeyCodes(void);
-void linenoiseMaskModeEnable(void);
-void linenoiseMaskModeDisable(void);
-
-/* Encoding functions. */
-typedef size_t(linenoisePrevCharLen)(const char * buf, size_t buf_len, size_t pos, size_t * col_len);
-typedef size_t(linenoiseNextCharLen)(const char * buf, size_t buf_len, size_t pos, size_t * col_len);
-typedef size_t(linenoiseReadCode)(int fd, char * buf, size_t buf_len, int * c);
-
-void linenoiseSetEncodingFunctions(linenoisePrevCharLen * prevCharLenFunc, linenoiseNextCharLen * nextCharLenFunc,
- linenoiseReadCode * readCodeFunc);
-
-#ifdef __cplusplus
-}
-#endif
-
-#endif /* __LINENOISE_H */
diff --git a/tools/run/run.cpp b/tools/run/run.cpp
deleted file mode 100644
index b90a7253c4..0000000000
--- a/tools/run/run.cpp
+++ /dev/null
@@ -1,1408 +0,0 @@
-#include "chat.h"
-#include "common.h"
-#include "llama-cpp.h"
-#include "log.h"
-
-#include "linenoise.cpp/linenoise.h"
-
-#define JSON_ASSERT GGML_ASSERT
-#include
-
-#if defined(_WIN32)
-# define WIN32_LEAN_AND_MEAN
-# ifndef NOMINMAX
-# define NOMINMAX
-# endif
-# include
-# include
-#else
-# include
-# include
-# include
-#endif
-
-#if defined(LLAMA_USE_CURL)
-# include
-#else
-# include "http.h"
-#endif
-
-#include
-
-#include
-#include
-#include
-#include
-#include
-#include
-#include
-#include
-#include
-#include
-
-#if defined(__unix__) || (defined(__APPLE__) && defined(__MACH__)) || defined(_WIN32)
-[[noreturn]] static void sigint_handler(int) {
- printf("\n" LOG_COL_DEFAULT);
- exit(0); // not ideal, but it's the only way to guarantee exit in all cases
-}
-#endif
-
-GGML_ATTRIBUTE_FORMAT(1, 2)
-static int printe(const char * fmt, ...) {
- va_list args;
- va_start(args, fmt);
- const int ret = vfprintf(stderr, fmt, args);
- va_end(args);
-
- return ret;
-}
-
-static std::string strftime_fmt(const char * fmt, const std::tm & tm) {
- std::ostringstream oss;
- oss << std::put_time(&tm, fmt);
-
- return oss.str();
-}
-
-class Opt {
- public:
- int init(int argc, const char ** argv) {
- ctx_params = llama_context_default_params();
- model_params = llama_model_default_params();
- context_size_default = ctx_params.n_batch;
- n_threads_default = ctx_params.n_threads;
- ngl_default = model_params.n_gpu_layers;
- common_params_sampling sampling;
- temperature_default = sampling.temp;
-
- if (argc < 2) {
- printe("Error: No arguments provided.\n");
- print_help();
- return 1;
- }
-
- // Parse arguments
- if (parse(argc, argv)) {
- printe("Error: Failed to parse arguments.\n");
- print_help();
- return 1;
- }
-
- // If help is requested, show help and exit
- if (help) {
- print_help();
- return 2;
- }
-
- ctx_params.n_batch = context_size >= 0 ? context_size : context_size_default;
- ctx_params.n_ctx = ctx_params.n_batch;
- ctx_params.n_threads = ctx_params.n_threads_batch = n_threads >= 0 ? n_threads : n_threads_default;
- model_params.n_gpu_layers = ngl >= 0 ? ngl : ngl_default;
- temperature = temperature >= 0 ? temperature : temperature_default;
-
- return 0; // Success
- }
-
- llama_context_params ctx_params;
- llama_model_params model_params;
- std::string model_;
- std::string chat_template_file;
- std::string user;
- bool use_jinja = false;
- int context_size = -1, ngl = -1, n_threads = -1;
- float temperature = -1;
- bool verbose = false;
-
- private:
- int context_size_default = -1, ngl_default = -1, n_threads_default = -1;
- float temperature_default = -1;
- bool help = false;
-
- bool parse_flag(const char ** argv, int i, const char * short_opt, const char * long_opt) {
- return strcmp(argv[i], short_opt) == 0 || strcmp(argv[i], long_opt) == 0;
- }
-
- int handle_option_with_value(int argc, const char ** argv, int & i, int & option_value) {
- if (i + 1 >= argc) {
- return 1;
- }
-
- option_value = std::atoi(argv[++i]);
-
- return 0;
- }
-
- int handle_option_with_value(int argc, const char ** argv, int & i, float & option_value) {
- if (i + 1 >= argc) {
- return 1;
- }
-
- option_value = std::atof(argv[++i]);
-
- return 0;
- }
-
- int handle_option_with_value(int argc, const char ** argv, int & i, std::string & option_value) {
- if (i + 1 >= argc) {
- return 1;
- }
-
- option_value = argv[++i];
-
- return 0;
- }
-
- int parse_options_with_value(int argc, const char ** argv, int & i, bool & options_parsing) {
- if (options_parsing && (strcmp(argv[i], "-c") == 0 || strcmp(argv[i], "--context-size") == 0)) {
- if (handle_option_with_value(argc, argv, i, context_size) == 1) {
- return 1;
- }
- } else if (options_parsing &&
- (strcmp(argv[i], "-n") == 0 || strcmp(argv[i], "-ngl") == 0 || strcmp(argv[i], "--ngl") == 0)) {
- if (handle_option_with_value(argc, argv, i, ngl) == 1) {
- return 1;
- }
- } else if (options_parsing && (strcmp(argv[i], "-t") == 0 || strcmp(argv[i], "--threads") == 0)) {
- if (handle_option_with_value(argc, argv, i, n_threads) == 1) {
- return 1;
- }
- } else if (options_parsing && strcmp(argv[i], "--temp") == 0) {
- if (handle_option_with_value(argc, argv, i, temperature) == 1) {
- return 1;
- }
- } else if (options_parsing && strcmp(argv[i], "--chat-template-file") == 0) {
- if (handle_option_with_value(argc, argv, i, chat_template_file) == 1) {
- return 1;
- }
- use_jinja = true;
- } else {
- return 2;
- }
-
- return 0;
- }
-
- int parse_options(const char ** argv, int & i, bool & options_parsing) {
- if (options_parsing && (parse_flag(argv, i, "-v", "--verbose") || parse_flag(argv, i, "-v", "--log-verbose"))) {
- verbose = true;
- } else if (options_parsing && strcmp(argv[i], "--jinja") == 0) {
- use_jinja = true;
- } else if (options_parsing && parse_flag(argv, i, "-h", "--help")) {
- help = true;
- return 0;
- } else if (options_parsing && strcmp(argv[i], "--") == 0) {
- options_parsing = false;
- } else {
- return 2;
- }
-
- return 0;
- }
-
- int parse_positional_args(const char ** argv, int & i, int & positional_args_i) {
- if (positional_args_i == 0) {
- if (!argv[i][0] || argv[i][0] == '-') {
- return 1;
- }
-
- ++positional_args_i;
- model_ = argv[i];
- } else if (positional_args_i == 1) {
- ++positional_args_i;
- user = argv[i];
- } else {
- user += " " + std::string(argv[i]);
- }
-
- return 0;
- }
-
- int parse(int argc, const char ** argv) {
- bool options_parsing = true;
- for (int i = 1, positional_args_i = 0; i < argc; ++i) {
- int ret = parse_options_with_value(argc, argv, i, options_parsing);
- if (ret == 0) {
- continue;
- } else if (ret == 1) {
- return ret;
- }
-
- ret = parse_options(argv, i, options_parsing);
- if (ret == 0) {
- continue;
- } else if (ret == 1) {
- return ret;
- }
-
- if (parse_positional_args(argv, i, positional_args_i)) {
- return 1;
- }
- }
-
- if (model_.empty()) {
- return 1;
- }
-
- return 0;
- }
-
- void print_help() const {
- printf(
- "Description:\n"
- " Runs a llm\n"
- "\n"
- "Usage:\n"
- " llama-run [options] model [prompt]\n"
- "\n"
- "Options:\n"
- " -c, --context-size \n"
- " Context size (default: %d)\n"
- " --chat-template-file \n"
- " Path to the file containing the chat template to use with the model.\n"
- " Only supports jinja templates and implicitly sets the --jinja flag.\n"
- " --jinja\n"
- " Use jinja templating for the chat template of the model\n"
- " -n, -ngl, --ngl \n"
- " Number of GPU layers (default: %d)\n"
- " --temp \n"
- " Temperature (default: %.1f)\n"
- " -t, --threads \n"
- " Number of threads to use during generation (default: %d)\n"
- " -v, --verbose, --log-verbose\n"
- " Set verbosity level to infinity (i.e. log all messages, useful for debugging)\n"
- " -h, --help\n"
- " Show help message\n"
- "\n"
- "Commands:\n"
- " model\n"
- " Model is a string with an optional prefix of \n"
- " huggingface:// (hf://), modelscope:// (ms://), ollama://, https:// or file://.\n"
- " If no protocol is specified and a file exists in the specified\n"
- " path, file:// is assumed, otherwise if a file does not exist in\n"
- " the specified path, ollama:// is assumed. Models that are being\n"
- " pulled are downloaded with .partial extension while being\n"
- " downloaded and then renamed as the file without the .partial\n"
- " extension when complete.\n"
- "\n"
- "Examples:\n"
- " llama-run llama3\n"
- " llama-run ollama://granite-code\n"
- " llama-run ollama://smollm:135m\n"
- " llama-run hf://QuantFactory/SmolLM-135M-GGUF/SmolLM-135M.Q2_K.gguf\n"
- " llama-run "
- "huggingface://bartowski/SmolLM-1.7B-Instruct-v0.2-GGUF/SmolLM-1.7B-Instruct-v0.2-IQ3_M.gguf\n"
- " llama-run ms://QuantFactory/SmolLM-135M-GGUF/SmolLM-135M.Q2_K.gguf\n"
- " llama-run "
- "modelscope://bartowski/SmolLM-1.7B-Instruct-v0.2-GGUF/SmolLM-1.7B-Instruct-v0.2-IQ3_M.gguf\n"
- " llama-run https://example.com/some-file1.gguf\n"
- " llama-run some-file2.gguf\n"
- " llama-run file://some-file3.gguf\n"
- " llama-run --ngl 999 some-file4.gguf\n"
- " llama-run --ngl 999 some-file5.gguf Hello World\n",
- context_size_default, ngl_default, temperature_default, n_threads_default);
- }
-};
-
-struct progress_data {
- size_t file_size = 0;
- std::chrono::steady_clock::time_point start_time = std::chrono::steady_clock::now();
- bool printed = false;
-};
-
-static int get_terminal_width() {
-#if defined(_WIN32)
- CONSOLE_SCREEN_BUFFER_INFO csbi;
- GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE), &csbi);
- return csbi.srWindow.Right - csbi.srWindow.Left + 1;
-#else
- struct winsize w;
- ioctl(STDOUT_FILENO, TIOCGWINSZ, &w);
- return w.ws_col;
-#endif
-}
-
-class File {
- public:
- FILE * file = nullptr;
-
- FILE * open(const std::string & filename, const char * mode) {
- file = ggml_fopen(filename.c_str(), mode);
-
- return file;
- }
-
- int lock() {
- if (file) {
-# ifdef _WIN32
- fd = _fileno(file);
- hFile = (HANDLE) _get_osfhandle(fd);
- if (hFile == INVALID_HANDLE_VALUE) {
- fd = -1;
-
- return 1;
- }
-
- OVERLAPPED overlapped = {};
- if (!LockFileEx(hFile, LOCKFILE_EXCLUSIVE_LOCK | LOCKFILE_FAIL_IMMEDIATELY, 0, MAXDWORD, MAXDWORD,
- &overlapped)) {
- fd = -1;
-
- return 1;
- }
-# else
- fd = fileno(file);
- if (flock(fd, LOCK_EX | LOCK_NB) != 0) {
- fd = -1;
-
- return 1;
- }
-# endif
- }
-
- return 0;
- }
-
- std::string to_string() {
- fseek(file, 0, SEEK_END);
- const size_t size = ftell(file);
- fseek(file, 0, SEEK_SET);
- std::string out;
- out.resize(size);
- const size_t read_size = fread(&out[0], 1, size, file);
- if (read_size != size) {
- printe("Error reading file: %s", strerror(errno));
- }
-
- return out;
- }
-
- ~File() {
- if (fd >= 0) {
-# ifdef _WIN32
- if (hFile != INVALID_HANDLE_VALUE) {
- OVERLAPPED overlapped = {};
- UnlockFileEx(hFile, 0, MAXDWORD, MAXDWORD, &overlapped);
- }
-# else
- flock(fd, LOCK_UN);
-# endif
- }
-
- if (file) {
- fclose(file);
- }
- }
-
- private:
- int fd = -1;
-# ifdef _WIN32
- HANDLE hFile = nullptr;
-# endif
-};
-
-class HttpClient {
- public:
- int init(const std::string & url, const std::vector & headers, const std::string & output_file,
- const bool progress, std::string * response_str = nullptr) {
- if (std::filesystem::exists(output_file)) {
- return 0;
- }
-
- std::string output_file_partial;
-
- if (!output_file.empty()) {
- output_file_partial = output_file + ".partial";
- }
-
- if (download(url, headers, output_file_partial, progress, response_str)) {
- return 1;
- }
-
- if (!output_file.empty()) {
- try {
- std::filesystem::rename(output_file_partial, output_file);
- } catch (const std::filesystem::filesystem_error & e) {
- printe("Failed to rename '%s' to '%s': %s\n", output_file_partial.c_str(), output_file.c_str(), e.what());
- return 1;
- }
- }
-
- return 0;
- }
-
-#ifdef LLAMA_USE_CURL
-
- ~HttpClient() {
- if (chunk) {
- curl_slist_free_all(chunk);
- }
-
- if (curl) {
- curl_easy_cleanup(curl);
- }
- }
-
- private:
- CURL * curl = nullptr;
- struct curl_slist * chunk = nullptr;
-
- int download(const std::string & url, const std::vector & headers, const std::string & output_file,
- const bool progress, std::string * response_str = nullptr) {
- curl = curl_easy_init();
- if (!curl) {
- return 1;
- }
-
- progress_data data;
- File out;
- if (!output_file.empty()) {
- if (!out.open(output_file, "ab")) {
- printe("Failed to open file for writing\n");
-
- return 1;
- }
-
- if (out.lock()) {
- printe("Failed to exclusively lock file\n");
-
- return 1;
- }
- }
-
- set_write_options(response_str, out);
- data.file_size = set_resume_point(output_file);
- set_progress_options(progress, data);
- set_headers(headers);
- CURLcode res = perform(url);
- if (res != CURLE_OK){
- printe("Fetching resource '%s' failed: %s\n", url.c_str(), curl_easy_strerror(res));
- return 1;
- }
-
- return 0;
- }
-
- void set_write_options(std::string * response_str, const File & out) {
- if (response_str) {
- curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, capture_data);
- curl_easy_setopt(curl, CURLOPT_WRITEDATA, response_str);
- } else {
- curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, write_data);
- curl_easy_setopt(curl, CURLOPT_WRITEDATA, out.file);
- }
- }
-
- size_t set_resume_point(const std::string & output_file) {
- size_t file_size = 0;
- if (std::filesystem::exists(output_file)) {
- file_size = std::filesystem::file_size(output_file);
- curl_easy_setopt(curl, CURLOPT_RESUME_FROM_LARGE, static_cast(file_size));
- }
-
- return file_size;
- }
-
- void set_progress_options(bool progress, progress_data & data) {
- if (progress) {
- curl_easy_setopt(curl, CURLOPT_NOPROGRESS, 0L);
- curl_easy_setopt(curl, CURLOPT_XFERINFODATA, &data);
- curl_easy_setopt(curl, CURLOPT_XFERINFOFUNCTION, update_progress);
- }
- }
-
- void set_headers(const std::vector & headers) {
- if (!headers.empty()) {
- if (chunk) {
- curl_slist_free_all(chunk);
- chunk = 0;
- }
-
- for (const auto & header : headers) {
- chunk = curl_slist_append(chunk, header.c_str());
- }
-
- curl_easy_setopt(curl, CURLOPT_HTTPHEADER, chunk);
- }
- }
-
- CURLcode perform(const std::string & url) {
- curl_easy_setopt(curl, CURLOPT_URL, url.c_str());
- curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION, 1L);
- curl_easy_setopt(curl, CURLOPT_DEFAULT_PROTOCOL, "https");
- curl_easy_setopt(curl, CURLOPT_FAILONERROR, 1L);
-#ifdef _WIN32
- curl_easy_setopt(curl, CURLOPT_SSL_OPTIONS, CURLSSLOPT_NATIVE_CA);
-#endif
- return curl_easy_perform(curl);
- }
-
-#else // LLAMA_USE_CURL is not defined
-
-#define curl_off_t long long // temporary hack
-
- private:
- // this is a direct translation of the cURL download() above
- int download(const std::string & url, const std::vector & headers_vec, const std::string & output_file,
- const bool progress, std::string * response_str = nullptr) {
- try {
- auto [cli, url_parts] = common_http_client(url);
-
- httplib::Headers headers;
- for (const auto & h : headers_vec) {
- size_t pos = h.find(':');
- if (pos != std::string::npos) {
- headers.emplace(h.substr(0, pos), h.substr(pos + 2));
- }
- }
-
- File out;
- if (!output_file.empty()) {
- if (!out.open(output_file, "ab")) {
- printe("Failed to open file for writing\n");
- return 1;
- }
- if (out.lock()) {
- printe("Failed to exclusively lock file\n");
- return 1;
- }
- }
-
- size_t resume_offset = 0;
- if (!output_file.empty() && std::filesystem::exists(output_file)) {
- resume_offset = std::filesystem::file_size(output_file);
- if (resume_offset > 0) {
- headers.emplace("Range", "bytes=" + std::to_string(resume_offset) + "-");
- }
- }
-
- progress_data data;
- data.file_size = resume_offset;
-
- long long total_size = 0;
- long long received_this_session = 0;
-
- auto response_handler =
- [&](const httplib::Response & response) {
- if (resume_offset > 0 && response.status != 206) {
- printe("\nServer does not support resuming. Restarting download.\n");
- out.file = freopen(output_file.c_str(), "wb", out.file);
- if (!out.file) {
- return false;
- }
- data.file_size = 0;
- }
- if (progress) {
- if (response.has_header("Content-Length")) {
- total_size = std::stoll(response.get_header_value("Content-Length"));
- } else if (response.has_header("Content-Range")) {
- auto range = response.get_header_value("Content-Range");
- auto slash = range.find('/');
- if (slash != std::string::npos) {
- total_size = std::stoll(range.substr(slash + 1));
- }
- }
- }
- return true;
- };
-
- auto content_receiver =
- [&](const char * chunk, size_t length) {
- if (out.file && fwrite(chunk, 1, length, out.file) != length) {
- return false;
- }
- if (response_str) {
- response_str->append(chunk, length);
- }
- received_this_session += length;
-
- if (progress && total_size > 0) {
- update_progress(&data, total_size, received_this_session, 0, 0);
- }
- return true;
- };
-
- auto res = cli.Get(url_parts.path, headers, response_handler, content_receiver);
-
- if (data.printed) {
- printe("\n");
- }
-
- if (!res) {
- auto err = res.error();
- printe("Fetching resource '%s' failed: %s\n", url.c_str(), httplib::to_string(err).c_str());
- return 1;
- }
-
- if (res->status >= 400) {
- printe("Fetching resource '%s' failed with status code: %d\n", url.c_str(), res->status);
- return 1;
- }
-
- } catch (const std::exception & e) {
- printe("HTTP request failed: %s\n", e.what());
- return 1;
- }
- return 0;
- }
-
-#endif // LLAMA_USE_CURL
-
- static std::string human_readable_time(double seconds) {
- int hrs = static_cast(seconds) / 3600;
- int mins = (static_cast(seconds) % 3600) / 60;
- int secs = static_cast(seconds) % 60;
-
- if (hrs > 0) {
- return string_format("%dh %02dm %02ds", hrs, mins, secs);
- } else if (mins > 0) {
- return string_format("%dm %02ds", mins, secs);
- } else {
- return string_format("%ds", secs);
- }
- }
-
- static std::string human_readable_size(curl_off_t size) {
- static const char * suffix[] = { "B", "KB", "MB", "GB", "TB" };
- char length = sizeof(suffix) / sizeof(suffix[0]);
- int i = 0;
- double dbl_size = size;
- if (size > 1024) {
- for (i = 0; (size / 1024) > 0 && i < length - 1; i++, size /= 1024) {
- dbl_size = size / 1024.0;
- }
- }
-
- return string_format("%.2f %s", dbl_size, suffix[i]);
- }
-
- static int update_progress(void * ptr, curl_off_t total_to_download, curl_off_t now_downloaded, curl_off_t,
- curl_off_t) {
- progress_data * data = static_cast(ptr);
- if (total_to_download <= 0) {
- return 0;
- }
-
- total_to_download += data->file_size;
- const curl_off_t now_downloaded_plus_file_size = now_downloaded + data->file_size;
- const curl_off_t percentage = calculate_percentage(now_downloaded_plus_file_size, total_to_download);
- std::string progress_prefix = generate_progress_prefix(percentage);
-
- const double speed = calculate_speed(now_downloaded, data->start_time);
- const double tim = (total_to_download - now_downloaded) / speed;
- std::string progress_suffix =
- generate_progress_suffix(now_downloaded_plus_file_size, total_to_download, speed, tim);
-
- int progress_bar_width = calculate_progress_bar_width(progress_prefix, progress_suffix);
- std::string progress_bar;
- generate_progress_bar(progress_bar_width, percentage, progress_bar);
-
- print_progress(progress_prefix, progress_bar, progress_suffix);
- data->printed = true;
-
- return 0;
- }
-
- static curl_off_t calculate_percentage(curl_off_t now_downloaded_plus_file_size, curl_off_t total_to_download) {
- return (now_downloaded_plus_file_size * 100) / total_to_download;
- }
-
- static std::string generate_progress_prefix(curl_off_t percentage) {
- return string_format("%3ld%% |", static_cast(percentage));
- }
-
- static double calculate_speed(curl_off_t now_downloaded, const std::chrono::steady_clock::time_point & start_time) {
- const auto now = std::chrono::steady_clock::now();
- const std::chrono::duration elapsed_seconds = now - start_time;
- return now_downloaded / elapsed_seconds.count();
- }
-
- static std::string generate_progress_suffix(curl_off_t now_downloaded_plus_file_size, curl_off_t total_to_download,
- double speed, double estimated_time) {
- const int width = 10;
- return string_format("%*s/%*s%*s/s%*s", width, human_readable_size(now_downloaded_plus_file_size).c_str(),
- width, human_readable_size(total_to_download).c_str(), width,
- human_readable_size(speed).c_str(), width, human_readable_time(estimated_time).c_str());
- }
-
- static int calculate_progress_bar_width(const std::string & progress_prefix, const std::string & progress_suffix) {
- int progress_bar_width = get_terminal_width() - progress_prefix.size() - progress_suffix.size() - 3;
- if (progress_bar_width < 1) {
- progress_bar_width = 1;
- }
-
- return progress_bar_width;
- }
-
- static std::string generate_progress_bar(int progress_bar_width, curl_off_t percentage,
- std::string & progress_bar) {
- const curl_off_t pos = (percentage * progress_bar_width) / 100;
- for (int i = 0; i < progress_bar_width; ++i) {
- progress_bar.append((i < pos) ? "█" : " ");
- }
-
- return progress_bar;
- }
-
- static void print_progress(const std::string & progress_prefix, const std::string & progress_bar,
- const std::string & progress_suffix) {
- printe("\r" LOG_CLR_TO_EOL "%s%s| %s", progress_prefix.c_str(), progress_bar.c_str(), progress_suffix.c_str());
- }
- // Function to write data to a file
- static size_t write_data(void * ptr, size_t size, size_t nmemb, void * stream) {
- FILE * out = static_cast(stream);
- return fwrite(ptr, size, nmemb, out);
- }
-
- // Function to capture data into a string
- static size_t capture_data(void * ptr, size_t size, size_t nmemb, void * stream) {
- std::string * str = static_cast(stream);
- str->append(static_cast(ptr), size * nmemb);
- return size * nmemb;
- }
-
-};
-
-class LlamaData {
- public:
- llama_model_ptr model;
- llama_sampler_ptr sampler;
- llama_context_ptr context;
- std::vector messages; // TODO: switch to common_chat_msg
- std::list msg_strs;
- std::vector fmtted;
-
- int init(Opt & opt) {
- model = initialize_model(opt);
- if (!model) {
- return 1;
- }
-
- context = initialize_context(model, opt);
- if (!context) {
- return 1;
- }
-
- sampler = initialize_sampler(opt);
-
- return 0;
- }
-
- private:
- int download(const std::string & url, const std::string & output_file, const bool progress,
- const std::vector & headers = {}, std::string * response_str = nullptr) {
- HttpClient http;
- if (http.init(url, headers, output_file, progress, response_str)) {
- return 1;
- }
-
- return 0;
- }
-
- // Helper function to handle model tag extraction and URL construction
- std::pair extract_model_and_tag(std::string & model, const std::string & base_url) {
- std::string model_tag = "latest";
- const size_t colon_pos = model.find(':');
- if (colon_pos != std::string::npos) {
- model_tag = model.substr(colon_pos + 1);
- model = model.substr(0, colon_pos);
- }
-
- std::string url = base_url + model + "/manifests/" + model_tag;
-
- return { model, url };
- }
-
- // Helper function to download and parse the manifest
- int download_and_parse_manifest(const std::string & url, const std::vector & headers,
- nlohmann::json & manifest) {
- std::string manifest_str;
- int ret = download(url, "", false, headers, &manifest_str);
- if (ret) {
- return ret;
- }
-
- manifest = nlohmann::json::parse(manifest_str);
-
- return 0;
- }
-
- int dl_from_endpoint(std::string & model_endpoint, std::string & model, const std::string & bn) {
- // Find the second occurrence of '/' after protocol string
- size_t pos = model.find('/');
- pos = model.find('/', pos + 1);
- std::string hfr, hff;
- std::vector headers = { "User-Agent: llama-cpp", "Accept: application/json" };
- std::string url;
-
- if (pos == std::string::npos) {
- auto [model_name, manifest_url] = extract_model_and_tag(model, model_endpoint + "v2/");
- hfr = model_name;
-
- nlohmann::json manifest;
- int ret = download_and_parse_manifest(manifest_url, headers, manifest);
- if (ret) {
- return ret;
- }
-
- hff = manifest["ggufFile"]["rfilename"];
- } else {
- hfr = model.substr(0, pos);
- hff = model.substr(pos + 1);
- }
-
- url = model_endpoint + hfr + "/resolve/main/" + hff;
-
- return download(url, bn, true, headers);
- }
-
- int modelscope_dl(std::string & model, const std::string & bn) {
- std::string model_endpoint = "https://modelscope.cn/models/";
- return dl_from_endpoint(model_endpoint, model, bn);
- }
-
- int huggingface_dl(std::string & model, const std::string & bn) {
- std::string model_endpoint = get_model_endpoint();
- return dl_from_endpoint(model_endpoint, model, bn);
- }
-
- int ollama_dl(std::string & model, const std::string & bn) {
- const std::vector headers = { "Accept: application/vnd.docker.distribution.manifest.v2+json" };
- if (model.find('/') == std::string::npos) {
- model = "library/" + model;
- }
-
- auto [model_name, manifest_url] = extract_model_and_tag(model, "https://registry.ollama.ai/v2/");
- nlohmann::json manifest;
- int ret = download_and_parse_manifest(manifest_url, {}, manifest);
- if (ret) {
- return ret;
- }
-
- std::string layer;
- for (const auto & l : manifest["layers"]) {
- if (l["mediaType"] == "application/vnd.ollama.image.model") {
- layer = l["digest"];
- break;
- }
- }
-
- std::string blob_url = "https://registry.ollama.ai/v2/" + model_name + "/blobs/" + layer;
-
- return download(blob_url, bn, true, headers);
- }
-
- int github_dl(const std::string & model, const std::string & bn) {
- std::string repository = model;
- std::string branch = "main";
- const size_t at_pos = model.find('@');
- if (at_pos != std::string::npos) {
- repository = model.substr(0, at_pos);
- branch = model.substr(at_pos + 1);
- }
-
- const std::vector repo_parts = string_split(repository, "/");
- if (repo_parts.size() < 3) {
- printe("Invalid GitHub repository format\n");
- return 1;
- }
-
- const std::string & org = repo_parts[0];
- const std::string & project = repo_parts[1];
- std::string url = "https://raw.githubusercontent.com/" + org + "/" + project + "/" + branch;
- for (size_t i = 2; i < repo_parts.size(); ++i) {
- url += "/" + repo_parts[i];
- }
-
- return download(url, bn, true);
- }
-
- int s3_dl(const std::string & model, const std::string & bn) {
- const size_t slash_pos = model.find('/');
- if (slash_pos == std::string::npos) {
- return 1;
- }
-
- const std::string bucket = model.substr(0, slash_pos);
- const std::string key = model.substr(slash_pos + 1);
- const char * access_key = std::getenv("AWS_ACCESS_KEY_ID");
- const char * secret_key = std::getenv("AWS_SECRET_ACCESS_KEY");
- if (!access_key || !secret_key) {
- printe("AWS credentials not found in environment\n");
- return 1;
- }
-
- // Generate AWS Signature Version 4 headers
- // (Implementation requires HMAC-SHA256 and date handling)
- // Get current timestamp
- const time_t now = time(nullptr);
- const tm tm = *gmtime(&now);
- const std::string date = strftime_fmt("%Y%m%d", tm);
- const std::string datetime = strftime_fmt("%Y%m%dT%H%M%SZ", tm);
- const std::vector headers = {
- "Authorization: AWS4-HMAC-SHA256 Credential=" + std::string(access_key) + "/" + date +
- "/us-east-1/s3/aws4_request",
- "x-amz-content-sha256: UNSIGNED-PAYLOAD", "x-amz-date: " + datetime
- };
-
- const std::string url = "https://" + bucket + ".s3.amazonaws.com/" + key;
-
- return download(url, bn, true, headers);
- }
-
- std::string basename(const std::string & path) {
- const size_t pos = path.find_last_of("/\\");
- if (pos == std::string::npos) {
- return path;
- }
-
- return path.substr(pos + 1);
- }
-
- int rm_until_substring(std::string & model_, const std::string & substring) {
- const std::string::size_type pos = model_.find(substring);
- if (pos == std::string::npos) {
- return 1;
- }
-
- model_ = model_.substr(pos + substring.size()); // Skip past the substring
- return 0;
- }
-
- int resolve_model(std::string & model_) {
- int ret = 0;
- if (string_starts_with(model_, "file://") || std::filesystem::exists(model_)) {
- rm_until_substring(model_, "://");
-
- return ret;
- }
-
- const std::string bn = basename(model_);
- if (string_starts_with(model_, "hf://") || string_starts_with(model_, "huggingface://") ||
- string_starts_with(model_, "hf.co/")) {
- rm_until_substring(model_, "hf.co/");
- rm_until_substring(model_, "://");
- ret = huggingface_dl(model_, bn);
- } else if (string_starts_with(model_, "ms://") || string_starts_with(model_, "modelscope://")) {
- rm_until_substring(model_, "://");
- ret = modelscope_dl(model_, bn);
- } else if ((string_starts_with(model_, "https://") || string_starts_with(model_, "http://")) &&
- !string_starts_with(model_, "https://ollama.com/library/")) {
- ret = download(model_, bn, true);
- } else if (string_starts_with(model_, "github:") || string_starts_with(model_, "github://")) {
- rm_until_substring(model_, "github:");
- rm_until_substring(model_, "://");
- ret = github_dl(model_, bn);
- } else if (string_starts_with(model_, "s3://")) {
- rm_until_substring(model_, "://");
- ret = s3_dl(model_, bn);
- } else { // ollama:// or nothing
- rm_until_substring(model_, "ollama.com/library/");
- rm_until_substring(model_, "://");
- ret = ollama_dl(model_, bn);
- }
-
- model_ = bn;
-
- return ret;
- }
-
- // Initializes the model and returns a unique pointer to it
- llama_model_ptr initialize_model(Opt & opt) {
- ggml_backend_load_all();
- resolve_model(opt.model_);
- printe("\r" LOG_CLR_TO_EOL "Loading model");
- llama_model_ptr model(llama_model_load_from_file(opt.model_.c_str(), opt.model_params));
- if (!model) {
- printe("%s: error: unable to load model from file: %s\n", __func__, opt.model_.c_str());
- }
-
- printe("\r" LOG_CLR_TO_EOL);
- return model;
- }
-
- // Initializes the context with the specified parameters
- llama_context_ptr initialize_context(const llama_model_ptr & model, const Opt & opt) {
- llama_context_ptr context(llama_init_from_model(model.get(), opt.ctx_params));
- if (!context) {
- printe("%s: error: failed to create the llama_context\n", __func__);
- }
-
- return context;
- }
-
- // Initializes and configures the sampler
- llama_sampler_ptr initialize_sampler(const Opt & opt) {
- llama_sampler_ptr sampler(llama_sampler_chain_init(llama_sampler_chain_default_params()));
- llama_sampler_chain_add(sampler.get(), llama_sampler_init_min_p(0.05f, 1));
- llama_sampler_chain_add(sampler.get(), llama_sampler_init_temp(opt.temperature));
- llama_sampler_chain_add(sampler.get(), llama_sampler_init_dist(LLAMA_DEFAULT_SEED));
-
- return sampler;
- }
-};
-
-// Add a message to `messages` and store its content in `msg_strs`
-static void add_message(const char * role, const std::string & text, LlamaData & llama_data) {
- llama_data.msg_strs.push_back(std::move(text));
- llama_data.messages.push_back({ role, llama_data.msg_strs.back().c_str() });
-}
-
-// Function to apply the chat template and resize `formatted` if needed
-static int apply_chat_template(const struct common_chat_templates * tmpls, LlamaData & llama_data, const bool append, bool use_jinja) {
- common_chat_templates_inputs inputs;
- for (const auto & msg : llama_data.messages) {
- common_chat_msg cmsg;
- cmsg.role = msg.role;
- cmsg.content = msg.content;
- inputs.messages.push_back(cmsg);
- }
- inputs.add_generation_prompt = append;
- inputs.use_jinja = use_jinja;
-
- auto chat_params = common_chat_templates_apply(tmpls, inputs);
- // TODO: use other params for tool calls.
- auto result = chat_params.prompt;
- llama_data.fmtted.resize(result.size() + 1);
- memcpy(llama_data.fmtted.data(), result.c_str(), result.size() + 1);
- return result.size();
-}
-
-// Function to tokenize the prompt
-static int tokenize_prompt(const llama_vocab * vocab, const std::string & prompt,
- std::vector & prompt_tokens, const LlamaData & llama_data) {
- const bool is_first = llama_memory_seq_pos_max(llama_get_memory(llama_data.context.get()), 0) == -1;
- int n_tokens = prompt.size() + 2 * is_first;
- prompt_tokens.resize(n_tokens);
- n_tokens = llama_tokenize(vocab, prompt.c_str(), prompt.size(),
- prompt_tokens.data(), prompt_tokens.size(),
- is_first, /*parse_special =*/true);
- if (n_tokens == std::numeric_limits::min()) {
- printe("tokenization failed: input too large\n");
- return -1;
- }
- if (n_tokens < 0) {
- prompt_tokens.resize(-n_tokens);
- int check = llama_tokenize(vocab, prompt.c_str(), prompt.size(),
- prompt_tokens.data(), prompt_tokens.size(),
- is_first, /*parse_special =*/true);
- if (check != -n_tokens) {
- printe("failed to tokenize the prompt (size mismatch)\n");
- return -1;
- }
- n_tokens = check;
- } else {
- prompt_tokens.resize(n_tokens);
- }
- return n_tokens;
-}
-
-// Check if we have enough space in the context to evaluate this batch
-static int check_context_size(const llama_context_ptr & ctx, const llama_batch & batch) {
- const int n_ctx = llama_n_ctx(ctx.get());
- const int n_ctx_used = llama_memory_seq_pos_max(llama_get_memory(ctx.get()), 0);
- if (n_ctx_used + batch.n_tokens > n_ctx) {
- printf(LOG_COL_DEFAULT "\n");
- printe("context size exceeded\n");
- return 1;
- }
-
- return 0;
-}
-
-// convert the token to a string
-static int convert_token_to_string(const llama_vocab * vocab, const llama_token token_id, std::string & piece) {
- char buf[256];
- int n = llama_token_to_piece(vocab, token_id, buf, sizeof(buf), 0, true);
- if (n < 0) {
- printe("failed to convert token to piece\n");
- return 1;
- }
-
- piece = std::string(buf, n);
- return 0;
-}
-
-static void print_word_and_concatenate_to_response(const std::string & piece, std::string & response) {
- printf("%s", piece.c_str());
- fflush(stdout);
- response += piece;
-}
-
-// helper function to evaluate a prompt and generate a response
-static int generate(LlamaData & llama_data, const std::string & prompt, std::string & response) {
- const llama_vocab * vocab = llama_model_get_vocab(llama_data.model.get());
-
- std::vector tokens;
- if (tokenize_prompt(vocab, prompt, tokens, llama_data) < 0) {
- return 1;
- }
-
- // prepare a batch for the prompt
- llama_batch batch = llama_batch_get_one(tokens.data(), tokens.size());
- llama_token new_token_id;
- while (true) {
- check_context_size(llama_data.context, batch);
- if (llama_decode(llama_data.context.get(), batch)) {
- printe("failed to decode\n");
- return 1;
- }
-
- // sample the next token, check is it an end of generation?
- new_token_id = llama_sampler_sample(llama_data.sampler.get(), llama_data.context.get(), -1);
- if (llama_vocab_is_eog(vocab, new_token_id)) {
- break;
- }
-
- std::string piece;
- if (convert_token_to_string(vocab, new_token_id, piece)) {
- return 1;
- }
-
- print_word_and_concatenate_to_response(piece, response);
-
- // prepare the next batch with the sampled token
- batch = llama_batch_get_one(&new_token_id, 1);
- }
-
- printf(LOG_COL_DEFAULT);
- return 0;
-}
-
-static int read_user_input(std::string & user_input) {
- static const char * prompt_prefix_env = std::getenv("LLAMA_PROMPT_PREFIX");
- static const char * prompt_prefix = prompt_prefix_env ? prompt_prefix_env : "> ";
-#ifdef WIN32
- printf("\r" LOG_CLR_TO_EOL LOG_COL_DEFAULT "%s", prompt_prefix);
-
- std::getline(std::cin, user_input);
- if (std::cin.eof()) {
- printf("\n");
- return 1;
- }
-#else
- std::unique_ptr line(const_cast(linenoise(prompt_prefix)), free);
- if (!line) {
- return 1;
- }
-
- user_input = line.get();
-#endif
-
- if (user_input == "/bye") {
- return 1;
- }
-
- if (user_input.empty()) {
- return 2;
- }
-
-#ifndef WIN32
- linenoiseHistoryAdd(line.get());
-#endif
-
- return 0; // Should have data in happy path
-}
-
-// Function to generate a response based on the prompt
-static int generate_response(LlamaData & llama_data, const std::string & prompt, std::string & response,
- const bool stdout_a_terminal) {
- // Set response color
- if (stdout_a_terminal) {
- printf(LOG_COL_YELLOW);
- }
-
- if (generate(llama_data, prompt, response)) {
- printe("failed to generate response\n");
- return 1;
- }
-
- // End response with color reset and newline
- printf("\n%s", stdout_a_terminal ? LOG_COL_DEFAULT : "");
- return 0;
-}
-
-// Helper function to apply the chat template and handle errors
-static int apply_chat_template_with_error_handling(const common_chat_templates * tmpls, LlamaData & llama_data, const bool append, int & output_length, bool use_jinja) {
- const int new_len = apply_chat_template(tmpls, llama_data, append, use_jinja);
- if (new_len < 0) {
- printe("failed to apply the chat template\n");
- return -1;
- }
-
- output_length = new_len;
- return 0;
-}
-
-// Helper function to handle user input
-static int handle_user_input(std::string & user_input, const std::string & user) {
- if (!user.empty()) {
- user_input = user;
- return 0; // No need for interactive input
- }
-
- return read_user_input(user_input); // Returns true if input ends the loop
-}
-
-static bool is_stdin_a_terminal() {
-#if defined(_WIN32)
- HANDLE hStdin = GetStdHandle(STD_INPUT_HANDLE);
- DWORD mode;
- return GetConsoleMode(hStdin, &mode);
-#else
- return isatty(STDIN_FILENO);
-#endif
-}
-
-static bool is_stdout_a_terminal() {
-#if defined(_WIN32)
- HANDLE hStdout = GetStdHandle(STD_OUTPUT_HANDLE);
- DWORD mode;
- return GetConsoleMode(hStdout, &mode);
-#else
- return isatty(STDOUT_FILENO);
-#endif
-}
-
-// Function to handle user input
-static int get_user_input(std::string & user_input, const std::string & user) {
- while (true) {
- const int ret = handle_user_input(user_input, user);
- if (ret == 1) {
- return 1;
- }
-
- if (ret == 2) {
- continue;
- }
-
- break;
- }
-
- return 0;
-}
-
-// Reads a chat template file to be used
-static std::string read_chat_template_file(const std::string & chat_template_file) {
- File file;
- if (!file.open(chat_template_file, "r")) {
- printe("Error opening chat template file '%s': %s", chat_template_file.c_str(), strerror(errno));
- return "";
- }
-
- return file.to_string();
-}
-
-static int process_user_message(const Opt & opt, const std::string & user_input, LlamaData & llama_data,
- const common_chat_templates_ptr & chat_templates, int & prev_len,
- const bool stdout_a_terminal) {
- add_message("user", opt.user.empty() ? user_input : opt.user, llama_data);
- int new_len;
- if (apply_chat_template_with_error_handling(chat_templates.get(), llama_data, true, new_len, opt.use_jinja) < 0) {
- return 1;
- }
-
- std::string prompt(llama_data.fmtted.begin() + prev_len, llama_data.fmtted.begin() + new_len);
- std::string response;
- if (generate_response(llama_data, prompt, response, stdout_a_terminal)) {
- return 1;
- }
-
- if (!opt.user.empty()) {
- return 2;
- }
-
- add_message("assistant", response, llama_data);
- if (apply_chat_template_with_error_handling(chat_templates.get(), llama_data, false, prev_len, opt.use_jinja) < 0) {
- return 1;
- }
-
- return 0;
-}
-
-// Main chat loop function
-static int chat_loop(LlamaData & llama_data, const Opt & opt) {
- int prev_len = 0;
- llama_data.fmtted.resize(llama_n_ctx(llama_data.context.get()));
- std::string chat_template;
- if (!opt.chat_template_file.empty()) {
- chat_template = read_chat_template_file(opt.chat_template_file);
- }
-
- common_chat_templates_ptr chat_templates = common_chat_templates_init(llama_data.model.get(), chat_template);
- static const bool stdout_a_terminal = is_stdout_a_terminal();
- while (true) {
- // Get user input
- std::string user_input;
- if (get_user_input(user_input, opt.user) == 1) {
- return 0;
- }
-
- const int ret = process_user_message(opt, user_input, llama_data, chat_templates, prev_len, stdout_a_terminal);
- if (ret == 1) {
- return 1;
- } else if (ret == 2) {
- break;
- }
- }
-
- return 0;
-}
-
-static void log_callback(const enum ggml_log_level level, const char * text, void * p) {
- const Opt * opt = static_cast(p);
- if (opt->verbose || level == GGML_LOG_LEVEL_ERROR) {
- printe("%s", text);
- }
-}
-
-static std::string read_pipe_data() {
- std::ostringstream result;
- result << std::cin.rdbuf(); // Read all data from std::cin
- return result.str();
-}
-
-static void ctrl_c_handling() {
-#if defined(__unix__) || (defined(__APPLE__) && defined(__MACH__))
- struct sigaction sigint_action;
- sigint_action.sa_handler = sigint_handler;
- sigemptyset(&sigint_action.sa_mask);
- sigint_action.sa_flags = 0;
- sigaction(SIGINT, &sigint_action, NULL);
-#elif defined(_WIN32)
- auto console_ctrl_handler = +[](DWORD ctrl_type) -> BOOL {
- return (ctrl_type == CTRL_C_EVENT) ? (sigint_handler(SIGINT), true) : false;
- };
- SetConsoleCtrlHandler(reinterpret_cast(console_ctrl_handler), true);
-#endif
-}
-
-int main(int argc, const char ** argv) {
- ctrl_c_handling();
- Opt opt;
- const int ret = opt.init(argc, argv);
- if (ret == 2) {
- return 0;
- } else if (ret) {
- return 1;
- }
-
- if (!is_stdin_a_terminal()) {
- if (!opt.user.empty()) {
- opt.user += "\n\n";
- }
-
- opt.user += read_pipe_data();
- }
-
- llama_log_set(log_callback, &opt);
- LlamaData llama_data;
- if (llama_data.init(opt)) {
- return 1;
- }
-
- if (chat_loop(llama_data, opt)) {
- return 1;
- }
-
- return 0;
-}