discovery.c 31.7 KB
Newer Older
1
2
3
/*
 * This file is part of the Nice GLib ICE library.
 *
4
5
6
 * (C) 2008-2009 Collabora Ltd.
 *  Contact: Youness Alaoui
 * (C) 2007-2009 Nokia Corporation. All rights reserved.
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
 *  Contact: Kai Vehmanen
 *
 * The contents of this file are subject to the Mozilla Public License Version
 * 1.1 (the "License"); you may not use this file except in compliance with
 * the License. You may obtain a copy of the License at
 * http://www.mozilla.org/MPL/
 *
 * Software distributed under the License is distributed on an "AS IS" basis,
 * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
 * for the specific language governing rights and limitations under the
 * License.
 *
 * The Original Code is the Nice GLib ICE library.
 *
 * The Initial Developers of the Original Code are Collabora Ltd and Nokia
 * Corporation. All Rights Reserved.
 *
 * Contributors:
25
 *   Youness Alaoui, Collabora Ltd.
26
27
28
29
30
31
32
33
34
35
36
37
38
 *   Kai Vehmanen, Nokia
 *
 * Alternatively, the contents of this file may be used under the terms of the
 * the GNU Lesser General Public License Version 2.1 (the "LGPL"), in which
 * case the provisions of LGPL are applicable instead of those above. If you
 * wish to allow use of your version of this file only under the terms of the
 * LGPL and not to allow others to use your version of this file under the
 * MPL, indicate your decision by deleting the provisions above and replace
 * them with the notice and other provisions required by the LGPL. If you do
 * not delete the provisions above, a recipient may use your version of this
 * file under either the MPL or the LGPL.
 */

39
/*
40
41
42
 * @file discovery.c
 * @brief ICE candidate discovery functions
 */
43

44
45
#ifdef HAVE_CONFIG_H
# include <config.h>
46
#endif
47

48
49
#include <glib.h>

50
51
52
#include <stdlib.h>
#include <string.h>
#include <errno.h>
53

54
55
#include "debug.h"

56
57
58
59
60
#include "agent.h"
#include "agent-priv.h"
#include "agent-signals-marshal.h"
#include "component.h"
#include "discovery.h"
61
62
#include "stun/usages/bind.h"
#include "stun/usages/turn.h"
63
#include "socket.h"
64

65
static inline int priv_timer_expired (GTimeVal *timer, GTimeVal *now)
66
67
68
69
70
71
{
  return (now->tv_sec == timer->tv_sec) ?
    now->tv_usec >= timer->tv_usec :
    now->tv_sec >= timer->tv_sec;
}

72
/*
73
 * Frees the CandidateDiscovery structure pointed to
74
75
76
77
78
79
 * by 'user data'. Compatible with g_slist_foreach().
 */
void discovery_free_item (gpointer data, gpointer user_data)
{
  CandidateDiscovery *cand = data;
  g_assert (user_data == NULL);
80
81
  g_free (cand->msn_turn_username);
  g_free (cand->msn_turn_password);
82
83
84
  g_slice_free (CandidateDiscovery, cand);
}

85
/*
86
87
88
89
 * Frees all discovery related resources for the agent.
 */
void discovery_free (NiceAgent *agent)
{
90

91
92
93
94
  g_slist_foreach (agent->discovery_list, discovery_free_item, NULL);
  g_slist_free (agent->discovery_list);
  agent->discovery_list = NULL;
  agent->discovery_unsched_items = 0;
95

96
97
98
99
100
  if (agent->discovery_timer_source != NULL) {
    g_source_destroy (agent->discovery_timer_source);
    g_source_unref (agent->discovery_timer_source);
    agent->discovery_timer_source = NULL;
  }
101
102
}

103
/*
104
 * Prunes the list of discovery processes for items related
Youness Alaoui's avatar
Youness Alaoui committed
105
 * to stream 'stream_id'.
106
107
108
 *
 * @return TRUE on success, FALSE on a fatal error
 */
109
void discovery_prune_stream (NiceAgent *agent, guint stream_id)
110
111
112
113
{
  GSList *i;

  for (i = agent->discovery_list; i ; ) {
114
115
    CandidateDiscovery *cand = i->data;
    GSList *next = i->next;
116
117

    if (cand->stream->id == stream_id) {
118
      agent->discovery_list = g_slist_remove (agent->discovery_list, cand);
119
120
      discovery_free_item (cand, NULL);
    }
121
    i = next;
122
123
  }

124
125
126
127
128
129
  if (agent->discovery_list == NULL) {
    /* noone using the timer anymore, clean it up */
    discovery_free (agent);
  }
}

130

131
/*
132
133
134
135
136
137
 * Frees the CandidateDiscovery structure pointed to
 * by 'user data'. Compatible with g_slist_foreach().
 */
