123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331 |
- package json
- import "core:mem"
- import "core:math/bits"
- import "core:runtime"
- import "core:strconv"
- import "core:strings"
- import "core:io"
- Marshal_Data_Error :: enum {
- Unsupported_Type,
- }
- Marshal_Error :: union {
- Marshal_Data_Error,
- io.Error,
- }
- marshal :: proc(v: any, allocator := context.allocator) -> (data: []byte, err: Marshal_Error) {
- b := strings.make_builder(allocator)
- defer if err != .None {
- strings.destroy_builder(&b)
- }
- marshal_to_builder(&b, v) or_return
-
- if len(b.buf) != 0 {
- data = b.buf[:]
- }
- return data, .None
- }
- marshal_to_builder :: proc(b: ^strings.Builder, v: any) -> Marshal_Error {
- return marshal_to_writer(strings.to_writer(b), v)
- }
- marshal_to_writer :: proc(w: io.Writer, v: any) -> (err: Marshal_Error) {
- if v == nil {
- io.write_string(w, "null") or_return
- return
- }
- ti := runtime.type_info_base(type_info_of(v.id))
- a := any{v.data, ti.id}
- switch info in ti.variant {
- case runtime.Type_Info_Named:
- unreachable()
- case runtime.Type_Info_Integer:
- buf: [21]byte
- u: u128
- switch i in a {
- case i8: u = u128(i)
- case i16: u = u128(i)
- case i32: u = u128(i)
- case i64: u = u128(i)
- case int: u = u128(i)
- case u8: u = u128(i)
- case u16: u = u128(i)
- case u32: u = u128(i)
- case u64: u = u128(i)
- case u128: u = u128(i)
- case uint: u = u128(i)
- case uintptr: u = u128(i)
- case i16le: u = u128(i)
- case i32le: u = u128(i)
- case i64le: u = u128(i)
- case u16le: u = u128(i)
- case u32le: u = u128(i)
- case u64le: u = u128(i)
- case u128le: u = u128(i)
- case i16be: u = u128(i)
- case i32be: u = u128(i)
- case i64be: u = u128(i)
- case u16be: u = u128(i)
- case u32be: u = u128(i)
- case u64be: u = u128(i)
- case u128be: u = u128(i)
- }
- s := strconv.append_bits_128(buf[:], u, 10, info.signed, 8*ti.size, "0123456789", nil)
- io.write_string(w, s) or_return
- case runtime.Type_Info_Rune:
- r := a.(rune)
- io.write_byte(w, '"') or_return
- io.write_escaped_rune(w, r, '"', true) or_return
- io.write_byte(w, '"') or_return
- case runtime.Type_Info_Float:
- switch f in a {
- case f16: io.write_f16(w, f) or_return
- case f32: io.write_f32(w, f) or_return
- case f64: io.write_f64(w, f) or_return
- case: return .Unsupported_Type
- }
- case runtime.Type_Info_Complex:
- r, i: f64
- switch z in a {
- case complex32: r, i = f64(real(z)), f64(imag(z))
- case complex64: r, i = f64(real(z)), f64(imag(z))
- case complex128: r, i = f64(real(z)), f64(imag(z))
- case: return .Unsupported_Type
- }
-
- io.write_byte(w, '[') or_return
- io.write_f64(w, r) or_return
- io.write_string(w, ", ") or_return
- io.write_f64(w, i) or_return
- io.write_byte(w, ']') or_return
- case runtime.Type_Info_Quaternion:
- return .Unsupported_Type
- case runtime.Type_Info_String:
- switch s in a {
- case string: io.write_quoted_string(w, s) or_return
- case cstring: io.write_quoted_string(w, string(s)) or_return
- }
- case runtime.Type_Info_Boolean:
- val: bool
- switch b in a {
- case bool: val = bool(b)
- case b8: val = bool(b)
- case b16: val = bool(b)
- case b32: val = bool(b)
- case b64: val = bool(b)
- }
- io.write_string(w, val ? "true" : "false") or_return
- case runtime.Type_Info_Any:
- return .Unsupported_Type
- case runtime.Type_Info_Type_Id:
- return .Unsupported_Type
- case runtime.Type_Info_Pointer:
- return .Unsupported_Type
- case runtime.Type_Info_Multi_Pointer:
- return .Unsupported_Type
- case runtime.Type_Info_Procedure:
- return .Unsupported_Type
- case runtime.Type_Info_Tuple:
- return .Unsupported_Type
- case runtime.Type_Info_Simd_Vector:
- return .Unsupported_Type
- case runtime.Type_Info_Relative_Pointer:
- return .Unsupported_Type
- case runtime.Type_Info_Relative_Slice:
- return .Unsupported_Type
-
- case runtime.Type_Info_Matrix:
- return .Unsupported_Type
- case runtime.Type_Info_Array:
- io.write_byte(w, '[') or_return
- for i in 0..<info.count {
- if i > 0 { io.write_string(w, ", ") or_return }
- data := uintptr(v.data) + uintptr(i*info.elem_size)
- marshal_to_writer(w, any{rawptr(data), info.elem.id}) or_return
- }
- io.write_byte(w, ']') or_return
-
- case runtime.Type_Info_Enumerated_Array:
- index := runtime.type_info_base(info.index).variant.(runtime.Type_Info_Enum)
- io.write_byte(w, '[') or_return
- for i in 0..<info.count {
- if i > 0 { io.write_string(w, ", ") or_return }
- data := uintptr(v.data) + uintptr(i*info.elem_size)
- marshal_to_writer(w, any{rawptr(data), info.elem.id}) or_return
- }
- io.write_byte(w, ']') or_return
-
- case runtime.Type_Info_Dynamic_Array:
- io.write_byte(w, '[') or_return
- array := cast(^mem.Raw_Dynamic_Array)v.data
- for i in 0..<array.len {
- if i > 0 { io.write_string(w, ", ") or_return }
- data := uintptr(array.data) + uintptr(i*info.elem_size)
- marshal_to_writer(w, any{rawptr(data), info.elem.id}) or_return
- }
- io.write_byte(w, ']') or_return
- case runtime.Type_Info_Slice:
- io.write_byte(w, '[') or_return
- slice := cast(^mem.Raw_Slice)v.data
- for i in 0..<slice.len {
- if i > 0 { io.write_string(w, ", ") or_return }
- data := uintptr(slice.data) + uintptr(i*info.elem_size)
- marshal_to_writer(w, any{rawptr(data), info.elem.id}) or_return
- }
- io.write_byte(w, ']') or_return
- case runtime.Type_Info_Map:
- m := (^mem.Raw_Map)(v.data)
- io.write_byte(w, '{') or_return
- if m != nil {
- if info.generated_struct == nil {
- return .Unsupported_Type
- }
- entries := &m.entries
- gs := runtime.type_info_base(info.generated_struct).variant.(runtime.Type_Info_Struct)
- ed := runtime.type_info_base(gs.types[1]).variant.(runtime.Type_Info_Dynamic_Array)
- entry_type := ed.elem.variant.(runtime.Type_Info_Struct)
- entry_size := ed.elem_size
- for i in 0..<entries.len {
- if i > 0 { io.write_string(w, ", ") or_return }
- data := uintptr(entries.data) + uintptr(i*entry_size)
- key := rawptr(data + entry_type.offsets[2])
- value := rawptr(data + entry_type.offsets[3])
- marshal_to_writer(w, any{key, info.key.id}) or_return
- io.write_string(w, ": ") or_return
- marshal_to_writer(w, any{value, info.value.id}) or_return
- }
- }
- io.write_byte(w, '}') or_return
- case runtime.Type_Info_Struct:
- io.write_byte(w, '{') or_return
- for name, i in info.names {
- if i > 0 { io.write_string(w, ", ") or_return }
- io.write_quoted_string(w, name) or_return
- io.write_string(w, ": ") or_return
- id := info.types[i].id
- data := rawptr(uintptr(v.data) + info.offsets[i])
- marshal_to_writer(w, any{data, id}) or_return
- }
- io.write_byte(w, '}') or_return
- case runtime.Type_Info_Union:
- tag_ptr := uintptr(v.data) + info.tag_offset
- tag_any := any{rawptr(tag_ptr), info.tag_type.id}
- tag: i64 = -1
- switch i in tag_any {
- case u8: tag = i64(i)
- case i8: tag = i64(i)
- case u16: tag = i64(i)
- case i16: tag = i64(i)
- case u32: tag = i64(i)
- case i32: tag = i64(i)
- case u64: tag = i64(i)
- case i64: tag = i64(i)
- case: panic("Invalid union tag type")
- }
- if v.data == nil || tag == 0 {
- io.write_string(w, "null") or_return
- } else {
- id := info.variants[tag-1].id
- return marshal_to_writer(w, any{v.data, id})
- }
- case runtime.Type_Info_Enum:
- return marshal_to_writer(w, any{v.data, info.base.id})
- case runtime.Type_Info_Bit_Set:
- is_bit_set_different_endian_to_platform :: proc(ti: ^runtime.Type_Info) -> bool {
- if ti == nil {
- return false
- }
- t := runtime.type_info_base(ti)
- #partial switch info in t.variant {
- case runtime.Type_Info_Integer:
- switch info.endianness {
- case .Platform: return false
- case .Little: return ODIN_ENDIAN != "little"
- case .Big: return ODIN_ENDIAN != "big"
- }
- }
- return false
- }
- bit_data: u64
- bit_size := u64(8*ti.size)
- do_byte_swap := is_bit_set_different_endian_to_platform(info.underlying)
- switch bit_size {
- case 0: bit_data = 0
- case 8:
- x := (^u8)(v.data)^
- bit_data = u64(x)
- case 16:
- x := (^u16)(v.data)^
- if do_byte_swap {
- x = bits.byte_swap(x)
- }
- bit_data = u64(x)
- case 32:
- x := (^u32)(v.data)^
- if do_byte_swap {
- x = bits.byte_swap(x)
- }
- bit_data = u64(x)
- case 64:
- x := (^u64)(v.data)^
- if do_byte_swap {
- x = bits.byte_swap(x)
- }
- bit_data = u64(x)
- case: panic("unknown bit_size size")
- }
- io.write_u64(w, bit_data) or_return
- return .Unsupported_Type
- }
- return
- }
|