blob: 41e538d26a30b5a40e1a2222db5562e9169c741a [file] [log] [blame]
Harald Welte781bd5d2011-12-06 22:23:52 +01001/* GSM/GPRS/3G authentication core infrastructure */
2
3/* (C) 2011 by Harald Welte <laforge@gnumonks.org>
4 *
5 * All Rights Reserved
6 *
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License along
18 * with this program; if not, write to the Free Software Foundation, Inc.,
19 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
20 *
21 */
22
23#include <osmocom/crypt/auth.h>
Maxaead05d2016-04-21 15:57:52 +020024#include <osmocom/core/bits.h>
Harald Welte781bd5d2011-12-06 22:23:52 +010025#include "milenage/common.h"
26#include "milenage/milenage.h"
27
Harald Welte96e2a002017-06-12 21:44:18 +020028/*! \addtogroup auth
29 * @{
30 */
31
Harald Welte781bd5d2011-12-06 22:23:52 +010032static int milenage_gen_vec(struct osmo_auth_vector *vec,
33 struct osmo_sub_auth_data *aud,
34 const uint8_t *_rand)
35{
36 size_t res_len = sizeof(vec->res);
Neels Hofmeyr82c9a0e2017-03-13 17:36:17 +010037 uint64_t next_sqn;
Harald Welte781bd5d2011-12-06 22:23:52 +010038 uint8_t sqn[6];
Neels Hofmeyrbb6f7b72017-03-13 17:27:17 +010039 uint64_t ind_mask;
40 uint64_t seq_1;
Harald Welte781bd5d2011-12-06 22:23:52 +010041 int rc;
42
Neels Hofmeyrbb6f7b72017-03-13 17:27:17 +010043 /* Determine next SQN, according to 3GPP TS 33.102:
44 * SQN consists of SEQ and a lower significant part of IND bits:
45 *
46 * |----------SEQ------------|
47 * |------------------------SQN-----------|
48 * |-----IND----|
49 *
50 * The IND part is used as "slots": e.g. a given HLR client will always
51 * get the same IND part, called ind here, with incrementing SEQ. In
52 * the USIM, each IND slot enforces that its SEQ are used in ascending
53 * order -- as long as that constraint is satisfied, the SQN may jump
54 * forwards and backwards. For example, for ind_bitlen == 5, asking the
55 * USIM for SQN = 32, 64, 33 is allowed, because 32 and 64 are
56 * SEQ || (ind == 0), and though 33 is below 64, it is ind == 1 and
57 * allowed. Not allowed would be 32, 96, 64, because 64 would go
58 * backwards after 96, both being ind == 0.
59 *
60 * From the last used SQN, we want to increment SEQ + 1, and then pick
61 * the matching IND part.
62 *
63 * IND size is suggested in TS 33.102 as 5 bits. SQN is 48 bits long.
64 * If ind_bitlen is passed too large here, the algorithms will break
65 * down. But at which point should we return an error? A sane limit
66 * seems to be ind_bitlen == 10, but to protect against failure,
67 * limiting ind_bitlen to 28 is enough, 28 being the number of bits
68 * suggested for the delta in 33.102, which is discussed to still
69 * require 2^15 > 32000 authentications to wrap the SQN back to the
70 * start.
71 *
72 * Note that if a caller with ind == 1 generates N vectors, the SQN
73 * stored after this will reflect SEQ + N. If then another caller with
74 * ind == 2 generates another N vectors, this will then use SEQ + N
75 * onwards and end up with SEQ + N + N. In other words, most of each
76 * SEQ's IND slots will remain unused. When looking at SQN being 48
77 * bits wide, after dropping ind_bitlen (say 5) from it, we will still
78 * have a sequence range of 2^43 = 8.8e12, eight trillion sequences,
79 * which is large enough to not bother further. With the maximum
80 * ind_bitlen of 28 enforced below, we still get more than 1 million
81 * sequences, which is also sufficiently large.
82 *
83 * An ind_bitlen of zero may be passed from legacy callers that are not
84 * aware of the IND extension. For these, below algorithm works out as
85 * before, simply incrementing SQN by 1.
86 *
87 * This is also a mechanism for tools like the osmo-auc-gen to directly
88 * request a given SQN to be used. With ind_bitlen == 0 the caller can
89 * be sure that this code will increment SQN by exactly one before
90 * generating a tuple, thus a caller would simply pass
91 * { .ind_bitlen = 0, .ind = 0, .sqn = (desired_sqn - 1) }
92 */
93
94 if (aud->u.umts.ind_bitlen > 28)
95 return -2;
96
97 seq_1 = 1LL << aud->u.umts.ind_bitlen;
98 ind_mask = ~(seq_1 - 1);
99
100 /* the ind index must not affect the SEQ part */
Neels Hofmeyre0cb0ee2017-03-16 05:30:11 +0100101 if (aud->u.umts.ind >= seq_1)
Neels Hofmeyrbb6f7b72017-03-13 17:27:17 +0100102 return -3;
103
Neels Hofmeyr82c9a0e2017-03-13 17:36:17 +0100104 /* keep the incremented SQN local until gsm_milenage() succeeded. */
Neels Hofmeyrbb6f7b72017-03-13 17:27:17 +0100105 next_sqn = ((aud->u.umts.sqn + seq_1) & ind_mask) + aud->u.umts.ind;
Neels Hofmeyr82c9a0e2017-03-13 17:36:17 +0100106
107 osmo_store64be_ext(next_sqn, sqn, 6);
Harald Welteaae23622011-12-07 11:35:02 +0100108 milenage_generate(aud->u.umts.opc, aud->u.umts.amf, aud->u.umts.k,
Harald Welte781bd5d2011-12-06 22:23:52 +0100109 sqn, _rand,
110 vec->autn, vec->ik, vec->ck, vec->res, &res_len);
111 vec->res_len = res_len;
Harald Welteaae23622011-12-07 11:35:02 +0100112 rc = gsm_milenage(aud->u.umts.opc, aud->u.umts.k, _rand, vec->sres, vec->kc);
Harald Welte781bd5d2011-12-06 22:23:52 +0100113 if (rc < 0)
114 return rc;
115
116 vec->auth_types = OSMO_AUTH_TYPE_UMTS | OSMO_AUTH_TYPE_GSM;
Neels Hofmeyr82c9a0e2017-03-13 17:36:17 +0100117
118 /* for storage in the caller's AUC database */
119 aud->u.umts.sqn = next_sqn;
Harald Welte781bd5d2011-12-06 22:23:52 +0100120
121 return 0;
122}
123
124static int milenage_gen_vec_auts(struct osmo_auth_vector *vec,
125 struct osmo_sub_auth_data *aud,
126 const uint8_t *auts, const uint8_t *rand_auts,
127 const uint8_t *_rand)
128{
129 uint8_t sqn_out[6];
Harald Weltea72e47b2012-03-21 09:03:16 +0100130 uint8_t gen_opc[16];
131 uint8_t *opc;
Harald Welte781bd5d2011-12-06 22:23:52 +0100132 int rc;
133
Harald Weltea72e47b2012-03-21 09:03:16 +0100134 /* Check if we only know OP and compute OPC if required */
135 if (aud->type == OSMO_AUTH_TYPE_UMTS && aud->u.umts.opc_is_op) {
136 rc = milenage_opc_gen(gen_opc, aud->u.umts.k,
137 aud->u.umts.opc);
138 if (rc < 0)
139 return rc;
140 opc = gen_opc;
141 } else
142 opc = aud->u.umts.opc;
143
144 rc = milenage_auts(opc, aud->u.umts.k, rand_auts, auts, sqn_out);
Harald Welte781bd5d2011-12-06 22:23:52 +0100145 if (rc < 0)
146 return rc;
147
Neels Hofmeyrbb6f7b72017-03-13 17:27:17 +0100148 /* Update our "largest used SQN" from the USIM -- milenage_gen_vec()
149 * below will increment SQN. */
Neels Hofmeyr82c9a0e2017-03-13 17:36:17 +0100150 aud->u.umts.sqn = osmo_load64be_ext(sqn_out, 6) >> 16;
Harald Welte781bd5d2011-12-06 22:23:52 +0100151
152 return milenage_gen_vec(vec, aud, _rand);
153}
154
155static struct osmo_auth_impl milenage_alg = {
156 .algo = OSMO_AUTH_ALG_MILENAGE,
157 .name = "MILENAGE (libosmogsm built-in)",
158 .priority = 1000,
159 .gen_vec = &milenage_gen_vec,
160 .gen_vec_auts = &milenage_gen_vec_auts,
161};
162
163static __attribute__((constructor)) void on_dso_load_milenage(void)
164{
165 osmo_auth_register(&milenage_alg);
166}
Harald Welte96e2a002017-06-12 21:44:18 +0200167
168/*! @} */