void refresh_free_item (gpointer data, gpointer user_data)
{
  CandidateRefresh *cand = data;
138
139
140
141
142
143
  NiceAgent *agent = cand->agent;
  uint8_t *username;
  size_t username_len;
  uint8_t *password;
  size_t password_len;
  size_t buffer_len = 0;
144
  StunUsageTurnCompatibility turn_compat = agent_to_turn_compatibility (agent);
145
146
147
148
149
150
151
152
153
154
155
156
157
158

  g_assert (user_data == NULL);

  if (cand->timer_source != NULL) {
    g_source_destroy (cand->timer_source);
    g_source_unref (cand->timer_source);
    cand->timer_source = NULL;
  }
  if (cand->tick_source != NULL) {
    g_source_destroy (cand->tick_source);
    g_source_unref (cand->tick_source);
    cand->tick_source = NULL;
  }

159
160
161
162
  username = (uint8_t *)cand->turn->username;
  username_len = (size_t) strlen (cand->turn->username);
  password = (uint8_t *)cand->turn->password;
  password_len = (size_t) strlen (cand->turn->password);
163

164
165
  if (turn_compat == STUN_USAGE_TURN_COMPATIBILITY_MSN ||
      turn_compat == STUN_USAGE_TURN_COMPATIBILITY_OC2007) {
166
167
168
169
    username = g_base64_decode ((gchar *)username, &username_len);
    password = g_base64_decode ((gchar *)password, &password_len);
  }

170
  buffer_len = stun_usage_turn_create_refresh (&cand->stun_agent,
171
172
173
174
      &cand->stun_message,  cand->stun_buffer, sizeof(cand->stun_buffer),
      cand->stun_resp_msg.buffer == NULL ? NULL : &cand->stun_resp_msg, 0,
      username, username_len,
      password, password_len,
175
      agent_to_turn_compatibility (agent));
176
177

  if (buffer_len > 0) {
178
179
180
181
182
183
184
    StunTransactionId id;

    /* forget the transaction since we don't care about the result and
     * we don't implement retransmissions/timeout */
    stun_message_id (&cand->stun_message, id);
    stun_agent_forget_transaction (&cand->stun_agent, id);

185
186
187
    /* send the refresh twice since we won't do retransmissions */
    nice_socket_send (cand->nicesock, &cand->server,
        buffer_len, (gchar *)cand->stun_buffer);
188
189
190
191
    if (!nice_socket_is_reliable (cand->nicesock)) {
      nice_socket_send (cand->nicesock, &cand->server,
          buffer_len, (gchar *)cand->stun_buffer);
    }
192
193
194

  }

195
196
  if (turn_compat == STUN_USAGE_TURN_COMPATIBILITY_MSN ||
      turn_compat == STUN_USAGE_TURN_COMPATIBILITY_OC2007) {
197
198
199
    g_free (username);
    g_free (password);
  }
200
201
202
203

  g_slice_free (CandidateRefresh, cand);
}

204
/*
205
206
207
208
 * Frees all discovery related resources for the agent.
 */
void refresh_free (NiceAgent *agent)
{
209
210
211
  g_slist_foreach (agent->refresh_list, refresh_free_item, NULL);
  g_slist_free (agent->refresh_list);
  agent->refresh_list = NULL;
212
213
}

214
/*
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
 * Prunes the list of discovery processes for items related
 * to stream 'stream_id'. 
 *
 * @return TRUE on success, FALSE on a fatal error
 */
void refresh_prune_stream (NiceAgent *agent, guint stream_id)
{
  GSList *i;

  for (i = agent->refresh_list; i ;) {
    CandidateRefresh *cand = i->data;
    GSList *next = i->next;

    if (cand->stream->id == stream_id) {
      agent->refresh_list = g_slist_remove (agent->refresh_list, cand);
      refresh_free_item (cand, NULL);
    }

    i = next;
  }

}

void refresh_cancel (CandidateRefresh *refresh)
{
  refresh->agent->refresh_list = g_slist_remove (refresh->agent->refresh_list,
      refresh);
  refresh_free_item (refresh, NULL);
}

245
/*
246
 * Adds a new local candidate. Implements the candidate pruning
247
 * defined in ICE spec section 4.1.3 "Eliminating Redundant
248
 * Candidates" (ID-19).
249
250
251
 */
static gboolean priv_add_local_candidate_pruned (Component *component, NiceCandidate *candidate)
{
252
  GSList *i;
253
254
255

  for (i = component->local_candidates; i ; i = i->next) {
    NiceCandidate *c = i->data;
256

257
258
    if (nice_address_equal (&c->base_addr, &candidate->base_addr) &&
	nice_address_equal (&c->addr, &candidate->addr)) {
259
      nice_debug ("Candidate %p (component-id %u) redundant, ignoring.", candidate, component->id);
260
261
262
263
      return FALSE;
    }
  }

264
265
  component->local_candidates = g_slist_append (component->local_candidates,
      candidate);
266
267
268
269

  return TRUE;
}

270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
static guint priv_highest_remote_foundation (Component *component)
{
  GSList *i;
  guint highest = 1;
  gchar foundation[NICE_CANDIDATE_MAX_FOUNDATION];

  for (highest = 1;; highest++) {
    gboolean taken = FALSE;

    g_snprintf (foundation, NICE_CANDIDATE_MAX_FOUNDATION, "%u", highest);
    for (i = component->remote_candidates; i; i = i->next) {
      NiceCandidate *cand = i->data;
      if (strncmp (foundation, cand->foundation,
              NICE_CANDIDATE_MAX_FOUNDATION) == 0) {
        taken = TRUE;
        break;
      }
    }
    if (!taken)
      return highest;
  }

  g_return_val_if_reached (highest);
}

