use struct for grammar elements and add Unicode support
This commit is contained in:
parent
18e6221ade
commit
f8baad235d
6 changed files with 416 additions and 251 deletions
|
@ -7,18 +7,45 @@
|
||||||
#include <exception>
|
#include <exception>
|
||||||
|
|
||||||
namespace grammar_parser {
|
namespace grammar_parser {
|
||||||
uint16_t get_symbol_id(parse_state & state, const char * src, size_t len) {
|
// NOTE: assumes valid utf8 (but checks for overrun)
|
||||||
uint16_t next_id = static_cast<uint16_t>(state.symbol_ids.size());
|
// copied from llama.cpp
|
||||||
|
std::pair<uint32_t, const char *> decode_utf8(const char * src) {
|
||||||
|
static const int lookup[] = { 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 2, 2, 3, 4 };
|
||||||
|
uint8_t first_byte = static_cast<uint8_t>(*src);
|
||||||
|
uint8_t highbits = first_byte >> 4;
|
||||||
|
int len = lookup[highbits];
|
||||||
|
uint8_t mask = (1 << (8 - len)) - 1;
|
||||||
|
uint32_t value = first_byte & mask;
|
||||||
|
const char * end = src + len; // may overrun!
|
||||||
|
const char * pos = src + 1;
|
||||||
|
for ( ; pos < end && *pos; pos++) {
|
||||||
|
value = (value << 6) + (static_cast<uint8_t>(*pos) & 0x3F);
|
||||||
|
}
|
||||||
|
return std::make_pair(value, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
uint32_t get_symbol_id(parse_state & state, const char * src, size_t len) {
|
||||||
|
uint32_t next_id = static_cast<uint32_t>(state.symbol_ids.size());
|
||||||
auto result = state.symbol_ids.insert(std::make_pair(std::string(src, len), next_id));
|
auto result = state.symbol_ids.insert(std::make_pair(std::string(src, len), next_id));
|
||||||
return result.first->second;
|
return result.first->second;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint16_t generate_symbol_id(parse_state & state, const std::string & base_name) {
|
uint32_t generate_symbol_id(parse_state & state, const std::string & base_name) {
|
||||||
uint16_t next_id = static_cast<uint16_t>(state.symbol_ids.size());
|
uint32_t next_id = static_cast<uint32_t>(state.symbol_ids.size());
|
||||||
state.symbol_ids[base_name + '_' + std::to_string(next_id)] = next_id;
|
state.symbol_ids[base_name + '_' + std::to_string(next_id)] = next_id;
|
||||||
return next_id;
|
return next_id;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void add_rule(
|
||||||
|
parse_state & state,
|
||||||
|
uint32_t rule_id,
|
||||||
|
const std::vector<llama_grammar_element> & rule) {
|
||||||
|
if (state.rules.size() <= rule_id) {
|
||||||
|
state.rules.resize(rule_id + 1);
|
||||||
|
}
|
||||||
|
state.rules[rule_id] = rule;
|
||||||
|
}
|
||||||
|
|
||||||
bool is_word_char(char c) {
|
bool is_word_char(char c) {
|
||||||
return ('a' <= c && c <= 'z') || ('A' <= c && c <= 'Z') || c == '-' || ('0' <= c && c <= '9');
|
return ('a' <= c && c <= 'z') || ('A' <= c && c <= 'Z') || c == '-' || ('0' <= c && c <= '9');
|
||||||
}
|
}
|
||||||
|
@ -60,9 +87,10 @@ namespace grammar_parser {
|
||||||
return pos;
|
return pos;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::pair<uint16_t, const char *> parse_char(const char * src) {
|
std::pair<uint32_t, const char *> parse_char(const char * src) {
|
||||||
if (*src == '\\') {
|
if (*src == '\\') {
|
||||||
char esc = src[1];
|
char esc = src[1];
|
||||||
|
// TODO: 16- and 32-bit escapes
|
||||||
if (esc == 'x') {
|
if (esc == 'x') {
|
||||||
int first = hex_to_int(src[2]);
|
int first = hex_to_int(src[2]);
|
||||||
if (first > -1) {
|
if (first > -1) {
|
||||||
|
@ -83,7 +111,8 @@ namespace grammar_parser {
|
||||||
}
|
}
|
||||||
throw std::runtime_error(std::string("unknown escape at ") + src);
|
throw std::runtime_error(std::string("unknown escape at ") + src);
|
||||||
} else if (*src) {
|
} else if (*src) {
|
||||||
return std::make_pair(*src, src + 1);
|
auto decoded = decode_utf8(src);
|
||||||
|
return std::make_pair(decoded.first, decoded.second);
|
||||||
}
|
}
|
||||||
throw std::runtime_error("unexpected end of input");
|
throw std::runtime_error("unexpected end of input");
|
||||||
}
|
}
|
||||||
|
@ -92,132 +121,101 @@ namespace grammar_parser {
|
||||||
parse_state & state,
|
parse_state & state,
|
||||||
const char * src,
|
const char * src,
|
||||||
const std::string & rule_name,
|
const std::string & rule_name,
|
||||||
uint16_t rule_id,
|
uint32_t rule_id,
|
||||||
bool is_nested);
|
bool is_nested);
|
||||||
|
|
||||||
const char * parse_sequence(
|
const char * parse_sequence(
|
||||||
parse_state & state,
|
parse_state & state,
|
||||||
const char * src,
|
const char * src,
|
||||||
const std::string & rule_name,
|
const std::string & rule_name,
|
||||||
std::vector<uint16_t> & outbuf,
|
std::vector<llama_grammar_element> & out_elements,
|
||||||
bool is_nested) {
|
bool is_nested) {
|
||||||
size_t out_start = outbuf.size();
|
size_t last_sym_start = out_elements.size();
|
||||||
|
|
||||||
// sequence size, will be replaced at end when known
|
|
||||||
outbuf.push_back(0);
|
|
||||||
|
|
||||||
size_t last_sym_start = outbuf.size();
|
|
||||||
const char * pos = src;
|
const char * pos = src;
|
||||||
while (*pos) {
|
while (*pos) {
|
||||||
if (*pos == '"') { // literal string
|
if (*pos == '"') { // literal string
|
||||||
pos++;
|
pos++;
|
||||||
last_sym_start = outbuf.size();
|
last_sym_start = out_elements.size();
|
||||||
while (*pos != '"') {
|
while (*pos != '"') {
|
||||||
auto char_pair = parse_char(pos);
|
auto char_pair = parse_char(pos);
|
||||||
pos = char_pair.second;
|
pos = char_pair.second;
|
||||||
|
out_elements.push_back({LLAMA_GRETYPE_CHAR, char_pair.first});
|
||||||
// each char of a literal is encoded as a "range" of char - char
|
|
||||||
outbuf.push_back(2);
|
|
||||||
outbuf.push_back(char_pair.first);
|
|
||||||
outbuf.push_back(char_pair.first);
|
|
||||||
}
|
}
|
||||||
pos = parse_space(pos + 1, is_nested);
|
pos = parse_space(pos + 1, is_nested);
|
||||||
} else if (*pos == '[') { // char range(s)
|
} else if (*pos == '[') { // char range(s)
|
||||||
pos++;
|
pos++;
|
||||||
last_sym_start = outbuf.size();
|
last_sym_start = out_elements.size();
|
||||||
// num chars in range - replaced at end of loop
|
|
||||||
outbuf.push_back(0);
|
|
||||||
while (*pos != ']') {
|
while (*pos != ']') {
|
||||||
auto char_pair = parse_char(pos);
|
auto char_pair = parse_char(pos);
|
||||||
pos = char_pair.second;
|
pos = char_pair.second;
|
||||||
|
enum llama_gretype type = last_sym_start < out_elements.size()
|
||||||
|
? LLAMA_GRETYPE_CHAR_ALT
|
||||||
|
: LLAMA_GRETYPE_CHAR;
|
||||||
|
|
||||||
outbuf.push_back(char_pair.first);
|
out_elements.push_back({type, char_pair.first});
|
||||||
if (pos[0] == '-' && pos[1] != ']') {
|
if (pos[0] == '-' && pos[1] != ']') {
|
||||||
auto endchar_pair = parse_char(pos + 1);
|
auto endchar_pair = parse_char(pos + 1);
|
||||||
pos = endchar_pair.second;
|
pos = endchar_pair.second;
|
||||||
outbuf.push_back(endchar_pair.first);
|
out_elements.push_back({LLAMA_GRETYPE_CHAR_RNG_UPPER, endchar_pair.first});
|
||||||
} else {
|
|
||||||
// chars that aren't part of a c1-c2 range are just doubled (i.e., c-c)
|
|
||||||
outbuf.push_back(char_pair.first);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// replace num chars with actual
|
|
||||||
outbuf[last_sym_start] = static_cast<uint16_t>(outbuf.size() - last_sym_start - 1);
|
|
||||||
pos = parse_space(pos + 1, is_nested);
|
pos = parse_space(pos + 1, is_nested);
|
||||||
} else if (is_word_char(*pos)) { // rule reference
|
} else if (is_word_char(*pos)) { // rule reference
|
||||||
const char * name_end = parse_name(pos);
|
const char * name_end = parse_name(pos);
|
||||||
uint16_t ref_rule_id = get_symbol_id(state, pos, name_end - pos);
|
uint32_t ref_rule_id = get_symbol_id(state, pos, name_end - pos);
|
||||||
pos = parse_space(name_end, is_nested);
|
pos = parse_space(name_end, is_nested);
|
||||||
last_sym_start = outbuf.size();
|
last_sym_start = out_elements.size();
|
||||||
outbuf.push_back(1);
|
out_elements.push_back({LLAMA_GRETYPE_RULE_REF, ref_rule_id});
|
||||||
outbuf.push_back(ref_rule_id);
|
|
||||||
} else if (*pos == '(') { // grouping
|
} else if (*pos == '(') { // grouping
|
||||||
// parse nested alternates into synthesized rule
|
// parse nested alternates into synthesized rule
|
||||||
pos = parse_space(pos + 1, true);
|
pos = parse_space(pos + 1, true);
|
||||||
uint16_t sub_rule_id = generate_symbol_id(state, rule_name);
|
uint32_t sub_rule_id = generate_symbol_id(state, rule_name);
|
||||||
pos = parse_alternates(state, pos, rule_name, sub_rule_id, true);
|
pos = parse_alternates(state, pos, rule_name, sub_rule_id, true);
|
||||||
last_sym_start = outbuf.size();
|
last_sym_start = out_elements.size();
|
||||||
// output reference to synthesized rule
|
// output reference to synthesized rule
|
||||||
outbuf.push_back(1);
|
out_elements.push_back({LLAMA_GRETYPE_RULE_REF, sub_rule_id});
|
||||||
outbuf.push_back(sub_rule_id);
|
|
||||||
if (*pos != ')') {
|
if (*pos != ')') {
|
||||||
throw std::runtime_error(std::string("expecting ')' at ") + pos);
|
throw std::runtime_error(std::string("expecting ')' at ") + pos);
|
||||||
}
|
}
|
||||||
pos = parse_space(pos + 1, is_nested);
|
pos = parse_space(pos + 1, is_nested);
|
||||||
} else if (*pos == '*' || *pos == '+' || *pos == '?') { // repetition operator
|
} else if (*pos == '*' || *pos == '+' || *pos == '?') { // repetition operator
|
||||||
if (outbuf.size() - out_start - 1 == 0) {
|
if (last_sym_start == out_elements.size()) {
|
||||||
throw std::runtime_error(std::string("expecting preceeding item to */+/? at ") + pos);
|
throw std::runtime_error(std::string("expecting preceeding item to */+/? at ") + pos);
|
||||||
}
|
}
|
||||||
std::vector<uint16_t> & out_grammar = state.out_grammar;
|
|
||||||
|
|
||||||
// apply transformation to previous symbol (last_sym_start -
|
// apply transformation to previous symbol (last_sym_start to end) according to
|
||||||
// end) according to rewrite rules:
|
// rewrite rules:
|
||||||
// S* --> S' ::= S S' |
|
// S* --> S' ::= S S' |
|
||||||
// S+ --> S' ::= S S' | S
|
// S+ --> S' ::= S S' | S
|
||||||
// S? --> S' ::= S |
|
// S? --> S' ::= S |
|
||||||
uint16_t sub_rule_id = generate_symbol_id(state, rule_name);
|
uint32_t sub_rule_id = generate_symbol_id(state, rule_name);
|
||||||
out_grammar.push_back(sub_rule_id);
|
std::vector<llama_grammar_element> sub_rule;
|
||||||
size_t sub_rule_start = out_grammar.size();
|
|
||||||
// placeholder for size of 1st alternate
|
|
||||||
out_grammar.push_back(0);
|
|
||||||
// add preceding symbol to generated rule
|
// add preceding symbol to generated rule
|
||||||
out_grammar.insert(out_grammar.end(), outbuf.begin() + last_sym_start, outbuf.end());
|
sub_rule.insert(
|
||||||
|
sub_rule.end(), out_elements.begin() + last_sym_start, out_elements.end());
|
||||||
if (*pos == '*' || *pos == '+') {
|
if (*pos == '*' || *pos == '+') {
|
||||||
// cause generated rule to recurse
|
// cause generated rule to recurse
|
||||||
out_grammar.push_back(1);
|
sub_rule.push_back({LLAMA_GRETYPE_RULE_REF, sub_rule_id});
|
||||||
out_grammar.push_back(sub_rule_id);
|
|
||||||
}
|
}
|
||||||
// apply actual size
|
// mark start of alternate def
|
||||||
out_grammar[sub_rule_start] = out_grammar.size() - sub_rule_start;
|
sub_rule.push_back({LLAMA_GRETYPE_ALT, 0});
|
||||||
// mark end of 1st alternate
|
|
||||||
out_grammar.push_back(0);
|
|
||||||
sub_rule_start = out_grammar.size();
|
|
||||||
// placeholder for size of 2nd alternate
|
|
||||||
out_grammar.push_back(0);
|
|
||||||
if (*pos == '+') {
|
if (*pos == '+') {
|
||||||
// add preceding symbol as alternate only for '+'
|
// add preceding symbol as alternate only for '+' (otherwise empty)
|
||||||
out_grammar.insert(out_grammar.end(), outbuf.begin() + last_sym_start, outbuf.end());
|
sub_rule.insert(
|
||||||
|
sub_rule.end(), out_elements.begin() + last_sym_start, out_elements.end());
|
||||||
}
|
}
|
||||||
// apply actual size of 2nd alternate
|
sub_rule.push_back({LLAMA_GRETYPE_END, 0});
|
||||||
out_grammar[sub_rule_start] = out_grammar.size() - sub_rule_start;
|
add_rule(state, sub_rule_id, sub_rule);
|
||||||
// mark end of 2nd alternate, then end of rule
|
|
||||||
out_grammar.push_back(0);
|
|
||||||
out_grammar.push_back(0);
|
|
||||||
|
|
||||||
// in original rule, replace previous symbol with reference to generated rule
|
// in original rule, replace previous symbol with reference to generated rule
|
||||||
outbuf.resize(last_sym_start);
|
out_elements.resize(last_sym_start);
|
||||||
outbuf.push_back(1);
|
out_elements.push_back({LLAMA_GRETYPE_RULE_REF, sub_rule_id});
|
||||||
outbuf.push_back(sub_rule_id);
|
|
||||||
|
|
||||||
pos = parse_space(pos + 1, is_nested);
|
pos = parse_space(pos + 1, is_nested);
|
||||||
} else {
|
} else {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// apply actual size of this alternate sequence
|
|
||||||
outbuf[out_start] = static_cast<uint16_t>(outbuf.size() - out_start);
|
|
||||||
// mark end of alternate
|
|
||||||
outbuf.push_back(0);
|
|
||||||
return pos;
|
return pos;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -225,17 +223,17 @@ namespace grammar_parser {
|
||||||
parse_state & state,
|
parse_state & state,
|
||||||
const char * src,
|
const char * src,
|
||||||
const std::string & rule_name,
|
const std::string & rule_name,
|
||||||
uint16_t rule_id,
|
uint32_t rule_id,
|
||||||
bool is_nested) {
|
bool is_nested) {
|
||||||
std::vector<uint16_t> outbuf;
|
std::vector<llama_grammar_element> rule;
|
||||||
const char * pos = parse_sequence(state, src, rule_name, outbuf, is_nested);
|
const char * pos = parse_sequence(state, src, rule_name, rule, is_nested);
|
||||||
while (*pos == '|') {
|
while (*pos == '|') {
|
||||||
|
rule.push_back({LLAMA_GRETYPE_ALT, 0});
|
||||||
pos = parse_space(pos + 1, true);
|
pos = parse_space(pos + 1, true);
|
||||||
pos = parse_sequence(state, pos, rule_name, outbuf, is_nested);
|
pos = parse_sequence(state, pos, rule_name, rule, is_nested);
|
||||||
}
|
}
|
||||||
state.out_grammar.push_back(rule_id);
|
rule.push_back({LLAMA_GRETYPE_END, 0});
|
||||||
state.out_grammar.insert(state.out_grammar.end(), outbuf.begin(), outbuf.end());
|
add_rule(state, rule_id, rule);
|
||||||
state.out_grammar.push_back(0);
|
|
||||||
return pos;
|
return pos;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -243,7 +241,7 @@ namespace grammar_parser {
|
||||||
const char * name_end = parse_name(src);
|
const char * name_end = parse_name(src);
|
||||||
const char * pos = parse_space(name_end, false);
|
const char * pos = parse_space(name_end, false);
|
||||||
size_t name_len = name_end - src;
|
size_t name_len = name_end - src;
|
||||||
uint16_t rule_id = get_symbol_id(state, src, name_len);
|
uint32_t rule_id = get_symbol_id(state, src, name_len);
|
||||||
const std::string name(src, name_len);
|
const std::string name(src, name_len);
|
||||||
|
|
||||||
if (!(pos[0] == ':' && pos[1] == ':' && pos[2] == '=')) {
|
if (!(pos[0] == ':' && pos[1] == ':' && pos[2] == '=')) {
|
||||||
|
@ -270,7 +268,6 @@ namespace grammar_parser {
|
||||||
while (*pos) {
|
while (*pos) {
|
||||||
pos = parse_rule(state, pos);
|
pos = parse_rule(state, pos);
|
||||||
}
|
}
|
||||||
state.out_grammar.push_back(0xffff);
|
|
||||||
return state;
|
return state;
|
||||||
} catch (const std::exception & err) {
|
} catch (const std::exception & err) {
|
||||||
fprintf(stderr, "%s: error parsing grammar: %s\n", __func__, err.what());
|
fprintf(stderr, "%s: error parsing grammar: %s\n", __func__, err.what());
|
||||||
|
@ -278,53 +275,131 @@ namespace grammar_parser {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const uint16_t * print_rule(
|
void print_grammar_char(FILE * file, uint32_t c) {
|
||||||
FILE * file,
|
if (0x20 <= c && c <= 0x7f) {
|
||||||
const uint16_t * base,
|
fprintf(file, "%c", static_cast<char>(c));
|
||||||
const uint16_t * src,
|
} else {
|
||||||
const std::map<uint16_t, std::string> & symbol_id_names) {
|
// cop out of encoding UTF-8
|
||||||
uint16_t rule_id = *src;
|
fprintf(file, "<U+%04X>", c);
|
||||||
fprintf(file, "<%zu>%s ::= ", src - base, symbol_id_names.at(rule_id).c_str());
|
}
|
||||||
const uint16_t * pos = src + 1;
|
}
|
||||||
while (*pos) {
|
|
||||||
if (pos - 1 > src) {
|
|
||||||
fprintf(file, "| ");
|
|
||||||
}
|
|
||||||
pos++; // sequence size, not needed here
|
|
||||||
while (*pos) {
|
|
||||||
if (*pos == 1) {
|
|
||||||
uint16_t ref_rule_id = pos[1];
|
|
||||||
fprintf(file, "<%zu>%s ", pos - base, symbol_id_names.at(ref_rule_id).c_str());
|
|
||||||
pos += 2;
|
|
||||||
} else {
|
|
||||||
fprintf(file, "<%zu>[", pos - base);
|
|
||||||
uint16_t num_chars = *pos;
|
|
||||||
pos++;
|
|
||||||
|
|
||||||
for (uint16_t i = 0; i < num_chars; i += 2) {
|
bool is_char_element(llama_grammar_element elem) {
|
||||||
fprintf(file, "%lc-", static_cast<wint_t>(pos[i])); // REVIEW
|
switch (elem.type) {
|
||||||
if (i + 1 < num_chars) {
|
case LLAMA_GRETYPE_CHAR: return true;
|
||||||
fprintf(file, "%lc", static_cast<wint_t>(pos[i + 1]));
|
case LLAMA_GRETYPE_CHAR_ALT: return true;
|
||||||
}
|
case LLAMA_GRETYPE_CHAR_RNG_UPPER: return true;
|
||||||
}
|
default: return false;
|
||||||
fprintf(file, "] ");
|
}
|
||||||
pos += num_chars;
|
}
|
||||||
}
|
|
||||||
|
void print_rule_binary(FILE * file, const std::vector<llama_grammar_element> & rule) {
|
||||||
|
for (auto elem : rule) {
|
||||||
|
switch (elem.type) {
|
||||||
|
case LLAMA_GRETYPE_END: fprintf(file, "END"); break;
|
||||||
|
case LLAMA_GRETYPE_ALT: fprintf(file, "ALT"); break;
|
||||||
|
case LLAMA_GRETYPE_RULE_REF: fprintf(file, "RULE_REF"); break;
|
||||||
|
case LLAMA_GRETYPE_CHAR: fprintf(file, "CHAR"); break;
|
||||||
|
case LLAMA_GRETYPE_CHAR_RNG_UPPER: fprintf(file, "CHAR_RNG_UPPER"); break;
|
||||||
|
case LLAMA_GRETYPE_CHAR_ALT: fprintf(file, "CHAR_RNG_UPPER"); break;
|
||||||
|
}
|
||||||
|
switch (elem.type) {
|
||||||
|
case LLAMA_GRETYPE_END:
|
||||||
|
case LLAMA_GRETYPE_ALT:
|
||||||
|
case LLAMA_GRETYPE_RULE_REF:
|
||||||
|
fprintf(file, "(%u) ", elem.value);
|
||||||
|
break;
|
||||||
|
case LLAMA_GRETYPE_CHAR:
|
||||||
|
case LLAMA_GRETYPE_CHAR_RNG_UPPER:
|
||||||
|
case LLAMA_GRETYPE_CHAR_ALT:
|
||||||
|
fprintf(file, "(\"");
|
||||||
|
print_grammar_char(file, elem.value);
|
||||||
|
fprintf(file, "\") ");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fprintf(file, "\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
void print_rule(
|
||||||
|
FILE * file,
|
||||||
|
uint32_t rule_id,
|
||||||
|
const std::vector<llama_grammar_element> & rule,
|
||||||
|
const std::map<uint32_t, std::string> & symbol_id_names) {
|
||||||
|
if (rule.empty() || rule.back().type != LLAMA_GRETYPE_END) {
|
||||||
|
throw std::runtime_error(
|
||||||
|
"malformed rule, does not end with LLAMA_GRETYPE_END: " + std::to_string(rule_id));
|
||||||
|
}
|
||||||
|
fprintf(file, "%s ::= ", symbol_id_names.at(rule_id).c_str());
|
||||||
|
for (size_t i = 0, end = rule.size() - 1; i < end; i++) {
|
||||||
|
llama_grammar_element elem = rule[i];
|
||||||
|
switch (elem.type) {
|
||||||
|
case LLAMA_GRETYPE_END:
|
||||||
|
throw std::runtime_error(
|
||||||
|
"unexpected end of rule: " + std::to_string(rule_id) + "," +
|
||||||
|
std::to_string(i));
|
||||||
|
case LLAMA_GRETYPE_ALT:
|
||||||
|
fprintf(file, "| ");
|
||||||
|
break;
|
||||||
|
case LLAMA_GRETYPE_RULE_REF:
|
||||||
|
fprintf(file, "%s ", symbol_id_names.at(elem.value).c_str());
|
||||||
|
break;
|
||||||
|
case LLAMA_GRETYPE_CHAR:
|
||||||
|
fprintf(file, "[");
|
||||||
|
print_grammar_char(file, elem.value);
|
||||||
|
break;
|
||||||
|
case LLAMA_GRETYPE_CHAR_RNG_UPPER:
|
||||||
|
if (i == 0 || !is_char_element(rule[i - 1])) {
|
||||||
|
throw std::runtime_error(
|
||||||
|
"LLAMA_GRETYPE_CHAR_RNG_UPPER without preceding char: " +
|
||||||
|
std::to_string(rule_id) + "," + std::to_string(i));
|
||||||
|
}
|
||||||
|
fprintf(file, "-");
|
||||||
|
print_grammar_char(file, elem.value);
|
||||||
|
break;
|
||||||
|
case LLAMA_GRETYPE_CHAR_ALT:
|
||||||
|
if (i == 0 || !is_char_element(rule[i - 1])) {
|
||||||
|
throw std::runtime_error(
|
||||||
|
"LLAMA_GRETYPE_CHAR_ALT without preceding char: " +
|
||||||
|
std::to_string(rule_id) + "," + std::to_string(i));
|
||||||
|
}
|
||||||
|
print_grammar_char(file, elem.value);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (is_char_element(elem)) {
|
||||||
|
switch (rule[i + 1].type) {
|
||||||
|
case LLAMA_GRETYPE_CHAR_ALT:
|
||||||
|
case LLAMA_GRETYPE_CHAR_RNG_UPPER:
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
fprintf(file, "] ");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
pos++;
|
|
||||||
}
|
}
|
||||||
fprintf(file, "\n");
|
fprintf(file, "\n");
|
||||||
return pos + 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void print_grammar(FILE * file, const parse_state & state) {
|
void print_grammar(FILE * file, const parse_state & state) {
|
||||||
std::map<uint16_t, std::string> symbol_id_names;
|
try {
|
||||||
for (auto kv : state.symbol_ids) {
|
std::map<uint32_t, std::string> symbol_id_names;
|
||||||
symbol_id_names[kv.second] = kv.first;
|
for (auto kv : state.symbol_ids) {
|
||||||
}
|
symbol_id_names[kv.second] = kv.first;
|
||||||
const uint16_t * pos = state.out_grammar.data();
|
}
|
||||||
while (*pos != 0xffff) {
|
for (size_t i = 0, end = state.rules.size(); i < end; i++) {
|
||||||
pos = print_rule(file, state.out_grammar.data(), pos, symbol_id_names);
|
// fprintf(file, "%zu: ", i);
|
||||||
|
// print_rule_binary(file, state.rules[i]);
|
||||||
|
print_rule(file, i, state.rules[i], symbol_id_names);
|
||||||
|
}
|
||||||
|
} catch (const std::exception & err) {
|
||||||
|
fprintf(stderr, "\n%s: error printing grammar: %s\n", __func__, err.what());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::vector<const llama_grammar_element *> parse_state::c_rules() {
|
||||||
|
std::vector<const llama_grammar_element *> ret;
|
||||||
|
for (const auto & rule : rules) {
|
||||||
|
ret.push_back(rule.data());
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,6 +10,7 @@
|
||||||
// space ::= [ \t\n]*
|
// space ::= [ \t\n]*
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
#include "llama.h"
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <map>
|
#include <map>
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
|
@ -17,8 +18,10 @@
|
||||||
|
|
||||||
namespace grammar_parser {
|
namespace grammar_parser {
|
||||||
struct parse_state {
|
struct parse_state {
|
||||||
std::map<std::string, uint16_t> symbol_ids;
|
std::map<std::string, uint32_t> symbol_ids;
|
||||||
std::vector<uint16_t> out_grammar;
|
std::vector<std::vector<llama_grammar_element>> rules;
|
||||||
|
|
||||||
|
std::vector<const llama_grammar_element *> c_rules();
|
||||||
};
|
};
|
||||||
|
|
||||||
parse_state parse(const char * src);
|
parse_state parse(const char * src);
|
||||||
|
|
|
@ -300,14 +300,16 @@ int main(int argc, char ** argv) {
|
||||||
if (!params.grammar.empty()) {
|
if (!params.grammar.empty()) {
|
||||||
parsed_grammar = grammar_parser::parse(params.grammar.c_str());
|
parsed_grammar = grammar_parser::parse(params.grammar.c_str());
|
||||||
// will be empty (default) if there are parse errors
|
// will be empty (default) if there are parse errors
|
||||||
if (parsed_grammar.out_grammar.empty()) {
|
if (parsed_grammar.rules.empty()) {
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
fprintf(stderr, "%s: grammar:\n", __func__);
|
fprintf(stderr, "%s: grammar:\n", __func__);
|
||||||
grammar_parser::print_grammar(stderr, parsed_grammar);
|
grammar_parser::print_grammar(stderr, parsed_grammar);
|
||||||
fprintf(stderr, "\n");
|
fprintf(stderr, "\n");
|
||||||
|
|
||||||
|
std::vector<const llama_grammar_element *> grammar_rules(parsed_grammar.c_rules());
|
||||||
grammar = llama_grammar_init(
|
grammar = llama_grammar_init(
|
||||||
parsed_grammar.out_grammar.data(), parsed_grammar.symbol_ids.at("root"));
|
grammar_rules.data(), grammar_rules.size(), parsed_grammar.symbol_ids.at("root"));
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: replace with ring-buffer
|
// TODO: replace with ring-buffer
|
||||||
|
@ -653,8 +655,12 @@ int main(int argc, char ** argv) {
|
||||||
// reset grammar state if we're restarting generation
|
// reset grammar state if we're restarting generation
|
||||||
if (grammar != NULL) {
|
if (grammar != NULL) {
|
||||||
llama_grammar_free(grammar);
|
llama_grammar_free(grammar);
|
||||||
|
|
||||||
|
std::vector<const llama_grammar_element *> grammar_rules(
|
||||||
|
parsed_grammar.c_rules());
|
||||||
grammar = llama_grammar_init(
|
grammar = llama_grammar_init(
|
||||||
parsed_grammar.out_grammar.data(), parsed_grammar.symbol_ids.at("root"));
|
grammar_rules.data(), grammar_rules.size(),
|
||||||
|
parsed_grammar.symbol_ids.at("root"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
is_interacting = false;
|
is_interacting = false;
|
||||||
|
|
7
grammars/japanese.gbnf
Normal file
7
grammars/japanese.gbnf
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
# A probably incorrect grammar for Japanese
|
||||||
|
root ::= jp-char+ ([ \t\n] jp-char+)*
|
||||||
|
jp-char ::= hiragana | katakana | punctuation | cjk
|
||||||
|
hiragana ::= [ぁ-ゟ]
|
||||||
|
katakana ::= [ァ-ヿ]
|
||||||
|
punctuation ::= [、-〾]
|
||||||
|
cjk ::= [一-鿿]
|
260
llama.cpp
260
llama.cpp
|
@ -1841,45 +1841,86 @@ static std::vector<llama_vocab::id> llama_tokenize(const llama_vocab & vocab, co
|
||||||
//
|
//
|
||||||
|
|
||||||
struct llama_grammar {
|
struct llama_grammar {
|
||||||
const std::vector<const uint16_t *> rules;
|
const std::vector<std::vector<llama_grammar_element>> rules;
|
||||||
std::vector<std::vector<const uint16_t *>> stacks;
|
std::vector<std::vector<const llama_grammar_element *>> stacks;
|
||||||
};
|
};
|
||||||
|
|
||||||
// transforms a grammar pushdown stack into N possible stacks, all terminating
|
// NOTE: assumes valid utf8 (but checks for overrun)
|
||||||
|
std::pair<uint32_t, const char *> decode_utf8(const char * src) {
|
||||||
|
static const int lookup[] = { 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 2, 2, 3, 4 };
|
||||||
|
uint8_t first_byte = static_cast<uint8_t>(*src);
|
||||||
|
uint8_t highbits = first_byte >> 4;
|
||||||
|
int len = lookup[highbits];
|
||||||
|
uint8_t mask = (1 << (8 - len)) - 1;
|
||||||
|
uint32_t value = first_byte & mask;
|
||||||
|
const char * end = src + len; // may overrun!
|
||||||
|
const char * pos = src + 1; // may overrun!
|
||||||
|
for ( ; pos < end && *pos; pos++) {
|
||||||
|
value = (value << 6) + (static_cast<uint8_t>(*pos) & 0x3F);
|
||||||
|
}
|
||||||
|
return std::make_pair(value, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
// returns true iff pos points to the end of one of the definitions of a rule
|
||||||
|
static bool llama_grammar_is_end_of_sequence(const llama_grammar_element * pos) {
|
||||||
|
switch (pos->type) {
|
||||||
|
case LLAMA_GRETYPE_END: return true;
|
||||||
|
case LLAMA_GRETYPE_ALT: return true;
|
||||||
|
default: return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// transforms a grammar pushdown stack into N possible stacks, all ending
|
||||||
// at a character range (terminal element)
|
// at a character range (terminal element)
|
||||||
static void llama_grammar_advance_stack(
|
static void llama_grammar_advance_stack(
|
||||||
const std::vector<const uint16_t *> & rules,
|
const std::vector<std::vector<llama_grammar_element>> & rules,
|
||||||
const std::vector<const uint16_t *> & stack,
|
const std::vector<const llama_grammar_element *> & stack,
|
||||||
std::vector<std::vector<const uint16_t *>> & new_stacks) {
|
std::vector<std::vector<const llama_grammar_element *>> & new_stacks) {
|
||||||
|
|
||||||
if (stack.empty()) {
|
if (stack.empty()) {
|
||||||
new_stacks.push_back(stack);
|
new_stacks.push_back(stack);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const uint16_t * pos = stack.back();
|
const llama_grammar_element * pos = stack.back();
|
||||||
|
|
||||||
if (*pos == 1) {
|
switch (pos->type) {
|
||||||
// rule reference, apply rule to stack
|
case LLAMA_GRETYPE_RULE_REF: {
|
||||||
const uint16_t * subpos = rules[pos[1]] + 1;
|
const size_t rule_id = static_cast<size_t>(pos->value);
|
||||||
while (*subpos) {
|
const llama_grammar_element * subpos = rules[rule_id].data();
|
||||||
// init new stack without the top (pos)
|
do {
|
||||||
std::vector<const uint16_t *> new_stack(stack.begin(), stack.end() - 1);
|
// init new stack without the top (pos)
|
||||||
if (pos[2]) {
|
std::vector<const llama_grammar_element *> new_stack(stack.begin(), stack.end() - 1);
|
||||||
// if the rule ref is followed by another element, add that to stack
|
if (!llama_grammar_is_end_of_sequence(pos + 1)) {
|
||||||
new_stack.push_back(pos + 2);
|
// if this rule ref is followed by another element, add that to stack
|
||||||
}
|
new_stack.push_back(pos + 1);
|
||||||
if (subpos[1]) {
|
}
|
||||||
// if this alternate is nonempty, add that to the stack
|
if (!llama_grammar_is_end_of_sequence(subpos)) {
|
||||||
new_stack.push_back(subpos + 1);
|
// if alternate is nonempty, add to stack
|
||||||
}
|
new_stack.push_back(subpos);
|
||||||
llama_grammar_advance_stack(rules, new_stack, new_stacks);
|
}
|
||||||
subpos += 1 + *subpos;
|
llama_grammar_advance_stack(rules, new_stack, new_stacks);
|
||||||
|
while (!llama_grammar_is_end_of_sequence(subpos)) {
|
||||||
|
// scan to end of alternate def
|
||||||
|
subpos++;
|
||||||
|
}
|
||||||
|
if (subpos->type == LLAMA_GRETYPE_ALT) {
|
||||||
|
// there's another alternate def of this rule to process
|
||||||
|
subpos++;
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} while (true);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
} else {
|
case LLAMA_GRETYPE_CHAR:
|
||||||
// rule element size > 1 -> character reference
|
new_stacks.push_back(stack);
|
||||||
LLAMA_ASSERT(*pos);
|
break;
|
||||||
new_stacks.push_back(stack);
|
default:
|
||||||
|
// end of alternate (LLAMA_GRETYPE_END, LLAMA_GRETYPE_ALT) or middle of char range
|
||||||
|
// (LLAMA_GRETYPE_CHAR_ALT, LLAMA_GRETYPE_CHAR_RNG_UPPER); stack should never be left on
|
||||||
|
// those
|
||||||
|
LLAMA_ASSERT(false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1887,39 +1928,43 @@ static void llama_grammar_advance_stack(
|
||||||
// be positioned at a character range (see `llama_grammar_advance_stack`), and
|
// be positioned at a character range (see `llama_grammar_advance_stack`), and
|
||||||
// produces the N possible stacks if the given char is accepted at those
|
// produces the N possible stacks if the given char is accepted at those
|
||||||
// positions
|
// positions
|
||||||
static std::vector<std::vector<const uint16_t *>> llama_grammar_accept(
|
static std::vector<std::vector<const llama_grammar_element *>> llama_grammar_accept(
|
||||||
const std::vector<const uint16_t *> & rules,
|
const std::vector<std::vector<llama_grammar_element>> & rules,
|
||||||
const std::vector<std::vector<const uint16_t *>> & stacks,
|
const std::vector<std::vector<const llama_grammar_element *>> & stacks,
|
||||||
const uint16_t chr) {
|
const uint32_t chr) {
|
||||||
|
|
||||||
std::vector<std::vector<const uint16_t *>> new_stacks;
|
std::vector<std::vector<const llama_grammar_element *>> new_stacks;
|
||||||
|
|
||||||
for (const auto & stack : stacks) {
|
for (const auto & stack : stacks) {
|
||||||
if (stack.empty()) {
|
if (stack.empty()) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
const uint16_t * pos = stack.back();
|
const llama_grammar_element * pos = stack.back();
|
||||||
const uint16_t num_chars = *pos;
|
LLAMA_ASSERT(pos->type == LLAMA_GRETYPE_CHAR);
|
||||||
LLAMA_ASSERT(num_chars > 1);
|
|
||||||
|
|
||||||
pos++; // skip num chars indicator
|
|
||||||
bool found = false;
|
bool found = false;
|
||||||
// loop over the inclusive char pairs to find a match on the given char
|
do {
|
||||||
for (int i = 0; i < num_chars; i += 2) {
|
bool matches_range;
|
||||||
if (pos[i] <= chr && (i + 1 == num_chars || chr <= pos[i + 1])) {
|
if (pos[1].type == LLAMA_GRETYPE_CHAR_RNG_UPPER) {
|
||||||
found = true;
|
// inclusive range, e.g. [a-z]
|
||||||
break;
|
matches_range = pos->value <= chr && chr <= pos[1].value;
|
||||||
|
pos += 2;
|
||||||
|
} else {
|
||||||
|
// exact char match, e.g. [a] or "a"
|
||||||
|
matches_range = pos->value == chr;
|
||||||
|
pos += 1;
|
||||||
}
|
}
|
||||||
}
|
found = found || matches_range;
|
||||||
|
} while (pos->type == LLAMA_GRETYPE_CHAR_ALT);
|
||||||
|
|
||||||
if (!found) {
|
if (!found) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
// advance past char range, updating top of stack to next element, if any
|
// update top of stack to next element, if any
|
||||||
pos += num_chars;
|
std::vector<const llama_grammar_element *> new_stack(stack.begin(), stack.end() - 1);
|
||||||
std::vector<const uint16_t *> new_stack(stack.begin(), stack.end() - 1);
|
if (!llama_grammar_is_end_of_sequence(pos)) {
|
||||||
if (*pos) {
|
|
||||||
new_stack.push_back(pos);
|
new_stack.push_back(pos);
|
||||||
}
|
}
|
||||||
llama_grammar_advance_stack(rules, new_stack, new_stacks);
|
llama_grammar_advance_stack(rules, new_stack, new_stacks);
|
||||||
|
@ -1930,8 +1975,8 @@ static std::vector<std::vector<const uint16_t *>> llama_grammar_accept(
|
||||||
|
|
||||||
// returns `true` if one of the pushdown stacks can accept the given char.
|
// returns `true` if one of the pushdown stacks can accept the given char.
|
||||||
static bool llama_grammar_peek(
|
static bool llama_grammar_peek(
|
||||||
const std::vector<std::vector<const uint16_t *>> & stacks,
|
const std::vector<std::vector<const llama_grammar_element *>> & stacks,
|
||||||
const uint16_t chr) {
|
const uint32_t chr) {
|
||||||
|
|
||||||
for (const auto & stack : stacks) {
|
for (const auto & stack : stacks) {
|
||||||
if (stack.empty()) {
|
if (stack.empty()) {
|
||||||
|
@ -1939,16 +1984,24 @@ static bool llama_grammar_peek(
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
const uint16_t * pos = stack.back();
|
const llama_grammar_element * pos = stack.back();
|
||||||
const uint16_t num_chars = *pos;
|
LLAMA_ASSERT(pos->type == LLAMA_GRETYPE_CHAR);
|
||||||
LLAMA_ASSERT(num_chars > 1);
|
|
||||||
|
|
||||||
pos++;
|
do {
|
||||||
for (int i = 0; i < num_chars; i += 2) {
|
if (pos[1].type == LLAMA_GRETYPE_CHAR_RNG_UPPER) {
|
||||||
if (pos[i] <= chr && (i + 1 == num_chars || chr <= pos[i + 1])) {
|
// inclusive range, e.g. [a-z]
|
||||||
return true;
|
if (pos->value <= chr && chr <= pos[1].value) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
pos += 2;
|
||||||
|
} else {
|
||||||
|
// exact char match, e.g. [a] or "a"
|
||||||
|
if (pos->value == chr) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
pos += 1;
|
||||||
}
|
}
|
||||||
}
|
} while (pos->type == LLAMA_GRETYPE_CHAR_ALT);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
|
@ -1959,45 +2012,44 @@ static bool llama_grammar_peek(
|
||||||
// grammar - external
|
// grammar - external
|
||||||
//
|
//
|
||||||
|
|
||||||
struct llama_grammar * llama_grammar_init(const uint16_t * src, uint16_t start_rule_id) {
|
struct llama_grammar * llama_grammar_init(
|
||||||
const uint16_t * pos = src;
|
const llama_grammar_element ** rules,
|
||||||
std::vector<const uint16_t *> rules;
|
size_t n_rules,
|
||||||
|
size_t start_rule_index) {
|
||||||
|
const llama_grammar_element * pos;
|
||||||
|
|
||||||
// build `rules` as list of pointers to rules embedded in binary grammar `src`
|
// copy rule definitions into vectors
|
||||||
while (*pos != 0xffff) {
|
std::vector<std::vector<llama_grammar_element>> vec_rules(n_rules);
|
||||||
uint16_t rule_id = *pos;
|
for (size_t i = 0; i < n_rules; i++) {
|
||||||
if (rules.size() <= rule_id) {
|
for (pos = rules[i]; pos->type != LLAMA_GRETYPE_END; pos++) {
|
||||||
rules.resize(rule_id + 1);
|
vec_rules[i].push_back(*pos);
|
||||||
}
|
}
|
||||||
rules[rule_id] = pos;
|
vec_rules[i].push_back({LLAMA_GRETYPE_END, 0});
|
||||||
// skip rule id
|
|
||||||
pos++;
|
|
||||||
// skip rule alternates
|
|
||||||
while (*pos) {
|
|
||||||
pos += 1 + *pos;
|
|
||||||
}
|
|
||||||
// skip 0 denoting end of rule
|
|
||||||
pos++;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const uint16_t * start_rule = rules[start_rule_id];
|
|
||||||
|
|
||||||
LLAMA_ASSERT(*start_rule == start_rule_id);
|
|
||||||
|
|
||||||
// loop over alternates of start rule to build initial stacks
|
// loop over alternates of start rule to build initial stacks
|
||||||
pos = start_rule + 1;
|
std::vector<std::vector<const llama_grammar_element *>> stacks;
|
||||||
std::vector<std::vector<const uint16_t *>> stacks;
|
pos = rules[start_rule_index];
|
||||||
while (*pos) {
|
do {
|
||||||
std::vector<const uint16_t *> stack;
|
std::vector<const llama_grammar_element *> stack;
|
||||||
if (pos[1]) {
|
if (!llama_grammar_is_end_of_sequence(pos)) {
|
||||||
// if alernate is nonempty, add to stack
|
// if alternate is nonempty, add to stack
|
||||||
stack.push_back(pos + 1);
|
stack.push_back(pos);
|
||||||
}
|
}
|
||||||
llama_grammar_advance_stack(rules, stack, stacks);
|
llama_grammar_advance_stack(vec_rules, stack, stacks);
|
||||||
pos += 1 + *pos;
|
while (!llama_grammar_is_end_of_sequence(pos)) {
|
||||||
}
|
// scan to end of alternate def
|
||||||
|
pos++;
|
||||||
|
}
|
||||||
|
if (pos->type == LLAMA_GRETYPE_ALT) {
|
||||||
|
// there's another alternate def of this rule to process
|
||||||
|
pos++;
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} while (true);
|
||||||
|
|
||||||
return new llama_grammar{ rules, stacks };
|
return new llama_grammar{ std::move(vec_rules), std::move(stacks) };
|
||||||
}
|
}
|
||||||
|
|
||||||
void llama_grammar_free(struct llama_grammar * grammar) {
|
void llama_grammar_free(struct llama_grammar * grammar) {
|
||||||
|
@ -2285,7 +2337,7 @@ void llama_sample_grammar(struct llama_context * ctx, llama_token_data_array * c
|
||||||
const int64_t t_start_sample_us = ggml_time_us();
|
const int64_t t_start_sample_us = ggml_time_us();
|
||||||
const llama_token eos = llama_token_eos();
|
const llama_token eos = llama_token_eos();
|
||||||
// since many llama tokens are prefixed with a single space, special case a lookahead on ' '
|
// since many llama tokens are prefixed with a single space, special case a lookahead on ' '
|
||||||
const auto stacks_after_space = llama_grammar_accept(grammar->rules, grammar->stacks, ' ');
|
const auto stacks_after_space = llama_grammar_accept(grammar->rules, grammar->stacks, U' ');
|
||||||
|
|
||||||
for (size_t i = 0; i < candidates->size; ++i) {
|
for (size_t i = 0; i < candidates->size; ++i) {
|
||||||
const llama_token id = candidates->data[i].id;
|
const llama_token id = candidates->data[i].id;
|
||||||
|
@ -2296,10 +2348,15 @@ void llama_sample_grammar(struct llama_context * ctx, llama_token_data_array * c
|
||||||
bool valid = false;
|
bool valid = false;
|
||||||
if (id == eos) {
|
if (id == eos) {
|
||||||
valid = llama_grammar_peek(grammar->stacks, 0);
|
valid = llama_grammar_peek(grammar->stacks, 0);
|
||||||
} else if (str[0] == ' ') {
|
} else {
|
||||||
valid = llama_grammar_peek(stacks_after_space, str[1]);
|
const auto decoded = decode_utf8(str);
|
||||||
} else if (str[0] != 0) {
|
const uint32_t chr = decoded.first;
|
||||||
valid = llama_grammar_peek(grammar->stacks, str[0]);
|
if (chr == U' ') {
|
||||||
|
const char * next = decoded.second;
|
||||||
|
valid = llama_grammar_peek(stacks_after_space, decode_utf8(next).first);
|
||||||
|
} else if (chr != 0) {
|
||||||
|
valid = llama_grammar_peek(grammar->stacks, chr);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!valid) {
|
if (!valid) {
|
||||||
|
@ -2451,13 +2508,15 @@ llama_token llama_grammar_accept_token(struct llama_context * ctx, struct llama_
|
||||||
const char * suffix = str;
|
const char * suffix = str;
|
||||||
|
|
||||||
// Find prefix of selected token that matches grammar, expecting at least 1 char
|
// Find prefix of selected token that matches grammar, expecting at least 1 char
|
||||||
auto new_stacks = llama_grammar_accept(grammar->rules, grammar->stacks, *suffix);
|
auto decoded = decode_utf8(suffix);
|
||||||
|
auto new_stacks = llama_grammar_accept(grammar->rules, grammar->stacks, decoded.first);
|
||||||
LLAMA_ASSERT(!new_stacks.empty());
|
LLAMA_ASSERT(!new_stacks.empty());
|
||||||
if (*suffix) {
|
if (*suffix) {
|
||||||
++suffix;
|
suffix = decoded.second;
|
||||||
for ( ; *suffix; ++suffix) {
|
for ( ; *suffix; suffix = decoded.second) {
|
||||||
new_stacks = llama_grammar_accept(grammar->rules, new_stacks, *suffix);
|
decoded = decode_utf8(suffix);
|
||||||
if (new_stacks.empty()) {
|
new_stacks = llama_grammar_accept(grammar->rules, new_stacks, decoded.first);
|
||||||
|
if (new_stacks.empty() ) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2480,8 +2539,9 @@ llama_token llama_grammar_accept_token(struct llama_context * ctx, struct llama_
|
||||||
// accept the first token of the matching prefix into the grammar
|
// accept the first token of the matching prefix into the grammar
|
||||||
llama_token first_prefix_token = tokens[0];
|
llama_token first_prefix_token = tokens[0];
|
||||||
const char * first_prefix_str = llama_token_to_str(ctx, first_prefix_token);
|
const char * first_prefix_str = llama_token_to_str(ctx, first_prefix_token);
|
||||||
for ( ; *first_prefix_str; ++first_prefix_str) {
|
for ( ; *first_prefix_str; first_prefix_str = decoded.second) {
|
||||||
grammar->stacks = llama_grammar_accept(grammar->rules, grammar->stacks, *first_prefix_str);
|
decoded = decode_utf8(first_prefix_str);
|
||||||
|
grammar->stacks = llama_grammar_accept(grammar->rules, grammar->stacks, decoded.first);
|
||||||
LLAMA_ASSERT(!grammar->stacks.empty());
|
LLAMA_ASSERT(!grammar->stacks.empty());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
58
llama.h
58
llama.h
|
@ -55,8 +55,6 @@ extern "C" {
|
||||||
|
|
||||||
struct llama_context;
|
struct llama_context;
|
||||||
|
|
||||||
struct llama_grammar;
|
|
||||||
|
|
||||||
typedef int llama_token;
|
typedef int llama_token;
|
||||||
|
|
||||||
typedef struct llama_token_data {
|
typedef struct llama_token_data {
|
||||||
|
@ -125,6 +123,37 @@ extern "C" {
|
||||||
bool quantize_output_tensor; // quantize output.weight
|
bool quantize_output_tensor; // quantize output.weight
|
||||||
} llama_model_quantize_params;
|
} llama_model_quantize_params;
|
||||||
|
|
||||||
|
// grammar types
|
||||||
|
struct llama_grammar;
|
||||||
|
|
||||||
|
// grammar element type
|
||||||
|
enum llama_gretype {
|
||||||
|
// end of rule definition
|
||||||
|
LLAMA_GRETYPE_END = 0,
|
||||||
|
|
||||||
|
// start of alternate definition for rule
|
||||||
|
LLAMA_GRETYPE_ALT = 1,
|
||||||
|
|
||||||
|
// non-terminal element: reference to rule
|
||||||
|
LLAMA_GRETYPE_RULE_REF = 2,
|
||||||
|
|
||||||
|
// terminal element: character (code point)
|
||||||
|
LLAMA_GRETYPE_CHAR = 3,
|
||||||
|
|
||||||
|
// modifies a preceding LLAMA_GRETYPE_CHAR or LLAMA_GRETYPE_CHAR_ALT to
|
||||||
|
// be an inclusive range ([a-z])
|
||||||
|
LLAMA_GRETYPE_CHAR_RNG_UPPER = 4,
|
||||||
|
|
||||||
|
// modifies a preceding LLAMA_GRETYPE_CHAR or
|
||||||
|
// LLAMA_GRETYPE_CHAR_RNG_UPPER to add an alternate char to match ([ab], [a-zA])
|
||||||
|
LLAMA_GRETYPE_CHAR_ALT = 5,
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef struct llama_grammar_element {
|
||||||
|
enum llama_gretype type;
|
||||||
|
uint32_t value; // Unicode code point or rule ID
|
||||||
|
} llama_grammar_element;
|
||||||
|
|
||||||
LLAMA_API struct llama_context_params llama_context_default_params();
|
LLAMA_API struct llama_context_params llama_context_default_params();
|
||||||
LLAMA_API struct llama_model_quantize_params llama_model_quantize_default_params();
|
LLAMA_API struct llama_model_quantize_params llama_model_quantize_default_params();
|
||||||
|
|
||||||
|
@ -243,26 +272,11 @@ extern "C" {
|
||||||
|
|
||||||
// Grammar
|
// Grammar
|
||||||
//
|
//
|
||||||
// Accepts a binary encoding of a context-free grammar. The returned struct can be used to
|
LLAMA_API struct llama_grammar * llama_grammar_init(
|
||||||
// constrain sampled tokens (see below).
|
const llama_grammar_element ** rules,
|
||||||
//
|
size_t n_rules,
|
||||||
// The binary format represents one or more production rules, each with one or more alternate
|
size_t start_rule_index);
|
||||||
// defininitions:
|
|
||||||
//
|
|
||||||
// (<rule_id: u16> (<alt_size: u16> <alt_size * u16>)+ 0000)+ FFFF
|
|
||||||
//
|
|
||||||
// rule_ids should be assigned sequentially from zero but may appear out of order. Each
|
|
||||||
// rule alternate is a sequence of zero or more symbols, each prefixed with size:
|
|
||||||
//
|
|
||||||
// (<sym_size: u16> <sym_size * u16>)* 0000
|
|
||||||
//
|
|
||||||
// A symbol of size 1 is interpreted as a rule reference (whose value is the single following
|
|
||||||
// u16). Symbols sized greater than 1 are interpreted as inclusive pairs of 16-bit chars to
|
|
||||||
// match. Note that symbol sizes greater than 7FFF are reserved for future use.
|
|
||||||
//
|
|
||||||
// The provided `src` must be kept valid for the lifetime of the `llama_grammar`.
|
|
||||||
//
|
|
||||||
LLAMA_API struct llama_grammar * llama_grammar_init(const uint16_t * src, uint16_t start_rule_id);
|
|
||||||
LLAMA_API void llama_grammar_free(struct llama_grammar * grammar);
|
LLAMA_API void llama_grammar_free(struct llama_grammar * grammar);
|
||||||
|
|
||||||
// Sampling functions
|
// Sampling functions
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue