~ubuntu-branches/ubuntu/wily/psi/wily

« back to all changes in this revision

Viewing changes to iris/src/irisnet/noncore/stunmessage.cpp

  • Committer: Bazaar Package Importer
  • Author(s): Jan Niehusmann
  • Date: 2009-09-25 17:49:51 UTC
  • mfrom: (6.1.3 sid)
  • Revision ID: james.westby@ubuntu.com-20090925174951-lvm7kdap82o8xhn3
Tags: 0.13-1
* Updated to upstream version 0.13
* Set Standards-Version to 3.8.3

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
/*
 
2
 * Copyright (C) 2009  Barracuda Networks, Inc.
 
3
 *
 
4
 * This library is free software; you can redistribute it and/or
 
5
 * modify it under the terms of the GNU Lesser General Public
 
6
 * License as published by the Free Software Foundation; either
 
7
 * version 2.1 of the License, or (at your option) any later version.
 
8
 *
 
9
 * This library is distributed in the hope that it will be useful,
 
10
 * but WITHANY WARRANTY; without even the implied warranty of
 
11
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 
12
 * Lesser General Public License for more details.
 
13
 *
 
14
 * You should have received a copy of the GNU Lesser General Public
 
15
 * License along with this library; if not, write to the Free Software
 
16
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
 
17
 * 02110-1301  USA
 
18
 *
 
19
 */
 
20
 
 
21
#include "stunmessage.h"
 
22
 
 
23
#include <QSharedData>
 
24
#include <QtCrypto>
 
25
 
 
26
#define ENSURE_D { if(!d) d = new Private; }
 