295
/*
296
297
 * Assings a foundation to the candidate.
 *
298
 * Implements the mechanism described in ICE sect
299
 * 4.1.1.3 "Computing Foundations" (ID-19).
300
 */
301
static void priv_assign_foundation (NiceAgent *agent, NiceCandidate *candidate)
302
{
303
304
305
306
307
308
309
310
311
  GSList *i, *j, *k;

  for (i = agent->streams; i; i = i->next) {
    Stream *stream = i->data;
    for (j = stream->components; j; j = j->next) {
      Component *component = j->data;
      for (k = component->local_candidates; k; k = k->next) {
	NiceCandidate *n = k->data;
	NiceAddress temp = n->base_addr;
312
313
314
315
316

	/* note: candidate must not on the local candidate list */
	g_assert (candidate != n);

	/* note: ports are not to be compared */
317
318
	nice_address_set_port (&temp,
               nice_address_get_port (&candidate->base_addr));
319

320
	if (candidate->type == n->type &&
321
            candidate->transport == n->transport &&
322
            candidate->stream_id == n->stream_id &&
323
324
	    nice_address_equal (&candidate->base_addr, &temp) &&
            !(agent->compatibility == NICE_COMPATIBILITY_GOOGLE &&
325
                n->type == NICE_CANDIDATE_TYPE_RELAYED)) {
326
327
328
329
	  /* note: currently only one STUN/TURN server per stream at a
	   *       time is supported, so there is no need to check
	   *       for candidates that would otherwise share the
	   *       foundation, but have different STUN/TURN servers */
330
331
	  g_strlcpy (candidate->foundation, n->foundation,
              NICE_CANDIDATE_MAX_FOUNDATION);
332
333
334
335
336
337
338
339
          if (n->username) {
            g_free (candidate->username);
            candidate->username = g_strdup (n->username);
          }
          if (n->password) {
            g_free (candidate->password);
            candidate->password = g_strdup (n->password);
          }
340
341
342
	  return;
	}
      }
343
344
    }
  }
Youness Alaoui's avatar
Youness Alaoui committed
345

346
347
  g_snprintf (candidate->foundation, NICE_CANDIDATE_MAX_FOUNDATION,
      "%u", agent->next_candidate_id++);
348
349
}

350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
static void priv_assign_remote_foundation (NiceAgent *agent, NiceCandidate *candidate)
{
  GSList *i, *j, *k;
  guint next_remote_id;
  Component *component = NULL;

  for (i = agent->streams; i; i = i->next) {
    Stream *stream = i->data;
    for (j = stream->components; j; j = j->next) {
      Component *c = j->data;

      if (c->id == candidate->component_id)
        component = c;

      for (k = c->remote_candidates; k; k = k->next) {
	NiceCandidate *n = k->data;
366
	NiceAddress temp = n->addr;
367
368
369
370
371
372
373
374
375
376

	/* note: candidate must not on the remote candidate list */
	g_assert (candidate != n);

	/* note: ports are not to be compared */
	nice_address_set_port (&temp,
               nice_address_get_port (&candidate->base_addr));

	if (candidate->type == n->type &&
            candidate->stream_id == n->stream_id &&
377
	    nice_address_equal (&candidate->addr, &temp)) {
378
379
380
381
	  /* note: currently only one STUN/TURN server per stream at a
	   *       time is supported, so there is no need to check
	   *       for candidates that would otherwise share the
	   *       foundation, but have different STUN/TURN servers */
382
383
	  g_strlcpy (candidate->foundation, n->foundation,
              NICE_CANDIDATE_MAX_FOUNDATION);
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
          if (n->username) {
            g_free (candidate->username);
            candidate->username = g_strdup (n->username);
          }
          if (n->password) {
            g_free (candidate->password);
            candidate->password = g_strdup (n->password);
          }
	  return;
	}
      }
    }
  }

  if (component) {
    next_remote_id = priv_highest_remote_foundation (component);
    g_snprintf (candidate->foundation, NICE_CANDIDATE_MAX_FOUNDATION,
        "%u", next_remote_id);
  }
}


406
static
407
408
void priv_generate_candidate_credentials (NiceAgent *agent,
    NiceCandidate *candidate)
409
410
{

Jakub Adam's avatar
Jakub Adam committed
411
412
  if (agent->compatibility == NICE_COMPATIBILITY_MSN ||
      agent->compatibility == NICE_COMPATIBILITY_OC2007) {
413
414
415
    guchar username[32];
    guchar password[16];

Olivier Crête's avatar
Olivier Crête committed
416
417
    g_free (candidate->username);
    g_free (candidate->password);
418
419
420
421
422
423
424

    nice_rng_generate_bytes (agent->rng, 32, (gchar *)username);
    nice_rng_generate_bytes (agent->rng, 16, (gchar *)password);

    candidate->username = g_base64_encode (username, 32);
    candidate->password = g_base64_encode (password, 16);

425
426
427
  } else if (agent->compatibility == NICE_COMPATIBILITY_GOOGLE) {
    gchar username[16];

Olivier Crête's avatar
Olivier Crête committed
428
429
    g_free (candidate->username);
    g_free (candidate->password);
430
431
432
433
    candidate->password = NULL;

    nice_rng_generate_bytes_print (agent->rng, 16, (gchar *)username);

434
    candidate->username = g_strndup (username, 16);
435
436
437
438
439
  }


}

