mirror of
https://github.com/urbit/shrub.git
synced 2024-12-23 19:05:48 +03:00
211 lines
5.6 KiB
Plaintext
211 lines
5.6 KiB
Plaintext
/- asn1
|
|
:: |der: distinguished encoding rules for ASN.1
|
|
::
|
|
:: DER is a tag-length-value binary encoding for ASN.1, designed
|
|
:: so that there is only one (distinguished) valid encoding for an
|
|
:: instance of a type.
|
|
::
|
|
|%
|
|
:: +en:der: encode +spec:asn1 to +octs (kindof)
|
|
::
|
|
++ en
|
|
=< |= a=spec:asn1
|
|
^- [len=@ud dat=@ux]
|
|
=/ b ~(ren raw a)
|
|
[(lent b) (rep 3 b)]
|
|
|%
|
|
:: +raw:en:der: door for encoding +spec:asn1 to list of bytes
|
|
::
|
|
++ raw
|
|
|_ pec=spec:asn1
|
|
:: +ren:raw:en:der: render +spec:asn1 to tag-length-value bytes
|
|
::
|
|
++ ren
|
|
^- (list @D)
|
|
=/ a lem
|
|
[tag (weld (len a) a)]
|
|
:: +tag:raw:en:der: tag byte
|
|
::
|
|
++ tag
|
|
^- @D
|
|
?- pec
|
|
[%int *] 2
|
|
[%bit *] 3
|
|
[%oct *] 4
|
|
[%nul *] 5
|
|
[%obj *] 6
|
|
[%seq *] 48 :: constructed: (con 0x20 16)
|
|
[%set *] 49 :: constructed: (con 0x20 17)
|
|
[%con *] ;: con
|
|
0x80 :: context-specifc
|
|
?:(imp.bes.pec 0 0x20) :: implicit?
|
|
(dis 0x1f tag.bes.pec) :: 5 bits of custom tag
|
|
==
|
|
==
|
|
:: +lem:raw:en:der: element bytes
|
|
::
|
|
++ lem
|
|
^- (list @D)
|
|
?- pec
|
|
:: unsigned only, interpreted as positive-signed and
|
|
:: rendered in big-endian byte order. negative-signed would
|
|
:: be two's complement
|
|
::
|
|
[%int *] =/ a (flop (rip 3 int.pec))
|
|
?~ a [0 ~]
|
|
?:((lte i.a 127) a [0 a])
|
|
:: padded to byte-width, must be already byte-aligned
|
|
::
|
|
[%bit *] =/ a (rip 3 bit.pec)
|
|
=/ b ~| %der-invalid-bit
|
|
?. =(0 (mod len.pec 8))
|
|
~|(%der-invalid-bit-alignment !!)
|
|
(sub (div len.pec 8) (lent a))
|
|
[0 (weld a (reap b 0))]
|
|
:: padded to byte-width
|
|
::
|
|
[%oct *] =/ a (rip 3 oct.pec)
|
|
=/ b ~| %der-invalid-oct
|
|
(sub len.pec (lent a))
|
|
(weld a (reap b 0))
|
|
::
|
|
[%nul *] ~
|
|
[%obj *] (rip 3 obj.pec)
|
|
::
|
|
[%seq *] %- zing
|
|
|- ^- (list (list @))
|
|
?~ seq.pec ~
|
|
:- ren(pec i.seq.pec)
|
|
$(seq.pec t.seq.pec)
|
|
:: presumed to be already deduplicated and sorted
|
|
::
|
|
[%set *] %- zing
|
|
|- ^- (list (list @))
|
|
?~ set.pec ~
|
|
:- ren(pec i.set.pec)
|
|
$(set.pec t.set.pec)
|
|
:: already constructed
|
|
::
|
|
[%con *] con.pec
|
|
==
|
|
:: +len:raw:en:der: length bytes
|
|
::
|
|
++ len
|
|
|= a=(list @D)
|
|
^- (list @D)
|
|
=/ b (lent a)
|
|
?: (lte b 127)
|
|
[b ~] :: note: big-endian
|
|
[(con 0x80 (met 3 b)) (flop (rip 3 b))]
|
|
--
|
|
--
|
|
:: +de:der: decode atom to +spec:asn1
|
|
::
|
|
++ de
|
|
|= [len=@ud dat=@ux]
|
|
^- (unit spec:asn1)
|
|
:: XX refactor into +parse
|
|
=/ a (rip 3 dat)
|
|
=/ b ~| %der-invalid-len
|
|
(sub len (lent a))
|
|
(rust `(list @D)`(weld a (reap b 0)) parse)
|
|
:: +parse:der: DER parser combinator
|
|
::
|
|
++ parse
|
|
=< ^- $-(nail (like spec:asn1))
|
|
;~ pose
|
|
(stag %int (bass 256 (sear int ;~(pfix (tag 2) till))))
|
|
(stag %bit (sear bit (boss 256 ;~(pfix (tag 3) till))))
|
|
(stag %oct (boss 256 ;~(pfix (tag 4) till)))
|
|
(stag %nul (cold ~ ;~(plug (tag 5) (tag 0))))
|
|
(stag %obj (^boss 256 ;~(pfix (tag 6) till)))
|
|
(stag %seq (sear recur ;~(pfix (tag 48) till)))
|
|
(stag %set (sear recur ;~(pfix (tag 49) till)))
|
|
(stag %con ;~(plug (sear context next) till))
|
|
==
|
|
|%
|
|
:: +tag:parse:der: parse tag byte
|
|
::
|
|
++ tag
|
|
|=(a=@D (just a))
|
|
:: +int:parse:der: sear unsigned big-endian bytes
|
|
::
|
|
++ int
|
|
|= a=(list @D)
|
|
^- (unit (list @D))
|
|
?~ a ~
|
|
?: ?=([@ ~] a) `a
|
|
?. =(0 i.a) `a
|
|
?.((gth i.t.a 127) ~ `t.a)
|
|
:: +bit:parse:der: convert bytewidth to bitwidth
|
|
::
|
|
++ bit
|
|
|= [len=@ud dat=@ux]
|
|
^- (unit [len=@ud dat=@ux])
|
|
?. =(0 (end 3 dat)) ~
|
|
:+ ~
|
|
(mul 8 (dec len))
|
|
(rsh 3 dat)
|
|
:: +recur:parse:der: parse bytes for a list of +spec:asn1
|
|
::
|
|
++ recur
|
|
|=(a=(list @) (rust a (star parse)))
|
|
:: +context:parse:der: decode context-specific tag byte
|
|
::
|
|
++ context
|
|
|= a=@D
|
|
^- (unit bespoke:asn1)
|
|
?. =(1 (cut 0 [7 1] a)) ~
|
|
:+ ~
|
|
=(1 (cut 0 [5 1] a))
|
|
(dis 0x1f a)
|
|
:: +boss:parse:der: shadowed to count as well
|
|
::
|
|
:: Use for parsing +octs more broadly?
|
|
::
|
|
++ boss
|
|
|* [wuc=@ tyd=rule]
|
|
%+ cook
|
|
|= waq=(list @)
|
|
:- (lent waq)
|
|
(reel waq |=([p=@ q=@] (add p (mul wuc q))))
|
|
tyd
|
|
:: +till:parse:der: parser combinator for len-prefixed bytes
|
|
::
|
|
:: advance until
|
|
::
|
|
++ till
|
|
|= tub=nail
|
|
^- (like (list @D))
|
|
?~ q.tub
|
|
(fail tub)
|
|
:: fuz: first byte - length, or length of the length
|
|
::
|
|
=* fuz i.q.tub
|
|
:: nex: offset of value bytes from fuz
|
|
:: len: length of value bytes
|
|
::
|
|
=/ [nex=@ len=@]
|
|
:: faz: meaningful bits in fuz
|
|
::
|
|
=/ faz (end [0 7] fuz)
|
|
?: =(0 (cut 0 [7 1] fuz))
|
|
[0 faz]
|
|
[faz (rep 3 (flop (scag faz t.q.tub)))]
|
|
?: ?& !=(0 nex)
|
|
!=(nex (met 3 len))
|
|
==
|
|
(fail tub)
|
|
:: zuf: value bytes
|
|
::
|
|
=/ zuf (swag [nex len] t.q.tub)
|
|
?. =(len (lent zuf))
|
|
(fail tub)
|
|
:: zaf: product nail
|
|
::
|
|
=/ zaf [p.p.tub (add +(nex) q.p.tub)]
|
|
[zaf `[zuf zaf (slag (add nex len) t.q.tub)]]
|
|
--
|
|
--
|
|
|