2023-07-03 19:22:44 +00:00
|
|
|
#include "common.h"
|
|
|
|
#include "llama.h"
|
|
|
|
|
|
|
|
#include "binding.h"
|
|
|
|
|
|
|
|
#include <cassert>
|
|
|
|
#include <cinttypes>
|
|
|
|
#include <cmath>
|
|
|
|
#include <cstdio>
|
|
|
|
#include <cstring>
|
|
|
|
#include <fstream>
|
|
|
|
#include <iostream>
|
|
|
|
#include <regex>
|
|
|
|
#include <sstream>
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
|
|
|
#if defined(__unix__) || (defined(__APPLE__) && defined(__MACH__))
|
|
|
|
#include <signal.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#elif defined(_WIN32)
|
|
|
|
#define WIN32_LEAN_AND_MEAN
|
|
|
|
#define NOMINMAX
|
|
|
|
#include <signal.h>
|
|
|
|
#include <windows.h>
|
|
|
|
#endif
|
|
|
|
|
2023-07-06 21:49:05 +00:00
|
|
|
#if defined(__unix__) || (defined(__APPLE__) && defined(__MACH__)) || \
|
2023-07-03 19:22:44 +00:00
|
|
|
defined(_WIN32)
|
|
|
|
void sigint_handler(int signo) {
|
|
|
|
if (signo == SIGINT) {
|
|
|
|
_exit(130);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2023-07-05 20:28:18 +00:00
|
|
|
int get_embeddings(void *params_ptr, void *state_pr, float *res_embeddings) {
|
|
|
|
gpt_params *params_p = (gpt_params *)params_ptr;
|
|
|
|
llama_context *ctx = (llama_context *)state_pr;
|
|
|
|
gpt_params params = *params_p;
|
|
|
|
|
|
|
|
if (params.seed <= 0) {
|
|
|
|
params.seed = time(NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::mt19937 rng(params.seed);
|
|
|
|
|
|
|
|
llama_init_backend(params.numa);
|
|
|
|
|
|
|
|
int n_past = 0;
|
|
|
|
|
|
|
|
// Add a space in front of the first character to match OG llama tokenizer
|
|
|
|
// behavior
|
|
|
|
params.prompt.insert(0, 1, ' ');
|
|
|
|
|
|
|
|
// tokenize the prompt
|
|
|
|
auto embd_inp = ::llama_tokenize(ctx, params.prompt, true);
|
|
|
|
|
|
|
|
// determine newline token
|
|
|
|
auto llama_token_newline = ::llama_tokenize(ctx, "\n", false);
|
|
|
|
|
|
|
|
if (embd_inp.size() > 0) {
|
|
|
|
if (llama_eval(ctx, embd_inp.data(), embd_inp.size(), n_past,
|
|
|
|
params.n_threads)) {
|
|
|
|
fprintf(stderr, "%s : failed to eval\n", __func__);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const int n_embd = llama_n_embd(ctx);
|
|
|
|
|
|
|
|
const auto embeddings = llama_get_embeddings(ctx);
|
|
|
|
|
|
|
|
for (int i = 0; i < n_embd; i++) {
|
|
|
|
res_embeddings[i] = embeddings[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int get_token_embeddings(void *params_ptr, void *state_pr, int *tokens,
|
|
|
|
int tokenSize, float *res_embeddings) {
|
|
|
|
gpt_params *params_p = (gpt_params *)params_ptr;
|
|
|
|
llama_context *ctx = (llama_context *)state_pr;
|
|
|
|
gpt_params params = *params_p;
|
|
|
|
|
|
|
|
for (int i = 0; i < tokenSize; i++) {
|
|
|
|
auto token_str = llama_token_to_str(ctx, tokens[i]);
|
|
|
|
if (token_str == nullptr) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
std::vector<std::string> my_vector;
|
|
|
|
std::string str_token(token_str); // create a new std::string from the char*
|
|
|
|
params_p->prompt += str_token;
|
|
|
|
}
|
|
|
|
|
|
|
|
return get_embeddings(params_ptr, state_pr, res_embeddings);
|
|
|
|
}
|
|
|
|
|
|
|
|
int eval(void *params_ptr, void *state_pr, char *text) {
|
|
|
|
gpt_params *params_p = (gpt_params *)params_ptr;
|
|
|
|
llama_context *ctx = (llama_context *)state_pr;
|
2023-07-03 19:22:44 +00:00
|
|
|
|
|
|
|
auto n_past = 0;
|
2023-07-05 20:28:18 +00:00
|
|
|
auto last_n_tokens_data =
|
|
|
|
std::vector<llama_token>(params_p->repeat_last_n, 0);
|
2023-07-03 19:22:44 +00:00
|
|
|
|
2023-07-05 20:28:18 +00:00
|
|
|
auto tokens = std::vector<llama_token>(params_p->n_ctx);
|
2023-07-03 19:22:44 +00:00
|
|
|
auto n_prompt_tokens =
|
|
|
|
llama_tokenize(ctx, text, tokens.data(), tokens.size(), true);
|
|
|
|
|
|
|
|
if (n_prompt_tokens < 1) {
|
|
|
|
fprintf(stderr, "%s : failed to tokenize prompt\n", __func__);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2023-07-05 20:28:18 +00:00
|
|
|
// evaluate prompt
|
2023-07-03 19:22:44 +00:00
|
|
|
return llama_eval(ctx, tokens.data(), n_prompt_tokens, n_past,
|
2023-07-05 20:28:18 +00:00
|
|
|
params_p->n_threads);
|
2023-07-03 19:22:44 +00:00
|
|
|
}
|
|
|
|
|
2023-07-05 20:28:18 +00:00
|
|
|
int llama_predict(void *params_ptr, void *state_pr, char *result, bool debug) {
|
|
|
|
gpt_params *params_p = (gpt_params *)params_ptr;
|
|
|
|
llama_context *ctx = (llama_context *)state_pr;
|
|
|
|
|
|
|
|
gpt_params params = *params_p;
|
2023-07-03 19:22:44 +00:00
|
|
|
|
|
|
|
const int n_ctx = llama_n_ctx(ctx);
|
|
|
|
|
2023-07-05 20:28:18 +00:00
|
|
|
if (params.seed <= 0) {
|
|
|
|
params.seed = time(NULL);
|
2023-07-03 19:22:44 +00:00
|
|
|
}
|
|
|
|
|
2023-07-05 20:28:18 +00:00
|
|
|
std::mt19937 rng(params.seed);
|
|
|
|
|
|
|
|
std::string path_session = params.path_prompt_cache;
|
2023-07-03 19:22:44 +00:00
|
|
|
std::vector<llama_token> session_tokens;
|
|
|
|
|
|
|
|
if (!path_session.empty()) {
|
|
|
|
if (debug) {
|
|
|
|
fprintf(stderr, "%s: attempting to load saved session from '%s'\n",
|
|
|
|
__func__, path_session.c_str());
|
|
|
|
}
|
|
|
|
// fopen to check for existing session
|
|
|
|
FILE *fp = std::fopen(path_session.c_str(), "rb");
|
|
|
|
if (fp != NULL) {
|
|
|
|
std::fclose(fp);
|
|
|
|
|
|
|
|
session_tokens.resize(n_ctx);
|
|
|
|
size_t n_token_count_out = 0;
|
|
|
|
if (!llama_load_session_file(
|
|
|
|
ctx, path_session.c_str(), session_tokens.data(),
|
|
|
|
session_tokens.capacity(), &n_token_count_out)) {
|
|
|
|
fprintf(stderr, "%s: error: failed to load session file '%s'\n",
|
|
|
|
__func__, path_session.c_str());
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
session_tokens.resize(n_token_count_out);
|
2023-07-05 20:28:18 +00:00
|
|
|
llama_set_rng_seed(ctx, params.seed);
|
2023-07-03 19:22:44 +00:00
|
|
|
if (debug) {
|
|
|
|
fprintf(stderr, "%s: loaded a session with prompt size of %d tokens\n",
|
|
|
|
__func__, (int)session_tokens.size());
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (debug) {
|
|
|
|
fprintf(stderr, "%s: session file does not exist, will create\n",
|
|
|
|
__func__);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<llama_token> embd_inp;
|
2023-07-05 20:28:18 +00:00
|
|
|
if (!params.prompt.empty() || session_tokens.empty()) {
|
2023-07-03 19:22:44 +00:00
|
|
|
// Add a space in front of the first character to match OG llama tokenizer
|
|
|
|
// behavior
|
2023-07-05 20:28:18 +00:00
|
|
|
params.prompt.insert(0, 1, ' ');
|
2023-07-03 19:22:44 +00:00
|
|
|
|
2023-07-05 20:28:18 +00:00
|
|
|
embd_inp = ::llama_tokenize(ctx, params.prompt, true);
|
2023-07-03 19:22:44 +00:00
|
|
|
} else {
|
|
|
|
embd_inp = session_tokens;
|
|
|
|
}
|
|
|
|
|
|
|
|
// debug message about similarity of saved session, if applicable
|
|
|
|
size_t n_matching_session_tokens = 0;
|
|
|
|
if (session_tokens.size()) {
|
|
|
|
for (llama_token id : session_tokens) {
|
|
|
|
if (n_matching_session_tokens >= embd_inp.size() ||
|
|
|
|
id != embd_inp[n_matching_session_tokens]) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
n_matching_session_tokens++;
|
|
|
|
}
|
|
|
|
if (debug) {
|
2023-07-05 20:28:18 +00:00
|
|
|
if (params.prompt.empty() &&
|
2023-07-03 19:22:44 +00:00
|
|
|
n_matching_session_tokens == embd_inp.size()) {
|
|
|
|
fprintf(stderr, "%s: using full prompt from session file\n", __func__);
|
|
|
|
} else if (n_matching_session_tokens >= embd_inp.size()) {
|
|
|
|
fprintf(stderr, "%s: session file has exact match for prompt!\n",
|
|
|
|
__func__);
|
|
|
|
} else if (n_matching_session_tokens < (embd_inp.size() / 2)) {
|
|
|
|
fprintf(stderr,
|
|
|
|
"%s: warning: session file has low similarity to prompt (%zu / "
|
|
|
|
"%zu tokens); will mostly be reevaluated\n",
|
|
|
|
__func__, n_matching_session_tokens, embd_inp.size());
|
|
|
|
} else {
|
|
|
|
fprintf(stderr, "%s: session file matches %zu / %zu tokens of prompt\n",
|
|
|
|
__func__, n_matching_session_tokens, embd_inp.size());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// if we will use the cache for the full prompt without reaching the end of
|
|
|
|
// the cache, force reevaluation of the last token token to recalculate the
|
|
|
|
// cached logits
|
|
|
|
if (!embd_inp.empty() && n_matching_session_tokens == embd_inp.size() &&
|
|
|
|
session_tokens.size() > embd_inp.size()) {
|
|
|
|
session_tokens.resize(embd_inp.size() - 1);
|
|
|
|
}
|
|
|
|
// number of tokens to keep when resetting context
|
2023-07-05 20:28:18 +00:00
|
|
|
if (params.n_keep < 0 || params.n_keep > (int)embd_inp.size()) {
|
|
|
|
params.n_keep = (int)embd_inp.size();
|
2023-07-03 19:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// determine newline token
|
|
|
|
auto llama_token_newline = ::llama_tokenize(ctx, "\n", false);
|
|
|
|
|
|
|
|
// TODO: replace with ring-buffer
|
|
|
|
std::vector<llama_token> last_n_tokens(n_ctx);
|
|
|
|
std::fill(last_n_tokens.begin(), last_n_tokens.end(), 0);
|
|
|
|
|
|
|
|
bool need_to_save_session =
|
|
|
|
!path_session.empty() && n_matching_session_tokens < embd_inp.size();
|
|
|
|
int n_past = 0;
|
2023-07-05 20:28:18 +00:00
|
|
|
int n_remain = params.n_predict;
|
2023-07-03 19:22:44 +00:00
|
|
|
int n_consumed = 0;
|
|
|
|
int n_session_consumed = 0;
|
|
|
|
|
|
|
|
std::vector<llama_token> embd;
|
|
|
|
std::string res = "";
|
|
|
|
|
|
|
|
// do one empty run to warm up the model
|
|
|
|
{
|
|
|
|
const std::vector<llama_token> tmp = {
|
|
|
|
llama_token_bos(),
|
|
|
|
};
|
2023-07-05 20:28:18 +00:00
|
|
|
llama_eval(ctx, tmp.data(), tmp.size(), 0, params.n_threads);
|
2023-07-03 19:22:44 +00:00
|
|
|
llama_reset_timings(ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (n_remain != 0) {
|
|
|
|
// predict
|
|
|
|
if (embd.size() > 0) {
|
|
|
|
// infinite text generation via context swapping
|
|
|
|
// if we run out of context:
|
|
|
|
// - take the n_keep first tokens from the original prompt (via n_past)
|
|
|
|
// - take half of the last (n_ctx - n_keep) tokens and recompute the
|
|
|
|
// logits in batches
|
|
|
|
if (n_past + (int)embd.size() > n_ctx) {
|
2023-07-05 20:28:18 +00:00
|
|
|
const int n_left = n_past - params.n_keep;
|
2023-07-03 19:22:44 +00:00
|
|
|
|
|
|
|
// always keep the first token - BOS
|
2023-07-05 20:28:18 +00:00
|
|
|
n_past = std::max(1, params.n_keep);
|
2023-07-03 19:22:44 +00:00
|
|
|
|
|
|
|
// insert n_left/2 tokens at the start of embd from last_n_tokens
|
|
|
|
embd.insert(embd.begin(),
|
|
|
|
last_n_tokens.begin() + n_ctx - n_left / 2 - embd.size(),
|
|
|
|
last_n_tokens.end() - embd.size());
|
|
|
|
|
|
|
|
// stop saving session if we run out of context
|
|
|
|
path_session.clear();
|
2023-07-05 20:28:18 +00:00
|
|
|
|
|
|
|
// printf("\n---\n");
|
|
|
|
// printf("resetting: '");
|
|
|
|
// for (int i = 0; i < (int) embd.size(); i++) {
|
|
|
|
// printf("%s", llama_token_to_str(ctx, embd[i]));
|
|
|
|
// }
|
|
|
|
// printf("'\n");
|
|
|
|
// printf("\n---\n");
|
2023-07-03 19:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// try to reuse a matching prefix from the loaded session instead of
|
|
|
|
// re-eval (via n_past)
|
|
|
|
if (n_session_consumed < (int)session_tokens.size()) {
|
|
|
|
size_t i = 0;
|
|
|
|
for (; i < embd.size(); i++) {
|
|
|
|
if (embd[i] != session_tokens[n_session_consumed]) {
|
|
|
|
session_tokens.resize(n_session_consumed);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
n_past++;
|
|
|
|
n_session_consumed++;
|
|
|
|
|
|
|
|
if (n_session_consumed >= (int)session_tokens.size()) {
|
|
|
|
++i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (i > 0) {
|
|
|
|
embd.erase(embd.begin(), embd.begin() + i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// evaluate tokens in batches
|
|
|
|
// embd is typically prepared beforehand to fit within a batch, but not
|
|
|
|
// always
|
2023-07-05 20:28:18 +00:00
|
|
|
for (int i = 0; i < (int)embd.size(); i += params.n_batch) {
|
2023-07-03 19:22:44 +00:00
|
|
|
int n_eval = (int)embd.size() - i;
|
2023-07-05 20:28:18 +00:00
|
|
|
if (n_eval > params.n_batch) {
|
|
|
|
n_eval = params.n_batch;
|
2023-07-03 19:22:44 +00:00
|
|
|
}
|
2023-07-05 20:28:18 +00:00
|
|
|
if (llama_eval(ctx, &embd[i], n_eval, n_past, params.n_threads)) {
|
2023-07-03 19:22:44 +00:00
|
|
|
fprintf(stderr, "%s : failed to eval\n", __func__);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
n_past += n_eval;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (embd.size() > 0 && !path_session.empty()) {
|
|
|
|
session_tokens.insert(session_tokens.end(), embd.begin(), embd.end());
|
|
|
|
n_session_consumed = session_tokens.size();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
embd.clear();
|
|
|
|
|
|
|
|
if ((int)embd_inp.size() <= n_consumed) {
|
|
|
|
// out of user input, sample next token
|
2023-07-05 20:28:18 +00:00
|
|
|
const float temp = params.temp;
|
2023-07-03 19:22:44 +00:00
|
|
|
const int32_t top_k =
|
2023-07-05 20:28:18 +00:00
|
|
|
params.top_k <= 0 ? llama_n_vocab(ctx) : params.top_k;
|
|
|
|
const float top_p = params.top_p;
|
|
|
|
const float tfs_z = params.tfs_z;
|
|
|
|
const float typical_p = params.typical_p;
|
2023-07-03 19:22:44 +00:00
|
|
|
const int32_t repeat_last_n =
|
2023-07-05 20:28:18 +00:00
|
|
|
params.repeat_last_n < 0 ? n_ctx : params.repeat_last_n;
|
|
|
|
const float repeat_penalty = params.repeat_penalty;
|
|
|
|
const float alpha_presence = params.presence_penalty;
|
|
|
|
const float alpha_frequency = params.frequency_penalty;
|
|
|
|
const int mirostat = params.mirostat;
|
|
|
|
const float mirostat_tau = params.mirostat_tau;
|
|
|
|
const float mirostat_eta = params.mirostat_eta;
|
|
|
|
const bool penalize_nl = params.penalize_nl;
|
2023-07-03 19:22:44 +00:00
|
|
|
|
|
|
|
// optionally save the session on first sample (for faster prompt loading
|
|
|
|
// next time)
|
|
|
|
if (!path_session.empty() && need_to_save_session &&
|
2023-07-05 20:28:18 +00:00
|
|
|
!params.prompt_cache_ro) {
|
2023-07-03 19:22:44 +00:00
|
|
|
need_to_save_session = false;
|
|
|
|
llama_save_session_file(ctx, path_session.c_str(),
|
|
|
|
session_tokens.data(), session_tokens.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
llama_token id = 0;
|
|
|
|
|
|
|
|
{
|
|
|
|
auto logits = llama_get_logits(ctx);
|
|
|
|
auto n_vocab = llama_n_vocab(ctx);
|
|
|
|
|
|
|
|
// Apply params.logit_bias map
|
2023-07-05 20:28:18 +00:00
|
|
|
for (auto it = params.logit_bias.begin(); it != params.logit_bias.end();
|
|
|
|
it++) {
|
2023-07-03 19:22:44 +00:00
|
|
|
logits[it->first] += it->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<llama_token_data> candidates;
|
|
|
|
candidates.reserve(n_vocab);
|
|
|
|
for (llama_token token_id = 0; token_id < n_vocab; token_id++) {
|
|
|
|
candidates.emplace_back(
|
|
|
|
llama_token_data{token_id, logits[token_id], 0.0f});
|
|
|
|
}
|
|
|
|
|
|
|
|
llama_token_data_array candidates_p = {candidates.data(),
|
|
|
|
candidates.size(), false};
|
|
|
|
|
|
|
|
// Apply penalties
|
|
|
|
float nl_logit = logits[llama_token_nl()];
|
|
|
|
auto last_n_repeat =
|
|
|
|
std::min(std::min((int)last_n_tokens.size(), repeat_last_n), n_ctx);
|
|
|
|
llama_sample_repetition_penalty(
|
|
|
|
ctx, &candidates_p,
|
|
|
|
last_n_tokens.data() + last_n_tokens.size() - last_n_repeat,
|
|
|
|
last_n_repeat, repeat_penalty);
|
|
|
|
llama_sample_frequency_and_presence_penalties(
|
|
|
|
ctx, &candidates_p,
|
|
|
|
last_n_tokens.data() + last_n_tokens.size() - last_n_repeat,
|
|
|
|
last_n_repeat, alpha_frequency, alpha_presence);
|
|
|
|
if (!penalize_nl) {
|
|
|
|
logits[llama_token_nl()] = nl_logit;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (temp <= 0) {
|
|
|
|
// Greedy sampling
|
|
|
|
id = llama_sample_token_greedy(ctx, &candidates_p);
|
|
|
|
} else {
|
|
|
|
if (mirostat == 1) {
|
|
|
|
static float mirostat_mu = 2.0f * mirostat_tau;
|
|
|
|
const int mirostat_m = 100;
|
|
|
|
llama_sample_temperature(ctx, &candidates_p, temp);
|
|
|
|
id = llama_sample_token_mirostat(ctx, &candidates_p, mirostat_tau,
|
|
|
|
mirostat_eta, mirostat_m,
|
|
|
|
&mirostat_mu);
|
|
|
|
} else if (mirostat == 2) {
|
|
|
|
static float mirostat_mu = 2.0f * mirostat_tau;
|
|
|
|
llama_sample_temperature(ctx, &candidates_p, temp);
|
|
|
|
id = llama_sample_token_mirostat_v2(
|
|
|
|
ctx, &candidates_p, mirostat_tau, mirostat_eta, &mirostat_mu);
|
|
|
|
} else {
|
|
|
|
// Temperature sampling
|
|
|
|
llama_sample_top_k(ctx, &candidates_p, top_k, 1);
|
|
|
|
llama_sample_tail_free(ctx, &candidates_p, tfs_z, 1);
|
|
|
|
llama_sample_typical(ctx, &candidates_p, typical_p, 1);
|
|
|
|
llama_sample_top_p(ctx, &candidates_p, top_p, 1);
|
|
|
|
llama_sample_temperature(ctx, &candidates_p, temp);
|
|
|
|
id = llama_sample_token(ctx, &candidates_p);
|
|
|
|
}
|
|
|
|
}
|
2023-07-05 20:28:18 +00:00
|
|
|
// printf("`%d`", candidates_p.size);
|
2023-07-03 19:22:44 +00:00
|
|
|
|
|
|
|
last_n_tokens.erase(last_n_tokens.begin());
|
|
|
|
last_n_tokens.push_back(id);
|
|
|
|
}
|
|
|
|
|
|
|
|
// add it to the context
|
|
|
|
embd.push_back(id);
|
|
|
|
|
|
|
|
// decrement remaining sampling budget
|
|
|
|
--n_remain;
|
|
|
|
|
|
|
|
// call the token callback, no need to check if one is actually
|
|
|
|
// registered, that will be handled on the Go side.
|
|
|
|
auto token_str = llama_token_to_str(ctx, id);
|
2023-07-05 20:28:18 +00:00
|
|
|
if (!tokenCallback(state_pr, (char *)token_str)) {
|
2023-07-03 19:22:44 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// some user input remains from prompt or interaction, forward it to
|
|
|
|
// processing
|
|
|
|
while ((int)embd_inp.size() > n_consumed) {
|
|
|
|
embd.push_back(embd_inp[n_consumed]);
|
|
|
|
last_n_tokens.erase(last_n_tokens.begin());
|
|
|
|
last_n_tokens.push_back(embd_inp[n_consumed]);
|
|
|
|
++n_consumed;
|
2023-07-05 20:28:18 +00:00
|
|
|
if ((int)embd.size() >= params.n_batch) {
|
2023-07-03 19:22:44 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto id : embd) {
|
|
|
|
res += llama_token_to_str(ctx, id);
|
|
|
|
}
|
|
|
|
|
|
|
|
// check for stop prompt
|
2023-07-05 20:28:18 +00:00
|
|
|
if (params.antiprompt.size()) {
|
2023-07-03 19:22:44 +00:00
|
|
|
std::string last_output;
|
|
|
|
for (auto id : last_n_tokens) {
|
|
|
|
last_output += llama_token_to_str(ctx, id);
|
|
|
|
}
|
|
|
|
// Check if each of the reverse prompts appears at the end of the output.
|
2023-07-05 20:28:18 +00:00
|
|
|
for (std::string &antiprompt : params.antiprompt) {
|
2023-07-03 19:22:44 +00:00
|
|
|
// size_t extra_padding = params.interactive ? 0 : 2;
|
|
|
|
size_t extra_padding = 2;
|
|
|
|
size_t search_start_pos =
|
|
|
|
last_output.length() >
|
|
|
|
static_cast<size_t>(antiprompt.length() + extra_padding)
|
|
|
|
? last_output.length() -
|
|
|
|
static_cast<size_t>(antiprompt.length() + extra_padding)
|
|
|
|
: 0;
|
|
|
|
|
|
|
|
if (last_output.find(antiprompt.c_str(), search_start_pos) !=
|
|
|
|
std::string::npos) {
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// end of text token
|
|
|
|
if (!embd.empty() && embd.back() == llama_token_eos()) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-05 20:28:18 +00:00
|
|
|
if (!path_session.empty() && params.prompt_cache_all &&
|
|
|
|
!params.prompt_cache_ro) {
|
2023-07-03 19:22:44 +00:00
|
|
|
if (debug) {
|
|
|
|
fprintf(stderr, "\n%s: saving final output to session file '%s'\n",
|
|
|
|
__func__, path_session.c_str());
|
|
|
|
}
|
|
|
|
llama_save_session_file(ctx, path_session.c_str(), session_tokens.data(),
|
|
|
|
session_tokens.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
end:
|
|
|
|
#if defined(_WIN32)
|
|
|
|
signal(SIGINT, SIG_DFL);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (debug) {
|
|
|
|
llama_print_timings(ctx);
|
|
|
|
llama_reset_timings(ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
strcpy(result, res.c_str());
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2023-07-05 20:28:18 +00:00
|
|
|
void llama_binding_free_model(void *state_ptr) {
|
|
|
|
llama_context *ctx = (llama_context *)state_ptr;
|
|
|
|
llama_free(ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
void llama_free_params(void *params_ptr) {
|
|
|
|
gpt_params *params = (gpt_params *)params_ptr;
|
|
|
|
delete params;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<std::string> create_vector(const char **strings, int count) {
|
|
|
|
std::vector<std::string> *vec = new std::vector<std::string>;
|
|
|
|
for (int i = 0; i < count; i++) {
|
|
|
|
vec->push_back(std::string(strings[i]));
|
|
|
|
}
|
|
|
|
return *vec;
|
|
|
|
}
|
|
|
|
|
|
|
|
void delete_vector(std::vector<std::string> *vec) { delete vec; }
|
|
|
|
|
|
|
|
int load_state(void *ctx, char *statefile, char *modes) {
|
|
|
|
llama_context *state = (llama_context *)ctx;
|
|
|
|
const llama_context *constState = static_cast<const llama_context *>(state);
|
|
|
|
const size_t state_size = llama_get_state_size(state);
|
|
|
|
uint8_t *state_mem = new uint8_t[state_size];
|
|
|
|
|
|
|
|
{
|
|
|
|
FILE *fp_read = fopen(statefile, modes);
|
|
|
|
if (state_size != llama_get_state_size(constState)) {
|
|
|
|
fprintf(stderr, "\n%s : failed to validate state size\n", __func__);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
const size_t ret = fread(state_mem, 1, state_size, fp_read);
|
|
|
|
if (ret != state_size) {
|
|
|
|
fprintf(stderr, "\n%s : failed to read state\n", __func__);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
llama_set_state_data(
|
|
|
|
state, state_mem); // could also read directly from memory mapped file
|
|
|
|
fclose(fp_read);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void save_state(void *ctx, char *dst, char *modes) {
|
|
|
|
llama_context *state = (llama_context *)ctx;
|
|
|
|
|
|
|
|
const size_t state_size = llama_get_state_size(state);
|
|
|
|
uint8_t *state_mem = new uint8_t[state_size];
|
|
|
|
|
|
|
|
// Save state (rng, logits, embedding and kv_cache) to file
|
|
|
|
{
|
|
|
|
FILE *fp_write = fopen(dst, modes);
|
|
|
|
llama_copy_state_data(
|
|
|
|
state, state_mem); // could also copy directly to memory mapped file
|
|
|
|
fwrite(state_mem, 1, state_size, fp_write);
|
|
|
|
fclose(fp_write);
|
|
|
|
}
|
|
|
|
}
|
2023-07-03 19:22:44 +00:00
|
|
|
|
|
|
|
void *llama_allocate_params(
|
|
|
|
const char *prompt, int seed, int threads, int tokens, int top_k,
|
|
|
|
float top_p, float temp, float repeat_penalty, int repeat_last_n,
|
|
|
|
bool ignore_eos, bool memory_f16, int n_batch, int n_keep,
|
|
|
|
const char **antiprompt, int antiprompt_count, float tfs_z, float typical_p,
|
|
|
|
float frequency_penalty, float presence_penalty, int mirostat,
|
|
|
|
float mirostat_eta, float mirostat_tau, bool penalize_nl,
|
2023-07-06 21:49:05 +00:00
|
|
|
const char *logit_bias, bool mlock, bool mmap, const char *maingpu,
|
|
|
|
const char *tensorsplit) {
|
2023-07-03 19:22:44 +00:00
|
|
|
gpt_params *params = new gpt_params;
|
|
|
|
params->seed = seed;
|
|
|
|
params->n_threads = threads;
|
|
|
|
params->n_predict = tokens;
|
|
|
|
params->repeat_last_n = repeat_last_n;
|
|
|
|
params->top_k = top_k;
|
|
|
|
params->top_p = top_p;
|
|
|
|
params->memory_f16 = memory_f16;
|
|
|
|
params->temp = temp;
|
|
|
|
params->use_mmap = mmap;
|
|
|
|
params->use_mlock = mlock;
|
|
|
|
params->repeat_penalty = repeat_penalty;
|
|
|
|
params->n_batch = n_batch;
|
|
|
|
params->n_keep = n_keep;
|
|
|
|
if (maingpu[0] != '\0') {
|
|
|
|
params->main_gpu = std::stoi(maingpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tensorsplit[0] != '\0') {
|
|
|
|
std::string arg_next = tensorsplit;
|
|
|
|
// split string by , and /
|
|
|
|
const std::regex regex{R"([,/]+)"};
|
|
|
|
std::sregex_token_iterator it{arg_next.begin(), arg_next.end(), regex, -1};
|
|
|
|
std::vector<std::string> split_arg{it, {}};
|
|
|
|
GGML_ASSERT(split_arg.size() <= LLAMA_MAX_DEVICES);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < LLAMA_MAX_DEVICES; ++i) {
|
|
|
|
if (i < split_arg.size()) {
|
|
|
|
params->tensor_split[i] = std::stof(split_arg[i]);
|
|
|
|
} else {
|
|
|
|
params->tensor_split[i] = 0.0f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ignore_eos) {
|
|
|
|
params->logit_bias[llama_token_eos()] = -INFINITY;
|
|
|
|
}
|
|
|
|
if (antiprompt_count > 0) {
|
2023-07-05 20:28:18 +00:00
|
|
|
params->antiprompt = create_vector(antiprompt, antiprompt_count);
|
2023-07-03 19:22:44 +00:00
|
|
|
}
|
|
|
|
params->tfs_z = tfs_z;
|
|
|
|
params->typical_p = typical_p;
|
|
|
|
params->presence_penalty = presence_penalty;
|
|
|
|
params->mirostat = mirostat;
|
|
|
|
params->mirostat_eta = mirostat_eta;
|
|
|
|
params->mirostat_tau = mirostat_tau;
|
|
|
|
params->penalize_nl = penalize_nl;
|
|
|
|
std::stringstream ss(logit_bias);
|
|
|
|
llama_token key;
|
|
|
|
char sign;
|
|
|
|
std::string value_str;
|
|
|
|
if (ss >> key && ss >> sign && std::getline(ss, value_str) &&
|
|
|
|
(sign == '+' || sign == '-')) {
|
|
|
|
params->logit_bias[key] =
|
|
|
|
std::stof(value_str) * ((sign == '-') ? -1.0f : 1.0f);
|
|
|
|
}
|
|
|
|
params->frequency_penalty = frequency_penalty;
|
|
|
|
params->prompt = prompt;
|
|
|
|
|
|
|
|
return params;
|
|
|
|
}
|
|
|
|
|
|
|
|
void *load_model(const char *fname, int n_ctx, int n_seed, bool memory_f16,
|
|
|
|
bool mlock, bool embeddings, bool mmap, bool low_vram,
|
|
|
|
bool vocab_only, int n_gpu_layers, int n_batch,
|
|
|
|
const char *maingpu, const char *tensorsplit, bool numa) {
|
2023-07-05 20:28:18 +00:00
|
|
|
// load the model
|
2023-07-03 19:22:44 +00:00
|
|
|
auto lparams = llama_context_default_params();
|
|
|
|
|
|
|
|
lparams.n_ctx = n_ctx;
|
|
|
|
lparams.seed = n_seed;
|
|
|
|
lparams.f16_kv = memory_f16;
|
|
|
|
lparams.embedding = embeddings;
|
|
|
|
lparams.use_mlock = mlock;
|
|
|
|
lparams.n_gpu_layers = n_gpu_layers;
|
|
|
|
lparams.use_mmap = mmap;
|
|
|
|
lparams.low_vram = low_vram;
|
|
|
|
lparams.vocab_only = vocab_only;
|
|
|
|
|
|
|
|
if (maingpu[0] != '\0') {
|
|
|
|
lparams.main_gpu = std::stoi(maingpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tensorsplit[0] != '\0') {
|
|
|
|
std::string arg_next = tensorsplit;
|
|
|
|
// split string by , and /
|
|
|
|
const std::regex regex{R"([,/]+)"};
|
|
|
|
std::sregex_token_iterator it{arg_next.begin(), arg_next.end(), regex, -1};
|
|
|
|
std::vector<std::string> split_arg{it, {}};
|
|
|
|
GGML_ASSERT(split_arg.size() <= LLAMA_MAX_DEVICES);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < LLAMA_MAX_DEVICES; ++i) {
|
|
|
|
if (i < split_arg.size()) {
|
|
|
|
lparams.tensor_split[i] = std::stof(split_arg[i]);
|
|
|
|
} else {
|
|
|
|
lparams.tensor_split[i] = 0.0f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
lparams.n_batch = n_batch;
|
|
|
|
|
|
|
|
llama_init_backend(numa);
|
2023-07-05 20:28:18 +00:00
|
|
|
void *res = nullptr;
|
|
|
|
try {
|
|
|
|
res = llama_init_from_file(fname, lparams);
|
|
|
|
} catch (std::runtime_error &e) {
|
|
|
|
fprintf(stderr, "failed %s", e.what());
|
|
|
|
return res;
|
2023-07-03 19:22:44 +00:00
|
|
|
}
|
|
|
|
|
2023-07-05 20:28:18 +00:00
|
|
|
return res;
|
|
|
|
}
|