440
/*
441
442
443
444
445
446
447
448
449
450
451
452
453
454
 * Creates a local host candidate for 'component_id' of stream
 * 'stream_id'.
 *
 * @return pointer to the created candidate, or NULL on error
 */
NiceCandidate *discovery_add_local_host_candidate (
  NiceAgent *agent,
  guint stream_id,
  guint component_id,
  NiceAddress *address)
{
  NiceCandidate *candidate;
  Component *component;
  Stream *stream;
455
  NiceSocket *udp_socket = NULL;
456
457
458
459
460

  if (!agent_find_component (agent, stream_id, component_id, &stream, &component))
    return NULL;

  candidate = nice_candidate_new (NICE_CANDIDATE_TYPE_HOST);
461
462
463
464
465
466
  candidate->stream_id = stream_id;
  candidate->component_id = component_id;
  candidate->addr = *address;
  candidate->base_addr = *address;
  if (agent->compatibility == NICE_COMPATIBILITY_GOOGLE) {
    candidate->priority = nice_candidate_jingle_priority (candidate);
Jakub Adam's avatar
Jakub Adam committed
467
468
  } else if (agent->compatibility == NICE_COMPATIBILITY_MSN ||
             agent->compatibility == NICE_COMPATIBILITY_OC2007)  {
469
470
471
472
    candidate->priority = nice_candidate_msn_priority (candidate);
  } else {
    candidate->priority = nice_candidate_ice_priority (candidate);
  }
473

474
475
476
477
478
479
  priv_generate_candidate_credentials (agent, candidate);
  priv_assign_foundation (agent, candidate);

  /* note: candidate username and password are left NULL as stream
     level ufrag/password are used */
  udp_socket = nice_udp_bsd_socket_new (address);
480
481
  if (!udp_socket)
    goto errors;
482
483


484
485
486
  _priv_set_socket_tos (agent, udp_socket, stream->tos);
  agent_attach_stream_component_socket (agent, stream,
      component, udp_socket);
487

488
489
490
  candidate->sockptr = udp_socket;
  candidate->addr = udp_socket->addr;
  candidate->base_addr = udp_socket->addr;
491

492
493
  if (!priv_add_local_candidate_pruned (component, candidate))
    goto errors;
494

495
  component->sockets = g_slist_append (component->sockets, udp_socket);
Youness Alaoui's avatar
Youness Alaoui committed
496

497
  return candidate;
498
499
500
501
502
503

errors:
  nice_candidate_free (candidate);
  if (udp_socket)
    nice_socket_free (udp_socket);
  return NULL;
504
505
}

506
/*
507
508
509
510
511
 * Creates a server reflexive candidate for 'component_id' of stream
 * 'stream_id'.
 *
 * @return pointer to the created candidate, or NULL on error
 */
Youness Alaoui's avatar
Youness Alaoui committed
512
NiceCandidate*
513
514
515
516
517
discovery_add_server_reflexive_candidate (
  NiceAgent *agent,
  guint stream_id,
  guint component_id,
  NiceAddress *address,
518
  NiceSocket *base_socket)
519
520
521
522
{
  NiceCandidate *candidate;
  Component *component;
  Stream *stream;
523
  gboolean result = FALSE;
524
525
526
527
528

  if (!agent_find_component (agent, stream_id, component_id, &stream, &component))
    return NULL;

  candidate = nice_candidate_new (NICE_CANDIDATE_TYPE_SERVER_REFLEXIVE);
529
530
531

  if (agent->compatibility == NICE_COMPATIBILITY_GOOGLE) {
    candidate->priority = nice_candidate_jingle_priority (candidate);
Jakub Adam's avatar
Jakub Adam committed
532
533
  } else if (agent->compatibility == NICE_COMPATIBILITY_MSN ||
             agent->compatibility == NICE_COMPATIBILITY_OC2007)  {
534
535
536
    candidate->priority = nice_candidate_msn_priority (candidate);
  } else {
    candidate->priority =  nice_candidate_ice_priority_full
537
        (NICE_CANDIDATE_TYPE_PREF_SERVER_REFLEXIVE, 0, component_id);
538
539
540
541
  }
  candidate->stream_id = stream_id;
  candidate->component_id = component_id;
  candidate->addr = *address;
542

543
544
545
  /* step: link to the base candidate+socket */
  candidate->sockptr = base_socket;
  candidate->base_addr = base_socket->addr;
546

547
548
  priv_generate_candidate_credentials (agent, candidate);
  priv_assign_foundation (agent, candidate);
549

550
551
552
553
554
555
556
  result = priv_add_local_candidate_pruned (component, candidate);
  if (result) {
    agent_signal_new_candidate (agent, candidate);
  }
  else {
    /* error: duplicate candidate */
    nice_candidate_free (candidate), candidate = NULL;
557
558
559
560
561
  }

  return candidate;
}

