mirror of
https://github.com/ruby/ruby.git
synced 2025-08-23 21:14:23 +02:00

I, Luke T. Shumaker, am the sole author of the added code.
I did not reference CVTUTF when writing it. I did reference the
Unicode standard (15.0.0), the Wikipedia article on UTF-8, and the
Wikipedia article on UTF-16. When I saw some tests fail, I did
reference the old deleted code (but a JSON-specific part, inherently
not as based on CVTUTF) to determine that script_safe should also
escape U+2028 and U+2029.
I targeted simplicity and clarity when writing the code--it can likely
be optimized. In my mind, the obvious next optimization is to have it
combine contiguous non-escaped characters into just one call to
fbuffer_append(), instead of calling fbuffer_append() for each
character.
Regarding the use of the "modern" types `uint32_t`, `uint16_t`, and
`bool`:
- ruby.h is guaranteed to give us uint32_t and uint16_t.
- Since Ruby 3.0.0, ruby.h is guaranteed to give us bool... but we
support down to Ruby 2.3. But, ruby.h is guaranteed to give us
HAVE_STDBOOL_H for the C99 stdbool.h; so use that to include
stdbool.h if we can, and if not then fall back to a copy of the
same bool definition that Ruby 3.0.5 uses with C89.
c96351f874
79 lines
2.4 KiB
C
79 lines
2.4 KiB
C
#ifndef _PARSER_H_
|
|
#define _PARSER_H_
|
|
|
|
#include "ruby.h"
|
|
|
|
#ifndef HAVE_RUBY_RE_H
|
|
#include "re.h"
|
|
#endif
|
|
|
|
#ifdef HAVE_RUBY_ST_H
|
|
#include "ruby/st.h"
|
|
#else
|
|
#include "st.h"
|
|
#endif
|
|
|
|
#ifndef MAYBE_UNUSED
|
|
# define MAYBE_UNUSED(x) x
|
|
#endif
|
|
|
|
#define option_given_p(opts, key) RTEST(rb_funcall(opts, i_key_p, 1, key))
|
|
|
|
typedef struct JSON_ParserStruct {
|
|
VALUE Vsource;
|
|
char *source;
|
|
long len;
|
|
char *memo;
|
|
VALUE create_id;
|
|
int max_nesting;
|
|
int allow_nan;
|
|
int parsing_name;
|
|
int symbolize_names;
|
|
int freeze;
|
|
VALUE object_class;
|
|
VALUE array_class;
|
|
VALUE decimal_class;
|
|
int create_additions;
|
|
VALUE match_string;
|
|
FBuffer *fbuffer;
|
|
} JSON_Parser;
|
|
|
|
#define GET_PARSER \
|
|
GET_PARSER_INIT; \
|
|
if (!json->Vsource) rb_raise(rb_eTypeError, "uninitialized instance")
|
|
#define GET_PARSER_INIT \
|
|
JSON_Parser *json; \
|
|
TypedData_Get_Struct(self, JSON_Parser, &JSON_Parser_type, json)
|
|
|
|
#define MinusInfinity "-Infinity"
|
|
#define EVIL 0x666
|
|
|
|
static uint32_t unescape_unicode(const unsigned char *p);
|
|
static int convert_UTF32_to_UTF8(char *buf, uint32_t ch);
|
|
static char *JSON_parse_object(JSON_Parser *json, char *p, char *pe, VALUE *result, int current_nesting);
|
|
static char *JSON_parse_value(JSON_Parser *json, char *p, char *pe, VALUE *result, int current_nesting);
|
|
static char *JSON_parse_integer(JSON_Parser *json, char *p, char *pe, VALUE *result);
|
|
static char *JSON_parse_float(JSON_Parser *json, char *p, char *pe, VALUE *result);
|
|
static char *JSON_parse_array(JSON_Parser *json, char *p, char *pe, VALUE *result, int current_nesting);
|
|
static VALUE json_string_unescape(char *string, char *stringEnd, int intern, int symbolize);
|
|
static char *JSON_parse_string(JSON_Parser *json, char *p, char *pe, VALUE *result);
|
|
static VALUE convert_encoding(VALUE source);
|
|
static VALUE cParser_initialize(int argc, VALUE *argv, VALUE self);
|
|
static VALUE cParser_parse(VALUE self);
|
|
static void JSON_mark(void *json);
|
|
static void JSON_free(void *json);
|
|
static VALUE cJSON_parser_s_allocate(VALUE klass);
|
|
static VALUE cParser_source(VALUE self);
|
|
#ifndef ZALLOC
|
|
#define ZALLOC(type) ((type *)ruby_zalloc(sizeof(type)))
|
|
static inline void *ruby_zalloc(size_t n)
|
|
{
|
|
void *p = ruby_xmalloc(n);
|
|
memset(p, 0, n);
|
|
return p;
|
|
}
|
|
#endif
|
|
|
|
static const rb_data_type_t JSON_Parser_type;
|
|
|
|
#endif
|