let parse_optional ?(pos=0) ?len buf =
let len =
match len with Some s -> s | None -> String.length buf in
let uint16 pos =
Binary_packing.unpack_unsigned_8 ~buf ~pos +
Binary_packing.unpack_unsigned_8 ~buf ~pos:(pos + 1) lsl 7 in
let from () = String.sub buf pos len in
dbg "from: %S" (from ());
let rec build ofs acc =
let error e = fail (`wrong_auxiliary_data (e, from ())) in
if ofs >= len then return acc
else (
if ofs + 2 >= len then error `out_of_bounds
else (
let tag = String.sub buf ofs 2 in
let typ = buf.[ofs + 2] in
let check_size_and_return n r =
if ofs + 2 + n >= len then error `out_of_bounds
else return (r, n) in
let parse_cCsSiIf pos typ =
begin match typ with
| 'i' ->
signed_int ~buf ~pos >>= fun v ->
check_size_and_return 4 (`int v)
| 'A' -> check_size_and_return 1 (`char buf.[pos])
| 'c' | 'C' -> check_size_and_return 1 (`int (Char.to_int buf.[pos]))
| 's' ->
check_size_and_return 2 (`int (
Binary_packing.unpack_signed_16
~byte_order:`Little_endian ~buf ~pos))
| 'S' -> check_size_and_return 2 (`int (uint16 pos))
| 'f' ->
let f =
Binary_packing.unpack_signed_32
~byte_order:`Little_endian ~buf ~pos |! Int32.float_of_bits in
check_size_and_return 4 (`float f)
| _ -> error (`unknown_type typ)
end
in
let pos = ofs + 3 in
begin match typ with
| 'A' -> check_size_and_return 1 (`char buf.[pos])
| 'Z' ->
begin match String.index_from buf pos '\000' with
| Some e -> return (`string String.(slice buf pos e), e - pos + 1)
| None -> error `null_terminated_string
end
| 'H' ->
begin match String.index_from buf pos '\000' with
| Some e -> return (`string String.(slice buf pos e), e - pos + 1)
| None -> error `null_terminated_hexarray
end
| 'B' ->
check_size_and_return 1 buf.[pos] >>= fun (array_type, _) ->
signed_int ~buf ~pos:(pos + 1) >>= fun i32 ->
check_size_and_return 4 i32 >>= fun (size, _) ->
(if size > 4000 then error (`array_size size) else return ())
>>= fun () ->
let arr = Array.create size (`char 'B') in
let rec loop p = function
| 0 -> return p
| n ->
parse_cCsSiIf p array_type
>>= fun (v, nb) ->
arr.(size - n) <- v;
loop (p + nb) (n - 1) in
loop (pos + 5) size
>>= fun newpos ->
return (`array (array_type, arr), newpos - pos)
| c -> parse_cCsSiIf pos c
end
>>= fun (v, nbread) ->
build (ofs + 3 + nbread) ((tag, typ, v) :: acc)
)
)
in
match build pos [] with
| Ok r -> return (List.rev r)
| Error (`wrong_auxiliary_data e) -> fail (`wrong_auxiliary_data e)
| Error (`wrong_int32 e) -> fail (`wrong_auxiliary_data (`wrong_int32 e, from ()))