562

563
/*
564
565
566
567
568
569
570
571
572
573
574
 * Creates a server reflexive candidate for 'component_id' of stream
 * 'stream_id'.
 *
 * @return pointer to the created candidate, or NULL on error
 */
NiceCandidate* 
discovery_add_relay_candidate (
  NiceAgent *agent,
  guint stream_id,
  guint component_id,
  NiceAddress *address,
575
576
  NiceSocket *base_socket,
  TurnServer *turn)
577
578
579
580
{
  NiceCandidate *candidate;
  Component *component;
  Stream *stream;
581
  NiceSocket *relay_socket = NULL;
582
583
584
585
586

  if (!agent_find_component (agent, stream_id, component_id, &stream, &component))
    return NULL;

  candidate = nice_candidate_new (NICE_CANDIDATE_TYPE_RELAYED);
587
588
589

  if (agent->compatibility == NICE_COMPATIBILITY_GOOGLE) {
    candidate->priority = nice_candidate_jingle_priority (candidate);
Jakub Adam's avatar
Jakub Adam committed
590
591
  } else if (agent->compatibility == NICE_COMPATIBILITY_MSN ||
             agent->compatibility == NICE_COMPATIBILITY_OC2007)  {
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
    candidate->priority = nice_candidate_msn_priority (candidate);
  } else {
    candidate->priority =  nice_candidate_ice_priority_full
        (NICE_CANDIDATE_TYPE_PREF_RELAYED, 0, component_id);
  }
  candidate->stream_id = stream_id;
  candidate->component_id = component_id;
  candidate->addr = *address;
  candidate->turn = turn;

  /* step: link to the base candidate+socket */
  relay_socket = nice_turn_socket_new (agent, address,
      base_socket, &turn->server,
      turn->username, turn->password,
      agent_to_turn_socket_compatibility (agent));
607
608
  if (!relay_socket)
    goto errors;
609

610
611
  candidate->sockptr = relay_socket;
  candidate->base_addr = base_socket->addr;
612

613
  priv_generate_candidate_credentials (agent, candidate);
614

615
616
617
618
  /* Google uses the turn username as the candidate username */
  if (agent->compatibility == NICE_COMPATIBILITY_GOOGLE) {
    g_free (candidate->username);
    candidate->username = g_strdup (turn->username);
619
620
  }

621
622
623
624
625
626
627
628
  priv_assign_foundation (agent, candidate);

  if (!priv_add_local_candidate_pruned (component, candidate))
    goto errors;

  component->sockets = g_slist_append (component->sockets, relay_socket);
  agent_signal_new_candidate (agent, candidate);

629
  return candidate;
630
631
632
633
634
635

errors:
  nice_candidate_free (candidate);
  if (relay_socket)
    nice_socket_free (relay_socket);
  return NULL;
636
637
}

638
/*
639
640
641
642
643
 * Creates a peer reflexive candidate for 'component_id' of stream
 * 'stream_id'.
 *
 * @return pointer to the created candidate, or NULL on error
 */
644
NiceCandidate*
645
646
647
648
649
discovery_add_peer_reflexive_candidate (
  NiceAgent *agent,
  guint stream_id,
  guint component_id,
  NiceAddress *address,
650
  NiceSocket *base_socket,
651
652
  NiceCandidate *local,
  NiceCandidate *remote)
653
654
655
656
{
  NiceCandidate *candidate;
  Component *component;
  Stream *stream;
657
  gboolean result;
658
659
660
661
662
663

  if (!agent_find_component (agent, stream_id, component_id, &stream, &component))
    return NULL;

  candidate = nice_candidate_new (NICE_CANDIDATE_TYPE_PEER_REFLEXIVE);

664
665
666
  candidate->transport = NICE_CANDIDATE_TRANSPORT_UDP;
  if (agent->compatibility == NICE_COMPATIBILITY_GOOGLE) {
    candidate->priority = nice_candidate_jingle_priority (candidate);
Jakub Adam's avatar
Jakub Adam committed
667
668
  } else if (agent->compatibility == NICE_COMPATIBILITY_MSN ||
             agent->compatibility == NICE_COMPATIBILITY_OC2007)  {
669
670
671
    candidate->priority = nice_candidate_msn_priority (candidate);
  } else {
    candidate->priority = nice_candidate_ice_priority_full
672
        (NICE_CANDIDATE_TYPE_PREF_PEER_REFLEXIVE, 0, component_id);
673
674
675
676
677
  }
  candidate->stream_id = stream_id;
  candidate->component_id = component_id;
  candidate->addr = *address;
  candidate->base_addr = base_socket->addr;
678

679

680
  priv_assign_foundation (agent, candidate);
681

Jakub Adam's avatar
Jakub Adam committed
682
683
  if ((agent->compatibility == NICE_COMPATIBILITY_MSN ||
       agent->compatibility == NICE_COMPATIBILITY_OC2007) &&
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
      remote && local) {
    guchar *new_username = NULL;
    guchar *decoded_local = NULL;
    guchar *decoded_remote = NULL;
    gsize local_size;
    gsize remote_size;
    g_free(candidate->username);
    g_free(candidate->password);

    decoded_local = g_base64_decode (local->username, &local_size);
    decoded_remote = g_base64_decode (remote->username, &remote_size);

    new_username = g_new0(guchar, local_size + remote_size);
    memcpy(new_username, decoded_local, local_size);
    memcpy(new_username + local_size, decoded_remote, remote_size);

    candidate->username = g_base64_encode (new_username, local_size + remote_size);
    g_free(new_username);
    g_free(decoded_local);
    g_free(decoded_remote);

    candidate->password = g_strdup(local->password);
  } else if (local) {
    g_free(candidate->username);
    g_free(candidate->password);

    candidate->username = g_strdup(local->username);
    candidate->password = g_strdup(local->password);
  }
713

714
715
716
  /* step: link to the base candidate+socket */
  candidate->sockptr = base_socket;
  candidate->base_addr = base_socket->addr;
717

718
719
720
721
  result = priv_add_local_candidate_pruned (component, candidate);
  if (result != TRUE) {
    /* error: memory allocation, or duplicate candidatet */
    nice_candidate_free (candidate), candidate = NULL;
722
723
724
725
726
727
  }

  return candidate;
}


