| #include "sampling.h" |
|
|
| #include "common.h" |
|
|
| #include <cmath> |
| #include <unordered_map> |
|
|
| |
| |
| template<typename T> |
| struct ring_buffer { |
| ring_buffer(size_t cap) : capacity(cap), data(cap) {} |
|
|
| T & front() { |
| if (sz == 0) { |
| throw std::runtime_error("ring buffer is empty"); |
| } |
| return data[first]; |
| } |
|
|
| const T & front() const { |
| if (sz == 0) { |
| throw std::runtime_error("ring buffer is empty"); |
| } |
| return data[first]; |
| } |
|
|
| T & back() { |
| if (sz == 0) { |
| throw std::runtime_error("ring buffer is empty"); |
| } |
| return data[pos]; |
| } |
|
|
| const T & back() const { |
| if (sz == 0) { |
| throw std::runtime_error("ring buffer is empty"); |
| } |
| return data[pos]; |
| } |
|
|
| void push_back(const T & value) { |
| if (sz == capacity) { |
| |
| first = (first + 1) % capacity; |
| } else { |
| sz++; |
| } |
| data[pos] = value; |
| pos = (pos + 1) % capacity; |
| } |
|
|
| T pop_front() { |
| if (sz == 0) { |
| throw std::runtime_error("ring buffer is empty"); |
| } |
| T value = data[first]; |
| first = (first + 1) % capacity; |
| sz--; |
| return value; |
| } |
|
|
| const T & rat(size_t i) const { |
| if (i >= sz) { |
| throw std::runtime_error("ring buffer: index out of bounds"); |
| } |
| return data[(first + sz - i - 1) % capacity]; |
| } |
|
|
| std::vector<T> to_vector() const { |
| std::vector<T> result; |
| result.reserve(sz); |
| for (size_t i = 0; i < sz; i++) { |
| result.push_back(data[(first + i) % capacity]); |
| } |
| return result; |
| } |
|
|
| void clear() { |
| |
| sz = 0; |
| first = 0; |
| pos = 0; |
| } |
|
|
| bool empty() const { |
| return sz == 0; |
| } |
|
|
| size_t size() const { |
| return sz; |
| } |
|
|
| size_t capacity = 0; |
| size_t sz = 0; |
| size_t first = 0; |
| size_t pos = 0; |
| std::vector<T> data; |
| }; |
|
|
| struct common_sampler { |
| common_params_sampling params; |
|
|
| struct llama_sampler * grmr; |
| struct llama_sampler * chain; |
|
|
| ring_buffer<llama_token> prev; |
|
|
| std::vector<llama_token_data> cur; |
|
|
| llama_token_data_array cur_p; |
|
|
| void set_logits(struct llama_context * ctx, int idx) { |
| const auto * logits = llama_get_logits_ith(ctx, idx); |
|
|
| const int n_vocab = llama_n_vocab(llama_get_model(ctx)); |
|
|
| cur.resize(n_vocab); |
|
|
| for (llama_token token_id = 0; token_id < n_vocab; token_id++) { |
| cur[token_id] = llama_token_data{token_id, logits[token_id], 0.0f}; |
| } |
|
|
| cur_p = { cur.data(), cur.size(), -1, false }; |
| } |
| }; |
|
|
| std::string common_params_sampling::print() const { |
| char result[1024]; |
|
|
| snprintf(result, sizeof(result), |
| "\trepeat_last_n = %d, repeat_penalty = %.3f, frequency_penalty = %.3f, presence_penalty = %.3f\n" |
| "\tdry_multiplier = %.3f, dry_base = %.3f, dry_allowed_length = %d, dry_penalty_last_n = %d\n" |
| "\ttop_k = %d, top_p = %.3f, min_p = %.3f, xtc_probability = %.3f, xtc_threshold = %.3f, typical_p = %.3f, temp = %.3f\n" |
| "\tmirostat = %d, mirostat_lr = %.3f, mirostat_ent = %.3f", |
| penalty_last_n, penalty_repeat, penalty_freq, penalty_present, |
| dry_multiplier, dry_base, dry_allowed_length, dry_penalty_last_n, |
| top_k, top_p, min_p, xtc_probability, xtc_threshold, typ_p, temp, |
| mirostat, mirostat_eta, mirostat_tau); |
|
|
| return std::string(result); |
| } |
|
|
| struct common_sampler * common_sampler_init(const struct llama_model * model, const struct common_params_sampling & params) { |
| llama_sampler_chain_params lparams = llama_sampler_chain_default_params(); |
|
|
| lparams.no_perf = params.no_perf; |
|
|
| auto * result = new common_sampler { |
| params, |
| llama_sampler_init_grammar(model, params.grammar.c_str(), "root"), |
| llama_sampler_chain_init(lparams), |
| ring_buffer<llama_token>(std::max(32, params.n_prev)), |
| {}, |
| {}, |
| }; |
|
|
| llama_sampler_chain_add(result->chain, |
| llama_sampler_init_logit_bias( |
| llama_n_vocab(model), |
| params.logit_bias.size(), |
| params.logit_bias.data())); |
|
|
| llama_sampler_chain_add(result->chain, |
| llama_sampler_init_penalties( |
| llama_n_vocab (model), |
| llama_token_eos(model), |
| llama_token_nl (model), |
| params.penalty_last_n, |
| params.penalty_repeat, |
| params.penalty_freq, |
| params.penalty_present, |
| params.penalize_nl, |
| params.ignore_eos)); |
|
|
| if (params.mirostat == 0) { |
| for (const auto & cnstr : params.samplers) { |
| switch (cnstr) { |
| case COMMON_SAMPLER_TYPE_DRY: |
| { |
| std::vector<const char*> c_breakers; |
| c_breakers.reserve(params.dry_sequence_breakers.size()); |
| for (const auto& str : params.dry_sequence_breakers) { |
| c_breakers.push_back(str.c_str()); |
| } |
|
|
| llama_sampler_chain_add(result->chain, llama_sampler_init_dry (model, params.dry_multiplier, params.dry_base, params.dry_allowed_length, params.dry_penalty_last_n, c_breakers.data(), c_breakers.size())); |
| } |
| break; |
| case COMMON_SAMPLER_TYPE_TOP_K: |
| llama_sampler_chain_add(result->chain, llama_sampler_init_top_k (params.top_k)); |
| break; |
| case COMMON_SAMPLER_TYPE_TOP_P: |
| llama_sampler_chain_add(result->chain, llama_sampler_init_top_p (params.top_p, params.min_keep)); |
| break; |
| case COMMON_SAMPLER_TYPE_MIN_P: |
| llama_sampler_chain_add(result->chain, llama_sampler_init_min_p (params.min_p, params.min_keep)); |
| break; |
| case COMMON_SAMPLER_TYPE_XTC: |
| llama_sampler_chain_add(result->chain, llama_sampler_init_xtc (params.xtc_probability, params.xtc_threshold, params.min_keep, params.seed)); |
| break; |
| case COMMON_SAMPLER_TYPE_TYPICAL_P: |
| llama_sampler_chain_add(result->chain, llama_sampler_init_typical (params.typ_p, params.min_keep)); |
| break; |
| case COMMON_SAMPLER_TYPE_TEMPERATURE: |
| llama_sampler_chain_add(result->chain, llama_sampler_init_temp_ext (params.temp, params.dynatemp_range, params.dynatemp_exponent)); |
| break; |
| case COMMON_SAMPLER_TYPE_INFILL: |
| llama_sampler_chain_add(result->chain, llama_sampler_init_infill (model)); |
| break; |
| default: |
| GGML_ASSERT(false && "unknown sampler type"); |
| } |
| } |
| llama_sampler_chain_add(result->chain, llama_sampler_init_dist(params.seed)); |
| } else if (params.mirostat == 1) { |
| llama_sampler_chain_add(result->chain, llama_sampler_init_temp(params.temp)); |
| llama_sampler_chain_add(result->chain, llama_sampler_init_mirostat(llama_n_vocab(model), params.seed, params.mirostat_tau, params.mirostat_eta, 100)); |
| } else if (params.mirostat == 2) { |
| llama_sampler_chain_add(result->chain, llama_sampler_init_temp(params.temp)); |
| llama_sampler_chain_add(result->chain, llama_sampler_init_mirostat_v2(params.seed, params.mirostat_tau, params.mirostat_eta)); |
| } else { |
| GGML_ASSERT(false && "unknown mirostat version"); |
| } |
|
|
| return result; |
| } |
|
|
| void common_sampler_free(struct common_sampler * gsmpl) { |
| if (gsmpl) { |
| llama_sampler_free(gsmpl->grmr); |
|
|
| llama_sampler_free(gsmpl->chain); |
|
|
| delete gsmpl; |
| } |
| } |
|
|
| void common_sampler_accept(struct common_sampler * gsmpl, llama_token token, bool accept_grammar) { |
| if (accept_grammar) { |
| llama_sampler_accept(gsmpl->grmr, token); |
| } |
|
|
| llama_sampler_accept(gsmpl->chain, token); |
|
|
| gsmpl->prev.push_back(token); |
| } |
|
|
| void common_sampler_reset(struct common_sampler * gsmpl) { |
| llama_sampler_reset(gsmpl->grmr); |
|
|
| llama_sampler_reset(gsmpl->chain); |
| } |
|
|
| struct common_sampler * common_sampler_clone(common_sampler * gsmpl) { |
| return new common_sampler { |
| gsmpl->params, |
| llama_sampler_clone(gsmpl->grmr), |
| llama_sampler_clone(gsmpl->chain), |
| gsmpl->prev, |
| gsmpl->cur, |
| gsmpl->cur_p, |
| }; |
| } |
|
|
| void common_perf_print(const struct llama_context * ctx, const struct common_sampler * gsmpl) { |
| |
|
|
| if (gsmpl) { |
| llama_perf_sampler_print(gsmpl->chain); |
| } |
| if (ctx) { |
| llama_perf_context_print(ctx); |
| } |
| } |
|
|
| llama_token common_sampler_sample(struct common_sampler * gsmpl, struct llama_context * ctx, int idx, bool grammar_first) { |
| gsmpl->set_logits(ctx, idx); |
|
|
| auto & grmr = gsmpl->grmr; |
| auto & chain = gsmpl->chain; |
| auto & cur_p = gsmpl->cur_p; |
|
|
| if (grammar_first) { |
| llama_sampler_apply(grmr, &cur_p); |
| } |
|
|
| llama_sampler_apply(chain, &cur_p); |
|
|
| GGML_ASSERT(cur_p.selected != -1 && "no selected token during sampling - check your sampling configuration"); |
|
|
| const llama_token id = cur_p.data[cur_p.selected].id; |
|
|
| if (grammar_first) { |
| return id; |
| } |
|
|
| |
| { |
| llama_token_data single_token_data = { id, 1.0f, 0.0f }; |
| llama_token_data_array single_token_data_array = { &single_token_data, 1, -1, false }; |
|
|
| llama_sampler_apply(grmr, &single_token_data_array); |
|
|
| const bool is_valid = single_token_data_array.data[0].logit != -INFINITY; |
| if (is_valid) { |
| return id; |
| } |
| } |
|
|
| |
| |
| gsmpl->set_logits(ctx, idx); |
|
|
| llama_sampler_apply(grmr, &cur_p); |
| llama_sampler_apply(chain, &cur_p); |
|
|
| GGML_ASSERT(cur_p.selected != -1 && "no selected token during re-sampling - check your sampling configuration"); |
|
|
| return cur_p.data[cur_p.selected].id; |
| } |
|
|
| std::vector<llama_token> common_sampler_sample_and_accept_n(struct common_sampler * gsmpl, struct llama_context * ctx, const std::vector<int> & idxs, const llama_tokens & draft, bool grammar_first) { |
| GGML_ASSERT(idxs.size() == draft.size() + 1 && "idxs.size() must be draft.size() + 1"); |
|
|
| std::vector<llama_token> result; |
| result.reserve(idxs.size()); |
|
|
| size_t i = 0; |
| for (; i < draft.size(); i++) { |
| const llama_token id = common_sampler_sample(gsmpl, ctx, idxs[i], grammar_first); |
|
|
| common_sampler_accept(gsmpl, id, true); |
|
|
| result.push_back(id); |
|
|
| if (draft[i] != id) { |
| break; |
| } |
| } |
|
|
| if (i == draft.size()) { |
| const llama_token id = common_sampler_sample(gsmpl, ctx, idxs[i], grammar_first); |
|
|
| common_sampler_accept(gsmpl, id, true); |
|
|
| result.push_back(id); |
| } |
|
|
| return result; |
| } |
|
|
| std::vector<llama_token> common_sampler_sample_and_accept_n(struct common_sampler * gsmpl, struct llama_context * ctx, const llama_tokens & draft, bool grammar_first) { |
| std::vector<int> idxs(draft.size() + 1); |
| for (size_t i = 0; i < idxs.size(); ++i) { |
| idxs[i] = i; |
| } |
|
|
| return common_sampler_sample_and_accept_n(gsmpl, ctx, idxs, draft, grammar_first); |
| } |
|
|
| uint32_t common_sampler_get_seed(const struct common_sampler * gsmpl) { |
| return llama_sampler_get_seed(gsmpl->chain); |
| } |
|
|
| |
|
|
| llama_token_data_array * common_sampler_get_candidates(struct common_sampler * gsmpl) { |
| return &gsmpl->cur_p; |
| } |
|
|
| llama_token common_sampler_last(const struct common_sampler * gsmpl) { |
| return gsmpl->prev.rat(0); |
| } |
|
|
| std::string common_sampler_print(const struct common_sampler * gsmpl) { |
| std::string result = "logits "; |
|
|
| for (int i = 0; i < llama_sampler_chain_n(gsmpl->chain); i++) { |
| const auto * smpl = llama_sampler_chain_get(gsmpl->chain, i); |
| result += std::string("-> ") + llama_sampler_name(smpl) + " "; |
| } |
|
|
| return result; |
| } |
|
|
| std::string common_sampler_prev_str(common_sampler * gsmpl, llama_context * ctx_main, int n) { |
| n = std::min(n, (int) gsmpl->prev.size()); |
|
|
| if (n <= 0) { |
| return ""; |
| } |
|
|
| std::string result; |
| result.reserve(8*n); |
|
|
| for (int i = n - 1; i >= 0; i--) { |
| const llama_token id = gsmpl->prev.rat(i); |
|
|
| GGML_ASSERT(id != LLAMA_TOKEN_NULL && "null token in the sampling history - should not happen"); |
|
|
| result += common_token_to_piece(ctx_main, id); |
| } |
|
|
| return result; |
| } |
|
|
| char common_sampler_type_to_chr(enum common_sampler_type cnstr) { |
| switch (cnstr) { |
| case COMMON_SAMPLER_TYPE_DRY: return 'd'; |
| case COMMON_SAMPLER_TYPE_TOP_K: return 'k'; |
| case COMMON_SAMPLER_TYPE_TYPICAL_P: return 'y'; |
| case COMMON_SAMPLER_TYPE_TOP_P: return 'p'; |
| case COMMON_SAMPLER_TYPE_MIN_P: return 'm'; |
| case COMMON_SAMPLER_TYPE_TEMPERATURE: return 't'; |
| case COMMON_SAMPLER_TYPE_XTC: return 'x'; |
| case COMMON_SAMPLER_TYPE_INFILL: return 'i'; |
| default : return '?'; |
| } |
| } |
|
|
| std::string common_sampler_type_to_str(enum common_sampler_type cnstr) { |
| switch (cnstr) { |
| case COMMON_SAMPLER_TYPE_DRY: return "dry"; |
| case COMMON_SAMPLER_TYPE_TOP_K: return "top_k"; |
| case COMMON_SAMPLER_TYPE_TYPICAL_P: return "typ_p"; |
| case COMMON_SAMPLER_TYPE_TOP_P: return "top_p"; |
| case COMMON_SAMPLER_TYPE_MIN_P: return "min_p"; |
| case COMMON_SAMPLER_TYPE_TEMPERATURE: return "temperature"; |
| case COMMON_SAMPLER_TYPE_XTC: return "xtc"; |
| case COMMON_SAMPLER_TYPE_INFILL: return "infill"; |
| default : return ""; |
| } |
| } |
|
|
| std::vector<common_sampler_type> common_sampler_types_from_names(const std::vector<std::string> & names, bool allow_alt_names) { |
| std::unordered_map<std::string, common_sampler_type> sampler_canonical_name_map { |
| { "dry", COMMON_SAMPLER_TYPE_DRY }, |
| { "top_k", COMMON_SAMPLER_TYPE_TOP_K }, |
| { "top_p", COMMON_SAMPLER_TYPE_TOP_P }, |
| { "typ_p", COMMON_SAMPLER_TYPE_TYPICAL_P }, |
| { "min_p", COMMON_SAMPLER_TYPE_MIN_P }, |
| { "temperature", COMMON_SAMPLER_TYPE_TEMPERATURE }, |
| { "xtc", COMMON_SAMPLER_TYPE_XTC }, |
| { "infill", COMMON_SAMPLER_TYPE_INFILL }, |
| }; |
|
|
| |
| |
| std::unordered_map<std::string, common_sampler_type> sampler_alt_name_map { |
| { "top-k", COMMON_SAMPLER_TYPE_TOP_K }, |
| { "top-p", COMMON_SAMPLER_TYPE_TOP_P }, |
| { "nucleus", COMMON_SAMPLER_TYPE_TOP_P }, |
| { "typical-p", COMMON_SAMPLER_TYPE_TYPICAL_P }, |
| { "typical", COMMON_SAMPLER_TYPE_TYPICAL_P }, |
| { "typ-p", COMMON_SAMPLER_TYPE_TYPICAL_P }, |
| { "typ", COMMON_SAMPLER_TYPE_TYPICAL_P }, |
| { "min-p", COMMON_SAMPLER_TYPE_MIN_P }, |
| { "temp", COMMON_SAMPLER_TYPE_TEMPERATURE }, |
| }; |
|
|
| std::vector<common_sampler_type> samplers; |
| samplers.reserve(names.size()); |
|
|
| for (const auto & name : names) { |
| auto sampler = sampler_canonical_name_map.find(name); |
| if (sampler != sampler_canonical_name_map.end()) { |
| samplers.push_back(sampler->second); |
| } else { |
| if (allow_alt_names) { |
| sampler = sampler_alt_name_map.find(name); |
| if (sampler != sampler_alt_name_map.end()) { |
| samplers.push_back(sampler->second); |
| } |
| } |
| } |
| } |
|
|
| return samplers; |
| } |
|
|
| std::vector<common_sampler_type> common_sampler_types_from_chars(const std::string & chars) { |
| std::unordered_map<char, common_sampler_type> sampler_name_map = { |
| { common_sampler_type_to_chr(COMMON_SAMPLER_TYPE_DRY), COMMON_SAMPLER_TYPE_DRY }, |
| { common_sampler_type_to_chr(COMMON_SAMPLER_TYPE_TOP_K), COMMON_SAMPLER_TYPE_TOP_K }, |
| { common_sampler_type_to_chr(COMMON_SAMPLER_TYPE_TYPICAL_P), COMMON_SAMPLER_TYPE_TYPICAL_P }, |
| { common_sampler_type_to_chr(COMMON_SAMPLER_TYPE_TOP_P), COMMON_SAMPLER_TYPE_TOP_P }, |
| { common_sampler_type_to_chr(COMMON_SAMPLER_TYPE_MIN_P), COMMON_SAMPLER_TYPE_MIN_P }, |
| { common_sampler_type_to_chr(COMMON_SAMPLER_TYPE_TEMPERATURE), COMMON_SAMPLER_TYPE_TEMPERATURE }, |
| { common_sampler_type_to_chr(COMMON_SAMPLER_TYPE_XTC), COMMON_SAMPLER_TYPE_XTC }, |
| { common_sampler_type_to_chr(COMMON_SAMPLER_TYPE_INFILL), COMMON_SAMPLER_TYPE_INFILL }, |
| }; |
|
|
| std::vector<common_sampler_type> samplers; |
| samplers.reserve(chars.size()); |
|
|
| for (const auto & c : chars) { |
| const auto sampler = sampler_name_map.find(c); |
| if (sampler != sampler_name_map.end()) { |
| samplers.push_back(sampler->second); |
| } |
| } |
|
|
| return samplers; |
| } |
|
|