27
 
 
28
namespace XMPP {
 
29
 
 
30
// some attribute types we need to explicitly support
 
31
enum
 
32
{
 
33
        AttribMessageIntegrity = 0x0008,
 
34
        AttribFingerprint      = 0x8028
 
35
};
 
36
 
 
37
// adapted from public domain source by Ross Williams and Eric Durbin
 
38
unsigned long crctable[256] =
 
39
{
 
40
        0x00000000L, 0x77073096L, 0xEE0E612CL, 0x990951BAL,
 
41
        0x076DC419L, 0x706AF48FL, 0xE963A535L, 0x9E6495A3L,
 
42
        0x0EDB8832L, 0x79DCB8A4L, 0xE0D5E91EL, 0x97D2D988L,
 
43
        0x09B64C2BL, 0x7EB17CBDL, 0xE7B82D07L, 0x90BF1D91L,
 
44
        0x1DB71064L, 0x6AB020F2L, 0xF3B97148L, 0x84BE41DEL,
 
45
        0x1ADAD47DL, 0x6DDDE4EBL, 0xF4D4B551L, 0x83D385C7L,
 
46
        0x136C9856L, 0x646BA8C0L, 0xFD62F97AL, 0x8A65C9ECL,
 
47
        0x14015C4FL, 0x63066CD9L, 0xFA0F3D63L, 0x8D080DF5L,
 
48
        0x3B6E20C8L, 0x4C69105EL, 0xD56041E4L, 0xA2677172L,
 
49
        0x3C03E4D1L, 0x4B04D447L, 0xD20D85FDL, 0xA50AB56BL,
 
50
        0x35B5A8FAL, 0x42B2986CL, 0xDBBBC9D6L, 0xACBCF940L,
 
51
        0x32D86CE3L, 0x45DF5C75L, 0xDCD60DCFL, 0xABD13D59L,
 
52
        0x26D930ACL, 0x51DE003AL, 0xC8D75180L, 0xBFD06116L,
 
53
        0x21B4F4B5L, 0x56B3C423L, 0xCFBA9599L, 0xB8BDA50FL,
 
54
        0x2802B89EL, 0x5F058808L, 0xC60CD9B2L, 0xB10BE924L,
 
55
        0x2F6F7C87L, 0x58684C11L, 0xC1611DABL, 0xB6662D3DL,
 
56
        0x76DC4190L, 0x01DB7106L, 0x98D220BCL, 0xEFD5102AL,
 
57
        0x71B18589L, 0x06B6B51FL, 0x9FBFE4A5L, 0xE8B8D433L,
 
58
        0x7807C9A2L, 0x0F00F934L, 0x9609A88EL, 0xE10E9818L,
 
59
        0x7F6A0DBBL, 0x086D3D2DL, 0x91646C97L, 0xE6635C01L,
 
60
        0x6B6B51F4L, 0x1C6C6162L, 0x856530D8L, 0xF262004EL,
 
61
        0x6C0695EDL, 0x1B01A57BL, 0x8208F4C1L, 0xF50FC457L,
 
62
        0x65B0D9C6L, 0x12B7E950L, 0x8BBEB8EAL, 0xFCB9887CL,
 
63
        0x62DD1DDFL, 0x15DA2D49L, 0x8CD37CF3L, 0xFBD44C65L,
 
64
        0x4DB26158L, 0x3AB551CEL, 0xA3BC0074L, 0xD4BB30E2L,
 
65
        0x4ADFA541L, 0x3DD895D7L, 0xA4D1C46DL, 0xD3D6F4FBL,
 
66
        0x4369E96AL, 0x346ED9FCL, 0xAD678846L, 0xDA60B8D0L,
 
67
        0x44042D73L, 0x33031DE5L, 0xAA0A4C5FL, 0xDD0D7CC9L,
 
68
        0x5005713CL, 0x270241AAL, 0xBE0B1010L, 0xC90C2086L,
 
69
        0x5768B525L, 0x206F85B3L, 0xB966D409L, 0xCE61E49FL,
 
70
        0x5EDEF90EL, 0x29D9C998L, 0xB0D09822L, 0xC7D7A8B4L,
 
71
        0x59B33D17L, 0x2EB40D81L, 0xB7BD5C3BL, 0xC0BA6CADL,
 
72
        0xEDB88320L, 0x9ABFB3B6L, 0x03B6E20CL, 0x74B1D29AL,
 
73
        0xEAD54739L, 0x9DD277AFL, 0x04DB2615L, 0x73DC1683L,
 
74
        0xE3630B12L, 0x94643B84L, 0x0D6D6A3EL, 0x7A6A5AA8L,
 
75
        0xE40ECF0BL, 0x9309FF9DL, 0x0A00AE27L, 0x7D079EB1L,
 
76
        0xF00F9344L, 0x8708A3D2L, 0x1E01F268L, 0x6906C2FEL,
 
77
        0xF762575DL, 0x806567CBL, 0x196C3671L, 0x6E6B06E7L,
 
78
        0xFED41B76L, 0x89D32BE0L, 0x10DA7A5AL, 0x67DD4ACCL,
 
79
        0xF9B9DF6FL, 0x8EBEEFF9L, 0x17B7BE43L, 0x60B08ED5L,
 
80
        0xD6D6A3E8L, 0xA1D1937EL, 0x38D8C2C4L, 0x4FDFF252L,
 
81
        0xD1BB67F1L, 0xA6BC5767L, 0x3FB506DDL, 0x48B2364BL,
 
82
        0xD80D2BDAL, 0xAF0A1B4CL, 0x36034AF6L, 0x41047A60L,
 
83
        0xDF60EFC3L, 0xA867DF55L, 0x316E8EEFL, 0x4669BE79L,
 
84
        0xCB61B38CL, 0xBC66831AL, 0x256FD2A0L, 0x5268E236L,
 
85
        0xCC0C7795L, 0xBB0B4703L, 0x220216B9L, 0x5505262FL,
 
86
        0xC5BA3BBEL, 0xB2BD0B28L, 0x2BB45A92L, 0x5CB36A04L,
 
87
        0xC2D7FFA7L, 0xB5D0CF31L, 0x2CD99E8BL, 0x5BDEAE1DL,
 
88
        0x9B64C2B0L, 0xEC63F226L, 0x756AA39CL, 0x026D930AL,
 
89
        0x9C0906A9L, 0xEB0E363FL, 0x72076785L, 0x05005713L,
 
90
        0x95BF4A82L, 0xE2B87A14L, 0x7BB12BAEL, 0x0CB61B38L,
 
91
        0x92D28E9BL, 0xE5D5BE0DL, 0x7CDCEFB7L, 0x0BDBDF21L,
 
92
        0x86D3D2D4L, 0xF1D4E242L, 0x68DDB3F8L, 0x1FDA836EL,
 
93
        0x81BE16CDL, 0xF6B9265BL, 0x6FB077E1L, 0x18B74777L,
 
94
        0x88085AE6L, 0xFF0F6A70L, 0x66063BCAL, 0x11010B5CL,
 
95
        0x8F659EFFL, 0xF862AE69L, 0x616BFFD3L, 0x166CCF45L,
 
96
        0xA00AE278L, 0xD70DD2EEL, 0x4E048354L, 0x3903B3C2L,
 
97
        0xA7672661L, 0xD06016F7L, 0x4969474DL, 0x3E6E77DBL,
 
98
        0xAED16A4AL, 0xD9D65ADCL, 0x40DF0B66L, 0x37D83BF0L,
 
99
        0xA9BCAE53L, 0xDEBB9EC5L, 0x47B2CF7FL, 0x30B5FFE9L,
 
100
        0xBDBDF21CL, 0xCABAC28AL, 0x53B39330L, 0x24B4A3A6L,
 
101
        0xBAD03605L, 0xCDD70693L, 0x54DE5729L, 0x23D967BFL,
 
102
        0xB3667A2EL, 0xC4614AB8L, 0x5D681B02L, 0x2A6F2B94L,
 
103
        0xB40BBE37L, 0xC30C8EA1L, 0x5A05DF1BL, 0x2D02EF8DL
 
104
};
 
105
 
 
106
class Crc32
 
107
{
 
108
private:
 
109
        quint32 result;
 
110
 
 
111
public:
 
112
        Crc32()
 
113
        {
 
114
                clear();
 
115
        }
 
116
 
 
117
        void clear()
 
118
        {
 
119
                result = 0xffffffff;
 
120
        }
 
121
 
 
122
        void update(const QByteArray &in)
 
123
        {
 
124
                for(int n = 0; n < in.size(); ++n)
 
125
                        result = (result >> 8) ^ (crctable[(result & 0xff) ^ (quint8)in[n]]);
 
126
        }
 
127
 
 
128
        quint32 final()
 
129
        {
 
130
                return result ^= 0xffffffff;
 
131
        }
 
132
 
 
133
        static quint32 process(const QByteArray &in)
 
134
        {
 
135
                Crc32 c;
 
136
                c.update(in);
 
137
                return c.final();
 
138
        }
 
139
};
 
140
 
 
141
static quint8 magic_cookie[4] = { 0x21, 0x12, 0xA4, 0x42 };
 
142
 
 
143
static quint16 read16(const quint8 *in)
 
144
{
 
145
        quint16 out = in[0];
 
146
        out <<= 8;
 
147
        out += in[1];
 
148
        return out;
 
149
}
 
150
 
 
151
static quint32 read32(const quint8 *in)
 
152
{
 
153
        quint32 out = in[0];
 
154
        out <<= 8;
 
155
        out += in[1];
 
156
        out <<= 8;
 
157
        out += in[2];
 
158
        out <<= 8;
 
159
        out += in[3];
 
160
        return out;
 
161
}
 
162
 
 
163
static void write16(quint8 *out, quint16 i)
 
164
{
 
165
        out[0] = (i >> 8) & 0xff;
 
166
        out[1] = i & 0xff;
 
167
}
 
168
 
 
169
static void write32(quint8 *out, quint32 i)
 
170
{
 
171
        out[0] = (i >> 24) & 0xff;
 
172
        out[1] = (i >> 16) & 0xff;
 
173
        out[2] = (i >> 8) & 0xff;
 
174
        out[3] = i & 0xff;
 
175
}
 
176
 
 
177
// do 3-field check of stun packet
 
178
// returns length of packet not counting the header, or -1 on error
 
179
static int check_and_get_length(const QByteArray &buf)
 
180
{
 
181
        // stun packets are at least 20 bytes
 
182
        if(buf.size() < 20)
 
183
                return -1;
 
184
 
 
185
        // minimal 3-field check
 
186
 
 
187
        // top 2 bits of packet must be 0
 
188
        if(buf[0] & 0xC0)
 
189
                return -1;
 
190
 
 
191
        const quint8 *p = (const quint8 *)buf.data();
 
192
        quint16 mlen = read16(p + 2);
 
193
 
 
194
        // bottom 2 bits of message length field must be 0
 
195
        if(mlen & 0x03)
 
196
                return -1;
 
197
 
 
198
        // (also, the message length should be a reasonable size)
 
199
        if(mlen + 20 > buf.size())
 
200
                return -1;
 
201
 
 
202
        // magic cookie must be set
 
203
        if(memcmp(p + 4, magic_cookie, 4) != 0)
 
204
                return -1;
 
205
 
 
206
        return mlen;
 
207
}
 
208
 
 
209
#define ATTRIBUTE_AREA_START  20
 
210
#define ATTRIBUTE_AREA_MAX    65535
 
211
#define ATTRIBUTE_VALUE_MAX   65531
 
212
 
 
213
// note: because the attribute area of the packet has a maximum size of
 
214
//   2^16-1, and each attribute itself has a 4 byte header, it follows that
 
215
//   the maximum size of an attribute's value is 2^16-5.  this means that,
 
216
//   even if padded with up to 3 bytes, the physical size of an attribute's
 
217
//   value will not overflow a 16-bit unsigned integer.
 
218
static quint16 round_up_length(quint16 in)
 
219
{
 
220
        Q_ASSERT(in <= ATTRIBUTE_VALUE_MAX);
 
221
        quint16 out = in;
 
222
        quint16 remainder = out % 4;
 
223
        if(remainder != 0)
 
224
                out += (4 - remainder);
 
225
        return out;
 
226
}
 
227
 
 
228
// buf    = entire stun packet
 
229
// offset = byte index of current attribute (first is offset=20)
 
230
// type   = take attribute type
 
231
// len    = take attribute value length (value is at offset + 4)
 
232
// returns offset of next attribute, -1 if no more
 
233
static int get_attribute_props(const QByteArray &buf, int offset, quint16 *type, int *len)
 
234
{
 
235
        Q_ASSERT(offset >= ATTRIBUTE_AREA_START);
 
236
 
 
237
        const quint8 *p = (const quint8 *)buf.data();
 
238
 
 
239
        // need at least 4 bytes for an attribute
 
240
        if(offset + 4 > buf.size())
 
241
                return -1;
 
242
 
 
243
        quint16 _type = read16(p + offset);
 
244
        offset += 2;
 
245
        quint16 _alen = read16(p + offset);
 
246
        offset += 2;
 
247
 
 
248
        // get physical length.  stun attributes are 4-byte aligned, and may
 
249
        //   contain 0-3 bytes of padding.
 
250
        quint16 plen = round_up_length(_alen);
 
251
        if(offset + plen > buf.size())
 
252
                return -1;
 
253
 
 
254
        *type = _type;
 
255
        *len = _alen;
 
256
        return offset + plen;
 
257
}
 
258
 
 
259
// buf    = entire stun packet
 
260
// type   = attribute type to find
 
261
// len    = take attribute value length (value is at offset + 4)
 
262
// next   = take offset of next attribute
 
263
// returns offset of found attribute, -1 if not found
 
264
static int find_attribute(const QByteArray &buf, quint16 type, int *len, int *next = 0)
 
265
{
 
266
        int at = ATTRIBUTE_AREA_START;
 
267
        quint16 _type;
 
268
        int _len;
 
269
        int _next;
 
270
 
 
271
        while(1)
 
272
        {
 
273
                _next = get_attribute_props(buf, at, &_type, &_len);
 
274
                if(_next == -1)
 
275
                        break;
 
276
                if(_type == type)
 
277
                {
 
278
                        *len = _len;
 
279
                        if(next)
 
280
                                *next = _next;
 
281
                        return at;
 
282
                }
 
283
                at = _next;
 
284
        }
 
285
 
 
286
        return -1;
 
287
}
 
288
 
 
289
// buf  = stun packet to append attribute to
 
290
// type = type of attribute
 
291
// len  = length of value
 
292
// returns offset of new attribute, or -1 if it can't fit
 
293
// note: attribute value is located at offset + 4 and is uninitialized
 
294
// note: padding following attribute is zeroed out
 
295
static int append_attribute_uninitialized(QByteArray *buf, quint16 type, int len)
 
296
{
 
297
        if(len > ATTRIBUTE_VALUE_MAX)
 
298
                return -1;
 
299
 
 
300
        quint16 alen = (quint16)len;
 
301
        quint16 plen = round_up_length(alen);
 
302
 
 
303
        if((buf->size() - ATTRIBUTE_AREA_START) + 4 + plen > ATTRIBUTE_AREA_MAX)
 
304
                return -1;
 
305
 
 
306
        int at = buf->size();
 
307
        buf->resize(buf->size() + 4 + plen);
 
308
        quint8 *p = (quint8 *)buf->data();
 
309
 
 
310
        write16(p + at, type);
 
311
        write16(p + at + 2, alen);
 
312
 
 
313
        // padding
 
314
        for(int n = 0; n < plen - alen; ++n)
 
315
                p[at + alen + n] = 0;
 
316
 
 
317
        return at;
 
318
}
 
319
 
 
320
static quint32 fingerprint_calc(const quint8 *buf, int size)
 
321
{
 
322
        QByteArray region = QByteArray::fromRawData((const char *)buf, size);
 
323
        return Crc32::process(region) ^ 0x5354554e;
 
324
}
 
325
 
 
326
static QByteArray message_integrity_calc(const quint8 *buf, int size, const QByteArray &key)
 
327
{
 
328
        QCA::MessageAuthenticationCode hmac("hmac(sha1)", key);
 
329
        QByteArray region = QByteArray::fromRawData((const char *)buf, size);
 
330
        QByteArray result = hmac.process(region).toByteArray();
 
331
        Q_ASSERT(result.size() == 20);
 
332
        return result;
 
333
}
 
334
 
 
335
// look for fingerprint attribute and confirm it
 
336
// buf = entire stun packet
 
337
// returns true if fingerprint attribute exists and is correct
 
338
static bool fingerprint_check(const QByteArray &buf)
 
339
{
 
340
        int at, len;
 
341
        at = find_attribute(buf, AttribFingerprint, &len);
 
342
        if(at == -1 || len != 4) // value must be 4 bytes
 
343
                return false;
 
344
 
 
345
        const quint8 *p = (const quint8 *)buf.data();
 
346
        quint32 fpval = read32(p + at + 4);
 
347
        quint32 fpcalc = fingerprint_calc(p, at);
 
348
        if(fpval == fpcalc)
 
349
                return true;
 
350
        else
 
351
                return false;
 
352
}
 
353
 
 
354
// copy the input buffer and prepare for message integrity checking.  the
 
355
//   packet is truncated after the message-integrity attribute (since nothing
 
356
//   after it is protected), and the packet length is adjusted in the header
 
357
//   accordingly.
 
358
// buf    = input stun packet
 
359
// out    = take output stun packet
 
360
// offset = take offset of message-integrity attribute
 
361
// returns true if message-integrity attribute exists and packet is prepared
 
362
// note: message-integrity value is at offset + 4 and is exactly 20 bytes
 
363
static bool message_integrity_prep(const QByteArray &buf, QByteArray *out, int *offset)
 
364
{
 
365
        int at, len, next;
 
366
        at = find_attribute(buf, AttribMessageIntegrity, &len, &next);
 
367
        if(at == -1 || len != 20) // value must be 20 bytes
 
368
                return false;
 
369
 
 
370
        // prepare new attribute area size
 
371
        int i = next - ATTRIBUTE_AREA_START;
 
372
 
 
373
        // new value must be divisible by 4
 
374
        if(i % 4 != 0)
 
375
                return false;
 
376
 
 
377
        // copy truncated packet
 
378
        *out = buf.mid(0, next);
 
379
 
 
380
        // set new length in header
 
381
        quint16 newlen = (quint16)i;
 
382
        write16((quint8 *)out->data() + 2, newlen);
 
383
 
 
384
        *offset = at;
 
385
        return true;
 
386
}
 
387
 
 
388
// confirm message integrity
 
389
// buf    = prepared stun packet (from message_integrity_prep())
 
390
// offset = offset of message-integrity attribute
 
391
// key    = the HMAC key
 
392
// returns true if correct
 
393
static bool message_integrity_check(const QByteArray &buf, int offset, const QByteArray &key)
 
394
{
 
395
        QByteArray mival = QByteArray::fromRawData(buf.data() + offset + 4, 20);
 
396
        QByteArray micalc = message_integrity_calc((const quint8 *)buf.data(), offset, key);
 
397
        if(mival == micalc)
 
398
                return true;
 
399
        else
 
400
                return false;
 
401
}
 
402
 
 
403
class StunMessage::Private : public QSharedData
 
404
{
 
405
public:
 
406
        StunMessage::Class mclass;
 
407
        quint16 method;
 
408
        quint8 magic[4];
 
409
        quint8 id[12];
 
410
        QList<Attribute> attribs;
 
411
 
 
412
        Private()
 
413
        {
 
414
                mclass = (StunMessage::Class)-1;
 
415
                method = 0;
 
416
                memcpy(magic, magic_cookie, 4);
 
417
                memset(id, 0, 12);
 
418
        }
 
419
};
 
420
 
 
421
StunMessage::StunMessage() :
 
422
        d(0)
 
423
{
 
424
}
 
425
 
 
426
StunMessage::StunMessage(const StunMessage &from) :
 
427
        d(from.d)
 
428
{
 
429
}
 
430
 
 
431
StunMessage::~StunMessage()
 
432
{
 
433
}
 
434
 
 
435
StunMessage & StunMessage::operator=(const StunMessage &from)
 
436
{
 
437
        d = from.d;
 
438
        return *this;
 
439
}
 
440
 
 
441
bool StunMessage::isNull() const
 
442
{
 
443
        return (d ? false: true);
 
444
}
 
445
 
 
446
StunMessage::Class StunMessage::mclass() const
 
447
{
 
448
        Q_ASSERT(d);
 
449
        return d->mclass;
 
450
}
 
451
 
 
452
quint16 StunMessage::method() const
 
453
{
 
454
        Q_ASSERT(d);
 
455
        return d->method;
 
456
}
 
457
 
 
458
const quint8 *StunMessage::magic() const
 
459
{
 
460
        Q_ASSERT(d);
 
461
        return d->magic;
 
462
}
 
463
 
 
464
const quint8 *StunMessage::id() const
 
465
{
 
466
        Q_ASSERT(d);
 
467
        return d->id;
 
468
}
 
469
 
 
470
QList<StunMessage::Attribute> StunMessage::attributes() const
 
471
{
 
472
        Q_ASSERT(d);
 
473
        return d->attribs;
 
474
}
 
475
 
 
476
QByteArray StunMessage::attribute(quint16 type) const
 
477
{
 
478
        Q_ASSERT(d);
 
479
 
 
480
        foreach(const Attribute &i, d->attribs)
 
481
        {
 
482
                if(i.type == type)
 
483
                        return i.value;
 
484
        }
 
485
        return QByteArray();
 
486
}
 
487
 
 
488
void StunMessage::setClass(Class mclass)
 
489
{
 
490
        ENSURE_D
 
491
        d->mclass = mclass;
 
492
}
 
493
 
 
494
void StunMessage::setMethod(quint16 method)
 
495
{
 
496
        ENSURE_D
 
497
        d->method = method;
 
498
}
 
499
 
 
500
void StunMessage::setMagic(const quint8 *magic)
 
501
{
 
502
        ENSURE_D
 
503
        memcpy(d->magic, magic, 4);
 
504
}
 
505
 
 
506
void StunMessage::setId(const quint8 *id)
 
507
{
 
508
        ENSURE_D
 
509
        memcpy(d->id, id, 12);
 
510
}
 
511
 
 
512
void StunMessage::setAttributes(const QList<Attribute> &attribs)
 
513
{
 
514
        ENSURE_D
 
515
        d->attribs = attribs;
 
516
}
 
517
 
 
518
QByteArray StunMessage::toBinary(int validationFlags, const QByteArray &key) const
 
519
{
 
520
        Q_ASSERT(d);
 
521
 
 
522
        // header
 
523
        QByteArray buf(20, 0);
 
524
        quint8 *p = (quint8 *)buf.data();
 
525
 
 
526
        quint8 classbits = 0;
 
527
        if(d->mclass == Request)
 
528
                classbits = 0; // 00
 
529
        else if(d->mclass == Indication)
 
530
                classbits = 1; // 01
 
531
        else if(d->mclass == SuccessResponse)
 
532
                classbits = 2; // 10
 
533
        else if(d->mclass == ErrorResponse)
 
534
                classbits = 3; // 11
 
535
        else
 
536
                Q_ASSERT(0);
 
537
 
 
538
        // method bits are split into 3 sections
 
539
        quint16 m1, m2, m3;
 
540
        m1 = d->method & 0x0f80; // M7-11
 
541
        m1 <<= 2;
 
542
        m2 = d->method & 0x0070; // M4-6
 
543
        m2 <<= 1;
 
544
        m3 = d->method & 0x000f; // M0-3
 
545
 
 
546
        // class bits are split into 2 sections
 
547
        quint16 c1, c2;
 
548
        c1 = classbits & 0x02; // C1
 
549
        c1 <<= 7;
 
550
        c2 = classbits & 0x01; // C0
 
551
        c2 <<= 4;
 
552
 
 
553
        quint16 type = m1 | m2 | m3 | c1 | c2;
 
554
        write16(p, type);
 
555
        write16(p + 2, 0);
 
556
        memcpy(p + 4, d->magic, 4);
 
557
        memcpy(p + 8, d->id, 12);
 
558
 
 
559
        foreach(const Attribute &i, d->attribs)
 
560
        {
 
561
                int at = append_attribute_uninitialized(&buf, i.type, i.value.size());
 
562
                if(at == -1)
 
563
                        return QByteArray();
 
564
 
 
565
                memcpy(buf.data() + at + 4, i.value.data(), i.value.size());
 
566
        }
 
567
 
 
568
        // set attribute area size
 
569
        write16(p + 2, buf.size() - ATTRIBUTE_AREA_START);
 
570
 
 
571
        if(validationFlags & MessageIntegrity)
 
572
        {
 
573
                quint16 alen = 20; // size of hmac(sha1)
 
574
                int at = append_attribute_uninitialized(&buf, AttribMessageIntegrity, alen);
 
575
                if(at == -1)
 
576
                        return QByteArray();
 
577
 
 
578
                p = (quint8 *)buf.data(); // follow the resize
 
579
 
 
580
                // set attribute area size to include the new attribute
 
581
                write16(p + 2, buf.size() - ATTRIBUTE_AREA_START);
 
582
 
 
583
                // now calculate the hash and fill in the value
 
584
                QByteArray result = message_integrity_calc(p, at, key);
 
585
                Q_ASSERT(result.size() == alen);
 
586
                memcpy(p + at + 4, result.data(), alen);
 
587
        }
 
588
 
 
589
        if(validationFlags & Fingerprint)
 
590
        {
 
591
                quint16 alen = 4; // size of crc32
 
592
                int at = append_attribute_uninitialized(&buf, AttribFingerprint, alen);
 
593
                if(at == -1)
 
594
                        return QByteArray();
 
595
 
 
596
                p = (quint8 *)buf.data(); // follow the resize
 
597
 
 
598
                // set attribute area size to include the new attribute
 
599
                write16(p + 2, buf.size() - ATTRIBUTE_AREA_START);
 
600
 
 
601
                // now calculate the fingerprint and fill in the value
 
602
                quint32 fpcalc = fingerprint_calc(p, at);
 
603
                write32(p + at + 4, fpcalc);
 
604
        }
 
605
 
 
606
        return buf;
 
607
}
 
608
 
 
609
StunMessage StunMessage::fromBinary(const QByteArray &a, ConvertResult *result, int validationFlags, const QByteArray &key)
 
610
{
 
611
        int mlen = check_and_get_length(a);
 
612
        if(mlen == -1)
 
613
        {
 
614
                if(result)
 
615
                        *result = ErrorFormat;
 
616
                return StunMessage();
 
617
        }
 
618
 
 
619
        if(validationFlags & Fingerprint)
 
620
        {
 
621
                if(!fingerprint_check(a))
 
622
                {
 
623
                        if(result)
 
624
                                *result = ErrorFingerprint;
 
625
                        return StunMessage();
 
626
                }
 
627
        }
 
628
 
 
629
        QByteArray in;
 
630
 
 
631
        if(validationFlags & MessageIntegrity)
 
632
        {
 
633
                int offset;
 
634
                if(!message_integrity_prep(a, &in, &offset))
 
635
                {
 
636
                        if(result)
 
637
                                *result = ErrorMessageIntegrity;
 
638
                        return StunMessage();
 
639
                }
 
640
 
 
641
                if(!message_integrity_check(in, offset, key))
 
642
                {
 
643
                        if(result)
 
644
                                *result = ErrorMessageIntegrity;
 
645
                        return StunMessage();
 
646
                }
 
647
        }
 
648
        else
 
649
                in = a;
 
650
 
 
651
        // all validating complete, now just parse the packet
 
652
 
 
653
        const quint8 *p = (const quint8 *)in.data();
 
654
 
 
655
        // method bits are split into 3 sections
 
656
        quint16 m1, m2, m3;
 
657
        m1 = p[0] & 0x3e; // M7-11
 
658
        m1 <<= 6;
 
659
        m2 = p[1] & 0xe0; // M4-6
 
660
        m2 >>= 1;
 
661
        m3 = p[1] & 0x0f; // M0-3
 
662
 
 
663
        // class bits are split into 2 sections
 
664
        quint8 c1, c2;
 
665
        c1 = p[0] & 0x01; // C1
 
666
        c1 <<= 1;
 
667
        c2 = p[1] & 0x10; // C0
 
668
        c2 >>= 4;
 
669
 
 
670
        quint16 method = m1 | m2 | m3;
 
671
        quint8 classbits = c1 | c2;
 
672
 
 
673
        Class mclass;
 
674
        if(classbits == 0) // 00
 
675
                mclass = Request;
 
676
        else if(classbits == 1) // 01
 
677
                mclass = Indication;
 
678
        else if(classbits == 2) // 10
 
679
                mclass = SuccessResponse;
 
680
        else // 11
 
681
                mclass = ErrorResponse;
 
682
 
 
683
        StunMessage out;
 
684
        out.setClass(mclass);
 
685
        out.setMethod(method);
 
686
        out.setMagic(p + 4);
 
687
        out.setId(p + 8);
 
688
 
 
689
        QList<Attribute> list;
 
690
        int at = ATTRIBUTE_AREA_START;
 
691
        while(1)
 
692
        {
 
693
                quint16 type;
 
694
                int len;
 
695
                int next;
 
696
 
 
697
                next = get_attribute_props(in, at, &type, &len);
 
698
                if(next == -1)
 
699
                        break;
 
700
 
 
701
                Attribute attrib;
 
702
                attrib.type = type;
 
703
                attrib.value = in.mid(at + 4, len);
 
704
                list += attrib;
 
705
 
 
706
                at = next;
 
707
        }
 
708
        out.setAttributes(list);
 
709
 
 
710
        if(result)
 
711
                *result = ConvertGood;
 
712
        return out;
 
713
}
 
714
 
 
715
bool StunMessage::isProbablyStun(const QByteArray &a)
 
716
{
 
717
        return (check_and_get_length(a) != -1 ? true: false);
 
718
}
 
719
 
 
720
}