728
/*
729
730
731
732
 * Adds a new peer reflexive candidate to the list of known
 * remote candidates. The candidate is however not paired with
 * existing local candidates.
 *
733
 * See ICE sect 7.2.1.3 "Learning Peer Reflexive Candidates" (ID-19).
734
735
736
737
738
739
740
 *
 * @return pointer to the created candidate, or NULL on error
 */
NiceCandidate *discovery_learn_remote_peer_reflexive_candidate (
  NiceAgent *agent,
  Stream *stream,
  Component *component,
741
  guint32 priority,
742
  const NiceAddress *remote_address,
743
  NiceSocket *udp_socket,
744
745
  NiceCandidate *local,
  NiceCandidate *remote)
746
747
748
{
  NiceCandidate *candidate;

749
750
751
  /* XXX: for use compiler */
  (void)udp_socket;

752
  candidate = nice_candidate_new (NICE_CANDIDATE_TYPE_PEER_REFLEXIVE);
753
754
755
756
757
758
759
760
761
762
763

  candidate->transport = NICE_CANDIDATE_TRANSPORT_UDP;
  candidate->addr = *remote_address;
  candidate->base_addr = *remote_address;

  /* if the check didn't contain the PRIORITY attribute, then the priority will
   * be 0, which is invalid... */
  if (priority != 0) {
    candidate->priority = priority;
  } else if (agent->compatibility == NICE_COMPATIBILITY_GOOGLE) {
    candidate->priority = nice_candidate_jingle_priority (candidate);
Jakub Adam's avatar
Jakub Adam committed
764
765
  } else if (agent->compatibility == NICE_COMPATIBILITY_MSN ||
             agent->compatibility == NICE_COMPATIBILITY_OC2007)  {
766
767
768
    candidate->priority = nice_candidate_msn_priority (candidate);
  } else {
    candidate->priority = nice_candidate_ice_priority_full
769
        (NICE_CANDIDATE_TYPE_PREF_PEER_REFLEXIVE, 0, component->id);
770
771
772
  }
  candidate->stream_id = stream->id;
  candidate->component_id = component->id;
773

774

775
  priv_assign_remote_foundation (agent, candidate);
776

Jakub Adam's avatar
Jakub Adam committed
777
778
  if ((agent->compatibility == NICE_COMPATIBILITY_MSN ||
       agent->compatibility == NICE_COMPATIBILITY_OC2007) &&
779
780
781
782
783
784
785
786
      remote && local) {
    guchar *new_username = NULL;
    guchar *decoded_local = NULL;
    guchar *decoded_remote = NULL;
    gsize local_size;
    gsize remote_size;
    g_free(candidate->username);
    g_free (candidate->password);
787

788
789
    decoded_local = g_base64_decode (local->username, &local_size);
    decoded_remote = g_base64_decode (remote->username, &remote_size);
790

791
792
793
    new_username = g_new0(guchar, local_size + remote_size);
    memcpy(new_username, decoded_remote, remote_size);
    memcpy(new_username + remote_size, decoded_local, local_size);
794

795
796
797
798
    candidate->username = g_base64_encode (new_username, local_size + remote_size);
    g_free(new_username);
    g_free(decoded_local);
    g_free(decoded_remote);
799

800
801
802
803
804
805
806
    candidate->password = g_strdup(remote->password);
  } else if (remote) {
    g_free (candidate->username);
    g_free (candidate->password);
    candidate->username = g_strdup(remote->username);
    candidate->password = g_strdup(remote->password);
  }
807

808
809
810
  candidate->sockptr = NULL; /* not stored for remote candidates */
  /* note: candidate username and password are left NULL as stream 
     level ufrag/password are used */
811

812
813
  component->remote_candidates = g_slist_append (component->remote_candidates,
      candidate);
814

815
  agent_signal_new_remote_candidate (agent, candidate);
816
817

  return candidate;
818
}
819

