author | haftmann |
Mon, 01 Mar 2010 13:40:23 +0100 | |
changeset 35416 | d8d7d1b785af |
parent 32960 | 69916a850301 |
child 35702 | fb7a386a15cb |
permissions | -rw-r--r-- |
3474 | 1 |
(* Title: HOL/Auth/TLS |
2 |
ID: $Id$ |
|
3 |
Author: Lawrence C Paulson, Cambridge University Computer Laboratory |
|
4 |
Copyright 1997 University of Cambridge |
|
5 |
||
3759
3d1ac6b82b28
Fixed ServerResume to check for ServerHello instead of making a new NB
paulson
parents:
3757
diff
changeset
|
6 |
Inductive relation "tls" for the TLS (Transport Layer Security) protocol. |
3672
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
7 |
This protocol is essentially the same as SSL 3.0. |
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
8 |
|
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
9 |
Abstracted from "The TLS Protocol, Version 1.0" by Tim Dierks and Christopher |
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
10 |
Allen, Transport Layer Security Working Group, 21 May 1997, |
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
11 |
INTERNET-DRAFT draft-ietf-tls-protocol-03.txt. Section numbers below refer |
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
12 |
to that memo. |
3474 | 13 |
|
14 |
An RSA cryptosystem is assumed, and X.509v3 certificates are abstracted down |
|
15 |
to the trivial form {A, publicKey(A)}privateKey(Server), where Server is a |
|
16 |
global signing authority. |
|
17 |
||
18 |
A is the client and B is the server, not to be confused with the constant |
|
19 |
Server, who is in charge of all public keys. |
|
20 |
||
3480
d59bbf053258
More realistic model: the Spy can compute clientK and serverK
paulson
parents:
3474
diff
changeset
|
21 |
The model assumes that no fraudulent certificates are present, but it does |
3519
ab0a9fbed4c0
Changing "lost" from a parameter of protocol definitions to a constant.
paulson
parents:
3515
diff
changeset
|
22 |
assume that some private keys are to the spy. |
3474 | 23 |
|
3745
4c5d3b1ddc75
Client, Server certificates now sent using the separate Certificate rule,
paulson
parents:
3729
diff
changeset
|
24 |
REMARK. The event "Notes A {|Agent B, Nonce PMS|}" appears in ClientKeyExch, |
3515
d8a71f6eaf40
Now uses the Notes constructor to distinguish the Client (who has chosen M)
paulson
parents:
3506
diff
changeset
|
25 |
CertVerify, ClientFinished to record that A knows M. It is a note from A to |
3745
4c5d3b1ddc75
Client, Server certificates now sent using the separate Certificate rule,
paulson
parents:
3729
diff
changeset
|
26 |
herself. Nobody else can see it. In ClientKeyExch, the Spy can substitute |
3515
d8a71f6eaf40
Now uses the Notes constructor to distinguish the Client (who has chosen M)
paulson
parents:
3506
diff
changeset
|
27 |
his own certificate for A's, but he cannot replace A's note by one for himself. |
d8a71f6eaf40
Now uses the Notes constructor to distinguish the Client (who has chosen M)
paulson
parents:
3506
diff
changeset
|
28 |
|
3672
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
29 |
The Note event avoids a weakness in the public-key model. Each |
3515
d8a71f6eaf40
Now uses the Notes constructor to distinguish the Client (who has chosen M)
paulson
parents:
3506
diff
changeset
|
30 |
agent's state is recorded as the trace of messages. When the true client (A) |
3672
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
31 |
invents PMS, he encrypts PMS with B's public key before sending it. The model |
3515
d8a71f6eaf40
Now uses the Notes constructor to distinguish the Client (who has chosen M)
paulson
parents:
3506
diff
changeset
|
32 |
does not distinguish the original occurrence of such a message from a replay. |
d8a71f6eaf40
Now uses the Notes constructor to distinguish the Client (who has chosen M)
paulson
parents:
3506
diff
changeset
|
33 |
In the shared-key model, the ability to encrypt implies the ability to |
d8a71f6eaf40
Now uses the Notes constructor to distinguish the Client (who has chosen M)
paulson
parents:
3506
diff
changeset
|
34 |
decrypt, so the problem does not arise. |
3685
5b8c0c8f576e
Full version of TLS including session resumption, but no Oops
paulson
parents:
3683
diff
changeset
|
35 |
|
3745
4c5d3b1ddc75
Client, Server certificates now sent using the separate Certificate rule,
paulson
parents:
3729
diff
changeset
|
36 |
Proofs would be simpler if ClientKeyExch included A's name within |
3685
5b8c0c8f576e
Full version of TLS including session resumption, but no Oops
paulson
parents:
3683
diff
changeset
|
37 |
Crypt KB (Nonce PMS). As things stand, there is much overlap between proofs |
5b8c0c8f576e
Full version of TLS including session resumption, but no Oops
paulson
parents:
3683
diff
changeset
|
38 |
about that message (which B receives) and the stronger event |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
39 |
Notes A {|Agent B, Nonce PMS|}. |
3474 | 40 |
*) |
41 |
||
13956 | 42 |
header{*The TLS Protocol: Transport Layer Security*} |
43 |
||
29888 | 44 |
theory TLS imports Public Nat_Int_Bij begin |
3474 | 45 |
|
35416
d8d7d1b785af
replaced a couple of constsdefs by definitions (also some old primrecs by modern ones)
haftmann
parents:
32960
diff
changeset
|
46 |
definition certificate :: "[agent,key] => msg" where |
13922 | 47 |
"certificate A KA == Crypt (priSK Server) {|Agent A, Key KA|}" |
48 |
||
49 |
text{*TLS apparently does not require separate keypairs for encryption and |
|
50 |
signature. Therefore, we formalize signature as encryption using the |
|
51 |
private encryption key.*} |
|
5653
204083e3f368
changed tags from 0, 1 to None, Some() to avoid special treatment of 0
paulson
parents:
5434
diff
changeset
|
52 |
|
6284
147db42c1009
tidying in conjuntion with the TISSEC paper; replaced (unit option)
paulson
parents:
5653
diff
changeset
|
53 |
datatype role = ClientRole | ServerRole |
147db42c1009
tidying in conjuntion with the TISSEC paper; replaced (unit option)
paulson
parents:
5653
diff
changeset
|
54 |
|
3474 | 55 |
consts |
3672
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
56 |
(*Pseudo-random function of Section 5*) |
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
57 |
PRF :: "nat*nat*nat => nat" |
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
58 |
|
3704 | 59 |
(*Client, server write keys are generated uniformly by function sessionK |
5653
204083e3f368
changed tags from 0, 1 to None, Some() to avoid special treatment of 0
paulson
parents:
5434
diff
changeset
|
60 |
to avoid duplicating their properties. They are distinguished by a |
204083e3f368
changed tags from 0, 1 to None, Some() to avoid special treatment of 0
paulson
parents:
5434
diff
changeset
|
61 |
tag (not a bool, to avoid the peculiarities of if-and-only-if). |
3704 | 62 |
Session keys implicitly include MAC secrets.*) |
6284
147db42c1009
tidying in conjuntion with the TISSEC paper; replaced (unit option)
paulson
parents:
5653
diff
changeset
|
63 |
sessionK :: "(nat*nat*nat) * role => key" |
3474 | 64 |
|
20768 | 65 |
abbreviation |
21404
eb85850d3eb7
more robust syntax for definition/abbreviation/notation;
wenzelm
parents:
20768
diff
changeset
|
66 |
clientK :: "nat*nat*nat => key" where |
20768 | 67 |
"clientK X == sessionK(X, ClientRole)" |
3677
f2569416d18b
Now with the sessionK constant and new events ClientAccepts and ServerAccepts
paulson
parents:
3676
diff
changeset
|
68 |
|
21404
eb85850d3eb7
more robust syntax for definition/abbreviation/notation;
wenzelm
parents:
20768
diff
changeset
|
69 |
abbreviation |
eb85850d3eb7
more robust syntax for definition/abbreviation/notation;
wenzelm
parents:
20768
diff
changeset
|
70 |
serverK :: "nat*nat*nat => key" where |
20768 | 71 |
"serverK X == sessionK(X, ServerRole)" |
72 |
||
3677
f2569416d18b
Now with the sessionK constant and new events ClientAccepts and ServerAccepts
paulson
parents:
3676
diff
changeset
|
73 |
|
14126
28824746d046
Tidying and replacement of some axioms by specifications
paulson
parents:
13956
diff
changeset
|
74 |
specification (PRF) |
28824746d046
Tidying and replacement of some axioms by specifications
paulson
parents:
13956
diff
changeset
|
75 |
inj_PRF: "inj PRF" |
13922 | 76 |
--{*the pseudo-random function is collision-free*} |
14126
28824746d046
Tidying and replacement of some axioms by specifications
paulson
parents:
13956
diff
changeset
|
77 |
apply (rule exI [of _ "%(x,y,z). nat2_to_nat(x, nat2_to_nat(y,z))"]) |
28824746d046
Tidying and replacement of some axioms by specifications
paulson
parents:
13956
diff
changeset
|
78 |
apply (simp add: inj_on_def) |
28824746d046
Tidying and replacement of some axioms by specifications
paulson
parents:
13956
diff
changeset
|
79 |
apply (blast dest!: nat2_to_nat_inj [THEN injD]) |
28824746d046
Tidying and replacement of some axioms by specifications
paulson
parents:
13956
diff
changeset
|
80 |
done |
3672
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
81 |
|
14126
28824746d046
Tidying and replacement of some axioms by specifications
paulson
parents:
13956
diff
changeset
|
82 |
specification (sessionK) |
28824746d046
Tidying and replacement of some axioms by specifications
paulson
parents:
13956
diff
changeset
|
83 |
inj_sessionK: "inj sessionK" |
13922 | 84 |
--{*sessionK is collision-free; also, no clientK clashes with any serverK.*} |
14126
28824746d046
Tidying and replacement of some axioms by specifications
paulson
parents:
13956
diff
changeset
|
85 |
apply (rule exI [of _ |
28824746d046
Tidying and replacement of some axioms by specifications
paulson
parents:
13956
diff
changeset
|
86 |
"%((x,y,z), r). nat2_to_nat(role_case 0 1 r, |
28824746d046
Tidying and replacement of some axioms by specifications
paulson
parents:
13956
diff
changeset
|
87 |
nat2_to_nat(x, nat2_to_nat(y,z)))"]) |
28824746d046
Tidying and replacement of some axioms by specifications
paulson
parents:
13956
diff
changeset
|
88 |
apply (simp add: inj_on_def split: role.split) |
28824746d046
Tidying and replacement of some axioms by specifications
paulson
parents:
13956
diff
changeset
|
89 |
apply (blast dest!: nat2_to_nat_inj [THEN injD]) |
28824746d046
Tidying and replacement of some axioms by specifications
paulson
parents:
13956
diff
changeset
|
90 |
done |
3677
f2569416d18b
Now with the sessionK constant and new events ClientAccepts and ServerAccepts
paulson
parents:
3676
diff
changeset
|
91 |
|
14126
28824746d046
Tidying and replacement of some axioms by specifications
paulson
parents:
13956
diff
changeset
|
92 |
axioms |
13922 | 93 |
--{*sessionK makes symmetric keys*} |
11287 | 94 |
isSym_sessionK: "sessionK nonces \<in> symKeys" |
3474 | 95 |
|
13922 | 96 |
--{*sessionK never clashes with a long-term symmetric key |
97 |
(they don't exist in TLS anyway)*} |
|
98 |
sessionK_neq_shrK [iff]: "sessionK nonces \<noteq> shrK A" |
|
99 |
||
3474 | 100 |
|
23746 | 101 |
inductive_set tls :: "event list set" |
102 |
where |
|
13922 | 103 |
Nil: --{*The initial, empty trace*} |
11287 | 104 |
"[] \<in> tls" |
3474 | 105 |
|
23746 | 106 |
| Fake: --{*The Spy may say anything he can say. The sender field is correct, |
13922 | 107 |
but agents don't use that information.*} |
11287 | 108 |
"[| evsf \<in> tls; X \<in> synth (analz (spies evsf)) |] |
109 |
==> Says Spy B X # evsf \<in> tls" |
|
3480
d59bbf053258
More realistic model: the Spy can compute clientK and serverK
paulson
parents:
3474
diff
changeset
|
110 |
|
23746 | 111 |
| SpyKeys: --{*The spy may apply @{term PRF} and @{term sessionK} |
13956 | 112 |
to available nonces*} |
11287 | 113 |
"[| evsSK \<in> tls; |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
114 |
{Nonce NA, Nonce NB, Nonce M} <= analz (spies evsSK) |] |
4421
88639289be39
Simplified SpyKeys and ClientKeyExch as suggested by James Margetson
paulson
parents:
4198
diff
changeset
|
115 |
==> Notes Spy {| Nonce (PRF(M,NA,NB)), |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
116 |
Key (sessionK((NA,NB,M),role)) |} # evsSK \<in> tls" |
3474 | 117 |
|
23746 | 118 |
| ClientHello: |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
119 |
--{*(7.4.1.2) |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
120 |
PA represents @{text CLIENT_VERSION}, @{text CIPHER_SUITES} and @{text COMPRESSION_METHODS}. |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
121 |
It is uninterpreted but will be confirmed in the FINISHED messages. |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
122 |
NA is CLIENT RANDOM, while SID is @{text SESSION_ID}. |
3676
cbaec955056b
Addition of SessionIDs to the Hello and Finished messages
paulson
parents:
3672
diff
changeset
|
123 |
UNIX TIME is omitted because the protocol doesn't use it. |
13956 | 124 |
May assume @{term "NA \<notin> range PRF"} because CLIENT RANDOM is |
125 |
28 bytes while MASTER SECRET is 48 bytes*} |
|
11287 | 126 |
"[| evsCH \<in> tls; Nonce NA \<notin> used evsCH; NA \<notin> range PRF |] |
3729
6be7cf5086ab
Renamed XA, XB to PA, PB and removed the certificate from Client Verify
paulson
parents:
3710
diff
changeset
|
127 |
==> Says A B {|Agent A, Nonce NA, Number SID, Number PA|} |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
128 |
# evsCH \<in> tls" |
3474 | 129 |
|
23746 | 130 |
| ServerHello: |
13922 | 131 |
--{*7.4.1.3 of the TLS Internet-Draft |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
132 |
PB represents @{text CLIENT_VERSION}, @{text CIPHER_SUITE} and @{text COMPRESSION_METHOD}. |
3672
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
133 |
SERVER CERTIFICATE (7.4.2) is always present. |
13956 | 134 |
@{text CERTIFICATE_REQUEST} (7.4.4) is implied.*} |
11287 | 135 |
"[| evsSH \<in> tls; Nonce NB \<notin> used evsSH; NB \<notin> range PRF; |
3729
6be7cf5086ab
Renamed XA, XB to PA, PB and removed the certificate from Client Verify
paulson
parents:
3710
diff
changeset
|
136 |
Says A' B {|Agent A, Nonce NA, Number SID, Number PA|} |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
137 |
\<in> set evsSH |] |
11287 | 138 |
==> Says B A {|Nonce NB, Number SID, Number PB|} # evsSH \<in> tls" |
3474 | 139 |
|
23746 | 140 |
| Certificate: |
13922 | 141 |
--{*SERVER (7.4.2) or CLIENT (7.4.6) CERTIFICATE.*} |
11287 | 142 |
"evsC \<in> tls ==> Says B A (certificate B (pubK B)) # evsC \<in> tls" |
3745
4c5d3b1ddc75
Client, Server certificates now sent using the separate Certificate rule,
paulson
parents:
3729
diff
changeset
|
143 |
|
23746 | 144 |
| ClientKeyExch: |
13922 | 145 |
--{*CLIENT KEY EXCHANGE (7.4.7). |
3672
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
146 |
The client, A, chooses PMS, the PREMASTER SECRET. |
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
147 |
She encrypts PMS using the supplied KB, which ought to be pubK B. |
13956 | 148 |
We assume @{term "PMS \<notin> range PRF"} because a clash betweem the PMS |
3672
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
149 |
and another MASTER SECRET is highly unlikely (even though |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
150 |
both items have the same length, 48 bytes). |
3672
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
151 |
The Note event records in the trace that she knows PMS |
13922 | 152 |
(see REMARK at top). *} |
11287 | 153 |
"[| evsCX \<in> tls; Nonce PMS \<notin> used evsCX; PMS \<notin> range PRF; |
154 |
Says B' A (certificate B KB) \<in> set evsCX |] |
|
3745
4c5d3b1ddc75
Client, Server certificates now sent using the separate Certificate rule,
paulson
parents:
3729
diff
changeset
|
155 |
==> Says A B (Crypt KB (Nonce PMS)) |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
156 |
# Notes A {|Agent B, Nonce PMS|} |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
157 |
# evsCX \<in> tls" |
3474 | 158 |
|
23746 | 159 |
| CertVerify: |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
160 |
--{*The optional Certificate Verify (7.4.8) message contains the |
3672
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
161 |
specific components listed in the security analysis, F.1.1.2. |
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
162 |
It adds the pre-master-secret, which is also essential! |
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
163 |
Checking the signature, which is the only use of A's certificate, |
13922 | 164 |
assures B of A's presence*} |
11287 | 165 |
"[| evsCV \<in> tls; |
166 |
Says B' A {|Nonce NB, Number SID, Number PB|} \<in> set evsCV; |
|
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
167 |
Notes A {|Agent B, Nonce PMS|} \<in> set evsCV |] |
3729
6be7cf5086ab
Renamed XA, XB to PA, PB and removed the certificate from Client Verify
paulson
parents:
3710
diff
changeset
|
168 |
==> Says A B (Crypt (priK A) (Hash{|Nonce NB, Agent B, Nonce PMS|})) |
11287 | 169 |
# evsCV \<in> tls" |
3474 | 170 |
|
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
171 |
--{*Finally come the FINISHED messages (7.4.8), confirming PA and PB |
3672
56e4365a0c99
TLS now with a distinction between premaster secret and master secret
paulson
parents:
3519
diff
changeset
|
172 |
among other things. The master-secret is PRF(PMS,NA,NB). |
13922 | 173 |
Either party may send its message first.*} |
3474 | 174 |
|
23746 | 175 |
| ClientFinished: |
13922 | 176 |
--{*The occurrence of Notes A {|Agent B, Nonce PMS|} stops the |
3515
d8a71f6eaf40
Now uses the Notes constructor to distinguish the Client (who has chosen M)
paulson
parents:
3506
diff
changeset
|
177 |
rule's applying when the Spy has satisfied the "Says A B" by |
d8a71f6eaf40
Now uses the Notes constructor to distinguish the Client (who has chosen M)
paulson
parents:
3506
diff
changeset
|
178 |
repaying messages sent by the true client; in that case, the |
6284
147db42c1009
tidying in conjuntion with the TISSEC paper; replaced (unit option)
paulson
parents:
5653
diff
changeset
|
179 |
Spy does not know PMS and could not send ClientFinished. One |
13956 | 180 |
could simply put @{term "A\<noteq>Spy"} into the rule, but one should not |
13922 | 181 |
expect the spy to be well-behaved.*} |
11287 | 182 |
"[| evsCF \<in> tls; |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
183 |
Says A B {|Agent A, Nonce NA, Number SID, Number PA|} |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
184 |
\<in> set evsCF; |
11287 | 185 |
Says B' A {|Nonce NB, Number SID, Number PB|} \<in> set evsCF; |
186 |
Notes A {|Agent B, Nonce PMS|} \<in> set evsCF; |
|
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
187 |
M = PRF(PMS,NA,NB) |] |
3474 | 188 |
==> Says A B (Crypt (clientK(NA,NB,M)) |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
189 |
(Hash{|Number SID, Nonce M, |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
190 |
Nonce NA, Number PA, Agent A, |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
191 |
Nonce NB, Number PB, Agent B|})) |
11287 | 192 |
# evsCF \<in> tls" |
3474 | 193 |
|
23746 | 194 |
| ServerFinished: |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
195 |
--{*Keeping A' and A'' distinct means B cannot even check that the |
13922 | 196 |
two messages originate from the same source. *} |
11287 | 197 |
"[| evsSF \<in> tls; |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
198 |
Says A' B {|Agent A, Nonce NA, Number SID, Number PA|} |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
199 |
\<in> set evsSF; |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
200 |
Says B A {|Nonce NB, Number SID, Number PB|} \<in> set evsSF; |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
201 |
Says A'' B (Crypt (pubK B) (Nonce PMS)) \<in> set evsSF; |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
202 |
M = PRF(PMS,NA,NB) |] |
3474 | 203 |
==> Says B A (Crypt (serverK(NA,NB,M)) |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
204 |
(Hash{|Number SID, Nonce M, |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
205 |
Nonce NA, Number PA, Agent A, |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
206 |
Nonce NB, Number PB, Agent B|})) |
11287 | 207 |
# evsSF \<in> tls" |
3474 | 208 |
|
23746 | 209 |
| ClientAccepts: |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
210 |
--{*Having transmitted ClientFinished and received an identical |
3677
f2569416d18b
Now with the sessionK constant and new events ClientAccepts and ServerAccepts
paulson
parents:
3676
diff
changeset
|
211 |
message encrypted with serverK, the client stores the parameters |
3687
fb7d096d7884
Simplified SpyKeys to use sessionK instead of clientK and serverK
paulson
parents:
3686
diff
changeset
|
212 |
needed to resume this session. The "Notes A ..." premise is |
13956 | 213 |
used to prove @{text Notes_master_imp_Crypt_PMS}.*} |
11287 | 214 |
"[| evsCA \<in> tls; |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
215 |
Notes A {|Agent B, Nonce PMS|} \<in> set evsCA; |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
216 |
M = PRF(PMS,NA,NB); |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
217 |
X = Hash{|Number SID, Nonce M, |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
218 |
Nonce NA, Number PA, Agent A, |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
219 |
Nonce NB, Number PB, Agent B|}; |
11287 | 220 |
Says A B (Crypt (clientK(NA,NB,M)) X) \<in> set evsCA; |
221 |
Says B' A (Crypt (serverK(NA,NB,M)) X) \<in> set evsCA |] |
|
222 |
==> |
|
223 |
Notes A {|Number SID, Agent A, Agent B, Nonce M|} # evsCA \<in> tls" |
|
3677
f2569416d18b
Now with the sessionK constant and new events ClientAccepts and ServerAccepts
paulson
parents:
3676
diff
changeset
|
224 |
|
23746 | 225 |
| ServerAccepts: |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
226 |
--{*Having transmitted ServerFinished and received an identical |
3677
f2569416d18b
Now with the sessionK constant and new events ClientAccepts and ServerAccepts
paulson
parents:
3676
diff
changeset
|
227 |
message encrypted with clientK, the server stores the parameters |
3687
fb7d096d7884
Simplified SpyKeys to use sessionK instead of clientK and serverK
paulson
parents:
3686
diff
changeset
|
228 |
needed to resume this session. The "Says A'' B ..." premise is |
13956 | 229 |
used to prove @{text Notes_master_imp_Crypt_PMS}.*} |
11287 | 230 |
"[| evsSA \<in> tls; |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
231 |
A \<noteq> B; |
11287 | 232 |
Says A'' B (Crypt (pubK B) (Nonce PMS)) \<in> set evsSA; |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
233 |
M = PRF(PMS,NA,NB); |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
234 |
X = Hash{|Number SID, Nonce M, |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
235 |
Nonce NA, Number PA, Agent A, |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
236 |
Nonce NB, Number PB, Agent B|}; |
11287 | 237 |
Says B A (Crypt (serverK(NA,NB,M)) X) \<in> set evsSA; |
238 |
Says A' B (Crypt (clientK(NA,NB,M)) X) \<in> set evsSA |] |
|
239 |
==> |
|
240 |
Notes B {|Number SID, Agent A, Agent B, Nonce M|} # evsSA \<in> tls" |
|
3677
f2569416d18b
Now with the sessionK constant and new events ClientAccepts and ServerAccepts
paulson
parents:
3676
diff
changeset
|
241 |
|
23746 | 242 |
| ClientResume: |
13956 | 243 |
--{*If A recalls the @{text SESSION_ID}, then she sends a FINISHED |
244 |
message using the new nonces and stored MASTER SECRET.*} |
|
11287 | 245 |
"[| evsCR \<in> tls; |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
246 |
Says A B {|Agent A, Nonce NA, Number SID, Number PA|}: set evsCR; |
11287 | 247 |
Says B' A {|Nonce NB, Number SID, Number PB|} \<in> set evsCR; |
248 |
Notes A {|Number SID, Agent A, Agent B, Nonce M|} \<in> set evsCR |] |
|
3685
5b8c0c8f576e
Full version of TLS including session resumption, but no Oops
paulson
parents:
3683
diff
changeset
|
249 |
==> Says A B (Crypt (clientK(NA,NB,M)) |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
250 |
(Hash{|Number SID, Nonce M, |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
251 |
Nonce NA, Number PA, Agent A, |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
252 |
Nonce NB, Number PB, Agent B|})) |
11287 | 253 |
# evsCR \<in> tls" |
3685
5b8c0c8f576e
Full version of TLS including session resumption, but no Oops
paulson
parents:
3683
diff
changeset
|
254 |
|
23746 | 255 |
| ServerResume: |
13956 | 256 |
--{*Resumption (7.3): If B finds the @{text SESSION_ID} then he can |
257 |
send a FINISHED message using the recovered MASTER SECRET*} |
|
11287 | 258 |
"[| evsSR \<in> tls; |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
259 |
Says A' B {|Agent A, Nonce NA, Number SID, Number PA|}: set evsSR; |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
260 |
Says B A {|Nonce NB, Number SID, Number PB|} \<in> set evsSR; |
11287 | 261 |
Notes B {|Number SID, Agent A, Agent B, Nonce M|} \<in> set evsSR |] |
3759
3d1ac6b82b28
Fixed ServerResume to check for ServerHello instead of making a new NB
paulson
parents:
3757
diff
changeset
|
262 |
==> Says B A (Crypt (serverK(NA,NB,M)) |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
263 |
(Hash{|Number SID, Nonce M, |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
264 |
Nonce NA, Number PA, Agent A, |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
265 |
Nonce NB, Number PB, Agent B|})) # evsSR |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
266 |
\<in> tls" |
3759
3d1ac6b82b28
Fixed ServerResume to check for ServerHello instead of making a new NB
paulson
parents:
3757
diff
changeset
|
267 |
|
23746 | 268 |
| Oops: |
13922 | 269 |
--{*The most plausible compromise is of an old session key. Losing |
3686
4b484805b4c4
First working version with Oops event for session keys
paulson
parents:
3685
diff
changeset
|
270 |
the MASTER SECRET or PREMASTER SECRET is more serious but |
13956 | 271 |
rather unlikely. The assumption @{term "A\<noteq>Spy"} is essential: |
272 |
otherwise the Spy could learn session keys merely by |
|
273 |
replaying messages!*} |
|
11287 | 274 |
"[| evso \<in> tls; A \<noteq> Spy; |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
275 |
Says A B (Crypt (sessionK((NA,NB,M),role)) X) \<in> set evso |] |
11287 | 276 |
==> Says A Spy (Key (sessionK((NA,NB,M),role))) # evso \<in> tls" |
277 |
||
278 |
(* |
|
279 |
Protocol goals: |
|
280 |
* M, serverK(NA,NB,M) and clientK(NA,NB,M) will be known only to the two |
|
281 |
parties (though A is not necessarily authenticated). |
|
282 |
||
283 |
* B upon receiving CertVerify knows that A is present (But this |
|
284 |
message is optional!) |
|
285 |
||
286 |
* A upon receiving ServerFinished knows that B is present |
|
287 |
||
288 |
* Each party who has received a FINISHED message can trust that the other |
|
289 |
party agrees on all message components, including PA and PB (thus foiling |
|
290 |
rollback attacks). |
|
291 |
*) |
|
292 |
||
293 |
declare Says_imp_knows_Spy [THEN analz.Inj, dest] |
|
294 |
declare parts.Body [dest] |
|
295 |
declare analz_into_parts [dest] |
|
296 |
declare Fake_parts_insert_in_Un [dest] |
|
297 |
||
298 |
||
13922 | 299 |
text{*Automatically unfold the definition of "certificate"*} |
11287 | 300 |
declare certificate_def [simp] |
301 |
||
13922 | 302 |
text{*Injectiveness of key-generating functions*} |
11287 | 303 |
declare inj_PRF [THEN inj_eq, iff] |
304 |
declare inj_sessionK [THEN inj_eq, iff] |
|
305 |
declare isSym_sessionK [simp] |
|
306 |
||
307 |
||
308 |
(*** clientK and serverK make symmetric keys; no clashes with pubK or priK ***) |
|
309 |
||
13922 | 310 |
lemma pubK_neq_sessionK [iff]: "publicKey b A \<noteq> sessionK arg" |
11287 | 311 |
by (simp add: symKeys_neq_imp_neq) |
312 |
||
313 |
declare pubK_neq_sessionK [THEN not_sym, iff] |
|
314 |
||
13922 | 315 |
lemma priK_neq_sessionK [iff]: "invKey (publicKey b A) \<noteq> sessionK arg" |
11287 | 316 |
by (simp add: symKeys_neq_imp_neq) |
317 |
||
318 |
declare priK_neq_sessionK [THEN not_sym, iff] |
|
319 |
||
320 |
lemmas keys_distinct = pubK_neq_sessionK priK_neq_sessionK |
|
321 |
||
322 |
||
13922 | 323 |
subsection{*Protocol Proofs*} |
11287 | 324 |
|
13922 | 325 |
text{*Possibility properties state that some traces run the protocol to the |
326 |
end. Four paths and 12 rules are considered.*} |
|
11287 | 327 |
|
328 |
||
329 |
(** These proofs assume that the Nonce_supply nonces |
|
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
330 |
(which have the form @ N. Nonce N \<notin> used evs) |
11287 | 331 |
lie outside the range of PRF. It seems reasonable, but as it is needed |
332 |
only for the possibility theorems, it is not taken as an axiom. |
|
333 |
**) |
|
334 |
||
335 |
||
13922 | 336 |
text{*Possibility property ending with ClientAccepts.*} |
11287 | 337 |
lemma "[| \<forall>evs. (@ N. Nonce N \<notin> used evs) \<notin> range PRF; A \<noteq> B |] |
338 |
==> \<exists>SID M. \<exists>evs \<in> tls. |
|
339 |
Notes A {|Number SID, Agent A, Agent B, Nonce M|} \<in> set evs" |
|
340 |
apply (intro exI bexI) |
|
341 |
apply (rule_tac [2] tls.Nil |
|
342 |
[THEN tls.ClientHello, THEN tls.ServerHello, |
|
343 |
THEN tls.Certificate, THEN tls.ClientKeyExch, |
|
344 |
THEN tls.ClientFinished, THEN tls.ServerFinished, |
|
13507 | 345 |
THEN tls.ClientAccepts], possibility, blast+) |
11287 | 346 |
done |
347 |
||
348 |
||
13922 | 349 |
text{*And one for ServerAccepts. Either FINISHED message may come first.*} |
11287 | 350 |
lemma "[| \<forall>evs. (@ N. Nonce N \<notin> used evs) \<notin> range PRF; A \<noteq> B |] |
351 |
==> \<exists>SID NA PA NB PB M. \<exists>evs \<in> tls. |
|
352 |
Notes B {|Number SID, Agent A, Agent B, Nonce M|} \<in> set evs" |
|
353 |
apply (intro exI bexI) |
|
354 |
apply (rule_tac [2] tls.Nil |
|
355 |
[THEN tls.ClientHello, THEN tls.ServerHello, |
|
356 |
THEN tls.Certificate, THEN tls.ClientKeyExch, |
|
357 |
THEN tls.ServerFinished, THEN tls.ClientFinished, |
|
13507 | 358 |
THEN tls.ServerAccepts], possibility, blast+) |
11287 | 359 |
done |
360 |
||
361 |
||
13922 | 362 |
text{*Another one, for CertVerify (which is optional)*} |
11287 | 363 |
lemma "[| \<forall>evs. (@ N. Nonce N \<notin> used evs) \<notin> range PRF; A \<noteq> B |] |
364 |
==> \<exists>NB PMS. \<exists>evs \<in> tls. |
|
365 |
Says A B (Crypt (priK A) (Hash{|Nonce NB, Agent B, Nonce PMS|})) |
|
366 |
\<in> set evs" |
|
367 |
apply (intro exI bexI) |
|
368 |
apply (rule_tac [2] tls.Nil |
|
369 |
[THEN tls.ClientHello, THEN tls.ServerHello, |
|
370 |
THEN tls.Certificate, THEN tls.ClientKeyExch, |
|
13507 | 371 |
THEN tls.CertVerify], possibility, blast+) |
11287 | 372 |
done |
373 |
||
374 |
||
13922 | 375 |
text{*Another one, for session resumption (both ServerResume and ClientResume). |
376 |
NO tls.Nil here: we refer to a previous session, not the empty trace.*} |
|
11287 | 377 |
lemma "[| evs0 \<in> tls; |
378 |
Notes A {|Number SID, Agent A, Agent B, Nonce M|} \<in> set evs0; |
|
379 |
Notes B {|Number SID, Agent A, Agent B, Nonce M|} \<in> set evs0; |
|
380 |
\<forall>evs. (@ N. Nonce N \<notin> used evs) \<notin> range PRF; |
|
381 |
A \<noteq> B |] |
|
382 |
==> \<exists>NA PA NB PB X. \<exists>evs \<in> tls. |
|
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
383 |
X = Hash{|Number SID, Nonce M, |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
384 |
Nonce NA, Number PA, Agent A, |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
385 |
Nonce NB, Number PB, Agent B|} & |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
386 |
Says A B (Crypt (clientK(NA,NB,M)) X) \<in> set evs & |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
387 |
Says B A (Crypt (serverK(NA,NB,M)) X) \<in> set evs" |
11287 | 388 |
apply (intro exI bexI) |
389 |
apply (rule_tac [2] tls.ClientHello |
|
390 |
[THEN tls.ServerHello, |
|
13507 | 391 |
THEN tls.ServerResume, THEN tls.ClientResume], possibility, blast+) |
11287 | 392 |
done |
393 |
||
394 |
||
13922 | 395 |
subsection{*Inductive proofs about tls*} |
11287 | 396 |
|
397 |
||
398 |
(** Theorems of the form X \<notin> parts (spies evs) imply that NOBODY |
|
399 |
sends messages containing X! **) |
|
400 |
||
13922 | 401 |
text{*Spy never sees a good agent's private key!*} |
11287 | 402 |
lemma Spy_see_priK [simp]: |
13922 | 403 |
"evs \<in> tls ==> (Key (privateKey b A) \<in> parts (spies evs)) = (A \<in> bad)" |
404 |
by (erule tls.induct, force, simp_all, blast) |
|
11287 | 405 |
|
406 |
lemma Spy_analz_priK [simp]: |
|
13922 | 407 |
"evs \<in> tls ==> (Key (privateKey b A) \<in> analz (spies evs)) = (A \<in> bad)" |
11287 | 408 |
by auto |
409 |
||
410 |
lemma Spy_see_priK_D [dest!]: |
|
13922 | 411 |
"[|Key (privateKey b A) \<in> parts (knows Spy evs); evs \<in> tls|] ==> A \<in> bad" |
11287 | 412 |
by (blast dest: Spy_see_priK) |
413 |
||
414 |
||
13922 | 415 |
text{*This lemma says that no false certificates exist. One might extend the |
11287 | 416 |
model to include bogus certificates for the agents, but there seems |
417 |
little point in doing so: the loss of their private keys is a worse |
|
13922 | 418 |
breach of security.*} |
11287 | 419 |
lemma certificate_valid: |
420 |
"[| certificate B KB \<in> parts (spies evs); evs \<in> tls |] ==> KB = pubK B" |
|
421 |
apply (erule rev_mp) |
|
13507 | 422 |
apply (erule tls.induct, force, simp_all, blast) |
11287 | 423 |
done |
424 |
||
425 |
lemmas CX_KB_is_pubKB = Says_imp_spies [THEN parts.Inj, THEN certificate_valid] |
|
426 |
||
427 |
||
13922 | 428 |
subsubsection{*Properties of items found in Notes*} |
11287 | 429 |
|
430 |
lemma Notes_Crypt_parts_spies: |
|
431 |
"[| Notes A {|Agent B, X|} \<in> set evs; evs \<in> tls |] |
|
432 |
==> Crypt (pubK B) X \<in> parts (spies evs)" |
|
433 |
apply (erule rev_mp) |
|
434 |
apply (erule tls.induct, |
|
435 |
frule_tac [7] CX_KB_is_pubKB, force, simp_all) |
|
436 |
apply (blast intro: parts_insertI) |
|
437 |
done |
|
438 |
||
13922 | 439 |
text{*C may be either A or B*} |
11287 | 440 |
lemma Notes_master_imp_Crypt_PMS: |
441 |
"[| Notes C {|s, Agent A, Agent B, Nonce(PRF(PMS,NA,NB))|} \<in> set evs; |
|
442 |
evs \<in> tls |] |
|
443 |
==> Crypt (pubK B) (Nonce PMS) \<in> parts (spies evs)" |
|
444 |
apply (erule rev_mp) |
|
445 |
apply (erule tls.induct, force, simp_all) |
|
13922 | 446 |
txt{*Fake*} |
11287 | 447 |
apply (blast intro: parts_insertI) |
13922 | 448 |
txt{*Client, Server Accept*} |
11287 | 449 |
apply (blast dest!: Notes_Crypt_parts_spies)+ |
450 |
done |
|
451 |
||
13922 | 452 |
text{*Compared with the theorem above, both premise and conclusion are stronger*} |
11287 | 453 |
lemma Notes_master_imp_Notes_PMS: |
454 |
"[| Notes A {|s, Agent A, Agent B, Nonce(PRF(PMS,NA,NB))|} \<in> set evs; |
|
455 |
evs \<in> tls |] |
|
456 |
==> Notes A {|Agent B, Nonce PMS|} \<in> set evs" |
|
457 |
apply (erule rev_mp) |
|
458 |
apply (erule tls.induct, force, simp_all) |
|
13922 | 459 |
txt{*ServerAccepts*} |
11287 | 460 |
apply blast |
461 |
done |
|
462 |
||
463 |
||
13922 | 464 |
subsubsection{*Protocol goal: if B receives CertVerify, then A sent it*} |
11287 | 465 |
|
13922 | 466 |
text{*B can check A's signature if he has received A's certificate.*} |
11287 | 467 |
lemma TrustCertVerify_lemma: |
468 |
"[| X \<in> parts (spies evs); |
|
469 |
X = Crypt (priK A) (Hash{|nb, Agent B, pms|}); |
|
470 |
evs \<in> tls; A \<notin> bad |] |
|
471 |
==> Says A B X \<in> set evs" |
|
472 |
apply (erule rev_mp, erule ssubst) |
|
13507 | 473 |
apply (erule tls.induct, force, simp_all, blast) |
11287 | 474 |
done |
475 |
||
13922 | 476 |
text{*Final version: B checks X using the distributed KA instead of priK A*} |
11287 | 477 |
lemma TrustCertVerify: |
478 |
"[| X \<in> parts (spies evs); |
|
479 |
X = Crypt (invKey KA) (Hash{|nb, Agent B, pms|}); |
|
480 |
certificate A KA \<in> parts (spies evs); |
|
481 |
evs \<in> tls; A \<notin> bad |] |
|
482 |
==> Says A B X \<in> set evs" |
|
483 |
by (blast dest!: certificate_valid intro!: TrustCertVerify_lemma) |
|
484 |
||
485 |
||
13922 | 486 |
text{*If CertVerify is present then A has chosen PMS.*} |
11287 | 487 |
lemma UseCertVerify_lemma: |
488 |
"[| Crypt (priK A) (Hash{|nb, Agent B, Nonce PMS|}) \<in> parts (spies evs); |
|
489 |
evs \<in> tls; A \<notin> bad |] |
|
490 |
==> Notes A {|Agent B, Nonce PMS|} \<in> set evs" |
|
491 |
apply (erule rev_mp) |
|
13507 | 492 |
apply (erule tls.induct, force, simp_all, blast) |
11287 | 493 |
done |
494 |
||
13922 | 495 |
text{*Final version using the distributed KA instead of priK A*} |
11287 | 496 |
lemma UseCertVerify: |
497 |
"[| Crypt (invKey KA) (Hash{|nb, Agent B, Nonce PMS|}) |
|
498 |
\<in> parts (spies evs); |
|
499 |
certificate A KA \<in> parts (spies evs); |
|
500 |
evs \<in> tls; A \<notin> bad |] |
|
501 |
==> Notes A {|Agent B, Nonce PMS|} \<in> set evs" |
|
502 |
by (blast dest!: certificate_valid intro!: UseCertVerify_lemma) |
|
503 |
||
504 |
||
505 |
lemma no_Notes_A_PRF [simp]: |
|
506 |
"evs \<in> tls ==> Notes A {|Agent B, Nonce (PRF x)|} \<notin> set evs" |
|
507 |
apply (erule tls.induct, force, simp_all) |
|
13922 | 508 |
txt{*ClientKeyExch: PMS is assumed to differ from any PRF.*} |
11287 | 509 |
apply blast |
510 |
done |
|
511 |
||
512 |
||
513 |
lemma MS_imp_PMS [dest!]: |
|
514 |
"[| Nonce (PRF (PMS,NA,NB)) \<in> parts (spies evs); evs \<in> tls |] |
|
515 |
==> Nonce PMS \<in> parts (spies evs)" |
|
516 |
apply (erule rev_mp) |
|
517 |
apply (erule tls.induct, force, simp_all) |
|
13922 | 518 |
txt{*Fake*} |
11287 | 519 |
apply (blast intro: parts_insertI) |
13922 | 520 |
txt{*Easy, e.g. by freshness*} |
11287 | 521 |
apply (blast dest: Notes_Crypt_parts_spies)+ |
522 |
done |
|
523 |
||
524 |
||
525 |
||
526 |
||
13922 | 527 |
subsubsection{*Unicity results for PMS, the pre-master-secret*} |
11287 | 528 |
|
13922 | 529 |
text{*PMS determines B.*} |
11287 | 530 |
lemma Crypt_unique_PMS: |
531 |
"[| Crypt(pubK B) (Nonce PMS) \<in> parts (spies evs); |
|
532 |
Crypt(pubK B') (Nonce PMS) \<in> parts (spies evs); |
|
533 |
Nonce PMS \<notin> analz (spies evs); |
|
534 |
evs \<in> tls |] |
|
535 |
==> B=B'" |
|
536 |
apply (erule rev_mp, erule rev_mp, erule rev_mp) |
|
537 |
apply (erule tls.induct, analz_mono_contra, force, simp_all (no_asm_simp)) |
|
13922 | 538 |
txt{*Fake, ClientKeyExch*} |
11287 | 539 |
apply blast+ |
540 |
done |
|
541 |
||
542 |
||
543 |
(** It is frustrating that we need two versions of the unicity results. |
|
544 |
But Notes A {|Agent B, Nonce PMS|} determines both A and B. Sometimes |
|
545 |
we have only the weaker assertion Crypt(pubK B) (Nonce PMS), which |
|
546 |
determines B alone, and only if PMS is secret. |
|
547 |
**) |
|
548 |
||
13922 | 549 |
text{*In A's internal Note, PMS determines A and B.*} |
11287 | 550 |
lemma Notes_unique_PMS: |
551 |
"[| Notes A {|Agent B, Nonce PMS|} \<in> set evs; |
|
552 |
Notes A' {|Agent B', Nonce PMS|} \<in> set evs; |
|
553 |
evs \<in> tls |] |
|
554 |
==> A=A' & B=B'" |
|
555 |
apply (erule rev_mp, erule rev_mp) |
|
556 |
apply (erule tls.induct, force, simp_all) |
|
13922 | 557 |
txt{*ClientKeyExch*} |
11287 | 558 |
apply (blast dest!: Notes_Crypt_parts_spies) |
559 |
done |
|
560 |
||
561 |
||
13922 | 562 |
subsection{*Secrecy Theorems*} |
11287 | 563 |
|
13956 | 564 |
text{*Key compromise lemma needed to prove @{term analz_image_keys}. |
13922 | 565 |
No collection of keys can help the spy get new private keys.*} |
11287 | 566 |
lemma analz_image_priK [rule_format]: |
567 |
"evs \<in> tls |
|
568 |
==> \<forall>KK. (Key(priK B) \<in> analz (Key`KK Un (spies evs))) = |
|
569 |
(priK B \<in> KK | B \<in> bad)" |
|
570 |
apply (erule tls.induct) |
|
571 |
apply (simp_all (no_asm_simp) |
|
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
572 |
del: image_insert |
11287 | 573 |
add: image_Un [THEN sym] |
574 |
insert_Key_image Un_assoc [THEN sym]) |
|
13922 | 575 |
txt{*Fake*} |
11287 | 576 |
apply spy_analz |
577 |
done |
|
578 |
||
579 |
||
13922 | 580 |
text{*slightly speeds up the big simplification below*} |
11287 | 581 |
lemma range_sessionkeys_not_priK: |
582 |
"KK <= range sessionK ==> priK B \<notin> KK" |
|
583 |
by blast |
|
584 |
||
585 |
||
13922 | 586 |
text{*Lemma for the trivial direction of the if-and-only-if*} |
11287 | 587 |
lemma analz_image_keys_lemma: |
588 |
"(X \<in> analz (G Un H)) --> (X \<in> analz H) ==> |
|
589 |
(X \<in> analz (G Un H)) = (X \<in> analz H)" |
|
590 |
by (blast intro: analz_mono [THEN subsetD]) |
|
591 |
||
592 |
(** Strangely, the following version doesn't work: |
|
593 |
\<forall>Z. (Nonce N \<in> analz (Key`(sessionK`Z) Un (spies evs))) = |
|
594 |
(Nonce N \<in> analz (spies evs))" |
|
595 |
**) |
|
596 |
||
597 |
lemma analz_image_keys [rule_format]: |
|
598 |
"evs \<in> tls ==> |
|
599 |
\<forall>KK. KK <= range sessionK --> |
|
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
600 |
(Nonce N \<in> analz (Key`KK Un (spies evs))) = |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
601 |
(Nonce N \<in> analz (spies evs))" |
11287 | 602 |
apply (erule tls.induct, frule_tac [7] CX_KB_is_pubKB) |
603 |
apply (safe del: iffI) |
|
604 |
apply (safe del: impI iffI intro!: analz_image_keys_lemma) |
|
605 |
apply (simp_all (no_asm_simp) (*faster*) |
|
606 |
del: image_insert imp_disjL (*reduces blow-up*) |
|
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
607 |
add: image_Un [THEN sym] Un_assoc [THEN sym] |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
608 |
insert_Key_singleton |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
609 |
range_sessionkeys_not_priK analz_image_priK) |
11287 | 610 |
apply (simp_all add: insert_absorb) |
13922 | 611 |
txt{*Fake*} |
11287 | 612 |
apply spy_analz |
613 |
done |
|
614 |
||
13922 | 615 |
text{*Knowing some session keys is no help in getting new nonces*} |
11287 | 616 |
lemma analz_insert_key [simp]: |
617 |
"evs \<in> tls ==> |
|
11655 | 618 |
(Nonce N \<in> analz (insert (Key (sessionK z)) (spies evs))) = |
11287 | 619 |
(Nonce N \<in> analz (spies evs))" |
620 |
by (simp del: image_insert |
|
621 |
add: insert_Key_singleton analz_image_keys) |
|
622 |
||
623 |
||
13922 | 624 |
subsubsection{*Protocol goal: serverK(Na,Nb,M) and clientK(Na,Nb,M) remain secure*} |
11287 | 625 |
|
626 |
(** Some lemmas about session keys, comprising clientK and serverK **) |
|
627 |
||
628 |
||
13922 | 629 |
text{*Lemma: session keys are never used if PMS is fresh. |
11287 | 630 |
Nonces don't have to agree, allowing session resumption. |
631 |
Converse doesn't hold; revealing PMS doesn't force the keys to be sent. |
|
13922 | 632 |
THEY ARE NOT SUITABLE AS SAFE ELIM RULES.*} |
11287 | 633 |
lemma PMS_lemma: |
634 |
"[| Nonce PMS \<notin> parts (spies evs); |
|
635 |
K = sessionK((Na, Nb, PRF(PMS,NA,NB)), role); |
|
636 |
evs \<in> tls |] |
|
637 |
==> Key K \<notin> parts (spies evs) & (\<forall>Y. Crypt K Y \<notin> parts (spies evs))" |
|
638 |
apply (erule rev_mp, erule ssubst) |
|
13922 | 639 |
apply (erule tls.induct, frule_tac [7] CX_KB_is_pubKB) |
11287 | 640 |
apply (force, simp_all (no_asm_simp)) |
13922 | 641 |
txt{*Fake*} |
11287 | 642 |
apply (blast intro: parts_insertI) |
13922 | 643 |
txt{*SpyKeys*} |
11287 | 644 |
apply blast |
13922 | 645 |
txt{*Many others*} |
11287 | 646 |
apply (force dest!: Notes_Crypt_parts_spies Notes_master_imp_Crypt_PMS)+ |
647 |
done |
|
648 |
||
649 |
lemma PMS_sessionK_not_spied: |
|
650 |
"[| Key (sessionK((Na, Nb, PRF(PMS,NA,NB)), role)) \<in> parts (spies evs); |
|
651 |
evs \<in> tls |] |
|
652 |
==> Nonce PMS \<in> parts (spies evs)" |
|
653 |
by (blast dest: PMS_lemma) |
|
654 |
||
655 |
lemma PMS_Crypt_sessionK_not_spied: |
|
656 |
"[| Crypt (sessionK((Na, Nb, PRF(PMS,NA,NB)), role)) Y |
|
657 |
\<in> parts (spies evs); evs \<in> tls |] |
|
658 |
==> Nonce PMS \<in> parts (spies evs)" |
|
659 |
by (blast dest: PMS_lemma) |
|
660 |
||
13922 | 661 |
text{*Write keys are never sent if M (MASTER SECRET) is secure. |
11287 | 662 |
Converse fails; betraying M doesn't force the keys to be sent! |
663 |
The strong Oops condition can be weakened later by unicity reasoning, |
|
664 |
with some effort. |
|
13956 | 665 |
NO LONGER USED: see @{text clientK_not_spied} and @{text serverK_not_spied}*} |
11287 | 666 |
lemma sessionK_not_spied: |
667 |
"[| \<forall>A. Says A Spy (Key (sessionK((NA,NB,M),role))) \<notin> set evs; |
|
668 |
Nonce M \<notin> analz (spies evs); evs \<in> tls |] |
|
669 |
==> Key (sessionK((NA,NB,M),role)) \<notin> parts (spies evs)" |
|
670 |
apply (erule rev_mp, erule rev_mp) |
|
671 |
apply (erule tls.induct, analz_mono_contra) |
|
672 |
apply (force, simp_all (no_asm_simp)) |
|
13922 | 673 |
txt{*Fake, SpyKeys*} |
11287 | 674 |
apply blast+ |
675 |
done |
|
676 |
||
677 |
||
13922 | 678 |
text{*If A sends ClientKeyExch to an honest B, then the PMS will stay secret.*} |
11287 | 679 |
lemma Spy_not_see_PMS: |
680 |
"[| Notes A {|Agent B, Nonce PMS|} \<in> set evs; |
|
681 |
evs \<in> tls; A \<notin> bad; B \<notin> bad |] |
|
682 |
==> Nonce PMS \<notin> analz (spies evs)" |
|
683 |
apply (erule rev_mp, erule tls.induct, frule_tac [7] CX_KB_is_pubKB) |
|
684 |
apply (force, simp_all (no_asm_simp)) |
|
13922 | 685 |
txt{*Fake*} |
11287 | 686 |
apply spy_analz |
13922 | 687 |
txt{*SpyKeys*} |
11287 | 688 |
apply force |
689 |
apply (simp_all add: insert_absorb) |
|
13922 | 690 |
txt{*ClientHello, ServerHello, ClientKeyExch: mostly freshness reasoning*} |
11287 | 691 |
apply (blast dest: Notes_Crypt_parts_spies) |
692 |
apply (blast dest: Notes_Crypt_parts_spies) |
|
693 |
apply (blast dest: Notes_Crypt_parts_spies) |
|
13956 | 694 |
txt{*ClientAccepts and ServerAccepts: because @{term "PMS \<notin> range PRF"}*} |
11287 | 695 |
apply force+ |
696 |
done |
|
697 |
||
698 |
||
13922 | 699 |
text{*If A sends ClientKeyExch to an honest B, then the MASTER SECRET |
700 |
will stay secret.*} |
|
11287 | 701 |
lemma Spy_not_see_MS: |
702 |
"[| Notes A {|Agent B, Nonce PMS|} \<in> set evs; |
|
703 |
evs \<in> tls; A \<notin> bad; B \<notin> bad |] |
|
704 |
==> Nonce (PRF(PMS,NA,NB)) \<notin> analz (spies evs)" |
|
705 |
apply (erule rev_mp, erule tls.induct, frule_tac [7] CX_KB_is_pubKB) |
|
706 |
apply (force, simp_all (no_asm_simp)) |
|
13922 | 707 |
txt{*Fake*} |
11287 | 708 |
apply spy_analz |
13922 | 709 |
txt{*SpyKeys: by secrecy of the PMS, Spy cannot make the MS*} |
11287 | 710 |
apply (blast dest!: Spy_not_see_PMS) |
711 |
apply (simp_all add: insert_absorb) |
|
13922 | 712 |
txt{*ClientAccepts and ServerAccepts: because PMS was already visible; |
713 |
others, freshness etc.*} |
|
11287 | 714 |
apply (blast dest: Notes_Crypt_parts_spies Spy_not_see_PMS |
715 |
Notes_imp_knows_Spy [THEN analz.Inj])+ |
|
716 |
done |
|
717 |
||
718 |
||
719 |
||
13922 | 720 |
subsubsection{*Weakening the Oops conditions for leakage of clientK*} |
11287 | 721 |
|
13922 | 722 |
text{*If A created PMS then nobody else (except the Spy in replays) |
723 |
would send a message using a clientK generated from that PMS.*} |
|
11287 | 724 |
lemma Says_clientK_unique: |
725 |
"[| Says A' B' (Crypt (clientK(Na,Nb,PRF(PMS,NA,NB))) Y) \<in> set evs; |
|
726 |
Notes A {|Agent B, Nonce PMS|} \<in> set evs; |
|
727 |
evs \<in> tls; A' \<noteq> Spy |] |
|
728 |
==> A = A'" |
|
729 |
apply (erule rev_mp, erule rev_mp) |
|
730 |
apply (erule tls.induct, frule_tac [7] CX_KB_is_pubKB) |
|
731 |
apply (force, simp_all) |
|
13922 | 732 |
txt{*ClientKeyExch*} |
11287 | 733 |
apply (blast dest!: PMS_Crypt_sessionK_not_spied) |
13922 | 734 |
txt{*ClientFinished, ClientResume: by unicity of PMS*} |
11287 | 735 |
apply (blast dest!: Notes_master_imp_Notes_PMS |
736 |
intro: Notes_unique_PMS [THEN conjunct1])+ |
|
737 |
done |
|
738 |
||
739 |
||
13922 | 740 |
text{*If A created PMS and has not leaked her clientK to the Spy, |
741 |
then it is completely secure: not even in parts!*} |
|
11287 | 742 |
lemma clientK_not_spied: |
743 |
"[| Notes A {|Agent B, Nonce PMS|} \<in> set evs; |
|
744 |
Says A Spy (Key (clientK(Na,Nb,PRF(PMS,NA,NB)))) \<notin> set evs; |
|
745 |
A \<notin> bad; B \<notin> bad; |
|
746 |
evs \<in> tls |] |
|
747 |
==> Key (clientK(Na,Nb,PRF(PMS,NA,NB))) \<notin> parts (spies evs)" |
|
748 |
apply (erule rev_mp, erule rev_mp) |
|
749 |
apply (erule tls.induct, frule_tac [7] CX_KB_is_pubKB) |
|
750 |
apply (force, simp_all (no_asm_simp)) |
|
13922 | 751 |
txt{*ClientKeyExch*} |
11287 | 752 |
apply blast |
13922 | 753 |
txt{*SpyKeys*} |
11287 | 754 |
apply (blast dest!: Spy_not_see_MS) |
13922 | 755 |
txt{*ClientKeyExch*} |
11287 | 756 |
apply (blast dest!: PMS_sessionK_not_spied) |
13922 | 757 |
txt{*Oops*} |
11287 | 758 |
apply (blast intro: Says_clientK_unique) |
759 |
done |
|
760 |
||
761 |
||
13922 | 762 |
subsubsection{*Weakening the Oops conditions for leakage of serverK*} |
11287 | 763 |
|
13922 | 764 |
text{*If A created PMS for B, then nobody other than B or the Spy would |
765 |
send a message using a serverK generated from that PMS.*} |
|
11287 | 766 |
lemma Says_serverK_unique: |
767 |
"[| Says B' A' (Crypt (serverK(Na,Nb,PRF(PMS,NA,NB))) Y) \<in> set evs; |
|
768 |
Notes A {|Agent B, Nonce PMS|} \<in> set evs; |
|
769 |
evs \<in> tls; A \<notin> bad; B \<notin> bad; B' \<noteq> Spy |] |
|
770 |
==> B = B'" |
|
771 |
apply (erule rev_mp, erule rev_mp) |
|
772 |
apply (erule tls.induct, frule_tac [7] CX_KB_is_pubKB) |
|
773 |
apply (force, simp_all) |
|
13922 | 774 |
txt{*ClientKeyExch*} |
11287 | 775 |
apply (blast dest!: PMS_Crypt_sessionK_not_spied) |
13922 | 776 |
txt{*ServerResume, ServerFinished: by unicity of PMS*} |
11287 | 777 |
apply (blast dest!: Notes_master_imp_Crypt_PMS |
778 |
dest: Spy_not_see_PMS Notes_Crypt_parts_spies Crypt_unique_PMS)+ |
|
779 |
done |
|
780 |
||
781 |
||
13922 | 782 |
text{*If A created PMS for B, and B has not leaked his serverK to the Spy, |
783 |
then it is completely secure: not even in parts!*} |
|
11287 | 784 |
lemma serverK_not_spied: |
785 |
"[| Notes A {|Agent B, Nonce PMS|} \<in> set evs; |
|
786 |
Says B Spy (Key(serverK(Na,Nb,PRF(PMS,NA,NB)))) \<notin> set evs; |
|
787 |
A \<notin> bad; B \<notin> bad; evs \<in> tls |] |
|
788 |
==> Key (serverK(Na,Nb,PRF(PMS,NA,NB))) \<notin> parts (spies evs)" |
|
789 |
apply (erule rev_mp, erule rev_mp) |
|
790 |
apply (erule tls.induct, frule_tac [7] CX_KB_is_pubKB) |
|
791 |
apply (force, simp_all (no_asm_simp)) |
|
13922 | 792 |
txt{*Fake*} |
11287 | 793 |
apply blast |
13922 | 794 |
txt{*SpyKeys*} |
11287 | 795 |
apply (blast dest!: Spy_not_see_MS) |
13922 | 796 |
txt{*ClientKeyExch*} |
11287 | 797 |
apply (blast dest!: PMS_sessionK_not_spied) |
13922 | 798 |
txt{*Oops*} |
11287 | 799 |
apply (blast intro: Says_serverK_unique) |
800 |
done |
|
801 |
||
802 |
||
13922 | 803 |
subsubsection{*Protocol goals: if A receives ServerFinished, then B is present |
11287 | 804 |
and has used the quoted values PA, PB, etc. Note that it is up to A |
13956 | 805 |
to compare PA with what she originally sent.*} |
11287 | 806 |
|
13922 | 807 |
text{*The mention of her name (A) in X assures A that B knows who she is.*} |
11287 | 808 |
lemma TrustServerFinished [rule_format]: |
809 |
"[| X = Crypt (serverK(Na,Nb,M)) |
|
810 |
(Hash{|Number SID, Nonce M, |
|
811 |
Nonce Na, Number PA, Agent A, |
|
812 |
Nonce Nb, Number PB, Agent B|}); |
|
813 |
M = PRF(PMS,NA,NB); |
|
814 |
evs \<in> tls; A \<notin> bad; B \<notin> bad |] |
|
815 |
==> Says B Spy (Key(serverK(Na,Nb,M))) \<notin> set evs --> |
|
816 |
Notes A {|Agent B, Nonce PMS|} \<in> set evs --> |
|
817 |
X \<in> parts (spies evs) --> Says B A X \<in> set evs" |
|
818 |
apply (erule ssubst)+ |
|
819 |
apply (erule tls.induct, frule_tac [7] CX_KB_is_pubKB) |
|
820 |
apply (force, simp_all (no_asm_simp)) |
|
13922 | 821 |
txt{*Fake: the Spy doesn't have the critical session key!*} |
11287 | 822 |
apply (blast dest: serverK_not_spied) |
13922 | 823 |
txt{*ClientKeyExch*} |
11287 | 824 |
apply (blast dest!: PMS_Crypt_sessionK_not_spied) |
825 |
done |
|
826 |
||
13922 | 827 |
text{*This version refers not to ServerFinished but to any message from B. |
11287 | 828 |
We don't assume B has received CertVerify, and an intruder could |
829 |
have changed A's identity in all other messages, so we can't be sure |
|
830 |
that B sends his message to A. If CLIENT KEY EXCHANGE were augmented |
|
13922 | 831 |
to bind A's identity with PMS, then we could replace A' by A below.*} |
11287 | 832 |
lemma TrustServerMsg [rule_format]: |
833 |
"[| M = PRF(PMS,NA,NB); evs \<in> tls; A \<notin> bad; B \<notin> bad |] |
|
834 |
==> Says B Spy (Key(serverK(Na,Nb,M))) \<notin> set evs --> |
|
835 |
Notes A {|Agent B, Nonce PMS|} \<in> set evs --> |
|
836 |
Crypt (serverK(Na,Nb,M)) Y \<in> parts (spies evs) --> |
|
837 |
(\<exists>A'. Says B A' (Crypt (serverK(Na,Nb,M)) Y) \<in> set evs)" |
|
838 |
apply (erule ssubst) |
|
839 |
apply (erule tls.induct, frule_tac [7] CX_KB_is_pubKB) |
|
840 |
apply (force, simp_all (no_asm_simp) add: ex_disj_distrib) |
|
13922 | 841 |
txt{*Fake: the Spy doesn't have the critical session key!*} |
11287 | 842 |
apply (blast dest: serverK_not_spied) |
13922 | 843 |
txt{*ClientKeyExch*} |
11287 | 844 |
apply (clarify, blast dest!: PMS_Crypt_sessionK_not_spied) |
13922 | 845 |
txt{*ServerResume, ServerFinished: by unicity of PMS*} |
11287 | 846 |
apply (blast dest!: Notes_master_imp_Crypt_PMS |
847 |
dest: Spy_not_see_PMS Notes_Crypt_parts_spies Crypt_unique_PMS)+ |
|
848 |
done |
|
849 |
||
850 |
||
13922 | 851 |
subsubsection{*Protocol goal: if B receives any message encrypted with clientK |
852 |
then A has sent it*} |
|
853 |
||
854 |
text{*ASSUMING that A chose PMS. Authentication is |
|
11287 | 855 |
assumed here; B cannot verify it. But if the message is |
13922 | 856 |
ClientFinished, then B can then check the quoted values PA, PB, etc.*} |
11287 | 857 |
|
858 |
lemma TrustClientMsg [rule_format]: |
|
859 |
"[| M = PRF(PMS,NA,NB); evs \<in> tls; A \<notin> bad; B \<notin> bad |] |
|
860 |
==> Says A Spy (Key(clientK(Na,Nb,M))) \<notin> set evs --> |
|
861 |
Notes A {|Agent B, Nonce PMS|} \<in> set evs --> |
|
862 |
Crypt (clientK(Na,Nb,M)) Y \<in> parts (spies evs) --> |
|
863 |
Says A B (Crypt (clientK(Na,Nb,M)) Y) \<in> set evs" |
|
864 |
apply (erule ssubst) |
|
865 |
apply (erule tls.induct, frule_tac [7] CX_KB_is_pubKB) |
|
866 |
apply (force, simp_all (no_asm_simp)) |
|
13922 | 867 |
txt{*Fake: the Spy doesn't have the critical session key!*} |
11287 | 868 |
apply (blast dest: clientK_not_spied) |
13922 | 869 |
txt{*ClientKeyExch*} |
11287 | 870 |
apply (blast dest!: PMS_Crypt_sessionK_not_spied) |
13922 | 871 |
txt{*ClientFinished, ClientResume: by unicity of PMS*} |
11287 | 872 |
apply (blast dest!: Notes_master_imp_Notes_PMS dest: Notes_unique_PMS)+ |
873 |
done |
|
874 |
||
875 |
||
13922 | 876 |
subsubsection{*Protocol goal: if B receives ClientFinished, and if B is able to |
11287 | 877 |
check a CertVerify from A, then A has used the quoted |
13956 | 878 |
values PA, PB, etc. Even this one requires A to be uncompromised.*} |
11287 | 879 |
lemma AuthClientFinished: |
880 |
"[| M = PRF(PMS,NA,NB); |
|
881 |
Says A Spy (Key(clientK(Na,Nb,M))) \<notin> set evs; |
|
882 |
Says A' B (Crypt (clientK(Na,Nb,M)) Y) \<in> set evs; |
|
883 |
certificate A KA \<in> parts (spies evs); |
|
884 |
Says A'' B (Crypt (invKey KA) (Hash{|nb, Agent B, Nonce PMS|})) |
|
885 |
\<in> set evs; |
|
886 |
evs \<in> tls; A \<notin> bad; B \<notin> bad |] |
|
887 |
==> Says A B (Crypt (clientK(Na,Nb,M)) Y) \<in> set evs" |
|
888 |
by (blast intro!: TrustClientMsg UseCertVerify) |
|
889 |
||
890 |
(*22/9/97: loads in 622s, which is 10 minutes 22 seconds*) |
|
891 |
(*24/9/97: loads in 672s, which is 11 minutes 12 seconds [stronger theorems]*) |
|
892 |
(*29/9/97: loads in 481s, after removing Certificate from ClientKeyExch*) |
|
893 |
(*30/9/97: loads in 476s, after removing unused theorems*) |
|
894 |
(*30/9/97: loads in 448s, after fixing ServerResume*) |
|
895 |
||
896 |
(*08/9/97: loads in 189s (pike), after much reorganization, |
|
897 |
back to 621s on albatross?*) |
|
898 |
||
899 |
(*10/2/99: loads in 139s (pike) |
|
900 |
down to 433s on albatross*) |
|
901 |
||
902 |
(*5/5/01: conversion to Isar script |
|
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
903 |
loads in 137s (perch) |
11287 | 904 |
the last ML version loaded in 122s on perch, a 600MHz machine: |
32960
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
905 |
twice as fast as pike. No idea why it's so much slower! |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
906 |
The Isar script is slower still, perhaps because simp_all simplifies |
69916a850301
eliminated hard tabulators, guessing at each author's individual tab-width;
wenzelm
parents:
29888
diff
changeset
|
907 |
the assumptions be default. |
11287 | 908 |
*) |
3474 | 909 |
|
910 |
end |