1*e0c4386eSCy Schubert# Copyright 2016 The OpenSSL Project Authors. All Rights Reserved. 2*e0c4386eSCy Schubert# 3*e0c4386eSCy Schubert# Licensed under the Apache License 2.0 (the "License"). You may not use 4*e0c4386eSCy Schubert# this file except in compliance with the License. You can obtain a copy 5*e0c4386eSCy Schubert# in the file LICENSE in the source distribution or at 6*e0c4386eSCy Schubert# https://www.openssl.org/source/license.html 7*e0c4386eSCy Schubert 8*e0c4386eSCy Schubertuse strict; 9*e0c4386eSCy Schubert 10*e0c4386eSCy Schubertpackage TLSProxy::Certificate; 11*e0c4386eSCy Schubert 12*e0c4386eSCy Schubertuse vars '@ISA'; 13*e0c4386eSCy Schubertpush @ISA, 'TLSProxy::Message'; 14*e0c4386eSCy Schubert 15*e0c4386eSCy Schubertsub new 16*e0c4386eSCy Schubert{ 17*e0c4386eSCy Schubert my $class = shift; 18*e0c4386eSCy Schubert my ($server, 19*e0c4386eSCy Schubert $data, 20*e0c4386eSCy Schubert $records, 21*e0c4386eSCy Schubert $startoffset, 22*e0c4386eSCy Schubert $message_frag_lens) = @_; 23*e0c4386eSCy Schubert 24*e0c4386eSCy Schubert my $self = $class->SUPER::new( 25*e0c4386eSCy Schubert $server, 26*e0c4386eSCy Schubert TLSProxy::Message::MT_CERTIFICATE, 27*e0c4386eSCy Schubert $data, 28*e0c4386eSCy Schubert $records, 29*e0c4386eSCy Schubert $startoffset, 30*e0c4386eSCy Schubert $message_frag_lens); 31*e0c4386eSCy Schubert 32*e0c4386eSCy Schubert $self->{first_certificate} = ""; 33*e0c4386eSCy Schubert $self->{extension_data} = ""; 34*e0c4386eSCy Schubert $self->{remaining_certdata} = ""; 35*e0c4386eSCy Schubert 36*e0c4386eSCy Schubert return $self; 37*e0c4386eSCy Schubert} 38*e0c4386eSCy Schubert 39*e0c4386eSCy Schubertsub parse 40*e0c4386eSCy Schubert{ 41*e0c4386eSCy Schubert my $self = shift; 42*e0c4386eSCy Schubert 43*e0c4386eSCy Schubert if (TLSProxy::Proxy->is_tls13()) { 44*e0c4386eSCy Schubert my $context_len = unpack('C', $self->data); 45*e0c4386eSCy Schubert my $context = substr($self->data, 1, $context_len); 46*e0c4386eSCy Schubert 47*e0c4386eSCy Schubert my $remdata = substr($self->data, 1 + $context_len); 48*e0c4386eSCy Schubert 49*e0c4386eSCy Schubert my ($hicertlistlen, $certlistlen) = unpack('Cn', $remdata); 50*e0c4386eSCy Schubert $certlistlen += ($hicertlistlen << 16); 51*e0c4386eSCy Schubert 52*e0c4386eSCy Schubert $remdata = substr($remdata, 3); 53*e0c4386eSCy Schubert 54*e0c4386eSCy Schubert die "Invalid Certificate List length" 55*e0c4386eSCy Schubert if length($remdata) != $certlistlen; 56*e0c4386eSCy Schubert 57*e0c4386eSCy Schubert my ($hicertlen, $certlen) = unpack('Cn', $remdata); 58*e0c4386eSCy Schubert $certlen += ($hicertlen << 16); 59*e0c4386eSCy Schubert 60*e0c4386eSCy Schubert die "Certificate too long" if ($certlen + 3) > $certlistlen; 61*e0c4386eSCy Schubert 62*e0c4386eSCy Schubert $remdata = substr($remdata, 3); 63*e0c4386eSCy Schubert 64*e0c4386eSCy Schubert my $certdata = substr($remdata, 0, $certlen); 65*e0c4386eSCy Schubert 66*e0c4386eSCy Schubert $remdata = substr($remdata, $certlen); 67*e0c4386eSCy Schubert 68*e0c4386eSCy Schubert my $extensions_len = unpack('n', $remdata); 69*e0c4386eSCy Schubert $remdata = substr($remdata, 2); 70*e0c4386eSCy Schubert 71*e0c4386eSCy Schubert die "Extensions too long" 72*e0c4386eSCy Schubert if ($certlen + 3 + $extensions_len + 2) > $certlistlen; 73*e0c4386eSCy Schubert 74*e0c4386eSCy Schubert my $extension_data = ""; 75*e0c4386eSCy Schubert if ($extensions_len != 0) { 76*e0c4386eSCy Schubert $extension_data = substr($remdata, 0, $extensions_len); 77*e0c4386eSCy Schubert 78*e0c4386eSCy Schubert if (length($extension_data) != $extensions_len) { 79*e0c4386eSCy Schubert die "Invalid extension length\n"; 80*e0c4386eSCy Schubert } 81*e0c4386eSCy Schubert } 82*e0c4386eSCy Schubert my %extensions = (); 83*e0c4386eSCy Schubert while (length($extension_data) >= 4) { 84*e0c4386eSCy Schubert my ($type, $size) = unpack("nn", $extension_data); 85*e0c4386eSCy Schubert my $extdata = substr($extension_data, 4, $size); 86*e0c4386eSCy Schubert $extension_data = substr($extension_data, 4 + $size); 87*e0c4386eSCy Schubert $extensions{$type} = $extdata; 88*e0c4386eSCy Schubert } 89*e0c4386eSCy Schubert $remdata = substr($remdata, $extensions_len); 90*e0c4386eSCy Schubert 91*e0c4386eSCy Schubert $self->context($context); 92*e0c4386eSCy Schubert $self->first_certificate($certdata); 93*e0c4386eSCy Schubert $self->extension_data(\%extensions); 94*e0c4386eSCy Schubert $self->remaining_certdata($remdata); 95*e0c4386eSCy Schubert 96*e0c4386eSCy Schubert print " Context:".$context."\n"; 97*e0c4386eSCy Schubert print " Certificate List Len:".$certlistlen."\n"; 98*e0c4386eSCy Schubert print " Certificate Len:".$certlen."\n"; 99*e0c4386eSCy Schubert print " Extensions Len:".$extensions_len."\n"; 100*e0c4386eSCy Schubert } else { 101*e0c4386eSCy Schubert my ($hicertlistlen, $certlistlen) = unpack('Cn', $self->data); 102*e0c4386eSCy Schubert $certlistlen += ($hicertlistlen << 16); 103*e0c4386eSCy Schubert 104*e0c4386eSCy Schubert my $remdata = substr($self->data, 3); 105*e0c4386eSCy Schubert 106*e0c4386eSCy Schubert die "Invalid Certificate List length" 107*e0c4386eSCy Schubert if length($remdata) != $certlistlen; 108*e0c4386eSCy Schubert 109*e0c4386eSCy Schubert my ($hicertlen, $certlen) = unpack('Cn', $remdata); 110*e0c4386eSCy Schubert $certlen += ($hicertlen << 16); 111*e0c4386eSCy Schubert 112*e0c4386eSCy Schubert die "Certificate too long" if ($certlen + 3) > $certlistlen; 113*e0c4386eSCy Schubert 114*e0c4386eSCy Schubert $remdata = substr($remdata, 3); 115*e0c4386eSCy Schubert 116*e0c4386eSCy Schubert my $certdata = substr($remdata, 0, $certlen); 117*e0c4386eSCy Schubert 118*e0c4386eSCy Schubert $remdata = substr($remdata, $certlen); 119*e0c4386eSCy Schubert 120*e0c4386eSCy Schubert $self->first_certificate($certdata); 121*e0c4386eSCy Schubert $self->remaining_certdata($remdata); 122*e0c4386eSCy Schubert 123*e0c4386eSCy Schubert print " Certificate List Len:".$certlistlen."\n"; 124*e0c4386eSCy Schubert print " Certificate Len:".$certlen."\n"; 125*e0c4386eSCy Schubert } 126*e0c4386eSCy Schubert} 127*e0c4386eSCy Schubert 128*e0c4386eSCy Schubert#Reconstruct the on-the-wire message data following changes 129*e0c4386eSCy Schubertsub set_message_contents 130*e0c4386eSCy Schubert{ 131*e0c4386eSCy Schubert my $self = shift; 132*e0c4386eSCy Schubert my $data; 133*e0c4386eSCy Schubert my $extensions = ""; 134*e0c4386eSCy Schubert 135*e0c4386eSCy Schubert if (TLSProxy::Proxy->is_tls13()) { 136*e0c4386eSCy Schubert foreach my $key (keys %{$self->extension_data}) { 137*e0c4386eSCy Schubert my $extdata = ${$self->extension_data}{$key}; 138*e0c4386eSCy Schubert $extensions .= pack("n", $key); 139*e0c4386eSCy Schubert $extensions .= pack("n", length($extdata)); 140*e0c4386eSCy Schubert $extensions .= $extdata; 141*e0c4386eSCy Schubert } 142*e0c4386eSCy Schubert $data = pack('C', length($self->context())); 143*e0c4386eSCy Schubert $data .= $self->context; 144*e0c4386eSCy Schubert my $certlen = length($self->first_certificate); 145*e0c4386eSCy Schubert my $certlistlen = $certlen + length($extensions) 146*e0c4386eSCy Schubert + length($self->remaining_certdata); 147*e0c4386eSCy Schubert my $hi = $certlistlen >> 16; 148*e0c4386eSCy Schubert $certlistlen = $certlistlen & 0xffff; 149*e0c4386eSCy Schubert $data .= pack('Cn', $hi, $certlistlen); 150*e0c4386eSCy Schubert $hi = $certlen >> 16; 151*e0c4386eSCy Schubert $certlen = $certlen & 0xffff; 152*e0c4386eSCy Schubert $data .= pack('Cn', $hi, $certlen); 153*e0c4386eSCy Schubert $data .= pack('n', length($extensions)); 154*e0c4386eSCy Schubert $data .= $extensions; 155*e0c4386eSCy Schubert $data .= $self->remaining_certdata(); 156*e0c4386eSCy Schubert $self->data($data); 157*e0c4386eSCy Schubert } else { 158*e0c4386eSCy Schubert my $certlen = length($self->first_certificate); 159*e0c4386eSCy Schubert my $certlistlen = $certlen + length($self->remaining_certdata); 160*e0c4386eSCy Schubert my $hi = $certlistlen >> 16; 161*e0c4386eSCy Schubert $certlistlen = $certlistlen & 0xffff; 162*e0c4386eSCy Schubert $data .= pack('Cn', $hi, $certlistlen); 163*e0c4386eSCy Schubert $hi = $certlen >> 16; 164*e0c4386eSCy Schubert $certlen = $certlen & 0xffff; 165*e0c4386eSCy Schubert $data .= pack('Cn', $hi, $certlen); 166*e0c4386eSCy Schubert $data .= $self->remaining_certdata(); 167*e0c4386eSCy Schubert $self->data($data); 168*e0c4386eSCy Schubert } 169*e0c4386eSCy Schubert} 170*e0c4386eSCy Schubert 171*e0c4386eSCy Schubert#Read/write accessors 172*e0c4386eSCy Schubertsub context 173*e0c4386eSCy Schubert{ 174*e0c4386eSCy Schubert my $self = shift; 175*e0c4386eSCy Schubert if (@_) { 176*e0c4386eSCy Schubert $self->{context} = shift; 177*e0c4386eSCy Schubert } 178*e0c4386eSCy Schubert return $self->{context}; 179*e0c4386eSCy Schubert} 180*e0c4386eSCy Schubertsub first_certificate 181*e0c4386eSCy Schubert{ 182*e0c4386eSCy Schubert my $self = shift; 183*e0c4386eSCy Schubert if (@_) { 184*e0c4386eSCy Schubert $self->{first_certificate} = shift; 185*e0c4386eSCy Schubert } 186*e0c4386eSCy Schubert return $self->{first_certificate}; 187*e0c4386eSCy Schubert} 188*e0c4386eSCy Schubertsub remaining_certdata 189*e0c4386eSCy Schubert{ 190*e0c4386eSCy Schubert my $self = shift; 191*e0c4386eSCy Schubert if (@_) { 192*e0c4386eSCy Schubert $self->{remaining_certdata} = shift; 193*e0c4386eSCy Schubert } 194*e0c4386eSCy Schubert return $self->{remaining_certdata}; 195*e0c4386eSCy Schubert} 196*e0c4386eSCy Schubertsub extension_data 197*e0c4386eSCy Schubert{ 198*e0c4386eSCy Schubert my $self = shift; 199*e0c4386eSCy Schubert if (@_) { 200*e0c4386eSCy Schubert $self->{extension_data} = shift; 201*e0c4386eSCy Schubert } 202*e0c4386eSCy Schubert return $self->{extension_data}; 203*e0c4386eSCy Schubert} 204*e0c4386eSCy Schubertsub set_extension 205*e0c4386eSCy Schubert{ 206*e0c4386eSCy Schubert my ($self, $ext_type, $ext_data) = @_; 207*e0c4386eSCy Schubert $self->{extension_data}{$ext_type} = $ext_data; 208*e0c4386eSCy Schubert} 209*e0c4386eSCy Schubertsub delete_extension 210*e0c4386eSCy Schubert{ 211*e0c4386eSCy Schubert my ($self, $ext_type) = @_; 212*e0c4386eSCy Schubert delete $self->{extension_data}{$ext_type}; 213*e0c4386eSCy Schubert} 214*e0c4386eSCy Schubert1; 215