820
/* 
821
822
823
824
825
826
827
828
 * Timer callback that handles scheduling new candidate discovery
 * processes (paced by the Ta timer), and handles running of the 
 * existing discovery processes.
 *
 * This function is designed for the g_timeout_add() interface.
 *
 * @return will return FALSE when no more pending timers.
 */
Youness Alaoui's avatar
Youness Alaoui committed
829
static gboolean priv_discovery_tick_unlocked (gpointer pointer)
830
831
832
833
834
{
  CandidateDiscovery *cand;
  NiceAgent *agent = pointer;
  GSList *i;
  int not_done = 0; /* note: track whether to continue timer */
835
  size_t buffer_len = 0;
836
837
838

  {
    static int tick_counter = 0;
839
    if (tick_counter++ % 50 == 0)
840
      nice_debug ("Agent %p : discovery tick #%d with list %p (1)", agent, tick_counter, agent->discovery_list);
841
842
843
844
845
846
847
848
849
850
  }

  for (i = agent->discovery_list; i ; i = i->next) {
    cand = i->data;

    if (cand->pending != TRUE) {
      cand->pending = TRUE;

      if (agent->discovery_unsched_items)
	--agent->discovery_unsched_items;
851
852
853
854

      {
        gchar tmpbuf[INET6_ADDRSTRLEN];
        nice_address_to_string (&cand->server, tmpbuf);
855
856
        nice_debug ("Agent %p : discovery - scheduling cand type %u addr %s.\n",
            agent, cand->type, tmpbuf);
857
858
      }
      if (nice_address_is_valid (&cand->server) &&
859
860
          (cand->type == NICE_CANDIDATE_TYPE_SERVER_REFLEXIVE ||
              cand->type == NICE_CANDIDATE_TYPE_RELAYED)) {
861

862
	agent_signal_component_state_change (agent,
863
864
865
					     cand->stream->id,
					     cand->component->id,
					     NICE_COMPONENT_STATE_GATHERING);
866

867
        if (cand->type == NICE_CANDIDATE_TYPE_SERVER_REFLEXIVE) {
868
          buffer_len = stun_usage_bind_create (&cand->stun_agent,
869
870
              &cand->stun_message, cand->stun_buffer, sizeof(cand->stun_buffer));
        } else if (cand->type == NICE_CANDIDATE_TYPE_RELAYED) {
871
872
873
874
          uint8_t *username = (uint8_t *)cand->turn->username;
          size_t username_len = (size_t) strlen (cand->turn->username);
          uint8_t *password = (uint8_t *)cand->turn->password;
          size_t password_len = (size_t) strlen (cand->turn->password);
875
876
          StunUsageTurnCompatibility turn_compat =
              agent_to_turn_compatibility (agent);
877

878
879
          if (turn_compat == STUN_USAGE_TURN_COMPATIBILITY_MSN ||
              turn_compat == STUN_USAGE_TURN_COMPATIBILITY_OC2007) {
880
881
882
883
            username = g_base64_decode ((gchar *)username, &username_len);
            password = g_base64_decode ((gchar *)password, &password_len);
          }

884
          buffer_len = stun_usage_turn_create (&cand->stun_agent,
885
              &cand->stun_message,  cand->stun_buffer, sizeof(cand->stun_buffer),
886
              cand->stun_resp_msg.buffer == NULL ? NULL : &cand->stun_resp_msg,
887
              STUN_USAGE_TURN_REQUEST_PORT_NORMAL,
888
              -1, -1,
889
890
              username, username_len,
              password, password_len,
891
              turn_compat);
892

893
894
          if (turn_compat == STUN_USAGE_TURN_COMPATIBILITY_MSN ||
              turn_compat == STUN_USAGE_TURN_COMPATIBILITY_OC2007) {
895
896
            g_free (cand->msn_turn_username);
            g_free (cand->msn_turn_password);
897
898
            cand->msn_turn_username = username;
            cand->msn_turn_password = password;
899
          }
900
        }
901
902

	if (buffer_len > 0) {
903
904
905
906
907
908
909
          if (nice_socket_is_reliable (cand->nicesock)) {
            stun_timer_start_reliable (&cand->timer,
                STUN_TIMER_DEFAULT_RELIABLE_TIMEOUT);
          } else {
            stun_timer_start (&cand->timer, 200,
                STUN_TIMER_DEFAULT_MAX_RETRANSMISSIONS);
          }
910
911

          /* send the conncheck */
912
          nice_socket_send (cand->nicesock, &cand->server,
913
914
              buffer_len, (gchar *)cand->stun_buffer);

915
916
917
	  /* case: success, start waiting for the result */
	  g_get_current_time (&cand->next_tick);

918
	} else {
919
920
	  /* case: error in starting discovery, start the next discovery */
	  cand->done = TRUE;
921
922
	  cand->stun_message.buffer = NULL;
	  cand->stun_message.buffer_len = 0;
923
	  continue;
924
925
	}
      }
926
      else
927
928
	/* allocate relayed candidates */
	g_assert_not_reached ();
929

930
931
      ++not_done; /* note: new discovery scheduled */
    }
Youness Alaoui's avatar
Youness Alaoui committed
932

933
934
935
936
937
    if (cand->done != TRUE) {
      GTimeVal now;

      g_get_current_time (&now);

938
      if (cand->stun_message.buffer == NULL) {
939
	nice_debug ("Agent %p : STUN discovery was cancelled, marking discovery done.", agent);
940
941
	cand->done = TRUE;
      }
942
      else if (priv_timer_expired (&cand->next_tick, &now)) {
943
944
        switch (stun_timer_refresh (&cand->timer)) {
          case STUN_USAGE_TIMER_RETURN_TIMEOUT:
945
946
947
948
949
950
951
952
953
954
955
956
957
958
            {
              /* Time out */
              /* case: error, abort processing */
              StunTransactionId id;

              stun_message_id (&cand->stun_message, id);
              stun_agent_forget_transaction (&cand->stun_agent, id);

              cand->done = TRUE;
              cand->stun_message.buffer = NULL;
              cand->stun_message.buffer_len = 0;
              nice_debug ("Agent %p : bind discovery timed out, aborting discovery item.", agent);
              break;
            }
959
          case STUN_USAGE_TIMER_RETURN_RETRANSMIT:
960
961
962
963
964
965
966
            {
              /* case: not ready complete, so schedule next timeout */
              unsigned int timeout = stun_timer_remainder (&cand->timer);

              stun_debug ("STUN transaction retransmitted (timeout %dms).\n",
                  timeout);

967
              /* retransmit */
968
              nice_socket_send (cand->nicesock, &cand->server,
969
970
971
972
973
974
975
                  stun_message_length (&cand->stun_message),
                  (gchar *)cand->stun_buffer);

              /* note: convert from milli to microseconds for g_time_val_add() */
              cand->next_tick = now;
              g_time_val_add (&cand->next_tick, timeout * 1000);

Youness Alaoui's avatar
Youness Alaoui committed
976
977
978
              ++not_done; /* note: retry later */
              break;
            }
979
          case STUN_USAGE_TIMER_RETURN_SUCCESS:
Youness Alaoui's avatar
Youness Alaoui committed
980
            {
981
982
              unsigned int timeout = stun_timer_remainder (&cand->timer);

Youness Alaoui's avatar
Youness Alaoui committed
983
984
985
              cand->next_tick = now;
              g_time_val_add (&cand->next_tick, timeout * 1000);

986
987
988
              ++not_done; /* note: retry later */
              break;
            }
989
	}
990

Youness Alaoui's avatar
Youness Alaoui committed
991
      } else {
992
	++not_done; /* note: discovery not expired yet */
Youness Alaoui's avatar
Youness Alaoui committed
993
      }
994
995
996
997
    }
  }

  if (not_done == 0) {
998
    nice_debug ("Agent %p : Candidate gathering FINISHED, stopping discovery timer.", agent);
999
1000
1001

    discovery_free (agent);

1002
    agent_gathering_done (agent);
1003

1004
1005
1006
1007
1008
1009
1010
    /* note: no pending timers, return FALSE to stop timer */
    return FALSE;
  }

  return TRUE;
}

