1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
open Sexplib.Std
open Result
type ('a, 'b) _result = [ `Ok of 'a | `Error of 'b ] [@@deriving sexp]
let result_of_sexp a b s = match _result_of_sexp a b s with
| `Ok x -> Ok x | `Error y -> Error y
let sexp_of_result a b r =
sexp_of__result a b (match r with Ok x -> `Ok x | Error y -> `Error y)
let _nbd_cmd_read = 0l
let _nbd_cmd_write = 1l
let _nbd_cmd_disc = 2l
let _nbd_cmd_flush = 3l
let _nbd_cmd_trim = 4l
let nbd_request_magic = 0x25609513l
let nbd_reply_magic = 0x67446698l
let nbd_flag_has_flags = 1
let nbd_flag_read_only = 2
let nbd_flag_send_flush = 4
let nbd_flag_send_fua = 8
let nbd_flag_rotational = 16
let nbd_flag_send_trim = 32
let nbd_flag_fixed_newstyle = 1
let nbd_flag_no_zeroes = 2
let nbd_flag_c_fixed_newstyle = 1
let nbd_flag_c_no_zeroes = 2
let zero buf =
for i = 0 to Cstruct.len buf - 1 do
Cstruct.set_uint8 buf i 0
done
module PerExportFlag = struct
type t =
| Read_only
| Send_flush
| Send_fua
| Rotational
| Send_trim
[@@deriving sexp]
let to_string t = Sexplib.Sexp.to_string (sexp_of_t t)
let of_int32 x =
let flags = Int32.to_int x in
let is_set i mask = i land mask = mask in
List.map snd
(List.filter (fun (mask,_) -> is_set flags mask)
[ nbd_flag_read_only, Read_only;
nbd_flag_send_flush, Send_flush;
nbd_flag_send_fua, Send_fua;
nbd_flag_rotational, Rotational;
nbd_flag_send_trim, Send_trim; ])
let to_int flags =
let one = function
| Read_only -> nbd_flag_read_only
| Send_flush -> nbd_flag_send_flush
| Send_fua -> nbd_flag_send_fua
| Rotational -> nbd_flag_rotational
| Send_trim -> nbd_flag_send_trim in
List.fold_left (lor) nbd_flag_has_flags (List.map one flags)
let to_int32 flags = Int32.of_int (to_int flags)
end
module GlobalFlag = struct
type t =
| Fixed_newstyle
| No_zeroes
[@@deriving sexp]
let to_string t = Sexplib.Sexp.to_string (sexp_of_t t)
let of_int flags =
let is_set i mask = i land mask = mask in
List.map snd
(List.filter (fun (mask,_) -> is_set flags mask)
[ nbd_flag_fixed_newstyle, Fixed_newstyle;
nbd_flag_no_zeroes, No_zeroes; ])
let to_int flags =
let one = function
| Fixed_newstyle -> nbd_flag_fixed_newstyle
| No_zeroes -> nbd_flag_no_zeroes in
List.fold_left (lor) 0 (List.map one flags)
end
module ClientFlag = struct
type t =
| Fixed_newstyle
| No_zeroes
[@@deriving sexp]
let to_string t = Sexplib.Sexp.to_string (sexp_of_t t)
let of_int32 flags =
let flags = Int32.to_int flags in
let is_set mask = mask land flags <> 0 in
List.map snd
(List.filter (fun (mask,_) -> is_set mask)
[ nbd_flag_c_fixed_newstyle, Fixed_newstyle;
nbd_flag_c_no_zeroes, No_zeroes; ])
let to_int32 flags =
let one = function
| Fixed_newstyle -> nbd_flag_c_fixed_newstyle
| No_zeroes -> nbd_flag_c_no_zeroes in
Int32.of_int (List.fold_left (lor) 0 (List.map one flags))
end
module Error = struct
type t = [
| `EPERM
| `EIO
| `ENOMEM
| `EINVAL
| `ENOSPC
| `Unknown of int32
] [@@deriving sexp]
let to_string t = Sexplib.Sexp.to_string (sexp_of_t t)
let of_int32 = function
| 1l -> `EPERM
| 5l -> `EIO
| 12l -> `ENOMEM
| 22l -> `EINVAL
| 28l -> `ENOSPC
| x -> `Unknown x
let to_int32 = function
| `EPERM -> 1l
| `EIO -> 5l
| `ENOMEM -> 12l
| `EINVAL -> 22l
| `ENOSPC -> 28l
| `Unknown x -> x
end
module Command = struct
type t =
| Read
| Write
| Disc
| Flush
| Trim
| Unknown of int32
[@@deriving sexp]
let to_string t = Sexplib.Sexp.to_string (sexp_of_t t)
let of_int32 = function
| 0l -> Read
| 1l -> Write
| 2l -> Disc
| 3l -> Flush
| 4l -> Trim
| c -> Unknown c
let to_int32 = function
| Read -> 0l
| Write -> 1l
| Disc -> 2l
| Flush -> 3l
| Trim -> 4l
| Unknown c -> c
end
module Option = struct
type t =
| ExportName
| Abort
| List
| StartTLS
| Unknown of int32
[@@deriving sexp]
let to_string t = Sexplib.Sexp.to_string (sexp_of_t t)
let of_int32 = function
| 1l -> ExportName
| 2l -> Abort
| 3l -> List
| 5l -> StartTLS
| c -> Unknown c
let to_int32 = function
| ExportName -> 1l
| Abort -> 2l
| List -> 3l
| StartTLS -> 5l
| Unknown c -> c
end
module OptionResponse = struct
type t =
| Ack
| Server
| Unsupported
| Policy
| Invalid
| Platform
| TlsReqd
| Unknown of int32
[@@deriving sexp]
let to_string t = Sexplib.Sexp.to_string (sexp_of_t t)
let of_int32 = function
| 1l -> Ack
| 2l -> Server
| -2147483647l -> Unsupported
| -2147483646l -> Policy
| -2147483645l -> Invalid
| -2147483644l -> Platform
| -2147483643l -> TlsReqd
| x -> Unknown x
let to_int32 = function
| Ack -> 1l
| Server -> 2l
| Unsupported -> -2147483647l
| Policy -> -2147483646l
| Invalid -> -2147483645l
| Platform -> -2147483644l
| TlsReqd -> -2147483643l
| Unknown x -> x
end
module Announcement = struct
type t = [ `V1 | `V2 ] [@@deriving sexp]
[%%cstruct
type t = {
passwd: uint8_t [@len 8];
magic: uint64_t;
} [@@big_endian]
]
let sizeof = sizeof_t
let expected_passwd = "NBDMAGIC"
let v1_magic = 0x00420281861253L
let v2_magic = 0x49484156454F5054L
let marshal buf t =
set_t_passwd expected_passwd 0 buf;
set_t_magic buf (match t with `V1 -> v1_magic | `V2 -> v2_magic)
let unmarshal buf =
let passwd = Cstruct.to_string (get_t_passwd buf) in
if passwd <> expected_passwd
then Error (Failure "Bad magic in negotiate")
else
let magic = get_t_magic buf in
if magic = v1_magic
then Ok `V1
else
if magic = v2_magic
then Ok `V2
else Error (Failure (Printf.sprintf "Bad magic; expected %Ld or %Ld got %Ld" v1_magic v2_magic magic))
end
module Negotiate = struct
type v1 = {
size: int64;
flags: PerExportFlag.t list;
} [@@deriving sexp]
type v2 = GlobalFlag.t list [@@deriving sexp]
type t =
| V1 of v1
| V2 of v2
[@@deriving sexp]
let to_string t = Sexplib.Sexp.to_string (sexp_of_t t)
[%%cstruct
type v1 = {
size: uint64_t;
flags: uint32_t;
padding: uint8_t [@len 124];
} [@@big_endian]
]
[%%cstruct
type v2 = {
flags: uint16_t;
} [@@big_endian]
]
let sizeof = function
| `V1 -> sizeof_v1
| `V2 -> sizeof_v2
let marshal buf t =
zero buf;
match t with
| V1 t ->
set_v1_size buf t.size;
set_v1_flags buf (PerExportFlag.to_int32 t.flags);
| V2 t ->
set_v2_flags buf (GlobalFlag.to_int t)
let unmarshal buf t =
match t with
| `V1 ->
let size = get_v1_size buf in
let flags = PerExportFlag.of_int32 (get_v1_flags buf) in
Ok (V1 { size; flags })
| `V2 ->
let flags = GlobalFlag.of_int (get_v2_flags buf) in
Ok (V2 flags)
end
module NegotiateResponse = struct
type t = ClientFlag.t list [@@deriving sexp]
let sizeof = 4
let marshal buf t =
Cstruct.BE.set_uint32 buf 0 (ClientFlag.to_int32 t)
let unmarshal buf =
ClientFlag.of_int32 (Cstruct.BE.get_uint32 buf 0)
end
: Option.t;
length: int32;
} [@@deriving sexp]
let = sizeof_t
let buf t =
set_t_magic buf Announcement.v2_magic;
set_t_ty buf (Option.to_int32 t.ty);
set_t_length buf t.length
let buf =
let open Rresult in
let magic = get_t_magic buf in
( if Announcement.v2_magic <> magic
then Error (Failure (Printf.sprintf "Bad reply magic: expected %Ld, got %Ld" Announcement.v2_magic magic))
else Ok () ) >>= fun () ->
let ty = Option.of_int32 (get_t_ty buf) in
let length = get_t_length buf in
Ok { ty; length }
end
module ExportName = struct
type t = string [@@deriving sexp]
let sizeof = String.length
let marshal buf x =
Cstruct.blit_from_string x 0 buf 0 (String.length x)
end
module DiskInfo = struct
type t = {
size: int64;
flags: PerExportFlag.t list
} [@@deriving sexp]
[%%cstruct
type t = {
size: uint64_t;
flags: uint16_t;
padding: uint8_t [@len 124];
} [@@big_endian]
]
let sizeof = sizeof_t
let unmarshal buf =
let size = get_t_size buf in
let flags = PerExportFlag.of_int32 (Int32.of_int (get_t_flags buf)) in
Ok { size; flags }
let marshal buf t =
set_t_size buf t.size;
set_t_flags buf (PerExportFlag.to_int t.flags)
end
: Option.t;
response_type: OptionResponse.t;
length: int32;
} [@@deriving sexp]
let t = Sexplib.Sexp.to_string (sexp_of_t t)
let = sizeof_t
let = 0x3e889045565a9L
let buf =
let open Rresult in
let magic = get_t_magic buf in
( if expected_magic <> magic
then Error (Failure (Printf.sprintf "Bad reply magic: expected %Ld, got %Ld" expected_magic magic))
else Ok () ) >>= fun () ->
let request_type = Option.of_int32 (get_t_request_type buf) in
let response_type = OptionResponse.of_int32 (get_t_response_type buf) in
let length = get_t_length buf in
Ok { request_type; response_type; length }
let buf t =
set_t_magic buf expected_magic;
set_t_request_type buf (Option.to_int32 t.request_type);
set_t_response_type buf (OptionResponse.to_int32 t.response_type);
set_t_length buf t.length
end
module Server = struct
type t = {
name: string;
} [@@deriving sexp]
[%%cstruct
type t = {
length: uint32_t;
} [@@big_endian]
]
let sizeof t = sizeof_t + (String.length t.name)
let unmarshal buf =
let length = Int32.to_int (get_t_length buf) in
let buf = Cstruct.shift buf sizeof_t in
let name = Cstruct.to_string (Cstruct.sub buf 0 length) in
Ok { name }
end
module Request = struct
type t = {
ty : Command.t;
handle : int64;
from : int64;
len : int32
} [@@deriving sexp]
let to_string t =
Printf.sprintf "{ Command = %s; handle = %Ld; from = %Ld; len = %ld }"
(Command.to_string t.ty) t.handle t.from t.len
[%%cstruct
type t = {
magic: uint32_t;
ty: uint32_t;
handle: uint64_t;
from: uint64_t;
len: uint32_t;
} [@@big_endian]
]
let unmarshal (buf: Cstruct.t) =
let open Rresult in
let magic = get_t_magic buf in
( if nbd_request_magic <> magic
then Error (Failure (Printf.sprintf "Bad request magic: expected %ld, got %ld" magic nbd_request_magic))
else Ok () ) >>= fun () ->
let ty = Command.of_int32 (get_t_ty buf) in
let handle = get_t_handle buf in
let from = get_t_from buf in
let len = get_t_len buf in
Ok { ty; handle; from; len }
let sizeof = sizeof_t
let marshal (buf: Cstruct.t) t =
set_t_magic buf nbd_request_magic;
set_t_ty buf (Command.to_int32 t.ty);
set_t_handle buf t.handle;
set_t_from buf t.from;
set_t_len buf t.len
end
module Reply = struct
type t = {
error : (unit, Error.t) result;
handle : int64;
} [@@deriving sexp]
let to_string t = Sexplib.Sexp.to_string (sexp_of_t t)
[%%cstruct
type t = {
magic: uint32_t;
error: uint32_t;
handle: uint64_t;
} [@@big_endian]
]
let unmarshal (buf: Cstruct.t) =
let open Rresult in
let magic = get_t_magic buf in
( if nbd_reply_magic <> magic
then Error (Failure (Printf.sprintf "Bad reply magic: expected %ld, got %ld" magic nbd_reply_magic))
else Ok () ) >>= fun () ->
let error = get_t_error buf in
let error = if error = 0l then Ok () else Error (Error.of_int32 error) in
let handle = get_t_handle buf in
Ok { error; handle }
let sizeof = sizeof_t
let marshal (buf: Cstruct.t) t =
set_t_magic buf nbd_reply_magic;
let error = match t.error with
| Ok () -> 0l
| Error e -> Error.to_int32 e in
set_t_error buf error;
set_t_handle buf t.handle
end