-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathporting.htm
1033 lines (522 loc) · 29.8 KB
/
porting.htm
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
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
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
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
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
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
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
<html>
<head>
<title>Porting SIP and Media Stack to Embedded Systems - pjsip.org</title>
<link rel="stylesheet" type="text/css" href="/style/style.css">
</head>
<body>
<!--#include file="header.html" -->
<p><a href="/">Home</a> --> <a href="/docs.htm">Documentations</a> --> Porting
Guide</p>
<table id="ContentSubTable" border="0" cellpadding="4" cellspacing="6" width="100%">
<tbody>
<tr>
<td>
<h1><a class="mozTocH1" name="mozTocId810898"></a>Porting SIP and Media Stack to Embedded Systems</h1>
<br>
<table style="text-align: left; width: 100%;" border="0" cellpadding="2" cellspacing="2">
<tbody>
<tr>
<td>Last Update: $Date: 2007-01-20 04:23:13 +0000 (Sat, 20 Jan 2007) $</td>
<td style="text-align: right;"><a href="/cgi-bin/showpage.cgi?porting">Print Friendly Page</a></td>
</tr>
</tbody>
</table>
<br>
<hr style="width: 100%; height: 2px;"> <br>
<p>PJLIB, PJLIB-UTIL, PJSIP and PJMEDIA libraries (or will be called just PJ libraries) have been designed specificly to be very portable and have very small footprint, to make it ideal to be used on embedded or even deeply embedded system development.</p>
<p>This article describes the challenges faced when porting PJ software to
these "non-standard" platforms and presents some strategies to maximize the
chance of successful porting efforts.</p>
<br>
<p><big><big>Table
of Contents</big></big></p>
<ul id="mozToc">
<!--mozToc h1 1 h2 2 h3 3--><li><a href="#mozTocId810898">Porting PJLIB, PJSIP, and PJMEDIA</a>
<ul>
<li><a href="#mozTocId136601">Porting
Considerations</a>
<ul>
<li><a href="#mozTocId86034">Target
CPU</a></li>
<li><a href="#mozTocId390047">Compiler
and Build System</a></li>
<li><a href="#mozTocId492338">Standard
LIBC Features</a></li>
<li><a href="#mozTocId426216">OS
Features</a></li>
<li><a href="#mozTocId152874">Networking
Features</a></li>
</ul>
</li>
<li><a href="#mozTocId50420">Porting
Strategies</a>
<ul>
<li><a href="#mozTocId565490">Fully
Porting PJLIB</a></li>
<li><a href="#mozTocId30930">Partially
Porting PJLIB</a></li>
</ul>
</li>
<li><a href="#mozTocId693552">Related
Pages</a></li>
</ul>
<p> </p>
</li>
</ul>
</td>
<td></td>
</tr>
<tr>
<td>
<h2><a class="mozTocH2" name="mozTocId136601"></a>Porting Considerations</h2>
<p>
This section describes components that need to be considered when porting PJ
libraries to new platform, especially the embedded or small-footprint systems.</p>
<p> </p>
<h3><a class="mozTocH3" name="mozTocId86034"></a>Target
CPU</h3>
<p>For this porting purpose, the target CPU determines the
characteristic of the basic data types (more specifically the width and
endianness) and the availability of floating point
co-processors. </p>
<h4><a class="mozTocH4" name="mozTocId8959281"></a>Instruction Set</h4>
<p>This normally is not something that we should worry about since all libraries are written on C language (there is no assembly instructions at all in the libraries).</p>
<h4><a class="mozTocH4" name="mozTocId895928"></a>Data Types</h4>
<p>PJLIB encapsulates all integral data types into the
corresponding pj_* data types in <a href="/pjlib/docs/html/types_8h-source.htm"><pj/types.h></a>.
In addition, 64bit integral data types are defined in the appropriate
compiler specific header file (e.g. <pj/compat/cc_gcc.h> for GCC,
<pj/compat/cc_msvc.h>for MSVC). </p>
<p>Currently the data type mapping is defined below.</p>
<table style="text-align: left; width: 100%;" border="1" cellpadding="4" cellspacing="1">
<tbody>
<tr>
<td style="font-weight: bold; text-align: center;">PJ data types</td>
<td style="font-weight: bold; text-align: center;">Description</td>
<td style="text-align: center;"><span style="font-weight: bold;">Availability</span></td>
<td style="font-weight: bold; text-align: center;">Mapped to..</td>
</tr>
<tr>
<td style="font-family: monospace;">pj_int8_t,
pj_uint8_t</td>
<td>8-bit
signed/unsigned integer</td>
<td>always
defined</td>
<td style="font-family: monospace;">char, unsigned
char</td>
</tr>
<tr>
<td style="font-family: monospace;">pj_int16_t, pj_uint16_t</td>
<td>16-bit
signed/unsigned integer</td>
<td>always
defined</td>
<td style="font-family: monospace;">short, unsigned
short</td>
</tr>
<tr>
<td style="font-family: monospace;">pj_int32_t,
pj_uint32_t</td>
<td>32-bit
signed/unsigned integer</td>
<td>always
defined</td>
<td style="font-family: monospace;">int, unsigned int</td>
</tr>
<tr>
<td style="font-family: monospace;">pj_int64_t,
pj_uint64_t</td>
<td>64-bit
signed/unsigned integer</td>
<td>only
when PJ_HAS_INT64 is defined (defined by compiler specific header file).</td>
<td style="font-family: monospace;">long
long/int64_t, unsigned long long/uint64_t</td>
</tr>
<tr>
<td>pj_ssize_t,
pj_size_t</td>
<td>At
least 32 bit, may be 64 bit.</td>
<td>always
defined</td>
<td>long,
size_t</td>
</tr>
<tr>
<td>pj_bool_t</td>
<td>Boolean</td>
<td>always
defined</td>
<td>int</td>
</tr>
<tr>
<td>pj_char_t</td>
<td>Native
character type for given platform.</td>
<td>always
defined</td>
<td>char
or wchar_t, depending on the value of PJ_NATIVE_STRING_IS_UNICODE.</td>
</tr>
<tr>
<td>(no PJ data type is defined)</td>
<td>Floating point</td>
<td>-</td>
<td>float</td>
</tr>
<tr>
<td>(no PJ data type is defined)</td>
<td>Double precision floating point</td>
<td>-</td>
<td>double</td>
</tr>
</tbody>
</table>
<p>The above mappings should work on most (if not all)
32-bit or 64-bit CPU and across different types of compilers, however <span style="font-weight: bold;">these
may not be accurate for 8-bit or 16-bit CPUs</span> (the <span style="font-family: monospace;">int/unsigned int</span>
data type may be less than 32-bit wide), therefore developer MUST
review and possibly fix these mappings before attempting to compile any
codes.</p>
<h4><a class="mozTocH4" name="mozTocId520438"></a>Endianness</h4>
<p>The PJLIB macros PJ_IS_LITTLE_ENDIAN and PJ_IS_BIG_ENDIAN MUST
be declared correctly according to the endianness of the target CPU.
These macros are declared by processor/machine specific header file
<pj/compat/m_*.h>.</p>
<h4><a class="mozTocH4" name="mozTocId977115"></a>Floating-Point Availability</h4>
<p>Floating point availability is controlled by
PJ_HAS_FLOATING_POINT macro, which by default is set to 1 in
<pj/config.h>. This can be overriden by declaring
PJ_HAS_FLOATING_POINT to zero in your <pj/config_site.h>.</p>
<p>PJLIB, PJLIB-UTIL, and PJSIP does not need any floating point
support. Even when floating point is disabled, everything should work
correctly. However, the PJ_HAS_FLOATING_POINT macro still MUST be
set accordingly in order for some fallback code to get activated.</p>
<p>However, the situation is different in PJMEDIA. Some PJMEDIA components
(such as tone generator, RTCP calculation, Speex codec, and AEC) do have fallback
algorithm implemented in fixed point, which get activated when PJ_HAS_FLOATING_POINT
macro is set to zero. However, some other components do not have the
alternative fixed point implementation, thus the floating point based
code will get used regardless of the floating point setting
(examples of such components are the resampling, PLC, and iLBC codec).</p>
<p> </p>
<h3><a class="mozTocH3" name="mozTocId390047"></a>Compiler
and Build System</h3>
The use of particular compiler will need to consider the following
issues.<br>
<h4><a class="mozTocH4" name="mozTocId135945"></a>Compiler Specific Features</h4>
<p>PJ currently supports GCC, MS Visual Studio, MS Embedded VC, Intel C
compiler (using gcc defines), and MetroWerks C compiler (works in
progress in Symbian port). To support new compiler, a new
<pj/compat/cc_*.h> needs to be created, and the <a href="/pjlib/docs/html/config_8h-source.htm"><pj/config.h></a>
file needs to be modified to include this new compiler specific header
file whenever the use of such compiler is detected.</p>
<p>PJ does not use any compiler specific constructs, so
supporting a new compiler should be straightforward.</p>
<h4><a class="mozTocH4" name="mozTocId5204381"></a>64bit Integral Data Type Availability</h4>
<p>A 64bit integer/unsigned integer data types are used quite heavily throughout the libraries for representing large integers. The availability of such data types are controlled by <tt>PJ_HAS_INT64</tt> macro.</p>
<p>On PJLIB and PJSIP libraries, normally the use of 64bit data types ar optional; when <tt>PJ_HAS_INT64</tt> macro is not declared, or when its value is zero, the 64bit data type will be replaced by multiple 32bit integers.</p>
<p>However, PJMEDIA uses 64bit data type as the frame timestamp (to avoid overflow), and this currently does not have the replacement 32bit algorithms. Many PJMEDIA components use the 64bit operations against frame timestamp, without providing alternative 32bit operations.</p>
<p>Most modern compilers such as <b>gcc</b> should have support for 64bit integral data types; however please check this when porting PJ software using other compilers. </p>
<h4><a class="mozTocH4" name="mozTocId25154"></a>Integrating into the Build System</h4>
<p>If the existing PJ Makefile based build system is to be used, then ideally the compiler
should support GCC syntax for specifying options (for example "/I" to
specify include search path). If this is not possible, when a new <span style="font-family: monospace;">"cc-*.mak"</span> file
need to be created in <span style="font-family: monospace;">$PJ-ROOT/build/</span>
directory, to tell the Makefile build system how to invoke various
options to the compiler (note: the use of compiler other than gcc family for
the Makefile has not been tested for some time, so few things may be broken). </p>
<p>Alternatively, you may create a new build system
altogether. For example, if the
compiler comes with its own build system (an IDE, for example), then
probably it's better to create new project files for building the
libraries instead of using PJ build system. This is the approach that
we use when porting PJ to Windows Mobile and Symbian.</p>
<p>When creating a new build system, please prefer to put
the build specific files (such as project files) under each <span style="font-family: monospace;">build/</span>
directory of each project, or in a sub-directory under <span style="font-family: monospace;">build/</span>
directory in each project, if at all possible. Also take care to create
another subdirectory under the project directory, to place the object
files, so that it is easy to add Subversion rule to ignore these
directories containing output files from being included in the source
control.</p>
<p>However, creating a new build system comes at a cost, that is it
would need to be maintained to keep it in sync with changes in main PJ
libraries. So please weight the convenience of having a build system
supported by/integrated with the target against the cost of maintaining
such build system when deciding whether choose this route.</p>
<h4><a class="mozTocH4" name="mozTocId518585"></a>Availability of <stdarg.h></h4>
Variable number of argument functionality is needed by only one place,
i.e. the logging functions declared in <pj/log.h> header file.
In most cases, <span style="font-family: monospace;"><stdarg.h></span>
header file should be available, since this is standard ANSI C feature. But
should this is not available, then probably we can get
away with disabling PJLIB logging functionality by defining PJ_LOG_MAX_LEVEL macro to
zero in your <pj/config_site.h>, but
definitely this practice is highly discouraged since having logging
is crucial when debugging the libraries!<br>
<p> </p>
<h3><a class="mozTocH3" name="mozTocId492338"></a>Standard
LIBC Features</h3>
Although higher level PJ libraries (such as PJSIP and PJMEDIA)
can work without LIBC (this has been proven in the past
with the porting to Linux kernel mode module target), in most cases
this means that either PJ will abstract LIBC or the similar LIBC
functionalities need to be implemented in PJLIB. <br>
<p>The LIBC header availabilities should be indicated by
the corresponding PJ_HAS_*_H macro in the appropriate OS specific
<pj/compat/os_*.h> header file (for example,
PJ_HAS_STRING_H indicates the availability of standard
<string.h> header file). </p>
<p>When some header files are missing, sometimes it is still
possible to recover the compilation if the same functionality is
defined by some other header files (for example, the standard string
manipulation functions are declared by <string.h>, but
some systems such as WinCE declare them in <stdlib.h>). In
other cases, when the functionality is really not available, then the
appropriate replacement implementation need to be provided (for example, isblank()
is implemented in <pj/compat/ctype.h> for targets that
lack it).</p>
Some of the most "problematic" LIBC features will be described below.<br>
<h4><a class="mozTocH4" name="mozTocId781063"></a>setjmp() and longjmp() </h4>
<p>Both <span style="font-family: monospace;">setjmp()</span>
and <span style="font-family: monospace;">longjmp()</span>
functions are needed to get the exception framework working. The <a href="/pjlib/docs/html/group__PJ__EXCEPT.htm">PJLIB
exception framework</a> is a TRY/CATCH mechanism for C programs,
and is normally used in scanners/parsers to raise syntax error
exception and in <a href="/pjlib/docs/html/group__PJ__POOL__GROUP.htm">PJLIB
Memory Pool</a> to raise memory allocation failure exception.</p>
This functionality is mandatory, and if setjmp()/longjmp() are not
available, they MUST be implemented in
<pj/compat/setjmp.h>.
<h4><a class="mozTocH4" name="mozTocId112805"></a>malloc()</h4>
Malloc is the default memory allocation backend for <a href="/pjlib/docs/html/group__PJ__POOL__GROUP.htm">PJLIB
Memory Pool</a>, but this can be changed by creating a new memory
pool policy (similar to <span style="font-family: monospace;">pj/pool_policy_malloc.c</span>)
and specify this policy when creating the pool factory.<br>
<p> </p>
<h3><a class="mozTocH3" name="mozTocId426216"></a>OS
Features</h3>
This definitely would be the hardest part of the porting efforts. Some
of the challenges to be expected are describesd below.<br>
<h4><a class="mozTocH4" name="mozTocId32272"></a>OS Dependent Features</h4>
OS dependent features needed by PJ are collected together in
<pj/os.h>, which are documented in <a href="/pjlib/docs/html/group__PJ__OS.htm">PJLIB OS Dependent</a>
documentation. PJLIB provides the implementation for Win32 and Posix
family OSes, so if the new OS supports one of these OS API, probably we
don't need to create a new os_*.c implementation. If this is not the
case, then a new os_*.c MUST be created.<br>
<p>Note that depending on the selected porting strategy
(described in next chapter), it may be possible to skip creating a full
os_*.c implementation for the new OS, and instead just create a dummy
implementation just to allow the library to link.</p>
<h4><a class="mozTocH4" name="mozTocId770829"></a>Threading</h4>
PJ software works with or without threads, as the libraries
do not create any threads on their own. But when threads are needed,
naturally all synchronization functions must be implemented for the new
OS.<br>
<p>It is perfectly possible to NOT use threads, for
example, to run the application on interrupt context, or to use some
kind of cooperative multitasking mechanism where a task does not
relinguish it's execution time until it has finished with the task. With PJSIP and PJMEDIA, it is also perfectly possible to use only single thread (the main thread) to run everything (including polling for the RTP/RTCP packets), provided that the sound device abstraction supports this. For these cases, we don't need to implement all the PJLIB OS features.</p>
<h4>Thread Specific Data</h4>
<p>The libraries need to store some data that is specific (and
private) for each thread. PJLIB provides this functionality by
abstracting the thread local storage (TLS) or thread specific storage
API that is system dependent into a uniform thread local storage API.</p>
<p>Even when multithreading is NOT used, thread local storage API must still be implemented in PJLIB. The TLS is used for example by the <A HREF="/pjlib/docs/html/group__PJ__EXCEPT.htm">PJLIB exception framework</A>, since an exception should only be cascaded to the appropriate handler on the same thread only. </p>
<p>In other cases, thread local storage or thread specific
storage API may not be available in the target OS. In these cases,
application developer would need to implement this functionality via a
homegrown mechanism.</p>
<h4>Synchronization Objects</h4>
<p>PJLIB provides abstractions for synchronization objects such
as semaphore, simple mutex, recursive mutex, and read-write mutex. All
of these objects are used by the upper layer libraries or application.</p>
<p>The Operating System normally would provide these
functionalities in the OS API. But even when it's not available,
normally we can emulate most mutex functionalities using semaphore
object.</p>
<p>More specificly, PJLIB provides a simple and elegant implementation of
read-write mutex for OS that doesn't provide this functionality (such
as Win32). </p>
<h4><a class="mozTocH4" name="mozTocId831428"></a>Unicode Support</h4>
<p>PJ can work in both ANSI and UNICODE systems, and in
fact, with PJLIB, it is possible to create a source files that would
build correctly in both ANSI and UNICODE variant of the same OS family (such as between Win32 and Windows CE).</p>
<p>The UNICODE support is indicated by
PJ_NATIVE_STRING_IS_UNICODE macro, which MUST be defined in the
appropriate <pj/compat/os_*.h> file. Some UNICODE utility
functions/macros are declared in <pj/unicode.h>.</p>
<p>PJLIB itself (and all other PJ libraries such as PJLIB-UTIL, PJSIP, and PJMEDIA) internally uses ANSI representation for all purposes, and only converts strings to Unicode when they interract with the Unicode operating systems.</p>
<p> </p>
<h3><a class="mozTocH3" name="mozTocId152874"></a>Networking
Features</h3>
<p>PJLIB defines rich abstractions for networking features.
These abstractions and the consideration when porting will be described
below.</p>
<h4><a class="mozTocH4" name="mozTocId624104"></a>Socket API</h4>
PJLIB <A HREF="/pjlib/docs/html/group__PJ__SOCK.htm">Socket API</A> provides abstraction for various socket backends using BSD like API, and the API is declared in <pj/sock.h>, and implementation is provided in pj/sock_bsd.c for BSD socket API. The abstraction has been designed to allow very different socket API backend to be used; for example, efforts are currently under way to implement socket API for Symbian OS directly on top of <b>RSocket</b> interface (instead of using the BSD abstraction).
<p>The PJLIB socket API is needed by the default SIP and
media transports. However this may not be needed if application
completely rewrites these SIP and media transports.</p>
<h4><a class="mozTocH4" name="mozTocId82122"></a>Address Resolution API</h4>
PJLIB provides address resolution API abstraction (<span style="font-family: monospace;">pj_gethostbyname()</span> and <span style="font-family: monospace;">pj_gethostip()</span> functions) in <pj/addr_resolv.h>, and implementation for BSD socket backend is provided in <span style="font-family: monospace;">pj/addr_resolv_sock.c</span>.<br>
<p>The address resolution functions are needed for two
purposes:</p>
<ul>
<li>for finding the API address of local host when
creating the SIP and media transports,</li>
<li>for SIP server resolution procedure, if RFC 3263
server resolution is not used.</li>
</ul>
If application works strictly with IP addresses only (no hostnames),
then the address resolution API may not be needed.
<h4><a class="mozTocH4" name="mozTocId536758"></a>Socket select() Abstraction</h4>
PJLIB provides abstraction for BSD select() system call in
<pj/sock_select.h>, and the implementation in <span style="font-family: monospace;">pj/sock_select.c</span>.
The select() abstraction is only used by select IOQUEUE (see below), so
if select IOQUEUE is not used, the select() abstraction is not needed.
<h4><a class="mozTocH4" name="mozTocId673808"></a>IOQUEUE</h4>
<p><a href="/pjlib/docs/html/group__PJ__IOQUEUE.htm">PJLIB
IOQUEUE</a> (I/O Queue) implements Proactor Pattern for
demultiplexing network events. The IOQUEUE can be implemented with
multiple backends, such as Windows NT IO Completion Port, Linux epoll,
or PJLIB's select() abstraction.</p>
<p>The IOQUEUE is mainly used to poll all sockets for
incoming packets, and standard PJSIP and PJMEDIA transports make use of
IOQUEUE. Because of this, it is possible to have all sockets in the
application (SIP sockets and RTP/RTCP sockets) register to one ioqueue,
then poll this ioqueue from a single place (and possibly with a single
thread).</p>
<p>However, for platforms that doesn't have socket
demultiplexing API (such as selec()), it is possible to NOT implement
IOQUEUE altogether, by implementing a custom SIP and media transport.</p>
<p> </p>
<h2><a class="mozTocH2" name="mozTocId50420"></a>Porting
Strategies</h2>
<p>This section describes the strategies to port PJ
libraries into a new platform. There are basically two approaches to
porting PJ software into new platform.</p>
<p>The "traditional" way is to completely port PJLIB for
the new
platform. Once PJLIB has been ported successfully, all other
libraries (PJLIB-UTIL, PJSIP, and PJMEDIA) should run on the new
platform without modifications, since they depend only to PJLIB.</p>
<p>The second approach is only to partially port PJLIB, but
some parts of PJSIP and PJMEDIA will need to be modified.</p>
<p>Both approaches will be described below.</p>
<h3><a class="mozTocH3" name="mozTocId565490"></a>Fully
Porting PJLIB</h3>
The "traditional" path to porting PJ software is to port
the whole PJLIB to the new platform. Since all other libraries and
applications only depend to PJLIB, these upper layer
libraries/applications would most likely be able to run without changes
on the new platform once PJLIB porting is complete. <br>
<p>To fully port PJLIB, developer need to provide all the
PJLIB features described above. In addition, PJLIB provides quite comprehensive tests (<span style="font-family: monospace;">pjlib-test</span>
application) to test PJLIB functionalities for the new platform. These
test should catch more than 90% of problems that may be had for the new
platform, so when <span style="font-family: monospace;">pjlib-test</span> reports that everything is okay, there is a very good chance that the porting effort has been successful.</p>
<p>This porting path is recommended for platforms that
provide
"modern" APIs such as Posix or Windows. If the target platform
provides Posix abstraction layer, then this path is recommended, since
PJLIB supports Posix API. If this is not available (or desired), then
some efforts would be required to
rewrite/build some features that are pretty "advanced", such as
threading, thread local storage, and network event demultiplexing API
(select() and the like).</p>
Since not all Operating Systems are able to provide such features, this
porting effort may not be feasible. In this case, the second approach
below may be used to port PJ software into such platforms.<br>
<p> </p>
<h3><a class="mozTocH3" name="mozTocId30930"></a>Partially
Porting PJLIB</h3>
<p>Many Operating Systems, especially deeply embedded OSes,
do not have
"advanced" features such as multithreading, thread local storage,
network event demultiplexing, and the likes. Or in other cases, these
features may not be desired, for design requirements reason or size consideration. For these
cases, fully porting PJLIB may not be the best option.</p>
<p>This section attempts to describe alternative way to
port PJ software into such platforms. However, readers please be aware
that this effort is still being tested, so the procedures described here may not be accurate at all.</p>
<p> </p>
<h4><a class="mozTocH4" name="mozTocId368534"></a>Design Considerations</h4>
<p>This kind of port requires specific application design
considerations, which will be described below.</p>
<dl>
<dt style="font-weight: bold;">Threading Strategy:
No Threading!</dt>
<dd>
<p>To simplify the porting, we would need to disable
threading. By disabling threading, we then could skip implementing the
bulk OS specific codes in <span style="font-family: monospace;">pj/os_*.c</span>
(things like threads and synchronization objects). Disabling threading
in this case does not mean that we should not use multitasking at all,
but rather to make sure that no task would be interrupted unless the
task is ready to yield when it thinks that it is safe to do so. This
concept is commonly called cooperative multitasking.</p>
</dd>
<dt style="font-weight: bold;">Create Active
Transports</dt>
<dd>
<p>The transport interfaces in both SIP and media stack
was designed around the idea that transports should be active (by
"active", it means that the stack should not need to poll these
transports to get events from the network). But "under the hood",
actually these transports would register their sockets to an IOQUEUE,
and application or SIP endpoint (pjsip_endpoint) would need to
regularly poll the ioqueue to retrieve the network events. </p>
<p>In other words, current framework requires full
implementation of IOQUEUE and PJLIB socket abstraction API, which may
not be available in the target platform. If IOQUEUE and socket API
porting is not desired, then developer would need to rewrite these SIP
and media transports to NOT use IOQUEUE and PJLIB socket. The SIP and
media transports API is quite straightforward; the interface would just
need to register a callback to be called when the stack need to send
packet, and the transport need to call some function in the stack
whenever a packet has arrived.</p>
<p>The transport object itself then may be written in
whatever API provided by the OS. It may even be called from some
interrupt that is triggered when incoming packets arrived.</p>
<p>By implementing the transports this way, there would
not be any need to implement PJLIB's IOQUEUE, select abstraction, and
socket abstraction API.</p>
</dd>
<dt style="font-weight: bold;"> Implement
PJLIB Dummy Functions</dt>
<dd>
<p>With this approach, many PJLIB functionalities would
not need to be implemented. However, those functions may still have to
be defined (otherwise link error will occur), so dummy implementations
need to be created.</p>
</dd>
<dt> </dt>
</dl>
<h4><a class="mozTocH4" name="mozTocId806544"></a>More Information</h4>
<p>This type of porting is still under investigation, thus
there's not much details can be revealed at the moment. More will be
written when there's more information available.</p>
<p>If you want to try this path, please contact me to
discuss things in more details. Thanks.</p>