Youness Alaoui's avatar
Youness Alaoui committed
1011
1012
1013
1014
1015
static gboolean priv_discovery_tick (gpointer pointer)
{
  NiceAgent *agent = pointer;
  gboolean ret;

1016
1017
1018
1019
1020
1021
1022
1023
  agent_lock();
  if (g_source_is_destroyed (g_main_current_source ())) {
    nice_debug ("Source was destroyed. "
        "Avoided race condition in priv_discovery_tick");
    agent_unlock ();
    return FALSE;
  }

Youness Alaoui's avatar
Youness Alaoui committed
1024
  ret = priv_discovery_tick_unlocked (pointer);
1025
1026
1027
1028
1029
1030
1031
  if (ret == FALSE) {
    if (agent->discovery_timer_source != NULL) {
      g_source_destroy (agent->discovery_timer_source);
      g_source_unref (agent->discovery_timer_source);
      agent->discovery_timer_source = NULL;
    }
  }
1032
  agent_unlock();
Youness Alaoui's avatar
Youness Alaoui committed
1033
1034
1035
1036

  return ret;
}

1037
/*
1038
1039
 * Initiates the candidate discovery process by starting
 * the necessary timers.
1040
1041
1042
1043
1044
1045
1046
1047
1048
 *
 * @pre agent->discovery_list != NULL  // unsched discovery items available
 */
void discovery_schedule (NiceAgent *agent)
{
  g_assert (agent->discovery_list != NULL);

  if (agent->discovery_unsched_items > 0) {
    
1049
    if (agent->discovery_timer_source == NULL) {
1050
      /* step: run first iteration immediately */
Youness Alaoui's avatar
Youness Alaoui committed
1051
      gboolean res = priv_discovery_tick_unlocked (agent);
1052
      if (res == TRUE) {
1053
        agent->discovery_timer_source = agent_timeout_add_with_context (agent, agent->timer_ta, priv_discovery_tick, agent);
1054
1055
      }
    }
1056
1057
  }
}