Cthulhu  0.2.10
Cthulhu compiler collection
parse.c
Go to the documentation of this file.
1 // SPDX-License-Identifier: LGPL-3.0-only
2 
3 #include "base/util.h"
4 #include "common.h"
5 
6 #include "config/config.h"
7 #include "core/macros.h"
8 #include "arena/arena.h"
9 #include "std/typed/vector.h"
10 #include "std/vector.h"
11 #include "std/str.h"
12 #include "std/map.h"
13 
14 #include "io/io.h"
15 
16 #include "interop/compile.h"
17 
18 #include "ap_bison.h" // IWYU pragma: keep
19 #include "ap_flex.h" // IWYU pragma: keep
20 
21 CT_CALLBACKS(kCallbacks, ap);
22 
23 static void push_single_arg(typevec_t *vec, const char *arg)
24 {
25  CTASSERT(vec != NULL);
26  CTASSERT(arg != NULL);
27 
28  // TODO: this is a bit of a bodge, i should think of a better
29  // way to extract flags from the command line
30  bool is_flag = arg[0] == '-' || arg[0] == '/';
31  size_t len = ctu_strlen(arg);
32 
33  if (is_flag)
34  {
35  size_t idx = 0;
36  for (size_t i = 0; i < len; i++)
37  {
38  if (arg[i] == ':' || arg[i] == '=')
39  {
40  idx = i + 1;
41  break;
42  }
43  }
44 
45  // we found an assignment so we need to split the string
46  if (idx > 0)
47  {
48  typevec_append(vec, arg, idx);
49 
50  typevec_append(vec, " \"", 2);
51  typevec_append(vec, arg + idx, len - idx);
52  typevec_push(vec, "\"");
53  }
54  else
55  {
56  // otherwise just paste it in
57  typevec_append(vec, arg, len);
58  }
59  }
60  else
61  {
62  // if its not a flag then we just need to wrap it in quotes
63  typevec_push(vec, "\"");
64  typevec_append(vec, arg, len);
65  typevec_push(vec, "\"");
66  }
67 }
68 
69 static char *join_args(int argc, const char **argv, arena_t *arena)
70 {
71  typevec_t *vec = typevec_new(sizeof(char), argc - 1, arena);
72  for (int i = 1; i < argc; i++)
73  {
74  const char *arg = argv[i];
75 
76  push_single_arg(vec, arg);
77 
78  typevec_push(vec, " ");
79  }
80 
81  typevec_push(vec, "\0");
82 
83  return typevec_data(vec);
84 }
85 
86 int ap_parse_common(ap_t *self, const char *text)
87 {
88  CTASSERT(self != NULL);
89  CTASSERT(text != NULL);
90 
91  io_t *io = io_string("<command>", text, self->arena);
92  scan_t *scan = scan_io("args", io, self->arena);
93 
94  scan_set_context(scan, self);
95  parse_result_t result = scan_buffer(scan, &kCallbacks);
96 
97  return result.result == eParseOk ? CT_EXIT_OK : CT_EXIT_ERROR;
98 }
99 
100 int ap_parse_args(ap_t *self, int argc, const char **argv)
101 {
102  char *args = join_args(argc, argv, self->arena);
103  return ap_parse_common(self, args);
104 }
105 
106 int ap_parse(ap_t *self, const char *str)
107 {
108  return ap_parse_common(self, str);
109 }
110 
111 static int get_option_type(const cfg_field_t *field)
112 {
113  switch (cfg_get_type(field))
114  {
115  case eConfigBool: return AP_BOOL_OPTION;
116  case eConfigInt: return AP_INT_OPTION;
117 
118  case eConfigEnum:
119  case eConfigFlags:
120  case eConfigString:
121  case eConfigVector:
122  return AP_STRING_OPTION;
123 
124  default:
125  CT_NEVER("unknown option type %d", cfg_get_type(field));
126  }
127 }
128 
129 static const char *get_lookup_name(const char *name, bool *negate, arena_t *arena)
130 {
131  CTASSERT(name != NULL);
132  CTASSERT(negate != NULL);
133 
134  // if the name ends with `-` then its a negation
135  // so we need to strip that off
136 
137  *negate = false;
138  size_t len = ctu_strlen(name);
139 
140  // something is wrong if the name is empty
141  CTASSERT(len > 0);
142 
143  if (name[len - 1] == '-')
144  {
145  *negate = true;
146  len -= 1;
147  return arena_strndup(name, len, arena);
148  }
149 
150  return name;
151 }
152 
153 int ap_get_opt(scan_t *scan, const char *name, ap_field_t *param, char **value)
154 {
155  CTASSERT(scan != NULL);
156  CTASSERT(name != NULL);
157  CTASSERT(param != NULL);
158 
159  ap_t *self = scan_get_context(scan);
160 
161  // if the name ends with `-` then its a negation
162  // so we need to strip that off
163 
164  bool negate = false;
165  const char *lookup = get_lookup_name(name, &negate, self->arena);
166 
167  cfg_field_t *result = map_get(self->name_lookup, lookup);
168  if (result == NULL)
169  {
170  *value = arena_strdup(name, self->arena);
171  vector_push(&self->unknown, *value);
172  return AP_ERROR;
173  }
174 
175  ap_field_t out = {
176  .field = result,
177  .negate = negate
178  };
179 
180  self->count += 1;
181  *param = out;
182  cfg_type_t type = get_option_type(result);
183 
184  if (negate && cfg_get_type(result) != eConfigBool)
185  {
186  ap_add_error(self, "cannot negate non-boolean flag: %s", name);
187  }
188 
189  return type;
190 }
CT_NODISCARD CT_SCAN_API scan_t * scan_io(const char *language, io_t *io, arena_t *arena)
create a scanner from an io source
Definition: scan.c:121
CT_SCAN_API void scan_set_context(scan_t *scan, void *value)
get the context of a scanner
Definition: scan.c:80
CT_NODISCARD CT_PUREFN CT_SCAN_API void * scan_get_context(const scan_t *scan)
get the context of a scanner
Definition: scan.c:88
int ap_parse(ap_t *self, const char *str)
parse a string
Definition: parse.c:106
int ap_parse_args(ap_t *self, int argc, const char **argv)
parse a command line
Definition: parse.c:100
CT_NODISCARD CT_PUREFN CT_BASE_API size_t ctu_strlen(const char *str)
get the length of a string not including the null terminator equivalent to strlen but with safety che...
Definition: util.c:87
STA_RET_STRING colour_t idx
Definition: colour.h:98
CT_PUREFN CT_CONFIG_API cfg_type_t cfg_get_type(const cfg_field_t *field)
get the type of a configuration field
Definition: reflect.c:6
cfg_type_t
the type of a configuration field
Definition: config.h:27
#define CT_EXIT_OK
no user errors or internal errors
Definition: macros.h:91
#define CT_EXIT_ERROR
the user has made an error
Definition: macros.h:92
CT_NODISCARD CT_PUREFN CT_STD_API void * map_get(const map_t *map, const void *key)
get a value from a map
Definition: map.c:324
CT_INTEROP_API parse_result_t scan_buffer(scan_t *scan, const scan_callbacks_t *callbacks)
parse the contents of a scanner into a language specific ast
Definition: compile.c:26
@ eParseOk
parse was successful
Definition: compile.h:76
CT_NODISCARD CT_IO_API io_t * io_string(const char *name, const char *string, arena_t *arena)
create an IO view of a string create a readonly IO view of a string
Definition: view.c:90
CT_NODISCARD CT_ARENA_API char * arena_strdup(const char *str, arena_t *arena)
allocate a copy of a string from a custom allocator
Definition: arena.c:95
CT_NODISCARD CT_ARENA_API char * arena_strndup(STA_READS(len) const char *str, size_t len, arena_t *arena)
allocate a copy of a string with a maximum length from a custom allocator
#define CT_NEVER(...)
assert that a code path is never reached
Definition: panic.h:136
#define CTASSERT(expr)
assert a condition, prints the condition as a message
Definition: panic.h:130
CT_STD_API void * typevec_push(typevec_t *vec, const void *src)
push a value onto the vector
Definition: vector.c:156
CT_NODISCARD CT_PUREFN CT_STD_API void * typevec_data(const typevec_t *vec)
get a pointer to the underlying data
Definition: vector.c:199
CT_STD_API void typevec_append(typevec_t *vec, const void *src, size_t len)
append multiple values onto the vector
Definition: vector.c:169
CT_NODISCARD CT_STD_API typevec_t * typevec_new(size_t width, size_t len, arena_t *arena)
create a new typed vector on the heap
Definition: vector.c:77
CT_STD_API void vector_push(vector_t **vector, void *value)
push a value onto the end of a vector
Definition: vector.c:108
CT_CALLBACKS(kCallbacks, ap)
int ap_parse_common(ap_t *self, const char *text)
Definition: parse.c:86
int ap_get_opt(scan_t *scan, const char *name, ap_field_t *param, char **value)
Definition: parse.c:153
cfg_field_t * field
Definition: common.h:64
argparse instance
Definition: common.h:26
an allocator object
Definition: arena.h:86
io object implementation
Definition: impl.h:122
arena_t * arena
the arena this object was allocated from
Definition: impl.h:133
parse_error_t result
Definition: compile.h:92
a source file scanner
Definition: scan.h:24
A vector with a fixed type size.
Definition: vector.h:24
void ap_add_error(ap_t *self, const char *fmt,...)
Definition: common.c:37