Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit 4f6f105

Browse files
authoredFeb 2, 2023
Merge pull request #306 from c-cube/to_hex_string
add to_hex_string
2 parents 2c9b3a0 + 9bf4589 commit 4f6f105

File tree

4 files changed

+76
-2
lines changed

4 files changed

+76
-2
lines changed
 

‎lib/cstruct.ml

+23
Original file line numberDiff line numberDiff line change
@@ -54,6 +54,7 @@ let err_sub t = err "Cstruct.sub: %a off=%d len=%d" pp_t t
5454
let err_shift t = err "Cstruct.shift %a %d" pp_t t
5555
let err_shiftv n = err "Cstruct.shiftv short by %d" n
5656
let err_copy t = err "Cstruct.copy %a off=%d len=%d" pp_t t
57+
let err_to_hex_string t = err "Cstruct.to_hex_string %a off=%d len=%d" pp_t t
5758
let err_blit_src src dst =
5859
err "Cstruct.blit src=%a dst=%a src-off=%d len=%d" pp_t src pp_t dst
5960
let err_blit_dst src dst =
@@ -401,6 +402,28 @@ let to_string ?(off=0) ?len:sz t =
401402
freshly-created value built by [to_bytes t]. *)
402403
copy t off len
403404

405+
let to_hex_string ?(off=0) ?len:sz t : string =
406+
let[@inline] nibble_to_char (i:int) : char =
407+
if i < 10 then
408+
Char.chr (i + Char.code '0')
409+
else
410+
Char.chr (i - 10 + Char.code 'a')
411+
in
412+
413+
let len = match sz with None -> length t - off | Some l -> l in
414+
if len < 0 || off < 0 || t.len - off < len then
415+
err_to_hex_string t off len
416+
else (
417+
let out = Bytes.create (2 * len) in
418+
for i=0 to len-1 do
419+
let c = Char.code @@ Bigarray.Array1.get t.buffer (i+t.off+off) in
420+
Bytes.set out (2*i) (nibble_to_char (c lsr 4));
421+
Bytes.set out (2*i+1) (nibble_to_char (c land 0xf));
422+
done;
423+
Bytes.unsafe_to_string out
424+
)
425+
426+
404427
let to_bytes ?off ?len t =
405428
Bytes.unsafe_of_string (to_string ?off ?len t)
406429

‎lib/cstruct.mli

+10-2
Original file line numberDiff line numberDiff line change
@@ -349,10 +349,18 @@ val split: ?start:int -> t -> int -> t * t
349349
val to_string: ?off:int -> ?len:int -> t -> string
350350
(** [to_string ~off ~len t] will allocate a fresh OCaml [string] and copy the
351351
contents of the cstruct starting at offset [off] (default [0]) of length
352-
[len] (default [Cstruct.len t - off]) into it, and return that string.
352+
[len] (default [Cstruct.length t - off]) into it, and return that string.
353353
354354
@raise Invalid_argument if [off] or [len] is negative, or
355-
[Cstruct.len str - off] < [len]. *)
355+
[Cstruct.length t - off] < [len]. *)
356+
357+
val to_hex_string : ?off:int -> ?len:int -> t -> string
358+
(** [to_hex_string ~off ~len t] is a fresh OCaml [string] containing
359+
the hex representation of [sub t off len]. It is therefore of length
360+
[2 * len]. This string can be read back into a Cstruct using {!of_hex}.
361+
@raise Invalid_argument if [off] or [len] is negative, or
362+
if [Cstruct.length t - off < len].
363+
@since 6.2 *)
356364

357365
val to_bytes: ?off:int -> ?len:int -> t -> bytes
358366
(** [to_bytes ~off ~len t] will allocate a fresh OCaml [bytes] and copy the

‎lib/cstruct_cap.mli

+7
Original file line numberDiff line numberDiff line change
@@ -265,6 +265,13 @@ val to_string : ?off:int -> ?len:int -> 'a rd t -> string
265265
@raise Invalid_argument if [off] and [len] does not designate a valid
266266
segment of [t]. *)
267267

268+
val to_hex_string : ?off:int -> ?len:int -> _ rd t -> string
269+
(** [to_hex_string ~off ~len t] is a fresh OCaml [string] containing
270+
the hex representation of [sub t off len]. See {!Cstruct.to_hex_string}.
271+
@raise Invalid_argument if [off] or [len] is negative, or
272+
if [Cstruct.length t - off < len].
273+
@since 6.2 *)
274+
268275
val of_hex : ?off:int -> ?len:int -> string -> rdwr t
269276
(** [of_hex ~off ~len s] allocates a buffer and copies the content of [s]
270277
starting at offset [off] (default [0]) of length [len] (default

‎lib_test/tests.ml

+36
Original file line numberDiff line numberDiff line change
@@ -166,6 +166,35 @@ let hexdump_small () =
166166
let hex_multiline =
167167
Cstruct.of_hex "000102030405060708090a0b0c0d0e0f101112"
168168

169+
let hex_to_string_empty () =
170+
let c = Cstruct.of_string "" in
171+
let s = Cstruct.to_hex_string c in
172+
assert_string_equal ~msg:"encoded" s ""
173+
174+
let hex_to_string_small () =
175+
let c = Cstruct.of_string "hello world \x00 !" in
176+
let s = Cstruct.to_hex_string c in
177+
assert_string_equal ~msg:"encoded" "68656c6c6f20776f726c6420002021" s;
178+
let c' = Cstruct.of_hex s in
179+
assert_cs_equal ~msg:"decoded again" c c'
180+
181+
let hex_to_string_small_slice () =
182+
let c = Cstruct.of_string "This_1s Not @ Dr1LL" in
183+
let s = Cstruct.to_hex_string ~off:2 ~len:11 c in
184+
assert_string_equal ~msg:"encoded" "69735f3173204e6f742040" s;
185+
let c' = Cstruct.of_hex s in
186+
assert_cs_equal ~msg:"decoded again" (Cstruct.sub c 2 11) c';
187+
assert_string_equal ~msg:"decoded as str" "is_1s Not @" (Cstruct.to_string c')
188+
189+
let hex_to_string_small_slice_of_slice () =
190+
let c = Cstruct.of_string "This_1s Not @ Dr1LL" in
191+
let c_slice = Cstruct.sub c 2 11 in
192+
let s = Cstruct.to_hex_string ~off:3 ~len:6 c_slice in
193+
assert_string_equal ~msg:"encoded" "3173204e6f74" s;
194+
let c' = Cstruct.of_hex s in
195+
assert_cs_equal ~msg:"decoded again" (Cstruct.sub c_slice 3 6) c';
196+
assert_string_equal ~msg:"decoded as str" "1s Not" (Cstruct.to_string c')
197+
169198
let hexdump_multiline () =
170199
test_hexdump
171200
hex_multiline
@@ -224,7 +253,14 @@ let suite = [
224253
"aligned", `Quick, hexdump_aligned;
225254
"aligned to half", `Quick, hexdump_aligned_to_half;
226255
"in box", `Quick, hexdump_in_box;
256+
];
257+
"hex_to_string", [
258+
"empty", `Quick, hex_to_string_empty;
259+
"small", `Quick, hex_to_string_small;
260+
"small_slice", `Quick, hex_to_string_small_slice;
261+
"small_slice_of_slice", `Quick, hex_to_string_small_slice_of_slice;
227262
]
263+
228264
]
229265

230266
let () = Alcotest.run "cstruct" (("bounds", Bounds.suite) :: suite)

0 commit comments

Comments
 (0)