From b3d890ef2526636fe8547283f8c54797b75dd9a9 Mon Sep 17 00:00:00 2001 From: Anthony Minessale Date: Fri, 7 Apr 2006 16:10:03 +0000 Subject: [PATCH] add srtp git-svn-id: http://svn.freeswitch.org/svn/freeswitch/trunk@1086 d0543943-73ff-0310-b7d9-9358b9ac24b2 --- libs/srtp/.cvsignore | 4 + libs/srtp/CHANGES | 223 + libs/srtp/CVS/Entries | 22 + libs/srtp/CVS/Repository | 1 + libs/srtp/CVS/Root | 1 + libs/srtp/LICENSE | 35 + libs/srtp/Makefile | 220 + libs/srtp/Makefile.in | 220 + libs/srtp/README | 172 + libs/srtp/TODO | 66 + libs/srtp/VERSION | 1 + libs/srtp/config.guess | 1447 ++++ libs/srtp/config.log | 1118 +++ libs/srtp/config.status | 991 +++ libs/srtp/config.sub | 1555 ++++ libs/srtp/config_in.h | 170 + libs/srtp/configure | 8609 ++++++++++++++++++++++ libs/srtp/configure.in | 207 + libs/srtp/crypto/.cvsignore | 1 + libs/srtp/crypto/CVS/Entries | 12 + libs/srtp/crypto/CVS/Repository | 1 + libs/srtp/crypto/CVS/Root | 1 + libs/srtp/crypto/Makefile | 130 + libs/srtp/crypto/Makefile.in | 130 + libs/srtp/crypto/VERSION | 1 + libs/srtp/crypto/ae_xfm/CVS/Entries | 2 + libs/srtp/crypto/ae_xfm/CVS/Repository | 1 + libs/srtp/crypto/ae_xfm/CVS/Root | 1 + libs/srtp/crypto/ae_xfm/xfm.c | 570 ++ libs/srtp/crypto/cipher/CVS/Entries | 6 + libs/srtp/crypto/cipher/CVS/Repository | 1 + libs/srtp/crypto/cipher/CVS/Root | 1 + libs/srtp/crypto/cipher/aes.c | 1951 +++++ libs/srtp/crypto/cipher/aes_cbc.c | 444 ++ libs/srtp/crypto/cipher/aes_icm.c | 509 ++ libs/srtp/crypto/cipher/cipher.c | 407 + libs/srtp/crypto/cipher/null_cipher.c | 152 + libs/srtp/crypto/hash/CVS/Entries | 5 + libs/srtp/crypto/hash/CVS/Repository | 1 + libs/srtp/crypto/hash/CVS/Root | 1 + libs/srtp/crypto/hash/auth.c | 173 + libs/srtp/crypto/hash/hmac.c | 269 + libs/srtp/crypto/hash/null_auth.c | 160 + libs/srtp/crypto/hash/sha1.c | 404 + libs/srtp/crypto/include/.cvsignore | 1 + libs/srtp/crypto/include/CVS/Entries | 29 + libs/srtp/crypto/include/CVS/Repository | 1 + libs/srtp/crypto/include/CVS/Root | 1 + libs/srtp/crypto/include/aes.h | 84 + libs/srtp/crypto/include/aes_cbc.h | 50 + libs/srtp/crypto/include/aes_icm.h | 56 + libs/srtp/crypto/include/alloc.h | 57 + libs/srtp/crypto/include/auth.h | 159 + libs/srtp/crypto/include/cipher.h | 218 + libs/srtp/crypto/include/config.h | 171 + libs/srtp/crypto/include/crypto.h | 17 + libs/srtp/crypto/include/crypto_kernel.h | 288 + libs/srtp/crypto/include/crypto_math.h | 273 + libs/srtp/crypto/include/crypto_types.h | 206 + libs/srtp/crypto/include/cryptoalg.h | 133 + libs/srtp/crypto/include/datatypes.h | 426 ++ libs/srtp/crypto/include/err.h | 174 + libs/srtp/crypto/include/gf2_8.h | 79 + libs/srtp/crypto/include/hmac.h | 78 + libs/srtp/crypto/include/integers.h | 147 + libs/srtp/crypto/include/kernel_compat.h | 84 + libs/srtp/crypto/include/key.h | 82 + libs/srtp/crypto/include/null_auth.h | 68 + libs/srtp/crypto/include/null_cipher.h | 80 + libs/srtp/crypto/include/prng.h | 54 + libs/srtp/crypto/include/rand_source.h | 91 + libs/srtp/crypto/include/rdb.h | 94 + libs/srtp/crypto/include/rdbx.h | 146 + libs/srtp/crypto/include/sha1.h | 108 + libs/srtp/crypto/include/stat.h | 69 + libs/srtp/crypto/include/xfm.h | 139 + libs/srtp/crypto/kernel/CVS/Entries | 5 + libs/srtp/crypto/kernel/CVS/Repository | 1 + libs/srtp/crypto/kernel/CVS/Root | 1 + libs/srtp/crypto/kernel/alloc.c | 119 + libs/srtp/crypto/kernel/crypto_kernel.c | 523 ++ libs/srtp/crypto/kernel/err.c | 148 + libs/srtp/crypto/kernel/key.c | 115 + libs/srtp/crypto/math/CVS/Entries | 5 + libs/srtp/crypto/math/CVS/Repository | 1 + libs/srtp/crypto/math/CVS/Root | 1 + libs/srtp/crypto/math/datatypes.c | 600 ++ libs/srtp/crypto/math/gf2_8.c | 83 + libs/srtp/crypto/math/math.c | 962 +++ libs/srtp/crypto/math/stat.c | 367 + libs/srtp/crypto/replay/CVS/Entries | 4 + libs/srtp/crypto/replay/CVS/Repository | 1 + libs/srtp/crypto/replay/CVS/Root | 1 + libs/srtp/crypto/replay/rdb.c | 137 + libs/srtp/crypto/replay/rdbx.c | 289 + libs/srtp/crypto/replay/ut_sim.c | 105 + libs/srtp/crypto/rng/CVS/Entries | 5 + libs/srtp/crypto/rng/CVS/Repository | 1 + libs/srtp/crypto/rng/CVS/Root | 1 + libs/srtp/crypto/rng/ctr_prng.c | 108 + libs/srtp/crypto/rng/prng.c | 180 + libs/srtp/crypto/rng/rand_linux_kernel.c | 65 + libs/srtp/crypto/rng/rand_source.c | 118 + libs/srtp/crypto/test/.cvsignore | 8 + libs/srtp/crypto/test/CVS/Entries | 11 + libs/srtp/crypto/test/CVS/Repository | 1 + libs/srtp/crypto/test/CVS/Root | 1 + libs/srtp/crypto/test/aes_calc.c | 111 + libs/srtp/crypto/test/auth_driver.c | 200 + libs/srtp/crypto/test/cipher_driver.c | 491 ++ libs/srtp/crypto/test/datatypes_driver.c | 237 + libs/srtp/crypto/test/env.c | 99 + libs/srtp/crypto/test/kernel_driver.c | 126 + libs/srtp/crypto/test/rand_gen.c | 140 + libs/srtp/crypto/test/sha1_driver.c | 533 ++ libs/srtp/crypto/test/stat_driver.c | 101 + libs/srtp/doc/.cvsignore | 3 + libs/srtp/doc/CVS/Entries | 11 + libs/srtp/doc/CVS/Repository | 1 + libs/srtp/doc/CVS/Root | 1 + libs/srtp/doc/Doxyfile | 1042 +++ libs/srtp/doc/Makefile | 44 + libs/srtp/doc/Makefile.in | 44 + libs/srtp/doc/crypto_kernel.txt | 76 + libs/srtp/doc/draft-irtf-cfrg-icm-00.txt | 1 + libs/srtp/doc/header.template | 112 + libs/srtp/doc/intro.txt | 395 + libs/srtp/doc/libsrtp.pdf | Bin 0 -> 355874 bytes libs/srtp/doc/references.txt | 21 + libs/srtp/doc/rfc3711.txt | 3139 ++++++++ libs/srtp/include/CVS/Entries | 4 + libs/srtp/include/CVS/Repository | 1 + libs/srtp/include/CVS/Root | 1 + libs/srtp/include/rtp.h | 125 + libs/srtp/include/srtp.h | 993 +++ libs/srtp/include/ut_sim.h | 80 + libs/srtp/install-sh | 251 + libs/srtp/srtp/CVS/Entries | 2 + libs/srtp/srtp/CVS/Repository | 1 + libs/srtp/srtp/CVS/Root | 1 + libs/srtp/srtp/srtp.c | 1779 +++++ libs/srtp/tables/.cvsignore | 1 + libs/srtp/tables/CVS/Entries | 3 + libs/srtp/tables/CVS/Repository | 1 + libs/srtp/tables/CVS/Root | 1 + libs/srtp/tables/aes_tables.c | 346 + libs/srtp/test/.cvsignore | 12 + libs/srtp/test/CVS/Entries | 9 + libs/srtp/test/CVS/Repository | 1 + libs/srtp/test/CVS/Root | 1 + libs/srtp/test/lfsr.c | 310 + libs/srtp/test/rdbx_driver.c | 306 + libs/srtp/test/replay_driver.c | 180 + libs/srtp/test/roc_driver.c | 165 + libs/srtp/test/rtp.c | 150 + libs/srtp/test/rtpw.c | 507 ++ libs/srtp/test/srtp_driver.c | 1274 ++++ libs/srtp/timing | 1 + libs/srtp/undos.sh | 10 + libs/srtp/update.sh | 15 + 160 files changed, 41485 insertions(+) create mode 100644 libs/srtp/.cvsignore create mode 100644 libs/srtp/CHANGES create mode 100644 libs/srtp/CVS/Entries create mode 100644 libs/srtp/CVS/Repository create mode 100644 libs/srtp/CVS/Root create mode 100644 libs/srtp/LICENSE create mode 100644 libs/srtp/Makefile create mode 100644 libs/srtp/Makefile.in create mode 100644 libs/srtp/README create mode 100644 libs/srtp/TODO create mode 100644 libs/srtp/VERSION create mode 100755 libs/srtp/config.guess create mode 100644 libs/srtp/config.log create mode 100755 libs/srtp/config.status create mode 100755 libs/srtp/config.sub create mode 100644 libs/srtp/config_in.h create mode 100755 libs/srtp/configure create mode 100644 libs/srtp/configure.in create mode 100644 libs/srtp/crypto/.cvsignore create mode 100644 libs/srtp/crypto/CVS/Entries create mode 100644 libs/srtp/crypto/CVS/Repository create mode 100644 libs/srtp/crypto/CVS/Root create mode 100644 libs/srtp/crypto/Makefile create mode 100644 libs/srtp/crypto/Makefile.in create mode 100644 libs/srtp/crypto/VERSION create mode 100644 libs/srtp/crypto/ae_xfm/CVS/Entries create mode 100644 libs/srtp/crypto/ae_xfm/CVS/Repository create mode 100644 libs/srtp/crypto/ae_xfm/CVS/Root create mode 100644 libs/srtp/crypto/ae_xfm/xfm.c create mode 100644 libs/srtp/crypto/cipher/CVS/Entries create mode 100644 libs/srtp/crypto/cipher/CVS/Repository create mode 100644 libs/srtp/crypto/cipher/CVS/Root create mode 100644 libs/srtp/crypto/cipher/aes.c create mode 100644 libs/srtp/crypto/cipher/aes_cbc.c create mode 100644 libs/srtp/crypto/cipher/aes_icm.c create mode 100644 libs/srtp/crypto/cipher/cipher.c create mode 100644 libs/srtp/crypto/cipher/null_cipher.c create mode 100644 libs/srtp/crypto/hash/CVS/Entries create mode 100644 libs/srtp/crypto/hash/CVS/Repository create mode 100644 libs/srtp/crypto/hash/CVS/Root create mode 100644 libs/srtp/crypto/hash/auth.c create mode 100644 libs/srtp/crypto/hash/hmac.c create mode 100644 libs/srtp/crypto/hash/null_auth.c create mode 100644 libs/srtp/crypto/hash/sha1.c create mode 100644 libs/srtp/crypto/include/.cvsignore create mode 100644 libs/srtp/crypto/include/CVS/Entries create mode 100644 libs/srtp/crypto/include/CVS/Repository create mode 100644 libs/srtp/crypto/include/CVS/Root create mode 100644 libs/srtp/crypto/include/aes.h create mode 100644 libs/srtp/crypto/include/aes_cbc.h create mode 100644 libs/srtp/crypto/include/aes_icm.h create mode 100644 libs/srtp/crypto/include/alloc.h create mode 100644 libs/srtp/crypto/include/auth.h create mode 100644 libs/srtp/crypto/include/cipher.h create mode 100644 libs/srtp/crypto/include/config.h create mode 100644 libs/srtp/crypto/include/crypto.h create mode 100644 libs/srtp/crypto/include/crypto_kernel.h create mode 100644 libs/srtp/crypto/include/crypto_math.h create mode 100644 libs/srtp/crypto/include/crypto_types.h create mode 100644 libs/srtp/crypto/include/cryptoalg.h create mode 100644 libs/srtp/crypto/include/datatypes.h create mode 100644 libs/srtp/crypto/include/err.h create mode 100644 libs/srtp/crypto/include/gf2_8.h create mode 100644 libs/srtp/crypto/include/hmac.h create mode 100644 libs/srtp/crypto/include/integers.h create mode 100644 libs/srtp/crypto/include/kernel_compat.h create mode 100644 libs/srtp/crypto/include/key.h create mode 100644 libs/srtp/crypto/include/null_auth.h create mode 100644 libs/srtp/crypto/include/null_cipher.h create mode 100644 libs/srtp/crypto/include/prng.h create mode 100644 libs/srtp/crypto/include/rand_source.h create mode 100644 libs/srtp/crypto/include/rdb.h create mode 100644 libs/srtp/crypto/include/rdbx.h create mode 100644 libs/srtp/crypto/include/sha1.h create mode 100644 libs/srtp/crypto/include/stat.h create mode 100644 libs/srtp/crypto/include/xfm.h create mode 100644 libs/srtp/crypto/kernel/CVS/Entries create mode 100644 libs/srtp/crypto/kernel/CVS/Repository create mode 100644 libs/srtp/crypto/kernel/CVS/Root create mode 100644 libs/srtp/crypto/kernel/alloc.c create mode 100644 libs/srtp/crypto/kernel/crypto_kernel.c create mode 100644 libs/srtp/crypto/kernel/err.c create mode 100644 libs/srtp/crypto/kernel/key.c create mode 100644 libs/srtp/crypto/math/CVS/Entries create mode 100644 libs/srtp/crypto/math/CVS/Repository create mode 100644 libs/srtp/crypto/math/CVS/Root create mode 100644 libs/srtp/crypto/math/datatypes.c create mode 100644 libs/srtp/crypto/math/gf2_8.c create mode 100644 libs/srtp/crypto/math/math.c create mode 100644 libs/srtp/crypto/math/stat.c create mode 100644 libs/srtp/crypto/replay/CVS/Entries create mode 100644 libs/srtp/crypto/replay/CVS/Repository create mode 100644 libs/srtp/crypto/replay/CVS/Root create mode 100644 libs/srtp/crypto/replay/rdb.c create mode 100644 libs/srtp/crypto/replay/rdbx.c create mode 100644 libs/srtp/crypto/replay/ut_sim.c create mode 100644 libs/srtp/crypto/rng/CVS/Entries create mode 100644 libs/srtp/crypto/rng/CVS/Repository create mode 100644 libs/srtp/crypto/rng/CVS/Root create mode 100644 libs/srtp/crypto/rng/ctr_prng.c create mode 100644 libs/srtp/crypto/rng/prng.c create mode 100644 libs/srtp/crypto/rng/rand_linux_kernel.c create mode 100644 libs/srtp/crypto/rng/rand_source.c create mode 100644 libs/srtp/crypto/test/.cvsignore create mode 100644 libs/srtp/crypto/test/CVS/Entries create mode 100644 libs/srtp/crypto/test/CVS/Repository create mode 100644 libs/srtp/crypto/test/CVS/Root create mode 100644 libs/srtp/crypto/test/aes_calc.c create mode 100644 libs/srtp/crypto/test/auth_driver.c create mode 100644 libs/srtp/crypto/test/cipher_driver.c create mode 100644 libs/srtp/crypto/test/datatypes_driver.c create mode 100644 libs/srtp/crypto/test/env.c create mode 100644 libs/srtp/crypto/test/kernel_driver.c create mode 100644 libs/srtp/crypto/test/rand_gen.c create mode 100644 libs/srtp/crypto/test/sha1_driver.c create mode 100644 libs/srtp/crypto/test/stat_driver.c create mode 100644 libs/srtp/doc/.cvsignore create mode 100644 libs/srtp/doc/CVS/Entries create mode 100644 libs/srtp/doc/CVS/Repository create mode 100644 libs/srtp/doc/CVS/Root create mode 100644 libs/srtp/doc/Doxyfile create mode 100644 libs/srtp/doc/Makefile create mode 100644 libs/srtp/doc/Makefile.in create mode 100644 libs/srtp/doc/crypto_kernel.txt create mode 100644 libs/srtp/doc/draft-irtf-cfrg-icm-00.txt create mode 100644 libs/srtp/doc/header.template create mode 100644 libs/srtp/doc/intro.txt create mode 100644 libs/srtp/doc/libsrtp.pdf create mode 100644 libs/srtp/doc/references.txt create mode 100644 libs/srtp/doc/rfc3711.txt create mode 100644 libs/srtp/include/CVS/Entries create mode 100644 libs/srtp/include/CVS/Repository create mode 100644 libs/srtp/include/CVS/Root create mode 100644 libs/srtp/include/rtp.h create mode 100644 libs/srtp/include/srtp.h create mode 100644 libs/srtp/include/ut_sim.h create mode 100755 libs/srtp/install-sh create mode 100644 libs/srtp/srtp/CVS/Entries create mode 100644 libs/srtp/srtp/CVS/Repository create mode 100644 libs/srtp/srtp/CVS/Root create mode 100644 libs/srtp/srtp/srtp.c create mode 100644 libs/srtp/tables/.cvsignore create mode 100644 libs/srtp/tables/CVS/Entries create mode 100644 libs/srtp/tables/CVS/Repository create mode 100644 libs/srtp/tables/CVS/Root create mode 100644 libs/srtp/tables/aes_tables.c create mode 100644 libs/srtp/test/.cvsignore create mode 100644 libs/srtp/test/CVS/Entries create mode 100644 libs/srtp/test/CVS/Repository create mode 100644 libs/srtp/test/CVS/Root create mode 100644 libs/srtp/test/lfsr.c create mode 100644 libs/srtp/test/rdbx_driver.c create mode 100644 libs/srtp/test/replay_driver.c create mode 100644 libs/srtp/test/roc_driver.c create mode 100644 libs/srtp/test/rtp.c create mode 100644 libs/srtp/test/rtpw.c create mode 100644 libs/srtp/test/srtp_driver.c create mode 100644 libs/srtp/timing create mode 100755 libs/srtp/undos.sh create mode 100755 libs/srtp/update.sh diff --git a/libs/srtp/.cvsignore b/libs/srtp/.cvsignore new file mode 100644 index 0000000000..e453ab7a96 --- /dev/null +++ b/libs/srtp/.cvsignore @@ -0,0 +1,4 @@ +Makefile +config.log +config.status +autom4te.cache diff --git a/libs/srtp/CHANGES b/libs/srtp/CHANGES new file mode 100644 index 0000000000..949c389334 --- /dev/null +++ b/libs/srtp/CHANGES @@ -0,0 +1,223 @@ +Changelog + +1.3.20 + + Lots of changes. Thanks to Jeff Chan for catching a memory leak and + helping track down the endian issues with the SSRCs. + +1.3.8 + + This is an interim release. Several little-endian bugs were identified + and fixed; this means that we can use intel/linux for development again. + + Cleaned up sha1 and hmac code significantly, got rid of some excess + functions and properly documented the fuctions in the .h files. + + Eliminated some vestigial files. + + There is a SIGBUS error in the AES encrypt function on sparc + (observed on both solaris and openbsd) with gcc 2.95. Was unable to + find bad pointer anywhere, so I'm wondering if it isn't a compiler + problem (there's a known problem whose profile it fits). It doesn't + appear on any other platform, even in the cipher_driver stress + tests. + + Planned changes + + Change interface to nonces (xtd_seq_num_t) so that it uses + network byte ordering, and is consistent with other arguments. + + +1.3.6 + + Changed /dev/random (in configure.in and crypto/rng/rand_source.c) to + /dev/urandom; the latter is non-blocking on all known platforms (which + corrects some programs that seem to hang) and is actually present on + Open BSD (unlike /dev/random, which only works in the presence of + hardware supported random number generation). + + Added machine/types.h case in include/integers.h. + +1.3.5 + + Removing srtp_t::template and stream_clone(). + + Adding a new policy structure, which will reflect a complete SRTP + policy (including SRTCP). + + This version is *incomplete* and will undergo more changes. It is + provided only as a basis for discussion. + +1.3.4 + + Removed tmmh.c and tmmh.h, which implemented version one of TMMH. + + Changed srtp_get_trailer_length() to act on streams rather than + sessions, and documented the macro SRTP_MAX_TRAILER_LEN, which should + usually be used rather than that function. + + Removed 'salt' from cipher input. + + Changed rdbx to use err.h error codes. + + Changed malloc() and free() to xalloc() and xfree; these functions + are defined in crypto/kernel/alloc.c and declared in + include/alloc.h. + + Added 'output' functions to cipher, in addition to 'encrypt' + functions. It is no longer necessary to zeroize a buffer before + encrypting in order to get keystream. + + Changed octet_string_hex_string() so that "times two" isn't needed + in its input. + + Added crypto_kernel_init() prior to command-line parsing, so that + kernel can be passed command-line arguments, such as "-d + debug_module". This was done to for the applications + test/srtp-driver, test/kernel-driver, and test/ust-driver. + + Improved srtp_init_aes_128_prf - wrote key derivation function + (srtp_kdf_t). + + Add the tag_len as an argument to the auth_compute() function, but + not the corresponding macro. This change allows the tag length for + a given auth func to be set to different values at initialization + time. Previously, the structure auth_t contained the + output_length, but that value was inaccessible from hmac_compute() + and other functions. + + Re-named files from a-b.c to a_b.c. in order to help portability. + + Re-named rijndael to aes (or aes_128 as appropriate). + + +1.2.1 + + Changes so that 1.2.0 compiles on cygwin-win2k. + + Added better error reporting system. If syslog is present on the + OS, then it is used. + + +1.2.0 Many improvements and additions, and a fex fixes + + Fixed endian issues in RTP header construction in the function + rtp_sendto() in srtp/rtp.c. + + Implemented RIJNDAEL decryption operation, adding the functions + rijndael_decrypt() and rijndael_expand_decryption_key(). Also + re-named rijndael_expand_key() to rijndael_expand_encryption_key() + for consistency. + + Implemented random number source using /dev/random, in the files + crypto/rng/rand_source.c and include/rand_source.h. + + Added index check to SEAL cipher (only values less than 2^32 are + allowed) + + Added test case for null_auth authentication function. + + Added a timing test which tests the effect of CPU cache thrash on + cipher throughput. The test is done by the function + cipher_test_throughput_array(); the function + cipher_array_alloc_init() creates an array of ciphers for use in + this test. This test can be accessed by using the -a flag to + the application cipher-driver in the test subdirectory. + + Added argument processing to ust-driver.c, and added that app to + the 'runtest' target in Makefile.in. + + A minor auth_t API change: last argument of auth_init() eliminated. + + +1.0.6 A small but important fix + + Fixed srtp_init_aes_128_prf() by adding octet_string_set_to_zero() + after buffer allocation. + + Eliminated references to no-longer-existing variables in debugging + code in srtp/srtp.c. This fixes the compilation failure that + occured when using PRINT_DEBUG in that file. + + Corrected spelling of Richard Priestley's name in credits. Sorry + Richard! + + +1.0.5 Many little fixes + + Fixed octet_string_set_to_zero(), which was writing one + more zero octet than it should. This bug caused srtp_protect() + and srtp_unprotect() to overwrite the byte that followed the + srtp packet. + + Changed sizeof(uint32_t) to srtp_get_trailer_length() in + srtp-driver.c. This is just defensive coding. + + Added NULL check to malloc in srtp_alloc(). + + +1.0.4 Many minor fixes and two big ones (thanks for the bug reports!) + + Removed 'ssrc' from the srtp_init_aes_128_prf() function argument + list. This is so that applications which do not a priori know the + ssrc which they will be receiving can still use libsrtp. Now the + SSRC value is gleaned from the rtp header and exored into the + counter mode offset in the srtp_protect() and srtp_unprotect() + functions, if that cipher is used. This change cascaed through + many other functions, including srtp_init_from_hex(), + srtp_sender_init() and srtp_receiver_init() in rtp.c, and also + changing the CLI to test/rtpw. In the future, another function + call will be added to the library that enables multiple ssrc/key + pairs to be installed into the same srtp session, so that libsrtp + works with multiple srtp senders. For now, this functionality is + lacking. + + Removed the GDOI interface to the rtpw demo program. This will be + added again at a later date, after the SRTP and GDOI distributions + stabilize. For now, I've left in the GDOI #defines and autoconf + definitions so that they'll be in place when needed. + + Updated tmmhv2_compute() so that it didn't assume any particular + alginment of the output tag. + + Changed bit field variables in srtp.h to unsigned char from + unsigned int in order to avoid a potential endianness issue. + + Fixed rdbx_estimate_index() to handle all input cases. This solves + the now notorious "abaft" bug in the rtpw demo app on linux/intel, + in which spurious replay protection failures happen after that word + is received. + + Added ntohs(hdr->seq) to srtp_protect and srtp_unprotect, removed + from rijndael_icm_set_segment(). + + Added error checking and handling to srtp_sender_init() and + srtp_receiver_init(). + + Changed srtp_alloc() so that it does what you'd expect: allocate an + srtp_ctx_t structure. This hides the library internals. + + +1.0.1 Many minor fixes + + Added cipher_driver_buffer_test(...) to test/cipher-driver.c. This + function checks that the byte-buffering functions used by a cipher + are correct. + + Fixed SunOS/Solaris build problems: added HAVE_SYS_INT_TYPES_H and + changed index_t to xtd_seq_num_t (see include/rdbx.h). + + Fixed SEAL3.0 output byte buffering, added byte-buffering test to + cipher/cipher-driver.c. + + Fixed roc-driver so that the non-sequential insertion test + automatically recovers from bad estimates. This was required to + prevent spurious failures. + + Made rdbx_estimate_index(...) function smarter, so that initial RTP + sequence numbers greater than 32,768 don't cause it to estimate the + rollover counter of 0xffffffff. + + +1.0.0 Initial release + diff --git a/libs/srtp/CVS/Entries b/libs/srtp/CVS/Entries new file mode 100644 index 0000000000..ad3e351897 --- /dev/null +++ b/libs/srtp/CVS/Entries @@ -0,0 +1,22 @@ +/.cvsignore/1.2/Thu Sep 29 12:23:16 2005// +/CHANGES/1.1.1.1/Wed Sep 21 22:51:36 2005// +/LICENSE/1.2/Fri Sep 23 19:34:11 2005// +/Makefile.in/1.14/Fri Mar 17 21:00:46 2006// +/README/1.3/Sun Oct 2 12:04:37 2005// +/TODO/1.2/Fri Sep 23 19:34:11 2005// +/VERSION/1.2/Fri Sep 23 19:34:11 2005// +/config.guess/1.2/Sun Oct 2 19:03:54 2005// +/config.sub/1.2/Sun Oct 2 19:03:53 2005// +/config_in.h/1.7/Mon Oct 3 15:24:08 2005// +/configure/1.9/Mon Oct 3 15:23:13 2005// +/configure.in/1.10/Mon Oct 3 15:19:02 2005// +/install-sh/1.1.1.1/Wed Sep 21 22:51:38 2005// +/timing/1.1.1.1/Wed Sep 21 22:51:38 2005// +/undos.sh/1.1.1.1/Wed Sep 21 22:51:38 2005// +/update.sh/1.1.1.1/Wed Sep 21 22:51:38 2005// +D/crypto//// +D/doc//// +D/include//// +D/srtp//// +D/tables//// +D/test//// diff --git a/libs/srtp/CVS/Repository b/libs/srtp/CVS/Repository new file mode 100644 index 0000000000..3f1f3cb4f5 --- /dev/null +++ b/libs/srtp/CVS/Repository @@ -0,0 +1 @@ +srtp diff --git a/libs/srtp/CVS/Root b/libs/srtp/CVS/Root new file mode 100644 index 0000000000..b4b74237d9 --- /dev/null +++ b/libs/srtp/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@cvs.sourceforge.net:/cvsroot/srtp diff --git a/libs/srtp/LICENSE b/libs/srtp/LICENSE new file mode 100644 index 0000000000..d1ff4e8559 --- /dev/null +++ b/libs/srtp/LICENSE @@ -0,0 +1,35 @@ +/* + * + * Copyright (c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ diff --git a/libs/srtp/Makefile b/libs/srtp/Makefile new file mode 100644 index 0000000000..1bc052ecbd --- /dev/null +++ b/libs/srtp/Makefile @@ -0,0 +1,220 @@ +# Makefile for secure rtp +# +# David A. McGrew +# Cisco Systems, Inc. + +# targets: +# +# runtest runs test applications +# test builds test applications +# libcrypt.a static library implementing crypto engine +# libsrtp.a static library implementing srtp +# clean removes objects, libs, and executables +# distribution cleans and builds a .tgz +# tags builds etags file from all .c and .h files + +.PHONY: all test build_table_apps + +all: test + +runtest: build_table_apps test + @echo "running libsrtp test applications..." + crypto/test/cipher_driver$(EXE) -v >/dev/null + crypto/test/kernel_driver$(EXE) -v >/dev/null + test/rdbx_driver$(EXE) -v >/dev/null + test/srtp_driver$(EXE) -v >/dev/null + test/roc_driver$(EXE) -v >/dev/null + test/replay_driver$(EXE) -v >/dev/null + @echo "libsrtp test applications passed." + $(MAKE) -C crypto runtest + +# makefile variables + +CC = gcc +INCDIR = -Icrypto/include -I$(srcdir)/include -I$(srcdir)/crypto/include +DEFS = -DHAVE_CONFIG_H +CPPFLAGS= +CFLAGS = -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops +LIBS = +LDFLAGS = -L. +COMPILE = $(CC) $(DEFS) $(INCDIR) $(CPPFLAGS) $(CFLAGS) +SRTPLIB = -lsrtp + +RANLIB = ranlib +INSTALL = /usr/bin/install -c + +# EXE defines the suffix on executables - it's .exe for Windows, and +# null on linux, bsd, and OS X and other OSes. +EXE = +# gdoi is the group domain of interpretation for isakmp, a group key +# management system which can provide keys for srtp +gdoi = +# Random source. +RNG_OBJS = rand_source.o + +srcdir = . +top_srcdir = . +top_builddir = + +prefix = /usr/local/freeswitch +exec_prefix = ${prefix} +includedir = ${prefix}/include +libdir = ${exec_prefix}/lib + + +# implicit rules for object files and test apps + +%.o: %.c + $(COMPILE) -c $< -o $@ + +%$(EXE): %.c + $(COMPILE) $(LDFLAGS) $< -o $@ $(SRTPLIB) $(LIBS) + + +# libcrypt.a (the crypto engine) +ciphers = crypto/cipher/cipher.o crypto/cipher/null_cipher.o \ + crypto/cipher/aes.o crypto/cipher/aes_icm.o \ + crypto/cipher/aes_cbc.o + +hashes = crypto/hash/null_auth.o crypto/hash/sha1.o \ + crypto/hash/hmac.o crypto/hash/auth.o # crypto/hash/tmmhv2.o + +replay = crypto/replay/rdb.o crypto/replay/rdbx.o \ + crypto/replay/ut_sim.o + +math = crypto/math/datatypes.o crypto/math/stat.o + +ust = crypto/ust/ust.o + +rng = crypto/rng/$(RNG_OBJS) crypto/rng/prng.o crypto/rng/ctr_prng.o + +err = crypto/kernel/err.o + +kernel = crypto/kernel/crypto_kernel.o crypto/kernel/alloc.o \ + crypto/kernel/key.o $(rng) $(err) # $(ust) + +cryptobj = $(ciphers) $(hashes) $(math) $(stat) $(kernel) $(replay) + +# libsrtp.a (implements srtp processing) + +srtpobj = srtp/srtp.o + +libsrtp.a: $(srtpobj) $(cryptobj) $(gdoi) + ar cr libsrtp.a $^ + $(RANLIB) libsrtp.a + +# libcryptomath.a contains general-purpose routines that are used to +# generate tables and verify cryptoalgorithm implementations - this +# library is not meant to be included in production code + +cryptomath = crypto/math/math.o crypto/math/gf2_8.o + +libcryptomath.a: $(cryptomath) + ar cr libcryptomath.a $(cryptomath) + $(RANLIB) libcryptomath.a + + +# test applications + +crypto_testapp = crypto/test/aes_calc$(EXE) crypto/test/cipher_driver$(EXE) \ + crypto/test/datatypes_driver$(EXE) crypto/test/kernel_driver$(EXE) \ + crypto/test/rand_gen$(EXE) crypto/test/sha1_driver$(EXE) \ + crypto/test/stat_driver$(EXE) + +testapp = $(crypto_testapp) test/srtp_driver$(EXE) test/replay_driver$(EXE) \ + test/roc_driver$(EXE) test/rdbx_driver$(EXE) test/rtpw$(EXE) + +$(testapp): libsrtp.a + +test/rtpw$(EXE): test/rtpw.c test/rtp.c + $(COMPILE) $(LDFLAGS) -o $@ $^ $(LIBS) $(SRTPLIB) + +test: $(testapp) + @echo "Build done. Please run '$(MAKE) runtest' to run self tests." + +memtest: test/srtp_driver + @test/srtp_driver -v -d "alloc" > tmp + @grep freed tmp | wc -l > freed + @grep allocated tmp | wc -l > allocated + @echo "checking for memory leaks (only works with --enable-stdout)" + cmp -s allocated freed + @echo "passed (same number of alloc() and dealloc() calls found)" + @rm freed allocated tmp + +# tables_apps are used to generate the tables used in the crypto +# implementations; these need only be generated during porting, not +# for building libsrtp or the test applications + +table_apps = tables/aes_tables + +build_table_apps: $(table_apps) + +# in the tables/ subdirectory, we use libcryptomath instead of libsrtp + +tables/%: tables/%.c libcryptomath.a + $(COMPILE) $(LDFLAGS) $< -o $@ $(LIBS) libcryptomath.a + +# the target 'plot' runs the timing test (test/srtp_driver -t) then +# uses gnuplot to produce plots of the results - see the script file +# 'timing' + +plot: test/srtp_driver + test/srtp_driver -t > timing.dat + + +# bookkeeping: tags, clean, and distribution + +tags: + etags */*.[ch] */*/*.[ch] + + +# documentation - the target libsrtpdoc builds a PDF file documenting +# libsrtp + +libsrtpdoc: + $(MAKE) -C doc + +.PHONY: clean superclean install + +install: + @if [ -d $(DESTDIR)$(includedir)/srtp ]; then \ + echo "you should run 'make uninstall' first"; exit 1; \ + fi + $(INSTALL) -d $(DESTDIR)$(includedir)/srtp + $(INSTALL) -d $(DESTDIR)$(libdir) + cp include/*.h $(DESTDIR)$(includedir)/srtp + cp crypto/include/*.h $(DESTDIR)$(includedir)/srtp + if [ -f libsrtp.a ]; then cp libsrtp.a $(DESTDIR)$(libdir)/; fi + +uninstall: + rm -rf $(DESTDIR)$(includedir)/srtp + rm -rf $(DESTDIR)$(libdir)/libsrtp.a + +clean: + rm -rf $(cryptobj) $(srtpobj) $(cryptomath) $(table_apps) TAGS \ + libcryptomath.a libsrtp.a core *.core test/core + for a in * */* */*/*; do \ + if [ -f "$$a~" ] ; then rm -f $$a~; fi; \ + done; + for a in $(testapp) $(table_apps); do rm -rf $$a$(EXE); done + rm -rf *.pict *.jpg *.dat + rm -rf freed allocated tmp + $(MAKE) -C doc clean + $(MAKE) -C crypto clean + + +superclean: clean + rm -rf crypto/include/config.h config.log config.cache config.status \ + Makefile .gdb_history test/.gdb_history .DS_Store + rm -rf autom4te.cache + +distname = srtp-$(shell cat VERSION) + +distribution: runtest superclean + if ! [ -f VERSION ]; then exit 1; fi + if [ -f ../$(distname).tgz ]; then \ + mv ../$(distname).tgz ../$(distname).tgz.bak; \ + fi + cd ..; tar cvzf $(distname).tgz srtp + +# EOF diff --git a/libs/srtp/Makefile.in b/libs/srtp/Makefile.in new file mode 100644 index 0000000000..d4234db00a --- /dev/null +++ b/libs/srtp/Makefile.in @@ -0,0 +1,220 @@ +# Makefile for secure rtp +# +# David A. McGrew +# Cisco Systems, Inc. + +# targets: +# +# runtest runs test applications +# test builds test applications +# libcrypt.a static library implementing crypto engine +# libsrtp.a static library implementing srtp +# clean removes objects, libs, and executables +# distribution cleans and builds a .tgz +# tags builds etags file from all .c and .h files + +.PHONY: all test build_table_apps + +all: test + +runtest: build_table_apps test + @echo "running libsrtp test applications..." + crypto/test/cipher_driver$(EXE) -v >/dev/null + crypto/test/kernel_driver$(EXE) -v >/dev/null + test/rdbx_driver$(EXE) -v >/dev/null + test/srtp_driver$(EXE) -v >/dev/null + test/roc_driver$(EXE) -v >/dev/null + test/replay_driver$(EXE) -v >/dev/null + @echo "libsrtp test applications passed." + $(MAKE) -C crypto runtest + +# makefile variables + +CC = @CC@ +INCDIR = -Icrypto/include -I$(srcdir)/include -I$(srcdir)/crypto/include +DEFS = @DEFS@ +CPPFLAGS= @CPPFLAGS@ +CFLAGS = @CFLAGS@ +LIBS = @LIBS@ +LDFLAGS = @LDFLAGS@ -L. +COMPILE = $(CC) $(DEFS) $(INCDIR) $(CPPFLAGS) $(CFLAGS) +SRTPLIB = -lsrtp + +RANLIB = @RANLIB@ +INSTALL = @INSTALL@ + +# EXE defines the suffix on executables - it's .exe for Windows, and +# null on linux, bsd, and OS X and other OSes. +EXE = @EXE@ +# gdoi is the group domain of interpretation for isakmp, a group key +# management system which can provide keys for srtp +gdoi = @GDOI_OBJS@ +# Random source. +RNG_OBJS = @RNG_OBJS@ + +srcdir = @srcdir@ +top_srcdir = @top_srcdir@ +top_builddir = @top_builddir@ +VPATH = @srcdir@ +prefix = @prefix@ +exec_prefix = @exec_prefix@ +includedir = @includedir@ +libdir = @libdir@ + + +# implicit rules for object files and test apps + +%.o: %.c + $(COMPILE) -c $< -o $@ + +%$(EXE): %.c + $(COMPILE) $(LDFLAGS) $< -o $@ $(SRTPLIB) $(LIBS) + + +# libcrypt.a (the crypto engine) +ciphers = crypto/cipher/cipher.o crypto/cipher/null_cipher.o \ + crypto/cipher/aes.o crypto/cipher/aes_icm.o \ + crypto/cipher/aes_cbc.o + +hashes = crypto/hash/null_auth.o crypto/hash/sha1.o \ + crypto/hash/hmac.o crypto/hash/auth.o # crypto/hash/tmmhv2.o + +replay = crypto/replay/rdb.o crypto/replay/rdbx.o \ + crypto/replay/ut_sim.o + +math = crypto/math/datatypes.o crypto/math/stat.o + +ust = crypto/ust/ust.o + +rng = crypto/rng/$(RNG_OBJS) crypto/rng/prng.o crypto/rng/ctr_prng.o + +err = crypto/kernel/err.o + +kernel = crypto/kernel/crypto_kernel.o crypto/kernel/alloc.o \ + crypto/kernel/key.o $(rng) $(err) # $(ust) + +cryptobj = $(ciphers) $(hashes) $(math) $(stat) $(kernel) $(replay) + +# libsrtp.a (implements srtp processing) + +srtpobj = srtp/srtp.o + +libsrtp.a: $(srtpobj) $(cryptobj) $(gdoi) + ar cr libsrtp.a $^ + $(RANLIB) libsrtp.a + +# libcryptomath.a contains general-purpose routines that are used to +# generate tables and verify cryptoalgorithm implementations - this +# library is not meant to be included in production code + +cryptomath = crypto/math/math.o crypto/math/gf2_8.o + +libcryptomath.a: $(cryptomath) + ar cr libcryptomath.a $(cryptomath) + $(RANLIB) libcryptomath.a + + +# test applications + +crypto_testapp = crypto/test/aes_calc$(EXE) crypto/test/cipher_driver$(EXE) \ + crypto/test/datatypes_driver$(EXE) crypto/test/kernel_driver$(EXE) \ + crypto/test/rand_gen$(EXE) crypto/test/sha1_driver$(EXE) \ + crypto/test/stat_driver$(EXE) + +testapp = $(crypto_testapp) test/srtp_driver$(EXE) test/replay_driver$(EXE) \ + test/roc_driver$(EXE) test/rdbx_driver$(EXE) test/rtpw$(EXE) + +$(testapp): libsrtp.a + +test/rtpw$(EXE): test/rtpw.c test/rtp.c + $(COMPILE) $(LDFLAGS) -o $@ $^ $(LIBS) $(SRTPLIB) + +test: $(testapp) + @echo "Build done. Please run '$(MAKE) runtest' to run self tests." + +memtest: test/srtp_driver + @test/srtp_driver -v -d "alloc" > tmp + @grep freed tmp | wc -l > freed + @grep allocated tmp | wc -l > allocated + @echo "checking for memory leaks (only works with --enable-stdout)" + cmp -s allocated freed + @echo "passed (same number of alloc() and dealloc() calls found)" + @rm freed allocated tmp + +# tables_apps are used to generate the tables used in the crypto +# implementations; these need only be generated during porting, not +# for building libsrtp or the test applications + +table_apps = tables/aes_tables + +build_table_apps: $(table_apps) + +# in the tables/ subdirectory, we use libcryptomath instead of libsrtp + +tables/%: tables/%.c libcryptomath.a + $(COMPILE) $(LDFLAGS) $< -o $@ $(LIBS) libcryptomath.a + +# the target 'plot' runs the timing test (test/srtp_driver -t) then +# uses gnuplot to produce plots of the results - see the script file +# 'timing' + +plot: test/srtp_driver + test/srtp_driver -t > timing.dat + + +# bookkeeping: tags, clean, and distribution + +tags: + etags */*.[ch] */*/*.[ch] + + +# documentation - the target libsrtpdoc builds a PDF file documenting +# libsrtp + +libsrtpdoc: + $(MAKE) -C doc + +.PHONY: clean superclean install + +install: + @if [ -d $(DESTDIR)$(includedir)/srtp ]; then \ + echo "you should run 'make uninstall' first"; exit 1; \ + fi + $(INSTALL) -d $(DESTDIR)$(includedir)/srtp + $(INSTALL) -d $(DESTDIR)$(libdir) + cp include/*.h $(DESTDIR)$(includedir)/srtp + cp crypto/include/*.h $(DESTDIR)$(includedir)/srtp + if [ -f libsrtp.a ]; then cp libsrtp.a $(DESTDIR)$(libdir)/; fi + +uninstall: + rm -rf $(DESTDIR)$(includedir)/srtp + rm -rf $(DESTDIR)$(libdir)/libsrtp.a + +clean: + rm -rf $(cryptobj) $(srtpobj) $(cryptomath) $(table_apps) TAGS \ + libcryptomath.a libsrtp.a core *.core test/core + for a in * */* */*/*; do \ + if [ -f "$$a~" ] ; then rm -f $$a~; fi; \ + done; + for a in $(testapp) $(table_apps); do rm -rf $$a$(EXE); done + rm -rf *.pict *.jpg *.dat + rm -rf freed allocated tmp + $(MAKE) -C doc clean + $(MAKE) -C crypto clean + + +superclean: clean + rm -rf crypto/include/config.h config.log config.cache config.status \ + Makefile .gdb_history test/.gdb_history .DS_Store + rm -rf autom4te.cache + +distname = srtp-$(shell cat VERSION) + +distribution: runtest superclean + if ! [ -f VERSION ]; then exit 1; fi + if [ -f ../$(distname).tgz ]; then \ + mv ../$(distname).tgz ../$(distname).tgz.bak; \ + fi + cd ..; tar cvzf $(distname).tgz srtp + +# EOF diff --git a/libs/srtp/README b/libs/srtp/README new file mode 100644 index 0000000000..02dc1f8eb5 --- /dev/null +++ b/libs/srtp/README @@ -0,0 +1,172 @@ +Secure RTP (SRTP) and UST Reference Implementations +David A. McGrew +Cisco Systems, Inc. +mcgrew@cisco.com + + +This package provides an implementation of the Secure Real-time +Transport Protocol (SRTP), the Universal Security Transform (UST), and +a supporting cryptographic kernel. These mechanisms are documented in +the Internet Drafts in the doc/ subdirectory. The SRTP API is +documented in include/srtp.h, and the library is in libsrtp.a (after +compilation). + + +Installation: + +./configure [ options ] # GNU autoconf script +make # or gmake if needed; use GNU make + +The configure script accepts the following options: + + --help provides a usage summary + --disable-debug compile without the runtime debugging system + --enable-syslog use syslog for error reporting + --disable-stdout use stdout for error reporting + --enable-console use /dev/console for error reporting + --gdoi use GDOI key management (disabled at present) + +By default, debbuging is enabled and stdout is used for debugging. +You can use the above configure options to have the debugging output +sent to syslog or the system console. Alternatively, you can define +ERR_REPORTING_FILE in include/conf.h to be any other file that can be +opened by libSRTP, and debug messages will be sent to it. + +This package has been tested on Mac OS X (powerpc-apple-darwin1.4), +Cygwin (i686-pc-cygwin), and Sparc (sparc-sun-solaris2.6). Previous +versions have been tested on Linux and OpenBSD on both x86 and sparc +platforms. + +A quick tour of this package: + +Makefile targets: all, clean, ... +README this file +CHANGES change log +VERSION version number of this package +LICENSE legal details (it's a BSD-like license) +crypto/ciphers/ ciphers (null, aes_icm, ...) +crypto/math/ crypto math routines +crypto/hash/ crypto hashing (hmac, tmmhv2, ...) +crypto/replay/ replay protection +doc/ documentation: rfcs, apis, and suchlike +include/ include files for all code in distribution +srtp/ secure real-time transport protocol implementation +tables/ apps for generating tables (useful in porting) +test/ test drivers + + +Applications + + Several test drivers and a simple and portable srtp application + are included in the test/ subdirectory. + + test driver function tested + ------------------------------------------------------------- + kernel_driver crypto kernel (ciphers, auth funcs, rng) + srtp_driver srtp in-memory tests (does not use the network) + rdbx_driver rdbx (extended replay database) + roc_driver extended sequence number functions + replay_driver replay database (n.b. not used in libsrtp) + cipher_driver ciphers + auth_driver hash functions + + The app rtpw is a simple rtp application which reads words from + /usr/dict/words and then sends them out one at a time using [s]rtp. + Manual srtp keying uses the -k option; automated key management + using gdoi will be added later. + +usage: rtpw [-d ]* [-k [-a][-e]] [-s | -r] dest_ip dest_port +or rtpw -l + + Either the -s (sender) or -r (receiver) option must be chosen. + + The values dest_ip, dest_port are the ip address and udp port to + which the dictionary will be sent, respectively. + + options: + + -s (s)rtp sender - causes app to send words + + -r (s)rtp receive - causes app to receve words + + -k use srtp master key , where the + key is a hexadecimal value (without the + leading "0x") + + -e encrypt/decrypt (for data confidentiality) + (requires use of -k option as well) + + -a message authentication + (requires use of -k option as well) + + -l list debug modules + + -d turn on debugging for module + + +In order to get random 30-byte values for use as key/salt pairs , you +can use the following bash function to format the output of +/dev/random (where that device is available). + +function randhex() { + cat /dev/random | od --read-bytes=32 --width=32 -x | awk '{ print $2 $3 $4 $5 $6 $7 $8 $9 $10 $11 $12 $13 $14 $15 $16 }' +} + + +An example of an SRTP session using two rtpw programs follows: + +set k=c1eec3717da76195bb878578790af71c4ee9f859e197a414a78d5abc7451 + +[sh1]$ test/rtpw -s -k $k -ea 0.0.0.0 9999 +Security services: confidentiality message authentication +set master key/salt to C1EEC3717DA76195BB878578790AF71C/4EE9F859E197A414A78D5ABC7451 +setting SSRC to 2078917053 +sending word: A +sending word: a +sending word: aa +sending word: aal +... + +[sh2]$ test/rtpw -r -k $k -ea 0.0.0.0 9999 +security services: confidentiality message authentication +set master key/salt to C1EEC3717DA76195BB878578790AF71C/4EE9F859E197A414A78D5ABC7451 +19 octets received from SSRC 2078917053 word: A +19 octets received from SSRC 2078917053 word: a +20 octets received from SSRC 2078917053 word: aa +21 octets received from SSRC 2078917053 word: aal +... + +Implementation Notes + + * The srtp_protect() function assumes that the buffer holding the + rtp packet has enough storage allocated that the authentication + tag can be written to the end of that packet. If this assumption + is not valid, memory corruption will ensue. + + * Automated tests for the crypto functions are provided through + the cipher_type_self_test() and auth_type_self_test() functions. + These functions should be used to test each port of this code + to a new platform. + + * Replay protection is contained in the crypto engine, and + tests for it are provided. + + * This implementation provides calls to initialize, protect, and + unprotect RTP packets, and makes as few as possible assumptions + about how these functions will be called. For example, the + caller is not expected to provide packets in order (though if + they're called more than 65k out of sequence, synchronization + will be lost). + + * The sequence number in the rtp packet is used as the low 16 bits + of the sender's local packet index. Note that RTP will start its + sequence number in a random place, and the SRTP layer just jumps + forward to that number at its first invocation. An earlier + version of this library used initial sequence numbers that are + less than 32,768; this trick is no longer required as the + rdbx_estimate_index(...) function has been made smarter. + + * The replay window is 128 bits in length, and is hard-coded to this + value for now. + + diff --git a/libs/srtp/TODO b/libs/srtp/TODO new file mode 100644 index 0000000000..18846e8ad0 --- /dev/null +++ b/libs/srtp/TODO @@ -0,0 +1,66 @@ +TODO List + +1.4.1 + + - document which fields are in NBO/HBO, and check for consistency. + + - move HAVE_U_LONG_LONG inside of datatypes.c, or some other + separate file + + - re-write configure.in to make cross-compilation easier + + - eliminate GENERIC_AESICM by generalizing the code a bit + +Older comments + + - add tests for key_limit_t datatype + + - move octet_get_weight() from datatypes.c to math.c (any other + funcs?) + +Changes and additions planned + + Make cipher and auth dealloc() functions zeroize the key-storage + areas before calling free(). + + Eliminate key_len from auth_init() + + Doucument internal APIs (cipher, auth, srtp_protect, ...) + + +SRTP options not (yet) included in this libaray: + + - the aes-f8-mode cipher + - the Master Key Index + - re-keying using the key derivation function (only the initial + use of the PRF has been implemented, as it's sufficient + for most uses) + + +(OLD) PLANNED CHANGES + + strip out test/lfsr.c + + Write new documentation!!! + + Fix the x86 assembly code in aes.c. + + Eliminate /* DAM */ - there's one in srtp.c + + Change debugging so that it can print more than one line. Or perhaps + just change it so that a single check of the debug-enabled flag is + needed. + + Improve interface between cipher and rdbx - perhaps generalize rdbx + into 'nonce' datatype. + + Make rijndael_icm accept variable sized keys. + + Add rdbx functions that allow different-sized explicit sequence + numbers to be used. + + Write uniform byte-buffering code for PRFs, preferably as macros. + + Consider eliminating low-level alloc functions in favor of len() + functions, so that there need not be multiple allocations within a + particular alloc() function. diff --git a/libs/srtp/VERSION b/libs/srtp/VERSION new file mode 100644 index 0000000000..9df886c42a --- /dev/null +++ b/libs/srtp/VERSION @@ -0,0 +1 @@ +1.4.2 diff --git a/libs/srtp/config.guess b/libs/srtp/config.guess new file mode 100755 index 0000000000..7d0185e019 --- /dev/null +++ b/libs/srtp/config.guess @@ -0,0 +1,1447 @@ +#! /bin/sh +# Attempt to guess a canonical system name. +# Copyright (C) 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999, +# 2000, 2001, 2002, 2003, 2004 Free Software Foundation, Inc. + +timestamp='2004-09-07' + +# This file is free software; you can redistribute it and/or modify it +# under the terms of the GNU General Public License as published by +# the Free Software Foundation; either version 2 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, but +# WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU +# General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program; if not, write to the Free Software +# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. +# +# As a special exception to the GNU General Public License, if you +# distribute this file as part of a program that contains a +# configuration script generated by Autoconf, you may include it under +# the same distribution terms that you use for the rest of that program. + +# Originally written by Per Bothner . +# Please send patches to . Submit a context +# diff and a properly formatted ChangeLog entry. +# +# This script attempts to guess a canonical system name similar to +# config.sub. If it succeeds, it prints the system name on stdout, and +# exits with 0. Otherwise, it exits with 1. +# +# The plan is that this can be called by configure scripts if you +# don't specify an explicit build system type. + +me=`echo "$0" | sed -e 's,.*/,,'` + +usage="\ +Usage: $0 [OPTION] + +Output the configuration name of the system \`$me' is run on. + +Operation modes: + -h, --help print this help, then exit + -t, --time-stamp print date of last modification, then exit + -v, --version print version number, then exit + +Report bugs and patches to ." + +version="\ +GNU config.guess ($timestamp) + +Originally written by Per Bothner. +Copyright (C) 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003, 2004 +Free Software Foundation, Inc. + +This is free software; see the source for copying conditions. There is NO +warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE." + +help=" +Try \`$me --help' for more information." + +# Parse command line +while test $# -gt 0 ; do + case $1 in + --time-stamp | --time* | -t ) + echo "$timestamp" ; exit 0 ;; + --version | -v ) + echo "$version" ; exit 0 ;; + --help | --h* | -h ) + echo "$usage"; exit 0 ;; + -- ) # Stop option processing + shift; break ;; + - ) # Use stdin as input. + break ;; + -* ) + echo "$me: invalid option $1$help" >&2 + exit 1 ;; + * ) + break ;; + esac +done + +if test $# != 0; then + echo "$me: too many arguments$help" >&2 + exit 1 +fi + +trap 'exit 1' 1 2 15 + +# CC_FOR_BUILD -- compiler used by this script. Note that the use of a +# compiler to aid in system detection is discouraged as it requires +# temporary files to be created and, as you can see below, it is a +# headache to deal with in a portable fashion. + +# Historically, `CC_FOR_BUILD' used to be named `HOST_CC'. We still +# use `HOST_CC' if defined, but it is deprecated. + +# Portable tmp directory creation inspired by the Autoconf team. + +set_cc_for_build=' +trap "exitcode=\$?; (rm -f \$tmpfiles 2>/dev/null; rmdir \$tmp 2>/dev/null) && exit \$exitcode" 0 ; +trap "rm -f \$tmpfiles 2>/dev/null; rmdir \$tmp 2>/dev/null; exit 1" 1 2 13 15 ; +: ${TMPDIR=/tmp} ; + { tmp=`(umask 077 && mktemp -d -q "$TMPDIR/cgXXXXXX") 2>/dev/null` && test -n "$tmp" && test -d "$tmp" ; } || + { test -n "$RANDOM" && tmp=$TMPDIR/cg$$-$RANDOM && (umask 077 && mkdir $tmp) ; } || + { tmp=$TMPDIR/cg-$$ && (umask 077 && mkdir $tmp) && echo "Warning: creating insecure temp directory" >&2 ; } || + { echo "$me: cannot create a temporary directory in $TMPDIR" >&2 ; exit 1 ; } ; +dummy=$tmp/dummy ; +tmpfiles="$dummy.c $dummy.o $dummy.rel $dummy" ; +case $CC_FOR_BUILD,$HOST_CC,$CC in + ,,) echo "int x;" > $dummy.c ; + for c in cc gcc c89 c99 ; do + if ($c -c -o $dummy.o $dummy.c) >/dev/null 2>&1 ; then + CC_FOR_BUILD="$c"; break ; + fi ; + done ; + if test x"$CC_FOR_BUILD" = x ; then + CC_FOR_BUILD=no_compiler_found ; + fi + ;; + ,,*) CC_FOR_BUILD=$CC ;; + ,*,*) CC_FOR_BUILD=$HOST_CC ;; +esac ;' + +# This is needed to find uname on a Pyramid OSx when run in the BSD universe. +# (ghazi@noc.rutgers.edu 1994-08-24) +if (test -f /.attbin/uname) >/dev/null 2>&1 ; then + PATH=$PATH:/.attbin ; export PATH +fi + +UNAME_MACHINE=`(uname -m) 2>/dev/null` || UNAME_MACHINE=unknown +UNAME_RELEASE=`(uname -r) 2>/dev/null` || UNAME_RELEASE=unknown +UNAME_SYSTEM=`(uname -s) 2>/dev/null` || UNAME_SYSTEM=unknown +UNAME_VERSION=`(uname -v) 2>/dev/null` || UNAME_VERSION=unknown + +# Note: order is significant - the case branches are not exclusive. + +case "${UNAME_MACHINE}:${UNAME_SYSTEM}:${UNAME_RELEASE}:${UNAME_VERSION}" in + *:NetBSD:*:*) + # NetBSD (nbsd) targets should (where applicable) match one or + # more of the tupples: *-*-netbsdelf*, *-*-netbsdaout*, + # *-*-netbsdecoff* and *-*-netbsd*. For targets that recently + # switched to ELF, *-*-netbsd* would select the old + # object file format. This provides both forward + # compatibility and a consistent mechanism for selecting the + # object file format. + # + # Note: NetBSD doesn't particularly care about the vendor + # portion of the name. We always set it to "unknown". + sysctl="sysctl -n hw.machine_arch" + UNAME_MACHINE_ARCH=`(/sbin/$sysctl 2>/dev/null || \ + /usr/sbin/$sysctl 2>/dev/null || echo unknown)` + case "${UNAME_MACHINE_ARCH}" in + armeb) machine=armeb-unknown ;; + arm*) machine=arm-unknown ;; + sh3el) machine=shl-unknown ;; + sh3eb) machine=sh-unknown ;; + *) machine=${UNAME_MACHINE_ARCH}-unknown ;; + esac + # The Operating System including object format, if it has switched + # to ELF recently, or will in the future. + case "${UNAME_MACHINE_ARCH}" in + arm*|i386|m68k|ns32k|sh3*|sparc|vax) + eval $set_cc_for_build + if echo __ELF__ | $CC_FOR_BUILD -E - 2>/dev/null \ + | grep __ELF__ >/dev/null + then + # Once all utilities can be ECOFF (netbsdecoff) or a.out (netbsdaout). + # Return netbsd for either. FIX? + os=netbsd + else + os=netbsdelf + fi + ;; + *) + os=netbsd + ;; + esac + # The OS release + # Debian GNU/NetBSD machines have a different userland, and + # thus, need a distinct triplet. However, they do not need + # kernel version information, so it can be replaced with a + # suitable tag, in the style of linux-gnu. + case "${UNAME_VERSION}" in + Debian*) + release='-gnu' + ;; + *) + release=`echo ${UNAME_RELEASE}|sed -e 's/[-_].*/\./'` + ;; + esac + # Since CPU_TYPE-MANUFACTURER-KERNEL-OPERATING_SYSTEM: + # contains redundant information, the shorter form: + # CPU_TYPE-MANUFACTURER-OPERATING_SYSTEM is used. + echo "${machine}-${os}${release}" + exit 0 ;; + amd64:OpenBSD:*:*) + echo x86_64-unknown-openbsd${UNAME_RELEASE} + exit 0 ;; + amiga:OpenBSD:*:*) + echo m68k-unknown-openbsd${UNAME_RELEASE} + exit 0 ;; + cats:OpenBSD:*:*) + echo arm-unknown-openbsd${UNAME_RELEASE} + exit 0 ;; + hp300:OpenBSD:*:*) + echo m68k-unknown-openbsd${UNAME_RELEASE} + exit 0 ;; + luna88k:OpenBSD:*:*) + echo m88k-unknown-openbsd${UNAME_RELEASE} + exit 0 ;; + mac68k:OpenBSD:*:*) + echo m68k-unknown-openbsd${UNAME_RELEASE} + exit 0 ;; + macppc:OpenBSD:*:*) + echo powerpc-unknown-openbsd${UNAME_RELEASE} + exit 0 ;; + mvme68k:OpenBSD:*:*) + echo m68k-unknown-openbsd${UNAME_RELEASE} + exit 0 ;; + mvme88k:OpenBSD:*:*) + echo m88k-unknown-openbsd${UNAME_RELEASE} + exit 0 ;; + mvmeppc:OpenBSD:*:*) + echo powerpc-unknown-openbsd${UNAME_RELEASE} + exit 0 ;; + sgi:OpenBSD:*:*) + echo mips64-unknown-openbsd${UNAME_RELEASE} + exit 0 ;; + sun3:OpenBSD:*:*) + echo m68k-unknown-openbsd${UNAME_RELEASE} + exit 0 ;; + *:OpenBSD:*:*) + echo ${UNAME_MACHINE}-unknown-openbsd${UNAME_RELEASE} + exit 0 ;; + *:ekkoBSD:*:*) + echo ${UNAME_MACHINE}-unknown-ekkobsd${UNAME_RELEASE} + exit 0 ;; + macppc:MirBSD:*:*) + echo powerppc-unknown-mirbsd${UNAME_RELEASE} + exit 0 ;; + *:MirBSD:*:*) + echo ${UNAME_MACHINE}-unknown-mirbsd${UNAME_RELEASE} + exit 0 ;; + alpha:OSF1:*:*) + case $UNAME_RELEASE in + *4.0) + UNAME_RELEASE=`/usr/sbin/sizer -v | awk '{print $3}'` + ;; + *5.*) + UNAME_RELEASE=`/usr/sbin/sizer -v | awk '{print $4}'` + ;; + esac + # According to Compaq, /usr/sbin/psrinfo has been available on + # OSF/1 and Tru64 systems produced since 1995. I hope that + # covers most systems running today. This code pipes the CPU + # types through head -n 1, so we only detect the type of CPU 0. + ALPHA_CPU_TYPE=`/usr/sbin/psrinfo -v | sed -n -e 's/^ The alpha \(.*\) processor.*$/\1/p' | head -n 1` + case "$ALPHA_CPU_TYPE" in + "EV4 (21064)") + UNAME_MACHINE="alpha" ;; + "EV4.5 (21064)") + UNAME_MACHINE="alpha" ;; + "LCA4 (21066/21068)") + UNAME_MACHINE="alpha" ;; + "EV5 (21164)") + UNAME_MACHINE="alphaev5" ;; + "EV5.6 (21164A)") + UNAME_MACHINE="alphaev56" ;; + "EV5.6 (21164PC)") + UNAME_MACHINE="alphapca56" ;; + "EV5.7 (21164PC)") + UNAME_MACHINE="alphapca57" ;; + "EV6 (21264)") + UNAME_MACHINE="alphaev6" ;; + "EV6.7 (21264A)") + UNAME_MACHINE="alphaev67" ;; + "EV6.8CB (21264C)") + UNAME_MACHINE="alphaev68" ;; + "EV6.8AL (21264B)") + UNAME_MACHINE="alphaev68" ;; + "EV6.8CX (21264D)") + UNAME_MACHINE="alphaev68" ;; + "EV6.9A (21264/EV69A)") + UNAME_MACHINE="alphaev69" ;; + "EV7 (21364)") + UNAME_MACHINE="alphaev7" ;; + "EV7.9 (21364A)") + UNAME_MACHINE="alphaev79" ;; + esac + # A Pn.n version is a patched version. + # A Vn.n version is a released version. + # A Tn.n version is a released field test version. + # A Xn.n version is an unreleased experimental baselevel. + # 1.2 uses "1.2" for uname -r. + echo ${UNAME_MACHINE}-dec-osf`echo ${UNAME_RELEASE} | sed -e 's/^[PVTX]//' | tr 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz'` + exit 0 ;; + Alpha\ *:Windows_NT*:*) + # How do we know it's Interix rather than the generic POSIX subsystem? + # Should we change UNAME_MACHINE based on the output of uname instead + # of the specific Alpha model? + echo alpha-pc-interix + exit 0 ;; + 21064:Windows_NT:50:3) + echo alpha-dec-winnt3.5 + exit 0 ;; + Amiga*:UNIX_System_V:4.0:*) + echo m68k-unknown-sysv4 + exit 0;; + *:[Aa]miga[Oo][Ss]:*:*) + echo ${UNAME_MACHINE}-unknown-amigaos + exit 0 ;; + *:[Mm]orph[Oo][Ss]:*:*) + echo ${UNAME_MACHINE}-unknown-morphos + exit 0 ;; + *:OS/390:*:*) + echo i370-ibm-openedition + exit 0 ;; + *:OS400:*:*) + echo powerpc-ibm-os400 + exit 0 ;; + arm:RISC*:1.[012]*:*|arm:riscix:1.[012]*:*) + echo arm-acorn-riscix${UNAME_RELEASE} + exit 0;; + SR2?01:HI-UX/MPP:*:* | SR8000:HI-UX/MPP:*:*) + echo hppa1.1-hitachi-hiuxmpp + exit 0;; + Pyramid*:OSx*:*:* | MIS*:OSx*:*:* | MIS*:SMP_DC-OSx*:*:*) + # akee@wpdis03.wpafb.af.mil (Earle F. Ake) contributed MIS and NILE. + if test "`(/bin/universe) 2>/dev/null`" = att ; then + echo pyramid-pyramid-sysv3 + else + echo pyramid-pyramid-bsd + fi + exit 0 ;; + NILE*:*:*:dcosx) + echo pyramid-pyramid-svr4 + exit 0 ;; + DRS?6000:unix:4.0:6*) + echo sparc-icl-nx6 + exit 0 ;; + DRS?6000:UNIX_SV:4.2*:7*) + case `/usr/bin/uname -p` in + sparc) echo sparc-icl-nx7 && exit 0 ;; + esac ;; + sun4H:SunOS:5.*:*) + echo sparc-hal-solaris2`echo ${UNAME_RELEASE}|sed -e 's/[^.]*//'` + exit 0 ;; + sun4*:SunOS:5.*:* | tadpole*:SunOS:5.*:*) + echo sparc-sun-solaris2`echo ${UNAME_RELEASE}|sed -e 's/[^.]*//'` + exit 0 ;; + i86pc:SunOS:5.*:*) + echo i386-pc-solaris2`echo ${UNAME_RELEASE}|sed -e 's/[^.]*//'` + exit 0 ;; + sun4*:SunOS:6*:*) + # According to config.sub, this is the proper way to canonicalize + # SunOS6. Hard to guess exactly what SunOS6 will be like, but + # it's likely to be more like Solaris than SunOS4. + echo sparc-sun-solaris3`echo ${UNAME_RELEASE}|sed -e 's/[^.]*//'` + exit 0 ;; + sun4*:SunOS:*:*) + case "`/usr/bin/arch -k`" in + Series*|S4*) + UNAME_RELEASE=`uname -v` + ;; + esac + # Japanese Language versions have a version number like `4.1.3-JL'. + echo sparc-sun-sunos`echo ${UNAME_RELEASE}|sed -e 's/-/_/'` + exit 0 ;; + sun3*:SunOS:*:*) + echo m68k-sun-sunos${UNAME_RELEASE} + exit 0 ;; + sun*:*:4.2BSD:*) + UNAME_RELEASE=`(sed 1q /etc/motd | awk '{print substr($5,1,3)}') 2>/dev/null` + test "x${UNAME_RELEASE}" = "x" && UNAME_RELEASE=3 + case "`/bin/arch`" in + sun3) + echo m68k-sun-sunos${UNAME_RELEASE} + ;; + sun4) + echo sparc-sun-sunos${UNAME_RELEASE} + ;; + esac + exit 0 ;; + aushp:SunOS:*:*) + echo sparc-auspex-sunos${UNAME_RELEASE} + exit 0 ;; + # The situation for MiNT is a little confusing. The machine name + # can be virtually everything (everything which is not + # "atarist" or "atariste" at least should have a processor + # > m68000). The system name ranges from "MiNT" over "FreeMiNT" + # to the lowercase version "mint" (or "freemint"). Finally + # the system name "TOS" denotes a system which is actually not + # MiNT. But MiNT is downward compatible to TOS, so this should + # be no problem. + atarist[e]:*MiNT:*:* | atarist[e]:*mint:*:* | atarist[e]:*TOS:*:*) + echo m68k-atari-mint${UNAME_RELEASE} + exit 0 ;; + atari*:*MiNT:*:* | atari*:*mint:*:* | atarist[e]:*TOS:*:*) + echo m68k-atari-mint${UNAME_RELEASE} + exit 0 ;; + *falcon*:*MiNT:*:* | *falcon*:*mint:*:* | *falcon*:*TOS:*:*) + echo m68k-atari-mint${UNAME_RELEASE} + exit 0 ;; + milan*:*MiNT:*:* | milan*:*mint:*:* | *milan*:*TOS:*:*) + echo m68k-milan-mint${UNAME_RELEASE} + exit 0 ;; + hades*:*MiNT:*:* | hades*:*mint:*:* | *hades*:*TOS:*:*) + echo m68k-hades-mint${UNAME_RELEASE} + exit 0 ;; + *:*MiNT:*:* | *:*mint:*:* | *:*TOS:*:*) + echo m68k-unknown-mint${UNAME_RELEASE} + exit 0 ;; + m68k:machten:*:*) + echo m68k-apple-machten${UNAME_RELEASE} + exit 0 ;; + powerpc:machten:*:*) + echo powerpc-apple-machten${UNAME_RELEASE} + exit 0 ;; + RISC*:Mach:*:*) + echo mips-dec-mach_bsd4.3 + exit 0 ;; + RISC*:ULTRIX:*:*) + echo mips-dec-ultrix${UNAME_RELEASE} + exit 0 ;; + VAX*:ULTRIX*:*:*) + echo vax-dec-ultrix${UNAME_RELEASE} + exit 0 ;; + 2020:CLIX:*:* | 2430:CLIX:*:*) + echo clipper-intergraph-clix${UNAME_RELEASE} + exit 0 ;; + mips:*:*:UMIPS | mips:*:*:RISCos) + eval $set_cc_for_build + sed 's/^ //' << EOF >$dummy.c +#ifdef __cplusplus +#include /* for printf() prototype */ + int main (int argc, char *argv[]) { +#else + int main (argc, argv) int argc; char *argv[]; { +#endif + #if defined (host_mips) && defined (MIPSEB) + #if defined (SYSTYPE_SYSV) + printf ("mips-mips-riscos%ssysv\n", argv[1]); exit (0); + #endif + #if defined (SYSTYPE_SVR4) + printf ("mips-mips-riscos%ssvr4\n", argv[1]); exit (0); + #endif + #if defined (SYSTYPE_BSD43) || defined(SYSTYPE_BSD) + printf ("mips-mips-riscos%sbsd\n", argv[1]); exit (0); + #endif + #endif + exit (-1); + } +EOF + $CC_FOR_BUILD -o $dummy $dummy.c \ + && $dummy `echo "${UNAME_RELEASE}" | sed -n 's/\([0-9]*\).*/\1/p'` \ + && exit 0 + echo mips-mips-riscos${UNAME_RELEASE} + exit 0 ;; + Motorola:PowerMAX_OS:*:*) + echo powerpc-motorola-powermax + exit 0 ;; + Motorola:*:4.3:PL8-*) + echo powerpc-harris-powermax + exit 0 ;; + Night_Hawk:*:*:PowerMAX_OS | Synergy:PowerMAX_OS:*:*) + echo powerpc-harris-powermax + exit 0 ;; + Night_Hawk:Power_UNIX:*:*) + echo powerpc-harris-powerunix + exit 0 ;; + m88k:CX/UX:7*:*) + echo m88k-harris-cxux7 + exit 0 ;; + m88k:*:4*:R4*) + echo m88k-motorola-sysv4 + exit 0 ;; + m88k:*:3*:R3*) + echo m88k-motorola-sysv3 + exit 0 ;; + AViiON:dgux:*:*) + # DG/UX returns AViiON for all architectures + UNAME_PROCESSOR=`/usr/bin/uname -p` + if [ $UNAME_PROCESSOR = mc88100 ] || [ $UNAME_PROCESSOR = mc88110 ] + then + if [ ${TARGET_BINARY_INTERFACE}x = m88kdguxelfx ] || \ + [ ${TARGET_BINARY_INTERFACE}x = x ] + then + echo m88k-dg-dgux${UNAME_RELEASE} + else + echo m88k-dg-dguxbcs${UNAME_RELEASE} + fi + else + echo i586-dg-dgux${UNAME_RELEASE} + fi + exit 0 ;; + M88*:DolphinOS:*:*) # DolphinOS (SVR3) + echo m88k-dolphin-sysv3 + exit 0 ;; + M88*:*:R3*:*) + # Delta 88k system running SVR3 + echo m88k-motorola-sysv3 + exit 0 ;; + XD88*:*:*:*) # Tektronix XD88 system running UTekV (SVR3) + echo m88k-tektronix-sysv3 + exit 0 ;; + Tek43[0-9][0-9]:UTek:*:*) # Tektronix 4300 system running UTek (BSD) + echo m68k-tektronix-bsd + exit 0 ;; + *:IRIX*:*:*) + echo mips-sgi-irix`echo ${UNAME_RELEASE}|sed -e 's/-/_/g'` + exit 0 ;; + ????????:AIX?:[12].1:2) # AIX 2.2.1 or AIX 2.1.1 is RT/PC AIX. + echo romp-ibm-aix # uname -m gives an 8 hex-code CPU id + exit 0 ;; # Note that: echo "'`uname -s`'" gives 'AIX ' + i*86:AIX:*:*) + echo i386-ibm-aix + exit 0 ;; + ia64:AIX:*:*) + if [ -x /usr/bin/oslevel ] ; then + IBM_REV=`/usr/bin/oslevel` + else + IBM_REV=${UNAME_VERSION}.${UNAME_RELEASE} + fi + echo ${UNAME_MACHINE}-ibm-aix${IBM_REV} + exit 0 ;; + *:AIX:2:3) + if grep bos325 /usr/include/stdio.h >/dev/null 2>&1; then + eval $set_cc_for_build + sed 's/^ //' << EOF >$dummy.c + #include + + main() + { + if (!__power_pc()) + exit(1); + puts("powerpc-ibm-aix3.2.5"); + exit(0); + } +EOF + $CC_FOR_BUILD -o $dummy $dummy.c && $dummy && exit 0 + echo rs6000-ibm-aix3.2.5 + elif grep bos324 /usr/include/stdio.h >/dev/null 2>&1; then + echo rs6000-ibm-aix3.2.4 + else + echo rs6000-ibm-aix3.2 + fi + exit 0 ;; + *:AIX:*:[45]) + IBM_CPU_ID=`/usr/sbin/lsdev -C -c processor -S available | sed 1q | awk '{ print $1 }'` + if /usr/sbin/lsattr -El ${IBM_CPU_ID} | grep ' POWER' >/dev/null 2>&1; then + IBM_ARCH=rs6000 + else + IBM_ARCH=powerpc + fi + if [ -x /usr/bin/oslevel ] ; then + IBM_REV=`/usr/bin/oslevel` + else + IBM_REV=${UNAME_VERSION}.${UNAME_RELEASE} + fi + echo ${IBM_ARCH}-ibm-aix${IBM_REV} + exit 0 ;; + *:AIX:*:*) + echo rs6000-ibm-aix + exit 0 ;; + ibmrt:4.4BSD:*|romp-ibm:BSD:*) + echo romp-ibm-bsd4.4 + exit 0 ;; + ibmrt:*BSD:*|romp-ibm:BSD:*) # covers RT/PC BSD and + echo romp-ibm-bsd${UNAME_RELEASE} # 4.3 with uname added to + exit 0 ;; # report: romp-ibm BSD 4.3 + *:BOSX:*:*) + echo rs6000-bull-bosx + exit 0 ;; + DPX/2?00:B.O.S.:*:*) + echo m68k-bull-sysv3 + exit 0 ;; + 9000/[34]??:4.3bsd:1.*:*) + echo m68k-hp-bsd + exit 0 ;; + hp300:4.4BSD:*:* | 9000/[34]??:4.3bsd:2.*:*) + echo m68k-hp-bsd4.4 + exit 0 ;; + 9000/[34678]??:HP-UX:*:*) + HPUX_REV=`echo ${UNAME_RELEASE}|sed -e 's/[^.]*.[0B]*//'` + case "${UNAME_MACHINE}" in + 9000/31? ) HP_ARCH=m68000 ;; + 9000/[34]?? ) HP_ARCH=m68k ;; + 9000/[678][0-9][0-9]) + if [ -x /usr/bin/getconf ]; then + sc_cpu_version=`/usr/bin/getconf SC_CPU_VERSION 2>/dev/null` + sc_kernel_bits=`/usr/bin/getconf SC_KERNEL_BITS 2>/dev/null` + case "${sc_cpu_version}" in + 523) HP_ARCH="hppa1.0" ;; # CPU_PA_RISC1_0 + 528) HP_ARCH="hppa1.1" ;; # CPU_PA_RISC1_1 + 532) # CPU_PA_RISC2_0 + case "${sc_kernel_bits}" in + 32) HP_ARCH="hppa2.0n" ;; + 64) HP_ARCH="hppa2.0w" ;; + '') HP_ARCH="hppa2.0" ;; # HP-UX 10.20 + esac ;; + esac + fi + if [ "${HP_ARCH}" = "" ]; then + eval $set_cc_for_build + sed 's/^ //' << EOF >$dummy.c + + #define _HPUX_SOURCE + #include + #include + + int main () + { + #if defined(_SC_KERNEL_BITS) + long bits = sysconf(_SC_KERNEL_BITS); + #endif + long cpu = sysconf (_SC_CPU_VERSION); + + switch (cpu) + { + case CPU_PA_RISC1_0: puts ("hppa1.0"); break; + case CPU_PA_RISC1_1: puts ("hppa1.1"); break; + case CPU_PA_RISC2_0: + #if defined(_SC_KERNEL_BITS) + switch (bits) + { + case 64: puts ("hppa2.0w"); break; + case 32: puts ("hppa2.0n"); break; + default: puts ("hppa2.0"); break; + } break; + #else /* !defined(_SC_KERNEL_BITS) */ + puts ("hppa2.0"); break; + #endif + default: puts ("hppa1.0"); break; + } + exit (0); + } +EOF + (CCOPTS= $CC_FOR_BUILD -o $dummy $dummy.c 2>/dev/null) && HP_ARCH=`$dummy` + test -z "$HP_ARCH" && HP_ARCH=hppa + fi ;; + esac + if [ ${HP_ARCH} = "hppa2.0w" ] + then + # avoid double evaluation of $set_cc_for_build + test -n "$CC_FOR_BUILD" || eval $set_cc_for_build + if echo __LP64__ | (CCOPTS= $CC_FOR_BUILD -E -) | grep __LP64__ >/dev/null + then + HP_ARCH="hppa2.0w" + else + HP_ARCH="hppa64" + fi + fi + echo ${HP_ARCH}-hp-hpux${HPUX_REV} + exit 0 ;; + ia64:HP-UX:*:*) + HPUX_REV=`echo ${UNAME_RELEASE}|sed -e 's/[^.]*.[0B]*//'` + echo ia64-hp-hpux${HPUX_REV} + exit 0 ;; + 3050*:HI-UX:*:*) + eval $set_cc_for_build + sed 's/^ //' << EOF >$dummy.c + #include + int + main () + { + long cpu = sysconf (_SC_CPU_VERSION); + /* The order matters, because CPU_IS_HP_MC68K erroneously returns + true for CPU_PA_RISC1_0. CPU_IS_PA_RISC returns correct + results, however. */ + if (CPU_IS_PA_RISC (cpu)) + { + switch (cpu) + { + case CPU_PA_RISC1_0: puts ("hppa1.0-hitachi-hiuxwe2"); break; + case CPU_PA_RISC1_1: puts ("hppa1.1-hitachi-hiuxwe2"); break; + case CPU_PA_RISC2_0: puts ("hppa2.0-hitachi-hiuxwe2"); break; + default: puts ("hppa-hitachi-hiuxwe2"); break; + } + } + else if (CPU_IS_HP_MC68K (cpu)) + puts ("m68k-hitachi-hiuxwe2"); + else puts ("unknown-hitachi-hiuxwe2"); + exit (0); + } +EOF + $CC_FOR_BUILD -o $dummy $dummy.c && $dummy && exit 0 + echo unknown-hitachi-hiuxwe2 + exit 0 ;; + 9000/7??:4.3bsd:*:* | 9000/8?[79]:4.3bsd:*:* ) + echo hppa1.1-hp-bsd + exit 0 ;; + 9000/8??:4.3bsd:*:*) + echo hppa1.0-hp-bsd + exit 0 ;; + *9??*:MPE/iX:*:* | *3000*:MPE/iX:*:*) + echo hppa1.0-hp-mpeix + exit 0 ;; + hp7??:OSF1:*:* | hp8?[79]:OSF1:*:* ) + echo hppa1.1-hp-osf + exit 0 ;; + hp8??:OSF1:*:*) + echo hppa1.0-hp-osf + exit 0 ;; + i*86:OSF1:*:*) + if [ -x /usr/sbin/sysversion ] ; then + echo ${UNAME_MACHINE}-unknown-osf1mk + else + echo ${UNAME_MACHINE}-unknown-osf1 + fi + exit 0 ;; + parisc*:Lites*:*:*) + echo hppa1.1-hp-lites + exit 0 ;; + C1*:ConvexOS:*:* | convex:ConvexOS:C1*:*) + echo c1-convex-bsd + exit 0 ;; + C2*:ConvexOS:*:* | convex:ConvexOS:C2*:*) + if getsysinfo -f scalar_acc + then echo c32-convex-bsd + else echo c2-convex-bsd + fi + exit 0 ;; + C34*:ConvexOS:*:* | convex:ConvexOS:C34*:*) + echo c34-convex-bsd + exit 0 ;; + C38*:ConvexOS:*:* | convex:ConvexOS:C38*:*) + echo c38-convex-bsd + exit 0 ;; + C4*:ConvexOS:*:* | convex:ConvexOS:C4*:*) + echo c4-convex-bsd + exit 0 ;; + CRAY*Y-MP:*:*:*) + echo ymp-cray-unicos${UNAME_RELEASE} | sed -e 's/\.[^.]*$/.X/' + exit 0 ;; + CRAY*[A-Z]90:*:*:*) + echo ${UNAME_MACHINE}-cray-unicos${UNAME_RELEASE} \ + | sed -e 's/CRAY.*\([A-Z]90\)/\1/' \ + -e y/ABCDEFGHIJKLMNOPQRSTUVWXYZ/abcdefghijklmnopqrstuvwxyz/ \ + -e 's/\.[^.]*$/.X/' + exit 0 ;; + CRAY*TS:*:*:*) + echo t90-cray-unicos${UNAME_RELEASE} | sed -e 's/\.[^.]*$/.X/' + exit 0 ;; + CRAY*T3E:*:*:*) + echo alphaev5-cray-unicosmk${UNAME_RELEASE} | sed -e 's/\.[^.]*$/.X/' + exit 0 ;; + CRAY*SV1:*:*:*) + echo sv1-cray-unicos${UNAME_RELEASE} | sed -e 's/\.[^.]*$/.X/' + exit 0 ;; + *:UNICOS/mp:*:*) + echo craynv-cray-unicosmp${UNAME_RELEASE} | sed -e 's/\.[^.]*$/.X/' + exit 0 ;; + F30[01]:UNIX_System_V:*:* | F700:UNIX_System_V:*:*) + FUJITSU_PROC=`uname -m | tr 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz'` + FUJITSU_SYS=`uname -p | tr 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz' | sed -e 's/\///'` + FUJITSU_REL=`echo ${UNAME_RELEASE} | sed -e 's/ /_/'` + echo "${FUJITSU_PROC}-fujitsu-${FUJITSU_SYS}${FUJITSU_REL}" + exit 0 ;; + 5000:UNIX_System_V:4.*:*) + FUJITSU_SYS=`uname -p | tr 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz' | sed -e 's/\///'` + FUJITSU_REL=`echo ${UNAME_RELEASE} | tr 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' 'abcdefghijklmnopqrstuvwxyz' | sed -e 's/ /_/'` + echo "sparc-fujitsu-${FUJITSU_SYS}${FUJITSU_REL}" + exit 0 ;; + i*86:BSD/386:*:* | i*86:BSD/OS:*:* | *:Ascend\ Embedded/OS:*:*) + echo ${UNAME_MACHINE}-pc-bsdi${UNAME_RELEASE} + exit 0 ;; + sparc*:BSD/OS:*:*) + echo sparc-unknown-bsdi${UNAME_RELEASE} + exit 0 ;; + *:BSD/OS:*:*) + echo ${UNAME_MACHINE}-unknown-bsdi${UNAME_RELEASE} + exit 0 ;; + *:FreeBSD:*:*) + echo ${UNAME_MACHINE}-unknown-freebsd`echo ${UNAME_RELEASE}|sed -e 's/[-(].*//'` + exit 0 ;; + i*:CYGWIN*:*) + echo ${UNAME_MACHINE}-pc-cygwin + exit 0 ;; + i*:MINGW*:*) + echo ${UNAME_MACHINE}-pc-mingw32 + exit 0 ;; + i*:PW*:*) + echo ${UNAME_MACHINE}-pc-pw32 + exit 0 ;; + x86:Interix*:[34]*) + echo i586-pc-interix${UNAME_RELEASE}|sed -e 's/\..*//' + exit 0 ;; + [345]86:Windows_95:* | [345]86:Windows_98:* | [345]86:Windows_NT:*) + echo i${UNAME_MACHINE}-pc-mks + exit 0 ;; + i*:Windows_NT*:* | Pentium*:Windows_NT*:*) + # How do we know it's Interix rather than the generic POSIX subsystem? + # It also conflicts with pre-2.0 versions of AT&T UWIN. Should we + # UNAME_MACHINE based on the output of uname instead of i386? + echo i586-pc-interix + exit 0 ;; + i*:UWIN*:*) + echo ${UNAME_MACHINE}-pc-uwin + exit 0 ;; + p*:CYGWIN*:*) + echo powerpcle-unknown-cygwin + exit 0 ;; + prep*:SunOS:5.*:*) + echo powerpcle-unknown-solaris2`echo ${UNAME_RELEASE}|sed -e 's/[^.]*//'` + exit 0 ;; + *:GNU:*:*) + # the GNU system + echo `echo ${UNAME_MACHINE}|sed -e 's,[-/].*$,,'`-unknown-gnu`echo ${UNAME_RELEASE}|sed -e 's,/.*$,,'` + exit 0 ;; + *:GNU/*:*:*) + # other systems with GNU libc and userland + echo ${UNAME_MACHINE}-unknown-`echo ${UNAME_SYSTEM} | sed 's,^[^/]*/,,' | tr '[A-Z]' '[a-z]'``echo ${UNAME_RELEASE}|sed -e 's/[-(].*//'`-gnu + exit 0 ;; + i*86:Minix:*:*) + echo ${UNAME_MACHINE}-pc-minix + exit 0 ;; + arm*:Linux:*:*) + echo ${UNAME_MACHINE}-unknown-linux-gnu + exit 0 ;; + cris:Linux:*:*) + echo cris-axis-linux-gnu + exit 0 ;; + crisv32:Linux:*:*) + echo crisv32-axis-linux-gnu + exit 0 ;; + frv:Linux:*:*) + echo frv-unknown-linux-gnu + exit 0 ;; + ia64:Linux:*:*) + echo ${UNAME_MACHINE}-unknown-linux-gnu + exit 0 ;; + m32r*:Linux:*:*) + echo ${UNAME_MACHINE}-unknown-linux-gnu + exit 0 ;; + m68*:Linux:*:*) + echo ${UNAME_MACHINE}-unknown-linux-gnu + exit 0 ;; + mips:Linux:*:*) + eval $set_cc_for_build + sed 's/^ //' << EOF >$dummy.c + #undef CPU + #undef mips + #undef mipsel + #if defined(__MIPSEL__) || defined(__MIPSEL) || defined(_MIPSEL) || defined(MIPSEL) + CPU=mipsel + #else + #if defined(__MIPSEB__) || defined(__MIPSEB) || defined(_MIPSEB) || defined(MIPSEB) + CPU=mips + #else + CPU= + #endif + #endif +EOF + eval `$CC_FOR_BUILD -E $dummy.c 2>/dev/null | grep ^CPU=` + test x"${CPU}" != x && echo "${CPU}-unknown-linux-gnu" && exit 0 + ;; + mips64:Linux:*:*) + eval $set_cc_for_build + sed 's/^ //' << EOF >$dummy.c + #undef CPU + #undef mips64 + #undef mips64el + #if defined(__MIPSEL__) || defined(__MIPSEL) || defined(_MIPSEL) || defined(MIPSEL) + CPU=mips64el + #else + #if defined(__MIPSEB__) || defined(__MIPSEB) || defined(_MIPSEB) || defined(MIPSEB) + CPU=mips64 + #else + CPU= + #endif + #endif +EOF + eval `$CC_FOR_BUILD -E $dummy.c 2>/dev/null | grep ^CPU=` + test x"${CPU}" != x && echo "${CPU}-unknown-linux-gnu" && exit 0 + ;; + ppc:Linux:*:*) + echo powerpc-unknown-linux-gnu + exit 0 ;; + ppc64:Linux:*:*) + echo powerpc64-unknown-linux-gnu + exit 0 ;; + alpha:Linux:*:*) + case `sed -n '/^cpu model/s/^.*: \(.*\)/\1/p' < /proc/cpuinfo` in + EV5) UNAME_MACHINE=alphaev5 ;; + EV56) UNAME_MACHINE=alphaev56 ;; + PCA56) UNAME_MACHINE=alphapca56 ;; + PCA57) UNAME_MACHINE=alphapca56 ;; + EV6) UNAME_MACHINE=alphaev6 ;; + EV67) UNAME_MACHINE=alphaev67 ;; + EV68*) UNAME_MACHINE=alphaev68 ;; + esac + objdump --private-headers /bin/sh | grep ld.so.1 >/dev/null + if test "$?" = 0 ; then LIBC="libc1" ; else LIBC="" ; fi + echo ${UNAME_MACHINE}-unknown-linux-gnu${LIBC} + exit 0 ;; + parisc:Linux:*:* | hppa:Linux:*:*) + # Look for CPU level + case `grep '^cpu[^a-z]*:' /proc/cpuinfo 2>/dev/null | cut -d' ' -f2` in + PA7*) echo hppa1.1-unknown-linux-gnu ;; + PA8*) echo hppa2.0-unknown-linux-gnu ;; + *) echo hppa-unknown-linux-gnu ;; + esac + exit 0 ;; + parisc64:Linux:*:* | hppa64:Linux:*:*) + echo hppa64-unknown-linux-gnu + exit 0 ;; + s390:Linux:*:* | s390x:Linux:*:*) + echo ${UNAME_MACHINE}-ibm-linux + exit 0 ;; + sh64*:Linux:*:*) + echo ${UNAME_MACHINE}-unknown-linux-gnu + exit 0 ;; + sh*:Linux:*:*) + echo ${UNAME_MACHINE}-unknown-linux-gnu + exit 0 ;; + sparc:Linux:*:* | sparc64:Linux:*:*) + echo ${UNAME_MACHINE}-unknown-linux-gnu + exit 0 ;; + x86_64:Linux:*:*) + echo x86_64-unknown-linux-gnu + exit 0 ;; + i*86:Linux:*:*) + # The BFD linker knows what the default object file format is, so + # first see if it will tell us. cd to the root directory to prevent + # problems with other programs or directories called `ld' in the path. + # Set LC_ALL=C to ensure ld outputs messages in English. + ld_supported_targets=`cd /; LC_ALL=C ld --help 2>&1 \ + | sed -ne '/supported targets:/!d + s/[ ][ ]*/ /g + s/.*supported targets: *// + s/ .*// + p'` + case "$ld_supported_targets" in + elf32-i386) + TENTATIVE="${UNAME_MACHINE}-pc-linux-gnu" + ;; + a.out-i386-linux) + echo "${UNAME_MACHINE}-pc-linux-gnuaout" + exit 0 ;; + coff-i386) + echo "${UNAME_MACHINE}-pc-linux-gnucoff" + exit 0 ;; + "") + # Either a pre-BFD a.out linker (linux-gnuoldld) or + # one that does not give us useful --help. + echo "${UNAME_MACHINE}-pc-linux-gnuoldld" + exit 0 ;; + esac + # Determine whether the default compiler is a.out or elf + eval $set_cc_for_build + sed 's/^ //' << EOF >$dummy.c + #include + #ifdef __ELF__ + # ifdef __GLIBC__ + # if __GLIBC__ >= 2 + LIBC=gnu + # else + LIBC=gnulibc1 + # endif + # else + LIBC=gnulibc1 + # endif + #else + #ifdef __INTEL_COMPILER + LIBC=gnu + #else + LIBC=gnuaout + #endif + #endif + #ifdef __dietlibc__ + LIBC=dietlibc + #endif +EOF + eval `$CC_FOR_BUILD -E $dummy.c 2>/dev/null | grep ^LIBC=` + test x"${LIBC}" != x && echo "${UNAME_MACHINE}-pc-linux-${LIBC}" && exit 0 + test x"${TENTATIVE}" != x && echo "${TENTATIVE}" && exit 0 + ;; + i*86:DYNIX/ptx:4*:*) + # ptx 4.0 does uname -s correctly, with DYNIX/ptx in there. + # earlier versions are messed up and put the nodename in both + # sysname and nodename. + echo i386-sequent-sysv4 + exit 0 ;; + i*86:UNIX_SV:4.2MP:2.*) + # Unixware is an offshoot of SVR4, but it has its own version + # number series starting with 2... + # I am not positive that other SVR4 systems won't match this, + # I just have to hope. -- rms. + # Use sysv4.2uw... so that sysv4* matches it. + echo ${UNAME_MACHINE}-pc-sysv4.2uw${UNAME_VERSION} + exit 0 ;; + i*86:OS/2:*:*) + # If we were able to find `uname', then EMX Unix compatibility + # is probably installed. + echo ${UNAME_MACHINE}-pc-os2-emx + exit 0 ;; + i*86:XTS-300:*:STOP) + echo ${UNAME_MACHINE}-unknown-stop + exit 0 ;; + i*86:atheos:*:*) + echo ${UNAME_MACHINE}-unknown-atheos + exit 0 ;; + i*86:syllable:*:*) + echo ${UNAME_MACHINE}-pc-syllable + exit 0 ;; + i*86:LynxOS:2.*:* | i*86:LynxOS:3.[01]*:* | i*86:LynxOS:4.0*:*) + echo i386-unknown-lynxos${UNAME_RELEASE} + exit 0 ;; + i*86:*DOS:*:*) + echo ${UNAME_MACHINE}-pc-msdosdjgpp + exit 0 ;; + i*86:*:4.*:* | i*86:SYSTEM_V:4.*:*) + UNAME_REL=`echo ${UNAME_RELEASE} | sed 's/\/MP$//'` + if grep Novell /usr/include/link.h >/dev/null 2>/dev/null; then + echo ${UNAME_MACHINE}-univel-sysv${UNAME_REL} + else + echo ${UNAME_MACHINE}-pc-sysv${UNAME_REL} + fi + exit 0 ;; + i*86:*:5:[78]*) + case `/bin/uname -X | grep "^Machine"` in + *486*) UNAME_MACHINE=i486 ;; + *Pentium) UNAME_MACHINE=i586 ;; + *Pent*|*Celeron) UNAME_MACHINE=i686 ;; + esac + echo ${UNAME_MACHINE}-unknown-sysv${UNAME_RELEASE}${UNAME_SYSTEM}${UNAME_VERSION} + exit 0 ;; + i*86:*:3.2:*) + if test -f /usr/options/cb.name; then + UNAME_REL=`sed -n 's/.*Version //p' /dev/null >/dev/null ; then + UNAME_REL=`(/bin/uname -X|grep Release|sed -e 's/.*= //')` + (/bin/uname -X|grep i80486 >/dev/null) && UNAME_MACHINE=i486 + (/bin/uname -X|grep '^Machine.*Pentium' >/dev/null) \ + && UNAME_MACHINE=i586 + (/bin/uname -X|grep '^Machine.*Pent *II' >/dev/null) \ + && UNAME_MACHINE=i686 + (/bin/uname -X|grep '^Machine.*Pentium Pro' >/dev/null) \ + && UNAME_MACHINE=i686 + echo ${UNAME_MACHINE}-pc-sco$UNAME_REL + else + echo ${UNAME_MACHINE}-pc-sysv32 + fi + exit 0 ;; + pc:*:*:*) + # Left here for compatibility: + # uname -m prints for DJGPP always 'pc', but it prints nothing about + # the processor, so we play safe by assuming i386. + echo i386-pc-msdosdjgpp + exit 0 ;; + Intel:Mach:3*:*) + echo i386-pc-mach3 + exit 0 ;; + paragon:*:*:*) + echo i860-intel-osf1 + exit 0 ;; + i860:*:4.*:*) # i860-SVR4 + if grep Stardent /usr/include/sys/uadmin.h >/dev/null 2>&1 ; then + echo i860-stardent-sysv${UNAME_RELEASE} # Stardent Vistra i860-SVR4 + else # Add other i860-SVR4 vendors below as they are discovered. + echo i860-unknown-sysv${UNAME_RELEASE} # Unknown i860-SVR4 + fi + exit 0 ;; + mini*:CTIX:SYS*5:*) + # "miniframe" + echo m68010-convergent-sysv + exit 0 ;; + mc68k:UNIX:SYSTEM5:3.51m) + echo m68k-convergent-sysv + exit 0 ;; + M680?0:D-NIX:5.3:*) + echo m68k-diab-dnix + exit 0 ;; + M68*:*:R3V[5678]*:*) + test -r /sysV68 && echo 'm68k-motorola-sysv' && exit 0 ;; + 3[345]??:*:4.0:3.0 | 3[34]??A:*:4.0:3.0 | 3[34]??,*:*:4.0:3.0 | 3[34]??/*:*:4.0:3.0 | 4400:*:4.0:3.0 | 4850:*:4.0:3.0 | SKA40:*:4.0:3.0 | SDS2:*:4.0:3.0 | SHG2:*:4.0:3.0 | S7501*:*:4.0:3.0) + OS_REL='' + test -r /etc/.relid \ + && OS_REL=.`sed -n 's/[^ ]* [^ ]* \([0-9][0-9]\).*/\1/p' < /etc/.relid` + /bin/uname -p 2>/dev/null | grep 86 >/dev/null \ + && echo i486-ncr-sysv4.3${OS_REL} && exit 0 + /bin/uname -p 2>/dev/null | /bin/grep entium >/dev/null \ + && echo i586-ncr-sysv4.3${OS_REL} && exit 0 ;; + 3[34]??:*:4.0:* | 3[34]??,*:*:4.0:*) + /bin/uname -p 2>/dev/null | grep 86 >/dev/null \ + && echo i486-ncr-sysv4 && exit 0 ;; + m68*:LynxOS:2.*:* | m68*:LynxOS:3.0*:*) + echo m68k-unknown-lynxos${UNAME_RELEASE} + exit 0 ;; + mc68030:UNIX_System_V:4.*:*) + echo m68k-atari-sysv4 + exit 0 ;; + TSUNAMI:LynxOS:2.*:*) + echo sparc-unknown-lynxos${UNAME_RELEASE} + exit 0 ;; + rs6000:LynxOS:2.*:*) + echo rs6000-unknown-lynxos${UNAME_RELEASE} + exit 0 ;; + PowerPC:LynxOS:2.*:* | PowerPC:LynxOS:3.[01]*:* | PowerPC:LynxOS:4.0*:*) + echo powerpc-unknown-lynxos${UNAME_RELEASE} + exit 0 ;; + SM[BE]S:UNIX_SV:*:*) + echo mips-dde-sysv${UNAME_RELEASE} + exit 0 ;; + RM*:ReliantUNIX-*:*:*) + echo mips-sni-sysv4 + exit 0 ;; + RM*:SINIX-*:*:*) + echo mips-sni-sysv4 + exit 0 ;; + *:SINIX-*:*:*) + if uname -p 2>/dev/null >/dev/null ; then + UNAME_MACHINE=`(uname -p) 2>/dev/null` + echo ${UNAME_MACHINE}-sni-sysv4 + else + echo ns32k-sni-sysv + fi + exit 0 ;; + PENTIUM:*:4.0*:*) # Unisys `ClearPath HMP IX 4000' SVR4/MP effort + # says + echo i586-unisys-sysv4 + exit 0 ;; + *:UNIX_System_V:4*:FTX*) + # From Gerald Hewes . + # How about differentiating between stratus architectures? -djm + echo hppa1.1-stratus-sysv4 + exit 0 ;; + *:*:*:FTX*) + # From seanf@swdc.stratus.com. + echo i860-stratus-sysv4 + exit 0 ;; + *:VOS:*:*) + # From Paul.Green@stratus.com. + echo hppa1.1-stratus-vos + exit 0 ;; + mc68*:A/UX:*:*) + echo m68k-apple-aux${UNAME_RELEASE} + exit 0 ;; + news*:NEWS-OS:6*:*) + echo mips-sony-newsos6 + exit 0 ;; + R[34]000:*System_V*:*:* | R4000:UNIX_SYSV:*:* | R*000:UNIX_SV:*:*) + if [ -d /usr/nec ]; then + echo mips-nec-sysv${UNAME_RELEASE} + else + echo mips-unknown-sysv${UNAME_RELEASE} + fi + exit 0 ;; + BeBox:BeOS:*:*) # BeOS running on hardware made by Be, PPC only. + echo powerpc-be-beos + exit 0 ;; + BeMac:BeOS:*:*) # BeOS running on Mac or Mac clone, PPC only. + echo powerpc-apple-beos + exit 0 ;; + BePC:BeOS:*:*) # BeOS running on Intel PC compatible. + echo i586-pc-beos + exit 0 ;; + SX-4:SUPER-UX:*:*) + echo sx4-nec-superux${UNAME_RELEASE} + exit 0 ;; + SX-5:SUPER-UX:*:*) + echo sx5-nec-superux${UNAME_RELEASE} + exit 0 ;; + SX-6:SUPER-UX:*:*) + echo sx6-nec-superux${UNAME_RELEASE} + exit 0 ;; + Power*:Rhapsody:*:*) + echo powerpc-apple-rhapsody${UNAME_RELEASE} + exit 0 ;; + *:Rhapsody:*:*) + echo ${UNAME_MACHINE}-apple-rhapsody${UNAME_RELEASE} + exit 0 ;; + *:Darwin:*:*) + UNAME_PROCESSOR=`uname -p` || UNAME_PROCESSOR=unknown + case $UNAME_PROCESSOR in + *86) UNAME_PROCESSOR=i686 ;; + unknown) UNAME_PROCESSOR=powerpc ;; + esac + echo ${UNAME_PROCESSOR}-apple-darwin${UNAME_RELEASE} + exit 0 ;; + *:procnto*:*:* | *:QNX:[0123456789]*:*) + UNAME_PROCESSOR=`uname -p` + if test "$UNAME_PROCESSOR" = "x86"; then + UNAME_PROCESSOR=i386 + UNAME_MACHINE=pc + fi + echo ${UNAME_PROCESSOR}-${UNAME_MACHINE}-nto-qnx${UNAME_RELEASE} + exit 0 ;; + *:QNX:*:4*) + echo i386-pc-qnx + exit 0 ;; + NSR-?:NONSTOP_KERNEL:*:*) + echo nsr-tandem-nsk${UNAME_RELEASE} + exit 0 ;; + *:NonStop-UX:*:*) + echo mips-compaq-nonstopux + exit 0 ;; + BS2000:POSIX*:*:*) + echo bs2000-siemens-sysv + exit 0 ;; + DS/*:UNIX_System_V:*:*) + echo ${UNAME_MACHINE}-${UNAME_SYSTEM}-${UNAME_RELEASE} + exit 0 ;; + *:Plan9:*:*) + # "uname -m" is not consistent, so use $cputype instead. 386 + # is converted to i386 for consistency with other x86 + # operating systems. + if test "$cputype" = "386"; then + UNAME_MACHINE=i386 + else + UNAME_MACHINE="$cputype" + fi + echo ${UNAME_MACHINE}-unknown-plan9 + exit 0 ;; + *:TOPS-10:*:*) + echo pdp10-unknown-tops10 + exit 0 ;; + *:TENEX:*:*) + echo pdp10-unknown-tenex + exit 0 ;; + KS10:TOPS-20:*:* | KL10:TOPS-20:*:* | TYPE4:TOPS-20:*:*) + echo pdp10-dec-tops20 + exit 0 ;; + XKL-1:TOPS-20:*:* | TYPE5:TOPS-20:*:*) + echo pdp10-xkl-tops20 + exit 0 ;; + *:TOPS-20:*:*) + echo pdp10-unknown-tops20 + exit 0 ;; + *:ITS:*:*) + echo pdp10-unknown-its + exit 0 ;; + SEI:*:*:SEIUX) + echo mips-sei-seiux${UNAME_RELEASE} + exit 0 ;; + *:DragonFly:*:*) + echo ${UNAME_MACHINE}-unknown-dragonfly`echo ${UNAME_RELEASE}|sed -e 's/[-(].*//'` + exit 0 ;; + *:*VMS:*:*) + UNAME_MACHINE=`(uname -p) 2>/dev/null` + case "${UNAME_MACHINE}" in + A*) echo alpha-dec-vms && exit 0 ;; + I*) echo ia64-dec-vms && exit 0 ;; + V*) echo vax-dec-vms && exit 0 ;; + esac +esac + +#echo '(No uname command or uname output not recognized.)' 1>&2 +#echo "${UNAME_MACHINE}:${UNAME_SYSTEM}:${UNAME_RELEASE}:${UNAME_VERSION}" 1>&2 + +eval $set_cc_for_build +cat >$dummy.c < +# include +#endif +main () +{ +#if defined (sony) +#if defined (MIPSEB) + /* BFD wants "bsd" instead of "newsos". Perhaps BFD should be changed, + I don't know.... */ + printf ("mips-sony-bsd\n"); exit (0); +#else +#include + printf ("m68k-sony-newsos%s\n", +#ifdef NEWSOS4 + "4" +#else + "" +#endif + ); exit (0); +#endif +#endif + +#if defined (__arm) && defined (__acorn) && defined (__unix) + printf ("arm-acorn-riscix"); exit (0); +#endif + +#if defined (hp300) && !defined (hpux) + printf ("m68k-hp-bsd\n"); exit (0); +#endif + +#if defined (NeXT) +#if !defined (__ARCHITECTURE__) +#define __ARCHITECTURE__ "m68k" +#endif + int version; + version=`(hostinfo | sed -n 's/.*NeXT Mach \([0-9]*\).*/\1/p') 2>/dev/null`; + if (version < 4) + printf ("%s-next-nextstep%d\n", __ARCHITECTURE__, version); + else + printf ("%s-next-openstep%d\n", __ARCHITECTURE__, version); + exit (0); +#endif + +#if defined (MULTIMAX) || defined (n16) +#if defined (UMAXV) + printf ("ns32k-encore-sysv\n"); exit (0); +#else +#if defined (CMU) + printf ("ns32k-encore-mach\n"); exit (0); +#else + printf ("ns32k-encore-bsd\n"); exit (0); +#endif +#endif +#endif + +#if defined (__386BSD__) + printf ("i386-pc-bsd\n"); exit (0); +#endif + +#if defined (sequent) +#if defined (i386) + printf ("i386-sequent-dynix\n"); exit (0); +#endif +#if defined (ns32000) + printf ("ns32k-sequent-dynix\n"); exit (0); +#endif +#endif + +#if defined (_SEQUENT_) + struct utsname un; + + uname(&un); + + if (strncmp(un.version, "V2", 2) == 0) { + printf ("i386-sequent-ptx2\n"); exit (0); + } + if (strncmp(un.version, "V1", 2) == 0) { /* XXX is V1 correct? */ + printf ("i386-sequent-ptx1\n"); exit (0); + } + printf ("i386-sequent-ptx\n"); exit (0); + +#endif + +#if defined (vax) +# if !defined (ultrix) +# include +# if defined (BSD) +# if BSD == 43 + printf ("vax-dec-bsd4.3\n"); exit (0); +# else +# if BSD == 199006 + printf ("vax-dec-bsd4.3reno\n"); exit (0); +# else + printf ("vax-dec-bsd\n"); exit (0); +# endif +# endif +# else + printf ("vax-dec-bsd\n"); exit (0); +# endif +# else + printf ("vax-dec-ultrix\n"); exit (0); +# endif +#endif + +#if defined (alliant) && defined (i860) + printf ("i860-alliant-bsd\n"); exit (0); +#endif + + exit (1); +} +EOF + +$CC_FOR_BUILD -o $dummy $dummy.c 2>/dev/null && $dummy && exit 0 + +# Apollos put the system type in the environment. + +test -d /usr/apollo && { echo ${ISP}-apollo-${SYSTYPE}; exit 0; } + +# Convex versions that predate uname can use getsysinfo(1) + +if [ -x /usr/convex/getsysinfo ] +then + case `getsysinfo -f cpu_type` in + c1*) + echo c1-convex-bsd + exit 0 ;; + c2*) + if getsysinfo -f scalar_acc + then echo c32-convex-bsd + else echo c2-convex-bsd + fi + exit 0 ;; + c34*) + echo c34-convex-bsd + exit 0 ;; + c38*) + echo c38-convex-bsd + exit 0 ;; + c4*) + echo c4-convex-bsd + exit 0 ;; + esac +fi + +cat >&2 < in order to provide the needed +information to handle your system. + +config.guess timestamp = $timestamp + +uname -m = `(uname -m) 2>/dev/null || echo unknown` +uname -r = `(uname -r) 2>/dev/null || echo unknown` +uname -s = `(uname -s) 2>/dev/null || echo unknown` +uname -v = `(uname -v) 2>/dev/null || echo unknown` + +/usr/bin/uname -p = `(/usr/bin/uname -p) 2>/dev/null` +/bin/uname -X = `(/bin/uname -X) 2>/dev/null` + +hostinfo = `(hostinfo) 2>/dev/null` +/bin/universe = `(/bin/universe) 2>/dev/null` +/usr/bin/arch -k = `(/usr/bin/arch -k) 2>/dev/null` +/bin/arch = `(/bin/arch) 2>/dev/null` +/usr/bin/oslevel = `(/usr/bin/oslevel) 2>/dev/null` +/usr/convex/getsysinfo = `(/usr/convex/getsysinfo) 2>/dev/null` + +UNAME_MACHINE = ${UNAME_MACHINE} +UNAME_RELEASE = ${UNAME_RELEASE} +UNAME_SYSTEM = ${UNAME_SYSTEM} +UNAME_VERSION = ${UNAME_VERSION} +EOF + +exit 1 + +# Local variables: +# eval: (add-hook 'write-file-hooks 'time-stamp) +# time-stamp-start: "timestamp='" +# time-stamp-format: "%:y-%02m-%02d" +# time-stamp-end: "'" +# End: diff --git a/libs/srtp/config.log b/libs/srtp/config.log new file mode 100644 index 0000000000..e98db7982b --- /dev/null +++ b/libs/srtp/config.log @@ -0,0 +1,1118 @@ +This file contains any messages produced by compilers while +running configure, to aid debugging if configure makes a mistake. + +It was created by configure, which was +generated by GNU Autoconf 2.59. Invocation command line was + + $ ./configure --prefix=/usr/local/freeswitch --enable-pic + +## --------- ## +## Platform. ## +## --------- ## + +hostname = drone1 +uname -m = x86_64 +uname -r = 2.6.13.4 +uname -s = Linux +uname -v = #2 SMP Mon Oct 31 12:11:19 UTC 2005 + +/usr/bin/uname -p = unknown +/bin/uname -X = unknown + +/bin/arch = x86_64 +/usr/bin/arch -k = unknown +/usr/convex/getsysinfo = unknown +hostinfo = unknown +/bin/machine = unknown +/usr/bin/oslevel = unknown +/bin/universe = unknown + +PATH: /sbin +PATH: /usr/sbin +PATH: /bin +PATH: /usr/bin +PATH: /usr/X11R6/bin +PATH: /opt/bin +PATH: /usr/local/bin +PATH: /usr/local/sbin + + +## ----------- ## +## Core tests. ## +## ----------- ## + +configure:1349: checking for ranlib +configure:1365: found /usr/bin/ranlib +configure:1376: result: ranlib +configure:1434: checking for gcc +configure:1450: found /usr/bin/gcc +configure:1460: result: gcc +configure:1704: checking for C compiler version +configure:1707: gcc --version &5 +gcc (GCC) 3.4.4 (CRUX) +Copyright (C) 2004 Free Software Foundation, Inc. +This is free software; see the source for copying conditions. There is NO +warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + +configure:1710: $? = 0 +configure:1712: gcc -v &5 +Reading specs from /usr/lib64/gcc/x86_64-unknown-linux-gnu/3.4.4/specs +Configured with: ../gcc-3.4.4/configure --prefix=/usr --libexecdir=/usr/lib64 --libdir=/usr/lib64 --with-slibdir=/lib64 --enable-languages=c,c++,objc --enable-threads=posix --enable-__cxa_atexit --enable-clocale=gnu --enable-shared --enable-multilib --disable-nls +Thread model: posix +gcc version 3.4.4 (CRUX) +configure:1715: $? = 0 +configure:1717: gcc -V &5 +gcc: `-V' option must have argument +configure:1720: $? = 1 +configure:1743: checking for C compiler default output file name +configure:1746: gcc -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:1749: $? = 0 +configure:1795: result: a.out +configure:1800: checking whether the C compiler works +configure:1806: ./a.out +configure:1809: $? = 0 +configure:1826: result: yes +configure:1833: checking whether we are cross compiling +configure:1835: result: no +configure:1838: checking for suffix of executables +configure:1840: gcc -o conftest -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:1843: $? = 0 +configure:1868: result: +configure:1874: checking for suffix of object files +configure:1895: gcc -c -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:1898: $? = 0 +configure:1920: result: o +configure:1924: checking whether we are using the GNU C compiler +configure:1948: gcc -c -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:1954: $? = 0 +configure:1958: test -z + || test ! -s conftest.err +configure:1961: $? = 0 +configure:1964: test -s conftest.o +configure:1967: $? = 0 +configure:1980: result: yes +configure:1986: checking whether gcc accepts -g +configure:2007: gcc -c -g conftest.c >&5 +configure:2013: $? = 0 +configure:2017: test -z + || test ! -s conftest.err +configure:2020: $? = 0 +configure:2023: test -s conftest.o +configure:2026: $? = 0 +configure:2037: result: yes +configure:2054: checking for gcc option to accept ANSI C +configure:2124: gcc -c -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:2130: $? = 0 +configure:2134: test -z + || test ! -s conftest.err +configure:2137: $? = 0 +configure:2140: test -s conftest.o +configure:2143: $? = 0 +configure:2161: result: none needed +configure:2179: gcc -c -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +conftest.c:2: error: parse error before "me" +configure:2185: $? = 1 +configure: failed program was: +| #ifndef __cplusplus +| choke me +| #endif +configure:2356: checking for a BSD-compatible install +configure:2411: result: /usr/bin/install -c +configure:2437: checking whether to build for Linux kernel context +configure:2451: result: no +configure:2455: checking for /dev/urandom +configure:2470: result: yes +configure:2500: checking which random device to use +configure:2514: result: /dev/urandom +configure:2530: checking how to run the C preprocessor +configure:2565: gcc -E conftest.c +configure:2571: $? = 0 +configure:2603: gcc -E conftest.c +conftest.c:10:28: ac_nonexistent.h: No such file or directory +configure:2609: $? = 1 +configure: failed program was: +| /* confdefs.h. */ +| +| #define PACKAGE_NAME "" +| #define PACKAGE_TARNAME "" +| #define PACKAGE_VERSION "" +| #define PACKAGE_STRING "" +| #define PACKAGE_BUGREPORT "" +| #define DEV_URANDOM "/dev/urandom" +| /* end confdefs.h. */ +| #include +configure:2648: result: gcc -E +configure:2672: gcc -E conftest.c +configure:2678: $? = 0 +configure:2710: gcc -E conftest.c +conftest.c:10:28: ac_nonexistent.h: No such file or directory +configure:2716: $? = 1 +configure: failed program was: +| /* confdefs.h. */ +| +| #define PACKAGE_NAME "" +| #define PACKAGE_TARNAME "" +| #define PACKAGE_VERSION "" +| #define PACKAGE_STRING "" +| #define PACKAGE_BUGREPORT "" +| #define DEV_URANDOM "/dev/urandom" +| /* end confdefs.h. */ +| #include +configure:2760: checking for egrep +configure:2770: result: grep -E +configure:2775: checking for ANSI C header files +configure:2800: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:2806: $? = 0 +configure:2810: test -z + || test ! -s conftest.err +configure:2813: $? = 0 +configure:2816: test -s conftest.o +configure:2819: $? = 0 +configure:2908: gcc -o conftest -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +conftest.c: In function `main': +conftest.c:27: warning: implicit declaration of function `exit' +configure:2911: $? = 0 +configure:2913: ./conftest +configure:2916: $? = 0 +configure:2931: result: yes +configure:2955: checking for sys/types.h +configure:2971: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:2977: $? = 0 +configure:2981: test -z + || test ! -s conftest.err +configure:2984: $? = 0 +configure:2987: test -s conftest.o +configure:2990: $? = 0 +configure:3001: result: yes +configure:2955: checking for sys/stat.h +configure:2971: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:2977: $? = 0 +configure:2981: test -z + || test ! -s conftest.err +configure:2984: $? = 0 +configure:2987: test -s conftest.o +configure:2990: $? = 0 +configure:3001: result: yes +configure:2955: checking for stdlib.h +configure:2971: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:2977: $? = 0 +configure:2981: test -z + || test ! -s conftest.err +configure:2984: $? = 0 +configure:2987: test -s conftest.o +configure:2990: $? = 0 +configure:3001: result: yes +configure:2955: checking for string.h +configure:2971: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:2977: $? = 0 +configure:2981: test -z + || test ! -s conftest.err +configure:2984: $? = 0 +configure:2987: test -s conftest.o +configure:2990: $? = 0 +configure:3001: result: yes +configure:2955: checking for memory.h +configure:2971: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:2977: $? = 0 +configure:2981: test -z + || test ! -s conftest.err +configure:2984: $? = 0 +configure:2987: test -s conftest.o +configure:2990: $? = 0 +configure:3001: result: yes +configure:2955: checking for strings.h +configure:2971: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:2977: $? = 0 +configure:2981: test -z + || test ! -s conftest.err +configure:2984: $? = 0 +configure:2987: test -s conftest.o +configure:2990: $? = 0 +configure:3001: result: yes +configure:2955: checking for inttypes.h +configure:2971: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:2977: $? = 0 +configure:2981: test -z + || test ! -s conftest.err +configure:2984: $? = 0 +configure:2987: test -s conftest.o +configure:2990: $? = 0 +configure:3001: result: yes +configure:2955: checking for stdint.h +configure:2971: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:2977: $? = 0 +configure:2981: test -z + || test ! -s conftest.err +configure:2984: $? = 0 +configure:2987: test -s conftest.o +configure:2990: $? = 0 +configure:3001: result: yes +configure:2955: checking for unistd.h +configure:2971: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:2977: $? = 0 +configure:2981: test -z + || test ! -s conftest.err +configure:2984: $? = 0 +configure:2987: test -s conftest.o +configure:2990: $? = 0 +configure:3001: result: yes +configure:3018: checking for stdlib.h +configure:3023: result: yes +configure:3168: checking for unistd.h +configure:3173: result: yes +configure:3327: checking byteswap.h usability +configure:3339: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:3345: $? = 0 +configure:3349: test -z + || test ! -s conftest.err +configure:3352: $? = 0 +configure:3355: test -s conftest.o +configure:3358: $? = 0 +configure:3368: result: yes +configure:3372: checking byteswap.h presence +configure:3382: gcc -E conftest.c +configure:3388: $? = 0 +configure:3408: result: yes +configure:3443: checking for byteswap.h +configure:3450: result: yes +configure:3468: checking for stdint.h +configure:3473: result: yes +configure:3627: checking sys/uio.h usability +configure:3639: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:3645: $? = 0 +configure:3649: test -z + || test ! -s conftest.err +configure:3652: $? = 0 +configure:3655: test -s conftest.o +configure:3658: $? = 0 +configure:3668: result: yes +configure:3672: checking sys/uio.h presence +configure:3682: gcc -E conftest.c +configure:3688: $? = 0 +configure:3708: result: yes +configure:3743: checking for sys/uio.h +configure:3750: result: yes +configure:3768: checking for inttypes.h +configure:3773: result: yes +configure:3918: checking for sys/types.h +configure:3923: result: yes +configure:4077: checking machine/types.h usability +configure:4089: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +conftest.c:61:27: machine/types.h: No such file or directory +configure:4095: $? = 1 +configure: failed program was: +| /* confdefs.h. */ +| +| #define PACKAGE_NAME "" +| #define PACKAGE_TARNAME "" +| #define PACKAGE_VERSION "" +| #define PACKAGE_STRING "" +| #define PACKAGE_BUGREPORT "" +| #define DEV_URANDOM "/dev/urandom" +| #define STDC_HEADERS 1 +| #define HAVE_SYS_TYPES_H 1 +| #define HAVE_SYS_STAT_H 1 +| #define HAVE_STDLIB_H 1 +| #define HAVE_STRING_H 1 +| #define HAVE_MEMORY_H 1 +| #define HAVE_STRINGS_H 1 +| #define HAVE_INTTYPES_H 1 +| #define HAVE_STDINT_H 1 +| #define HAVE_UNISTD_H 1 +| #define HAVE_STDLIB_H 1 +| #define HAVE_UNISTD_H 1 +| #define HAVE_BYTESWAP_H 1 +| #define HAVE_STDINT_H 1 +| #define HAVE_SYS_UIO_H 1 +| #define HAVE_INTTYPES_H 1 +| #define HAVE_SYS_TYPES_H 1 +| /* end confdefs.h. */ +| #include +| #if HAVE_SYS_TYPES_H +| # include +| #endif +| #if HAVE_SYS_STAT_H +| # include +| #endif +| #if STDC_HEADERS +| # include +| # include +| #else +| # if HAVE_STDLIB_H +| # include +| # endif +| #endif +| #if HAVE_STRING_H +| # if !STDC_HEADERS && HAVE_MEMORY_H +| # include +| # endif +| # include +| #endif +| #if HAVE_STRINGS_H +| # include +| #endif +| #if HAVE_INTTYPES_H +| # include +| #else +| # if HAVE_STDINT_H +| # include +| # endif +| #endif +| #if HAVE_UNISTD_H +| # include +| #endif +| #include +configure:4118: result: no +configure:4122: checking machine/types.h presence +configure:4132: gcc -E conftest.c +conftest.c:27:27: machine/types.h: No such file or directory +configure:4138: $? = 1 +configure: failed program was: +| /* confdefs.h. */ +| +| #define PACKAGE_NAME "" +| #define PACKAGE_TARNAME "" +| #define PACKAGE_VERSION "" +| #define PACKAGE_STRING "" +| #define PACKAGE_BUGREPORT "" +| #define DEV_URANDOM "/dev/urandom" +| #define STDC_HEADERS 1 +| #define HAVE_SYS_TYPES_H 1 +| #define HAVE_SYS_STAT_H 1 +| #define HAVE_STDLIB_H 1 +| #define HAVE_STRING_H 1 +| #define HAVE_MEMORY_H 1 +| #define HAVE_STRINGS_H 1 +| #define HAVE_INTTYPES_H 1 +| #define HAVE_STDINT_H 1 +| #define HAVE_UNISTD_H 1 +| #define HAVE_STDLIB_H 1 +| #define HAVE_UNISTD_H 1 +| #define HAVE_BYTESWAP_H 1 +| #define HAVE_STDINT_H 1 +| #define HAVE_SYS_UIO_H 1 +| #define HAVE_INTTYPES_H 1 +| #define HAVE_SYS_TYPES_H 1 +| /* end confdefs.h. */ +| #include +configure:4158: result: no +configure:4193: checking for machine/types.h +configure:4200: result: no +configure:4227: checking sys/int_types.h usability +configure:4239: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +conftest.c:61:27: sys/int_types.h: No such file or directory +configure:4245: $? = 1 +configure: failed program was: +| /* confdefs.h. */ +| +| #define PACKAGE_NAME "" +| #define PACKAGE_TARNAME "" +| #define PACKAGE_VERSION "" +| #define PACKAGE_STRING "" +| #define PACKAGE_BUGREPORT "" +| #define DEV_URANDOM "/dev/urandom" +| #define STDC_HEADERS 1 +| #define HAVE_SYS_TYPES_H 1 +| #define HAVE_SYS_STAT_H 1 +| #define HAVE_STDLIB_H 1 +| #define HAVE_STRING_H 1 +| #define HAVE_MEMORY_H 1 +| #define HAVE_STRINGS_H 1 +| #define HAVE_INTTYPES_H 1 +| #define HAVE_STDINT_H 1 +| #define HAVE_UNISTD_H 1 +| #define HAVE_STDLIB_H 1 +| #define HAVE_UNISTD_H 1 +| #define HAVE_BYTESWAP_H 1 +| #define HAVE_STDINT_H 1 +| #define HAVE_SYS_UIO_H 1 +| #define HAVE_INTTYPES_H 1 +| #define HAVE_SYS_TYPES_H 1 +| /* end confdefs.h. */ +| #include +| #if HAVE_SYS_TYPES_H +| # include +| #endif +| #if HAVE_SYS_STAT_H +| # include +| #endif +| #if STDC_HEADERS +| # include +| # include +| #else +| # if HAVE_STDLIB_H +| # include +| # endif +| #endif +| #if HAVE_STRING_H +| # if !STDC_HEADERS && HAVE_MEMORY_H +| # include +| # endif +| # include +| #endif +| #if HAVE_STRINGS_H +| # include +| #endif +| #if HAVE_INTTYPES_H +| # include +| #else +| # if HAVE_STDINT_H +| # include +| # endif +| #endif +| #if HAVE_UNISTD_H +| # include +| #endif +| #include +configure:4268: result: no +configure:4272: checking sys/int_types.h presence +configure:4282: gcc -E conftest.c +conftest.c:27:27: sys/int_types.h: No such file or directory +configure:4288: $? = 1 +configure: failed program was: +| /* confdefs.h. */ +| +| #define PACKAGE_NAME "" +| #define PACKAGE_TARNAME "" +| #define PACKAGE_VERSION "" +| #define PACKAGE_STRING "" +| #define PACKAGE_BUGREPORT "" +| #define DEV_URANDOM "/dev/urandom" +| #define STDC_HEADERS 1 +| #define HAVE_SYS_TYPES_H 1 +| #define HAVE_SYS_STAT_H 1 +| #define HAVE_STDLIB_H 1 +| #define HAVE_STRING_H 1 +| #define HAVE_MEMORY_H 1 +| #define HAVE_STRINGS_H 1 +| #define HAVE_INTTYPES_H 1 +| #define HAVE_STDINT_H 1 +| #define HAVE_UNISTD_H 1 +| #define HAVE_STDLIB_H 1 +| #define HAVE_UNISTD_H 1 +| #define HAVE_BYTESWAP_H 1 +| #define HAVE_STDINT_H 1 +| #define HAVE_SYS_UIO_H 1 +| #define HAVE_INTTYPES_H 1 +| #define HAVE_SYS_TYPES_H 1 +| /* end confdefs.h. */ +| #include +configure:4308: result: no +configure:4343: checking for sys/int_types.h +configure:4350: result: no +configure:4380: checking sys/socket.h usability +configure:4392: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:4398: $? = 0 +configure:4402: test -z + || test ! -s conftest.err +configure:4405: $? = 0 +configure:4408: test -s conftest.o +configure:4411: $? = 0 +configure:4421: result: yes +configure:4425: checking sys/socket.h presence +configure:4435: gcc -E conftest.c +configure:4441: $? = 0 +configure:4461: result: yes +configure:4496: checking for sys/socket.h +configure:4503: result: yes +configure:4380: checking netinet/in.h usability +configure:4392: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:4398: $? = 0 +configure:4402: test -z + || test ! -s conftest.err +configure:4405: $? = 0 +configure:4408: test -s conftest.o +configure:4411: $? = 0 +configure:4421: result: yes +configure:4425: checking netinet/in.h presence +configure:4435: gcc -E conftest.c +configure:4441: $? = 0 +configure:4461: result: yes +configure:4496: checking for netinet/in.h +configure:4503: result: yes +configure:4380: checking arpa/inet.h usability +configure:4392: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:4398: $? = 0 +configure:4402: test -z + || test ! -s conftest.err +configure:4405: $? = 0 +configure:4408: test -s conftest.o +configure:4411: $? = 0 +configure:4421: result: yes +configure:4425: checking arpa/inet.h presence +configure:4435: gcc -E conftest.c +configure:4441: $? = 0 +configure:4461: result: yes +configure:4496: checking for arpa/inet.h +configure:4503: result: yes +configure:4530: checking windows.h usability +configure:4542: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +conftest.c:64:21: windows.h: No such file or directory +configure:4548: $? = 1 +configure: failed program was: +| /* confdefs.h. */ +| +| #define PACKAGE_NAME "" +| #define PACKAGE_TARNAME "" +| #define PACKAGE_VERSION "" +| #define PACKAGE_STRING "" +| #define PACKAGE_BUGREPORT "" +| #define DEV_URANDOM "/dev/urandom" +| #define STDC_HEADERS 1 +| #define HAVE_SYS_TYPES_H 1 +| #define HAVE_SYS_STAT_H 1 +| #define HAVE_STDLIB_H 1 +| #define HAVE_STRING_H 1 +| #define HAVE_MEMORY_H 1 +| #define HAVE_STRINGS_H 1 +| #define HAVE_INTTYPES_H 1 +| #define HAVE_STDINT_H 1 +| #define HAVE_UNISTD_H 1 +| #define HAVE_STDLIB_H 1 +| #define HAVE_UNISTD_H 1 +| #define HAVE_BYTESWAP_H 1 +| #define HAVE_STDINT_H 1 +| #define HAVE_SYS_UIO_H 1 +| #define HAVE_INTTYPES_H 1 +| #define HAVE_SYS_TYPES_H 1 +| #define HAVE_SYS_SOCKET_H 1 +| #define HAVE_NETINET_IN_H 1 +| #define HAVE_ARPA_INET_H 1 +| /* end confdefs.h. */ +| #include +| #if HAVE_SYS_TYPES_H +| # include +| #endif +| #if HAVE_SYS_STAT_H +| # include +| #endif +| #if STDC_HEADERS +| # include +| # include +| #else +| # if HAVE_STDLIB_H +| # include +| # endif +| #endif +| #if HAVE_STRING_H +| # if !STDC_HEADERS && HAVE_MEMORY_H +| # include +| # endif +| # include +| #endif +| #if HAVE_STRINGS_H +| # include +| #endif +| #if HAVE_INTTYPES_H +| # include +| #else +| # if HAVE_STDINT_H +| # include +| # endif +| #endif +| #if HAVE_UNISTD_H +| # include +| #endif +| #include +configure:4571: result: no +configure:4575: checking windows.h presence +configure:4585: gcc -E conftest.c +conftest.c:30:21: windows.h: No such file or directory +configure:4591: $? = 1 +configure: failed program was: +| /* confdefs.h. */ +| +| #define PACKAGE_NAME "" +| #define PACKAGE_TARNAME "" +| #define PACKAGE_VERSION "" +| #define PACKAGE_STRING "" +| #define PACKAGE_BUGREPORT "" +| #define DEV_URANDOM "/dev/urandom" +| #define STDC_HEADERS 1 +| #define HAVE_SYS_TYPES_H 1 +| #define HAVE_SYS_STAT_H 1 +| #define HAVE_STDLIB_H 1 +| #define HAVE_STRING_H 1 +| #define HAVE_MEMORY_H 1 +| #define HAVE_STRINGS_H 1 +| #define HAVE_INTTYPES_H 1 +| #define HAVE_STDINT_H 1 +| #define HAVE_UNISTD_H 1 +| #define HAVE_STDLIB_H 1 +| #define HAVE_UNISTD_H 1 +| #define HAVE_BYTESWAP_H 1 +| #define HAVE_STDINT_H 1 +| #define HAVE_SYS_UIO_H 1 +| #define HAVE_INTTYPES_H 1 +| #define HAVE_SYS_TYPES_H 1 +| #define HAVE_SYS_SOCKET_H 1 +| #define HAVE_NETINET_IN_H 1 +| #define HAVE_ARPA_INET_H 1 +| /* end confdefs.h. */ +| #include +configure:4611: result: no +configure:4646: checking for windows.h +configure:4653: result: no +configure:4830: checking syslog.h usability +configure:4842: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:4848: $? = 0 +configure:4852: test -z + || test ! -s conftest.err +configure:4855: $? = 0 +configure:4858: test -s conftest.o +configure:4861: $? = 0 +configure:4871: result: yes +configure:4875: checking syslog.h presence +configure:4885: gcc -E conftest.c +configure:4891: $? = 0 +configure:4911: result: yes +configure:4946: checking for syslog.h +configure:4953: result: yes +configure:4967: checking for int8_t +configure:4991: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:4997: $? = 0 +configure:5001: test -z + || test ! -s conftest.err +configure:5004: $? = 0 +configure:5007: test -s conftest.o +configure:5010: $? = 0 +configure:5021: result: yes +configure:5031: checking for uint8_t +configure:5055: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:5061: $? = 0 +configure:5065: test -z + || test ! -s conftest.err +configure:5068: $? = 0 +configure:5071: test -s conftest.o +configure:5074: $? = 0 +configure:5085: result: yes +configure:5095: checking for int16_t +configure:5119: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:5125: $? = 0 +configure:5129: test -z + || test ! -s conftest.err +configure:5132: $? = 0 +configure:5135: test -s conftest.o +configure:5138: $? = 0 +configure:5149: result: yes +configure:5159: checking for uint16_t +configure:5183: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:5189: $? = 0 +configure:5193: test -z + || test ! -s conftest.err +configure:5196: $? = 0 +configure:5199: test -s conftest.o +configure:5202: $? = 0 +configure:5213: result: yes +configure:5223: checking for int32_t +configure:5247: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:5253: $? = 0 +configure:5257: test -z + || test ! -s conftest.err +configure:5260: $? = 0 +configure:5263: test -s conftest.o +configure:5266: $? = 0 +configure:5277: result: yes +configure:5287: checking for uint32_t +configure:5311: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:5317: $? = 0 +configure:5321: test -z + || test ! -s conftest.err +configure:5324: $? = 0 +configure:5327: test -s conftest.o +configure:5330: $? = 0 +configure:5341: result: yes +configure:5351: checking for uint64_t +configure:5375: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:5381: $? = 0 +configure:5385: test -z + || test ! -s conftest.err +configure:5388: $? = 0 +configure:5391: test -s conftest.o +configure:5394: $? = 0 +configure:5405: result: yes +configure:5416: checking for unsigned long +configure:5440: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:5446: $? = 0 +configure:5450: test -z + || test ! -s conftest.err +configure:5453: $? = 0 +configure:5456: test -s conftest.o +configure:5459: $? = 0 +configure:5470: result: yes +configure:5473: checking size of unsigned long +configure:5792: gcc -o conftest -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:5795: $? = 0 +configure:5797: ./conftest +configure:5800: $? = 0 +configure:5823: result: 8 +configure:5830: checking for unsigned long long +configure:5854: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:5860: $? = 0 +configure:5864: test -z + || test ! -s conftest.err +configure:5867: $? = 0 +configure:5870: test -s conftest.o +configure:5873: $? = 0 +configure:5884: result: yes +configure:5887: checking size of unsigned long long +configure:6206: gcc -o conftest -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:6209: $? = 0 +configure:6211: ./conftest +configure:6214: $? = 0 +configure:6237: result: 8 +configure:6245: checking for an ANSI C-conforming const +configure:6312: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +conftest.c: In function `main': +conftest.c:67: warning: unused variable `s' +conftest.c:87: warning: unused variable `foo' +conftest.c:48: warning: unused variable `x' +conftest.c:54: warning: unused variable `zero' +conftest.c:84: warning: 'b' might be used uninitialized in this function +conftest.c:66: warning: 't' might be used uninitialized in this function +configure:6318: $? = 0 +configure:6322: test -z + || test ! -s conftest.err +configure:6325: $? = 0 +configure:6328: test -s conftest.o +configure:6331: $? = 0 +configure:6342: result: yes +configure:6352: checking for inline +configure:6373: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:6379: $? = 0 +configure:6383: test -z + || test ! -s conftest.err +configure:6386: $? = 0 +configure:6389: test -s conftest.o +configure:6392: $? = 0 +configure:6404: result: inline +configure:6423: checking for size_t +configure:6447: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:6453: $? = 0 +configure:6457: test -z + || test ! -s conftest.err +configure:6460: $? = 0 +configure:6463: test -s conftest.o +configure:6466: $? = 0 +configure:6477: result: yes +configure:6496: checking for socket +configure:6553: gcc -o conftest -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:6559: $? = 0 +configure:6563: test -z + || test ! -s conftest.err +configure:6566: $? = 0 +configure:6569: test -s conftest +configure:6572: $? = 0 +configure:6584: result: yes +configure:6496: checking for inet_aton +configure:6553: gcc -o conftest -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:6559: $? = 0 +configure:6563: test -z + || test ! -s conftest.err +configure:6566: $? = 0 +configure:6569: test -s conftest +configure:6572: $? = 0 +configure:6584: result: yes +configure:6496: checking for usleep +configure:6553: gcc -o conftest -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:6559: $? = 0 +configure:6563: test -z + || test ! -s conftest.err +configure:6566: $? = 0 +configure:6569: test -s conftest +configure:6572: $? = 0 +configure:6584: result: yes +configure:6730: checking whether byte ordering is bigendian +configure:6757: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +configure:6763: $? = 0 +configure:6767: test -z + || test ! -s conftest.err +configure:6770: $? = 0 +configure:6773: test -s conftest.o +configure:6776: $? = 0 +configure:6800: gcc -c -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops conftest.c >&5 +conftest.c: In function `main': +conftest.c:50: error: `not' undeclared (first use in this function) +conftest.c:50: error: (Each undeclared identifier is reported only once +conftest.c:50: error: for each function it appears in.) +conftest.c:50: error: parse error before "big" +configure:6806: $? = 1 +configure: failed program was: +| /* confdefs.h. */ +| +| #define PACKAGE_NAME "" +| #define PACKAGE_TARNAME "" +| #define PACKAGE_VERSION "" +| #define PACKAGE_STRING "" +| #define PACKAGE_BUGREPORT "" +| #define DEV_URANDOM "/dev/urandom" +| #define STDC_HEADERS 1 +| #define HAVE_SYS_TYPES_H 1 +| #define HAVE_SYS_STAT_H 1 +| #define HAVE_STDLIB_H 1 +| #define HAVE_STRING_H 1 +| #define HAVE_MEMORY_H 1 +| #define HAVE_STRINGS_H 1 +| #define HAVE_INTTYPES_H 1 +| #define HAVE_STDINT_H 1 +| #define HAVE_UNISTD_H 1 +| #define HAVE_STDLIB_H 1 +| #define HAVE_UNISTD_H 1 +| #define HAVE_BYTESWAP_H 1 +| #define HAVE_STDINT_H 1 +| #define HAVE_SYS_UIO_H 1 +| #define HAVE_INTTYPES_H 1 +| #define HAVE_SYS_TYPES_H 1 +| #define HAVE_SYS_SOCKET_H 1 +| #define HAVE_NETINET_IN_H 1 +| #define HAVE_ARPA_INET_H 1 +| #define HAVE_SYSLOG_H 1 +| #define HAVE_INT8_T 1 +| #define HAVE_UINT8_T 1 +| #define HAVE_INT16_T 1 +| #define HAVE_UINT16_T 1 +| #define HAVE_INT32_T 1 +| #define HAVE_UINT32_T 1 +| #define HAVE_UINT64_T 1 +| #define SIZEOF_UNSIGNED_LONG 8 +| #define SIZEOF_UNSIGNED_LONG_LONG 8 +| #define HAVE_SOCKET 1 +| #define HAVE_INET_ATON 1 +| #define HAVE_USLEEP 1 +| /* end confdefs.h. */ +| #include +| #include +| +| int +| main () +| { +| #if BYTE_ORDER != BIG_ENDIAN +| not big endian +| #endif +| +| ; +| return 0; +| } +configure:6941: result: no +configure:6967: checking build system type +configure:6985: result: x86_64-unknown-linux-gnu +configure:6993: checking host system type +configure:7007: result: x86_64-unknown-linux-gnu +configure:7053: checking whether to compile in debugging +configure:7069: result: yes +configure:7072: checking whether to use ISMAcryp code +configure:7088: result: no +configure:7091: checking whether to use syslog for error reporting +configure:7107: result: no +configure:7110: checking whether to use stdout for error reporting +configure:7126: result: yes +configure:7129: checking whether to use /dev/console for error reporting +configure:7150: result: no +configure:7153: checking whether to use GDOI key management +configure:7171: result: no +configure:7273: creating ./config.status + +## ---------------------- ## +## Running config.status. ## +## ---------------------- ## + +This file was extended by config.status, which was +generated by GNU Autoconf 2.59. Invocation command line was + + CONFIG_FILES = + CONFIG_HEADERS = + CONFIG_LINKS = + CONFIG_COMMANDS = + $ ./config.status + +on drone1 + +config.status:664: creating Makefile +config.status:664: creating crypto/Makefile +config.status:664: creating doc/Makefile +config.status:768: creating crypto/include/config.h +config.status:942: crypto/include/config.h is unchanged + +## ---------------- ## +## Cache variables. ## +## ---------------- ## + +ac_cv_build=x86_64-unknown-linux-gnu +ac_cv_build_alias=x86_64-unknown-linux-gnu +ac_cv_c_bigendian=no +ac_cv_c_compiler_gnu=yes +ac_cv_c_const=yes +ac_cv_c_inline=inline +ac_cv_env_CC_set= +ac_cv_env_CC_value= +ac_cv_env_CFLAGS_set= +ac_cv_env_CFLAGS_value= +ac_cv_env_CPPFLAGS_set= +ac_cv_env_CPPFLAGS_value= +ac_cv_env_CPP_set= +ac_cv_env_CPP_value= +ac_cv_env_LDFLAGS_set= +ac_cv_env_LDFLAGS_value= +ac_cv_env_build_alias_set= +ac_cv_env_build_alias_value= +ac_cv_env_host_alias_set= +ac_cv_env_host_alias_value= +ac_cv_env_target_alias_set= +ac_cv_env_target_alias_value= +ac_cv_exeext= +ac_cv_file__dev_urandom=yes +ac_cv_func_inet_aton=yes +ac_cv_func_socket=yes +ac_cv_func_usleep=yes +ac_cv_header_arpa_inet_h=yes +ac_cv_header_byteswap_h=yes +ac_cv_header_inttypes_h=yes +ac_cv_header_machine_types_h=no +ac_cv_header_memory_h=yes +ac_cv_header_netinet_in_h=yes +ac_cv_header_stdc=yes +ac_cv_header_stdint_h=yes +ac_cv_header_stdlib_h=yes +ac_cv_header_string_h=yes +ac_cv_header_strings_h=yes +ac_cv_header_sys_int_types_h=no +ac_cv_header_sys_socket_h=yes +ac_cv_header_sys_stat_h=yes +ac_cv_header_sys_types_h=yes +ac_cv_header_sys_uio_h=yes +ac_cv_header_syslog_h=yes +ac_cv_header_unistd_h=yes +ac_cv_header_windows_h=no +ac_cv_host=x86_64-unknown-linux-gnu +ac_cv_host_alias=x86_64-unknown-linux-gnu +ac_cv_objext=o +ac_cv_path_install='/usr/bin/install -c' +ac_cv_prog_CPP='gcc -E' +ac_cv_prog_ac_ct_CC=gcc +ac_cv_prog_ac_ct_RANLIB=ranlib +ac_cv_prog_cc_g=yes +ac_cv_prog_cc_stdc= +ac_cv_prog_egrep='grep -E' +ac_cv_sizeof_unsigned_long=8 +ac_cv_sizeof_unsigned_long_long=8 +ac_cv_type_int16_t=yes +ac_cv_type_int32_t=yes +ac_cv_type_int8_t=yes +ac_cv_type_size_t=yes +ac_cv_type_uint16_t=yes +ac_cv_type_uint32_t=yes +ac_cv_type_uint64_t=yes +ac_cv_type_uint8_t=yes +ac_cv_type_unsigned_long=yes +ac_cv_type_unsigned_long_long=yes + +## ----------------- ## +## Output variables. ## +## ----------------- ## + +CC='gcc' +CFLAGS='-fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops' +CPP='gcc -E' +CPPFLAGS='' +DEFS='-DHAVE_CONFIG_H' +ECHO_C='' +ECHO_N='-n' +ECHO_T='' +EGREP='grep -E' +EXE='' +EXEEXT='' +GDOI_OBJS='' +INSTALL_DATA='${INSTALL} -m 644' +INSTALL_PROGRAM='${INSTALL}' +INSTALL_SCRIPT='${INSTALL}' +LDFLAGS='' +LIBOBJS='' +LIBS='' +LTLIBOBJS='' +OBJEXT='o' +PACKAGE_BUGREPORT='' +PACKAGE_NAME='' +PACKAGE_STRING='' +PACKAGE_TARNAME='' +PACKAGE_VERSION='' +PATH_SEPARATOR=':' +RANLIB='ranlib' +RNG_OBJS='rand_source.o' +SHELL='/bin/sh' +ac_ct_CC='gcc' +ac_ct_RANLIB='ranlib' +bindir='${exec_prefix}/bin' +build='x86_64-unknown-linux-gnu' +build_alias='' +build_cpu='x86_64' +build_os='linux-gnu' +build_vendor='unknown' +datadir='${prefix}/share' +exec_prefix='${prefix}' +host='x86_64-unknown-linux-gnu' +host_alias='' +host_cpu='x86_64' +host_os='linux-gnu' +host_vendor='unknown' +includedir='${prefix}/include' +infodir='${prefix}/info' +libdir='${exec_prefix}/lib' +libexecdir='${exec_prefix}/libexec' +localstatedir='${prefix}/var' +mandir='${prefix}/man' +oldincludedir='/usr/include' +prefix='/usr/local/freeswitch' +program_transform_name='s,x,x,' +sbindir='${exec_prefix}/sbin' +sharedstatedir='${prefix}/com' +sysconfdir='${prefix}/etc' +target_alias='' + +## ----------- ## +## confdefs.h. ## +## ----------- ## + +#define CPU_CISC 1 +#define DEV_URANDOM "/dev/urandom" +#define ENABLE_DEBUGGING 1 +#define ERR_REPORTING_STDOUT 1 +#define HAVE_ARPA_INET_H 1 +#define HAVE_BYTESWAP_H 1 +#define HAVE_INET_ATON 1 +#define HAVE_INT16_T 1 +#define HAVE_INT32_T 1 +#define HAVE_INT8_T 1 +#define HAVE_INTTYPES_H 1 +#define HAVE_INTTYPES_H 1 +#define HAVE_MEMORY_H 1 +#define HAVE_NETINET_IN_H 1 +#define HAVE_SOCKET 1 +#define HAVE_STDINT_H 1 +#define HAVE_STDINT_H 1 +#define HAVE_STDLIB_H 1 +#define HAVE_STDLIB_H 1 +#define HAVE_STRINGS_H 1 +#define HAVE_STRING_H 1 +#define HAVE_SYSLOG_H 1 +#define HAVE_SYS_SOCKET_H 1 +#define HAVE_SYS_STAT_H 1 +#define HAVE_SYS_TYPES_H 1 +#define HAVE_SYS_TYPES_H 1 +#define HAVE_SYS_UIO_H 1 +#define HAVE_UINT16_T 1 +#define HAVE_UINT32_T 1 +#define HAVE_UINT64_T 1 +#define HAVE_UINT8_T 1 +#define HAVE_UNISTD_H 1 +#define HAVE_UNISTD_H 1 +#define HAVE_USLEEP 1 +#define PACKAGE_BUGREPORT "" +#define PACKAGE_NAME "" +#define PACKAGE_STRING "" +#define PACKAGE_TARNAME "" +#define PACKAGE_VERSION "" +#define SIZEOF_UNSIGNED_LONG 8 +#define SIZEOF_UNSIGNED_LONG_LONG 8 +#define STDC_HEADERS 1 + +configure: exit 0 diff --git a/libs/srtp/config.status b/libs/srtp/config.status new file mode 100755 index 0000000000..8fa9c0b7de --- /dev/null +++ b/libs/srtp/config.status @@ -0,0 +1,991 @@ +#! /bin/sh +# Generated by configure. +# Run this file to recreate the current configuration. +# Compiler output produced by configure, useful for debugging +# configure, is in config.log if it exists. + +debug=false +ac_cs_recheck=false +ac_cs_silent=false +SHELL=${CONFIG_SHELL-/bin/sh} +## --------------------- ## +## M4sh Initialization. ## +## --------------------- ## + +# Be Bourne compatible +if test -n "${ZSH_VERSION+set}" && (emulate sh) >/dev/null 2>&1; then + emulate sh + NULLCMD=: + # Zsh 3.x and 4.x performs word splitting on ${1+"$@"}, which + # is contrary to our usage. Disable this feature. + alias -g '${1+"$@"}'='"$@"' +elif test -n "${BASH_VERSION+set}" && (set -o posix) >/dev/null 2>&1; then + set -o posix +fi +DUALCASE=1; export DUALCASE # for MKS sh + +# Support unset when possible. +if ( (MAIL=60; unset MAIL) || exit) >/dev/null 2>&1; then + as_unset=unset +else + as_unset=false +fi + + +# Work around bugs in pre-3.0 UWIN ksh. +$as_unset ENV MAIL MAILPATH +PS1='$ ' +PS2='> ' +PS4='+ ' + +# NLS nuisances. +for as_var in \ + LANG LANGUAGE LC_ADDRESS LC_ALL LC_COLLATE LC_CTYPE LC_IDENTIFICATION \ + LC_MEASUREMENT LC_MESSAGES LC_MONETARY LC_NAME LC_NUMERIC LC_PAPER \ + LC_TELEPHONE LC_TIME +do + if (set +x; test -z "`(eval $as_var=C; export $as_var) 2>&1`"); then + eval $as_var=C; export $as_var + else + $as_unset $as_var + fi +done + +# Required to use basename. +if expr a : '\(a\)' >/dev/null 2>&1; then + as_expr=expr +else + as_expr=false +fi + +if (basename /) >/dev/null 2>&1 && test "X`basename / 2>&1`" = "X/"; then + as_basename=basename +else + as_basename=false +fi + + +# Name of the executable. +as_me=`$as_basename "$0" || +$as_expr X/"$0" : '.*/\([^/][^/]*\)/*$' \| \ + X"$0" : 'X\(//\)$' \| \ + X"$0" : 'X\(/\)$' \| \ + . : '\(.\)' 2>/dev/null || +echo X/"$0" | + sed '/^.*\/\([^/][^/]*\)\/*$/{ s//\1/; q; } + /^X\/\(\/\/\)$/{ s//\1/; q; } + /^X\/\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + + +# PATH needs CR, and LINENO needs CR and PATH. +# Avoid depending upon Character Ranges. +as_cr_letters='abcdefghijklmnopqrstuvwxyz' +as_cr_LETTERS='ABCDEFGHIJKLMNOPQRSTUVWXYZ' +as_cr_Letters=$as_cr_letters$as_cr_LETTERS +as_cr_digits='0123456789' +as_cr_alnum=$as_cr_Letters$as_cr_digits + +# The user is always right. +if test "${PATH_SEPARATOR+set}" != set; then + echo "#! /bin/sh" >conf$$.sh + echo "exit 0" >>conf$$.sh + chmod +x conf$$.sh + if (PATH="/nonexistent;."; conf$$.sh) >/dev/null 2>&1; then + PATH_SEPARATOR=';' + else + PATH_SEPARATOR=: + fi + rm -f conf$$.sh +fi + + + as_lineno_1=$LINENO + as_lineno_2=$LINENO + as_lineno_3=`(expr $as_lineno_1 + 1) 2>/dev/null` + test "x$as_lineno_1" != "x$as_lineno_2" && + test "x$as_lineno_3" = "x$as_lineno_2" || { + # Find who we are. Look in the path if we contain no path at all + # relative or not. + case $0 in + *[\\/]* ) as_myself=$0 ;; + *) as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + test -r "$as_dir/$0" && as_myself=$as_dir/$0 && break +done + + ;; + esac + # We did not find ourselves, most probably we were run as `sh COMMAND' + # in which case we are not to be found in the path. + if test "x$as_myself" = x; then + as_myself=$0 + fi + if test ! -f "$as_myself"; then + { { echo "$as_me:$LINENO: error: cannot find myself; rerun with an absolute path" >&5 +echo "$as_me: error: cannot find myself; rerun with an absolute path" >&2;} + { (exit 1); exit 1; }; } + fi + case $CONFIG_SHELL in + '') + as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in /bin$PATH_SEPARATOR/usr/bin$PATH_SEPARATOR$PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + for as_base in sh bash ksh sh5; do + case $as_dir in + /*) + if ("$as_dir/$as_base" -c ' + as_lineno_1=$LINENO + as_lineno_2=$LINENO + as_lineno_3=`(expr $as_lineno_1 + 1) 2>/dev/null` + test "x$as_lineno_1" != "x$as_lineno_2" && + test "x$as_lineno_3" = "x$as_lineno_2" ') 2>/dev/null; then + $as_unset BASH_ENV || test "${BASH_ENV+set}" != set || { BASH_ENV=; export BASH_ENV; } + $as_unset ENV || test "${ENV+set}" != set || { ENV=; export ENV; } + CONFIG_SHELL=$as_dir/$as_base + export CONFIG_SHELL + exec "$CONFIG_SHELL" "$0" ${1+"$@"} + fi;; + esac + done +done +;; + esac + + # Create $as_me.lineno as a copy of $as_myself, but with $LINENO + # uniformly replaced by the line number. The first 'sed' inserts a + # line-number line before each line; the second 'sed' does the real + # work. The second script uses 'N' to pair each line-number line + # with the numbered line, and appends trailing '-' during + # substitution so that $LINENO is not a special case at line end. + # (Raja R Harinath suggested sed '=', and Paul Eggert wrote the + # second 'sed' script. Blame Lee E. McMahon for sed's syntax. :-) + sed '=' <$as_myself | + sed ' + N + s,$,-, + : loop + s,^\(['$as_cr_digits']*\)\(.*\)[$]LINENO\([^'$as_cr_alnum'_]\),\1\2\1\3, + t loop + s,-$,, + s,^['$as_cr_digits']*\n,, + ' >$as_me.lineno && + chmod +x $as_me.lineno || + { { echo "$as_me:$LINENO: error: cannot create $as_me.lineno; rerun with a POSIX shell" >&5 +echo "$as_me: error: cannot create $as_me.lineno; rerun with a POSIX shell" >&2;} + { (exit 1); exit 1; }; } + + # Don't try to exec as it changes $[0], causing all sort of problems + # (the dirname of $[0] is not the place where we might find the + # original and so on. Autoconf is especially sensible to this). + . ./$as_me.lineno + # Exit status is that of the last command. + exit +} + + +case `echo "testing\c"; echo 1,2,3`,`echo -n testing; echo 1,2,3` in + *c*,-n*) ECHO_N= ECHO_C=' +' ECHO_T=' ' ;; + *c*,* ) ECHO_N=-n ECHO_C= ECHO_T= ;; + *) ECHO_N= ECHO_C='\c' ECHO_T= ;; +esac + +if expr a : '\(a\)' >/dev/null 2>&1; then + as_expr=expr +else + as_expr=false +fi + +rm -f conf$$ conf$$.exe conf$$.file +echo >conf$$.file +if ln -s conf$$.file conf$$ 2>/dev/null; then + # We could just check for DJGPP; but this test a) works b) is more generic + # and c) will remain valid once DJGPP supports symlinks (DJGPP 2.04). + if test -f conf$$.exe; then + # Don't use ln at all; we don't have any links + as_ln_s='cp -p' + else + as_ln_s='ln -s' + fi +elif ln conf$$.file conf$$ 2>/dev/null; then + as_ln_s=ln +else + as_ln_s='cp -p' +fi +rm -f conf$$ conf$$.exe conf$$.file + +if mkdir -p . 2>/dev/null; then + as_mkdir_p=: +else + test -d ./-p && rmdir ./-p + as_mkdir_p=false +fi + +as_executable_p="test -f" + +# Sed expression to map a string onto a valid CPP name. +as_tr_cpp="eval sed 'y%*$as_cr_letters%P$as_cr_LETTERS%;s%[^_$as_cr_alnum]%_%g'" + +# Sed expression to map a string onto a valid variable name. +as_tr_sh="eval sed 'y%*+%pp%;s%[^_$as_cr_alnum]%_%g'" + + +# IFS +# We need space, tab and new line, in precisely that order. +as_nl=' +' +IFS=" $as_nl" + +# CDPATH. +$as_unset CDPATH + +exec 6>&1 + +# Open the log real soon, to keep \$[0] and so on meaningful, and to +# report actual input values of CONFIG_FILES etc. instead of their +# values after options handling. Logging --version etc. is OK. +exec 5>>config.log +{ + echo + sed 'h;s/./-/g;s/^.../## /;s/...$/ ##/;p;x;p;x' <<_ASBOX +## Running $as_me. ## +_ASBOX +} >&5 +cat >&5 <<_CSEOF + +This file was extended by $as_me, which was +generated by GNU Autoconf 2.59. Invocation command line was + + CONFIG_FILES = $CONFIG_FILES + CONFIG_HEADERS = $CONFIG_HEADERS + CONFIG_LINKS = $CONFIG_LINKS + CONFIG_COMMANDS = $CONFIG_COMMANDS + $ $0 $@ + +_CSEOF +echo "on `(hostname || uname -n) 2>/dev/null | sed 1q`" >&5 +echo >&5 +config_files=" Makefile crypto/Makefile doc/Makefile" +config_headers=" crypto/include/config.h:config_in.h" + +ac_cs_usage="\ +\`$as_me' instantiates files from templates according to the +current configuration. + +Usage: $0 [OPTIONS] [FILE]... + + -h, --help print this help, then exit + -V, --version print version number, then exit + -q, --quiet do not print progress messages + -d, --debug don't remove temporary files + --recheck update $as_me by reconfiguring in the same conditions + --file=FILE[:TEMPLATE] + instantiate the configuration file FILE + --header=FILE[:TEMPLATE] + instantiate the configuration header FILE + +Configuration files: +$config_files + +Configuration headers: +$config_headers + +Report bugs to ." +ac_cs_version="\ +config.status +configured by ./configure, generated by GNU Autoconf 2.59, + with options \"'--prefix=/usr/local/freeswitch' '--enable-pic'\" + +Copyright (C) 2003 Free Software Foundation, Inc. +This config.status script is free software; the Free Software Foundation +gives unlimited permission to copy, distribute and modify it." +srcdir=. +INSTALL="/usr/bin/install -c" +# If no file are specified by the user, then we need to provide default +# value. By we need to know if files were specified by the user. +ac_need_defaults=: +while test $# != 0 +do + case $1 in + --*=*) + ac_option=`expr "x$1" : 'x\([^=]*\)='` + ac_optarg=`expr "x$1" : 'x[^=]*=\(.*\)'` + ac_shift=: + ;; + -*) + ac_option=$1 + ac_optarg=$2 + ac_shift=shift + ;; + *) # This is not an option, so the user has probably given explicit + # arguments. + ac_option=$1 + ac_need_defaults=false;; + esac + + case $ac_option in + # Handling of the options. + -recheck | --recheck | --rechec | --reche | --rech | --rec | --re | --r) + ac_cs_recheck=: ;; + --version | --vers* | -V ) + echo "$ac_cs_version"; exit 0 ;; + --he | --h) + # Conflict between --help and --header + { { echo "$as_me:$LINENO: error: ambiguous option: $1 +Try \`$0 --help' for more information." >&5 +echo "$as_me: error: ambiguous option: $1 +Try \`$0 --help' for more information." >&2;} + { (exit 1); exit 1; }; };; + --help | --hel | -h ) + echo "$ac_cs_usage"; exit 0 ;; + --debug | --d* | -d ) + debug=: ;; + --file | --fil | --fi | --f ) + $ac_shift + CONFIG_FILES="$CONFIG_FILES $ac_optarg" + ac_need_defaults=false;; + --header | --heade | --head | --hea ) + $ac_shift + CONFIG_HEADERS="$CONFIG_HEADERS $ac_optarg" + ac_need_defaults=false;; + -q | -quiet | --quiet | --quie | --qui | --qu | --q \ + | -silent | --silent | --silen | --sile | --sil | --si | --s) + ac_cs_silent=: ;; + + # This is an error. + -*) { { echo "$as_me:$LINENO: error: unrecognized option: $1 +Try \`$0 --help' for more information." >&5 +echo "$as_me: error: unrecognized option: $1 +Try \`$0 --help' for more information." >&2;} + { (exit 1); exit 1; }; } ;; + + *) ac_config_targets="$ac_config_targets $1" ;; + + esac + shift +done + +ac_configure_extra_args= + +if $ac_cs_silent; then + exec 6>/dev/null + ac_configure_extra_args="$ac_configure_extra_args --silent" +fi + +if $ac_cs_recheck; then + echo "running /bin/sh ./configure " '--prefix=/usr/local/freeswitch' '--enable-pic' $ac_configure_extra_args " --no-create --no-recursion" >&6 + exec /bin/sh ./configure '--prefix=/usr/local/freeswitch' '--enable-pic' $ac_configure_extra_args --no-create --no-recursion +fi + +for ac_config_target in $ac_config_targets +do + case "$ac_config_target" in + # Handling of arguments. + "Makefile" ) CONFIG_FILES="$CONFIG_FILES Makefile" ;; + "crypto/Makefile" ) CONFIG_FILES="$CONFIG_FILES crypto/Makefile" ;; + "doc/Makefile" ) CONFIG_FILES="$CONFIG_FILES doc/Makefile" ;; + "crypto/include/config.h" ) CONFIG_HEADERS="$CONFIG_HEADERS crypto/include/config.h:config_in.h" ;; + *) { { echo "$as_me:$LINENO: error: invalid argument: $ac_config_target" >&5 +echo "$as_me: error: invalid argument: $ac_config_target" >&2;} + { (exit 1); exit 1; }; };; + esac +done + +# If the user did not use the arguments to specify the items to instantiate, +# then the envvar interface is used. Set only those that are not. +# We use the long form for the default assignment because of an extremely +# bizarre bug on SunOS 4.1.3. +if $ac_need_defaults; then + test "${CONFIG_FILES+set}" = set || CONFIG_FILES=$config_files + test "${CONFIG_HEADERS+set}" = set || CONFIG_HEADERS=$config_headers +fi + +# Have a temporary directory for convenience. Make it in the build tree +# simply because there is no reason to put it here, and in addition, +# creating and moving files from /tmp can sometimes cause problems. +# Create a temporary directory, and hook for its removal unless debugging. +$debug || +{ + trap 'exit_status=$?; rm -rf $tmp && exit $exit_status' 0 + trap '{ (exit 1); exit 1; }' 1 2 13 15 +} + +# Create a (secure) tmp directory for tmp files. + +{ + tmp=`(umask 077 && mktemp -d -q "./confstatXXXXXX") 2>/dev/null` && + test -n "$tmp" && test -d "$tmp" +} || +{ + tmp=./confstat$$-$RANDOM + (umask 077 && mkdir $tmp) +} || +{ + echo "$me: cannot create a temporary directory in ." >&2 + { (exit 1); exit 1; } +} + + +# +# CONFIG_FILES section. +# + +# No need to generate the scripts if there are no CONFIG_FILES. +# This happens for instance when ./config.status config.h +if test -n "$CONFIG_FILES"; then + # Protect against being on the right side of a sed subst in config.status. + sed 's/,@/@@/; s/@,/@@/; s/,;t t$/@;t t/; /@;t t$/s/[\\&,]/\\&/g; + s/@@/,@/; s/@@/@,/; s/@;t t$/,;t t/' >$tmp/subs.sed <<\CEOF +s,@SHELL@,/bin/sh,;t t +s,@PATH_SEPARATOR@,:,;t t +s,@PACKAGE_NAME@,,;t t +s,@PACKAGE_TARNAME@,,;t t +s,@PACKAGE_VERSION@,,;t t +s,@PACKAGE_STRING@,,;t t +s,@PACKAGE_BUGREPORT@,,;t t +s,@exec_prefix@,${prefix},;t t +s,@prefix@,/usr/local/freeswitch,;t t +s,@program_transform_name@,s,x,x,,;t t +s,@bindir@,${exec_prefix}/bin,;t t +s,@sbindir@,${exec_prefix}/sbin,;t t +s,@libexecdir@,${exec_prefix}/libexec,;t t +s,@datadir@,${prefix}/share,;t t +s,@sysconfdir@,${prefix}/etc,;t t +s,@sharedstatedir@,${prefix}/com,;t t +s,@localstatedir@,${prefix}/var,;t t +s,@libdir@,${exec_prefix}/lib,;t t +s,@includedir@,${prefix}/include,;t t +s,@oldincludedir@,/usr/include,;t t +s,@infodir@,${prefix}/info,;t t +s,@mandir@,${prefix}/man,;t t +s,@build_alias@,,;t t +s,@host_alias@,,;t t +s,@target_alias@,,;t t +s,@DEFS@,-DHAVE_CONFIG_H,;t t +s,@ECHO_C@,,;t t +s,@ECHO_N@,-n,;t t +s,@ECHO_T@,,;t t +s,@LIBS@,,;t t +s,@RANLIB@,ranlib,;t t +s,@ac_ct_RANLIB@,ranlib,;t t +s,@CC@,gcc,;t t +s,@CFLAGS@,-fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops,;t t +s,@LDFLAGS@,,;t t +s,@CPPFLAGS@,,;t t +s,@ac_ct_CC@,gcc,;t t +s,@EXEEXT@,,;t t +s,@OBJEXT@,o,;t t +s,@INSTALL_PROGRAM@,${INSTALL},;t t +s,@INSTALL_SCRIPT@,${INSTALL},;t t +s,@INSTALL_DATA@,${INSTALL} -m 644,;t t +s,@RNG_OBJS@,rand_source.o,;t t +s,@CPP@,gcc -E,;t t +s,@EGREP@,grep -E,;t t +s,@build@,x86_64-unknown-linux-gnu,;t t +s,@build_cpu@,x86_64,;t t +s,@build_vendor@,unknown,;t t +s,@build_os@,linux-gnu,;t t +s,@host@,x86_64-unknown-linux-gnu,;t t +s,@host_cpu@,x86_64,;t t +s,@host_vendor@,unknown,;t t +s,@host_os@,linux-gnu,;t t +s,@EXE@,,;t t +s,@GDOI_OBJS@,,;t t +s,@LIBOBJS@,,;t t +s,@LTLIBOBJS@,,;t t +CEOF + + # Split the substitutions into bite-sized pieces for seds with + # small command number limits, like on Digital OSF/1 and HP-UX. + ac_max_sed_lines=48 + ac_sed_frag=1 # Number of current file. + ac_beg=1 # First line for current file. + ac_end=$ac_max_sed_lines # Line after last line for current file. + ac_more_lines=: + ac_sed_cmds= + while $ac_more_lines; do + if test $ac_beg -gt 1; then + sed "1,${ac_beg}d; ${ac_end}q" $tmp/subs.sed >$tmp/subs.frag + else + sed "${ac_end}q" $tmp/subs.sed >$tmp/subs.frag + fi + if test ! -s $tmp/subs.frag; then + ac_more_lines=false + else + # The purpose of the label and of the branching condition is to + # speed up the sed processing (if there are no `@' at all, there + # is no need to browse any of the substitutions). + # These are the two extra sed commands mentioned above. + (echo ':t + /@[a-zA-Z_][a-zA-Z_0-9]*@/!b' && cat $tmp/subs.frag) >$tmp/subs-$ac_sed_frag.sed + if test -z "$ac_sed_cmds"; then + ac_sed_cmds="sed -f $tmp/subs-$ac_sed_frag.sed" + else + ac_sed_cmds="$ac_sed_cmds | sed -f $tmp/subs-$ac_sed_frag.sed" + fi + ac_sed_frag=`expr $ac_sed_frag + 1` + ac_beg=$ac_end + ac_end=`expr $ac_end + $ac_max_sed_lines` + fi + done + if test -z "$ac_sed_cmds"; then + ac_sed_cmds=cat + fi +fi # test -n "$CONFIG_FILES" + +for ac_file in : $CONFIG_FILES; do test "x$ac_file" = x: && continue + # Support "outfile[:infile[:infile...]]", defaulting infile="outfile.in". + case $ac_file in + - | *:- | *:-:* ) # input from stdin + cat >$tmp/stdin + ac_file_in=`echo "$ac_file" | sed 's,[^:]*:,,'` + ac_file=`echo "$ac_file" | sed 's,:.*,,'` ;; + *:* ) ac_file_in=`echo "$ac_file" | sed 's,[^:]*:,,'` + ac_file=`echo "$ac_file" | sed 's,:.*,,'` ;; + * ) ac_file_in=$ac_file.in ;; + esac + + # Compute @srcdir@, @top_srcdir@, and @INSTALL@ for subdirectories. + ac_dir=`(dirname "$ac_file") 2>/dev/null || +$as_expr X"$ac_file" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$ac_file" : 'X\(//\)[^/]' \| \ + X"$ac_file" : 'X\(//\)$' \| \ + X"$ac_file" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$ac_file" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + { if $as_mkdir_p; then + mkdir -p "$ac_dir" + else + as_dir="$ac_dir" + as_dirs= + while test ! -d "$as_dir"; do + as_dirs="$as_dir $as_dirs" + as_dir=`(dirname "$as_dir") 2>/dev/null || +$as_expr X"$as_dir" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$as_dir" : 'X\(//\)[^/]' \| \ + X"$as_dir" : 'X\(//\)$' \| \ + X"$as_dir" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$as_dir" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + done + test ! -n "$as_dirs" || mkdir $as_dirs + fi || { { echo "$as_me:$LINENO: error: cannot create directory \"$ac_dir\"" >&5 +echo "$as_me: error: cannot create directory \"$ac_dir\"" >&2;} + { (exit 1); exit 1; }; }; } + + ac_builddir=. + +if test "$ac_dir" != .; then + ac_dir_suffix=/`echo "$ac_dir" | sed 's,^\.[\\/],,'` + # A "../" for each directory in $ac_dir_suffix. + ac_top_builddir=`echo "$ac_dir_suffix" | sed 's,/[^\\/]*,../,g'` +else + ac_dir_suffix= ac_top_builddir= +fi + +case $srcdir in + .) # No --srcdir option. We are building in place. + ac_srcdir=. + if test -z "$ac_top_builddir"; then + ac_top_srcdir=. + else + ac_top_srcdir=`echo $ac_top_builddir | sed 's,/$,,'` + fi ;; + [\\/]* | ?:[\\/]* ) # Absolute path. + ac_srcdir=$srcdir$ac_dir_suffix; + ac_top_srcdir=$srcdir ;; + *) # Relative path. + ac_srcdir=$ac_top_builddir$srcdir$ac_dir_suffix + ac_top_srcdir=$ac_top_builddir$srcdir ;; +esac + +# Do not use `cd foo && pwd` to compute absolute paths, because +# the directories may not exist. +case `pwd` in +.) ac_abs_builddir="$ac_dir";; +*) + case "$ac_dir" in + .) ac_abs_builddir=`pwd`;; + [\\/]* | ?:[\\/]* ) ac_abs_builddir="$ac_dir";; + *) ac_abs_builddir=`pwd`/"$ac_dir";; + esac;; +esac +case $ac_abs_builddir in +.) ac_abs_top_builddir=${ac_top_builddir}.;; +*) + case ${ac_top_builddir}. in + .) ac_abs_top_builddir=$ac_abs_builddir;; + [\\/]* | ?:[\\/]* ) ac_abs_top_builddir=${ac_top_builddir}.;; + *) ac_abs_top_builddir=$ac_abs_builddir/${ac_top_builddir}.;; + esac;; +esac +case $ac_abs_builddir in +.) ac_abs_srcdir=$ac_srcdir;; +*) + case $ac_srcdir in + .) ac_abs_srcdir=$ac_abs_builddir;; + [\\/]* | ?:[\\/]* ) ac_abs_srcdir=$ac_srcdir;; + *) ac_abs_srcdir=$ac_abs_builddir/$ac_srcdir;; + esac;; +esac +case $ac_abs_builddir in +.) ac_abs_top_srcdir=$ac_top_srcdir;; +*) + case $ac_top_srcdir in + .) ac_abs_top_srcdir=$ac_abs_builddir;; + [\\/]* | ?:[\\/]* ) ac_abs_top_srcdir=$ac_top_srcdir;; + *) ac_abs_top_srcdir=$ac_abs_builddir/$ac_top_srcdir;; + esac;; +esac + + + case $INSTALL in + [\\/$]* | ?:[\\/]* ) ac_INSTALL=$INSTALL ;; + *) ac_INSTALL=$ac_top_builddir$INSTALL ;; + esac + + if test x"$ac_file" != x-; then + { echo "$as_me:$LINENO: creating $ac_file" >&5 +echo "$as_me: creating $ac_file" >&6;} + rm -f "$ac_file" + fi + # Let's still pretend it is `configure' which instantiates (i.e., don't + # use $as_me), people would be surprised to read: + # /* config.h. Generated by config.status. */ + if test x"$ac_file" = x-; then + configure_input= + else + configure_input="$ac_file. " + fi + configure_input=$configure_input"Generated from `echo $ac_file_in | + sed 's,.*/,,'` by configure." + + # First look for the input files in the build tree, otherwise in the + # src tree. + ac_file_inputs=`IFS=: + for f in $ac_file_in; do + case $f in + -) echo $tmp/stdin ;; + [\\/$]*) + # Absolute (can't be DOS-style, as IFS=:) + test -f "$f" || { { echo "$as_me:$LINENO: error: cannot find input file: $f" >&5 +echo "$as_me: error: cannot find input file: $f" >&2;} + { (exit 1); exit 1; }; } + echo "$f";; + *) # Relative + if test -f "$f"; then + # Build tree + echo "$f" + elif test -f "$srcdir/$f"; then + # Source tree + echo "$srcdir/$f" + else + # /dev/null tree + { { echo "$as_me:$LINENO: error: cannot find input file: $f" >&5 +echo "$as_me: error: cannot find input file: $f" >&2;} + { (exit 1); exit 1; }; } + fi;; + esac + done` || { (exit 1); exit 1; } + sed "/^[ ]*VPATH[ ]*=/{ +s/:*\$(srcdir):*/:/; +s/:*\${srcdir}:*/:/; +s/:*@srcdir@:*/:/; +s/^\([^=]*=[ ]*\):*/\1/; +s/:*$//; +s/^[^=]*=[ ]*$//; +} + +:t +/@[a-zA-Z_][a-zA-Z_0-9]*@/!b +s,@configure_input@,$configure_input,;t t +s,@srcdir@,$ac_srcdir,;t t +s,@abs_srcdir@,$ac_abs_srcdir,;t t +s,@top_srcdir@,$ac_top_srcdir,;t t +s,@abs_top_srcdir@,$ac_abs_top_srcdir,;t t +s,@builddir@,$ac_builddir,;t t +s,@abs_builddir@,$ac_abs_builddir,;t t +s,@top_builddir@,$ac_top_builddir,;t t +s,@abs_top_builddir@,$ac_abs_top_builddir,;t t +s,@INSTALL@,$ac_INSTALL,;t t +" $ac_file_inputs | (eval "$ac_sed_cmds") >$tmp/out + rm -f $tmp/stdin + if test x"$ac_file" != x-; then + mv $tmp/out $ac_file + else + cat $tmp/out + rm -f $tmp/out + fi + +done + +# +# CONFIG_HEADER section. +# + +# These sed commands are passed to sed as "A NAME B NAME C VALUE D", where +# NAME is the cpp macro being defined and VALUE is the value it is being given. +# +# ac_d sets the value in "#define NAME VALUE" lines. +ac_dA='s,^\([ ]*\)#\([ ]*define[ ][ ]*\)' +ac_dB='[ ].*$,\1#\2' +ac_dC=' ' +ac_dD=',;t' +# ac_u turns "#undef NAME" without trailing blanks into "#define NAME VALUE". +ac_uA='s,^\([ ]*\)#\([ ]*\)undef\([ ][ ]*\)' +ac_uB='$,\1#\2define\3' +ac_uC=' ' +ac_uD=',;t' + +for ac_file in : $CONFIG_HEADERS; do test "x$ac_file" = x: && continue + # Support "outfile[:infile[:infile...]]", defaulting infile="outfile.in". + case $ac_file in + - | *:- | *:-:* ) # input from stdin + cat >$tmp/stdin + ac_file_in=`echo "$ac_file" | sed 's,[^:]*:,,'` + ac_file=`echo "$ac_file" | sed 's,:.*,,'` ;; + *:* ) ac_file_in=`echo "$ac_file" | sed 's,[^:]*:,,'` + ac_file=`echo "$ac_file" | sed 's,:.*,,'` ;; + * ) ac_file_in=$ac_file.in ;; + esac + + test x"$ac_file" != x- && { echo "$as_me:$LINENO: creating $ac_file" >&5 +echo "$as_me: creating $ac_file" >&6;} + + # First look for the input files in the build tree, otherwise in the + # src tree. + ac_file_inputs=`IFS=: + for f in $ac_file_in; do + case $f in + -) echo $tmp/stdin ;; + [\\/$]*) + # Absolute (can't be DOS-style, as IFS=:) + test -f "$f" || { { echo "$as_me:$LINENO: error: cannot find input file: $f" >&5 +echo "$as_me: error: cannot find input file: $f" >&2;} + { (exit 1); exit 1; }; } + # Do quote $f, to prevent DOS paths from being IFS'd. + echo "$f";; + *) # Relative + if test -f "$f"; then + # Build tree + echo "$f" + elif test -f "$srcdir/$f"; then + # Source tree + echo "$srcdir/$f" + else + # /dev/null tree + { { echo "$as_me:$LINENO: error: cannot find input file: $f" >&5 +echo "$as_me: error: cannot find input file: $f" >&2;} + { (exit 1); exit 1; }; } + fi;; + esac + done` || { (exit 1); exit 1; } + # Remove the trailing spaces. + sed 's/[ ]*$//' $ac_file_inputs >$tmp/in + + # Handle all the #define templates only if necessary. + if grep "^[ ]*#[ ]*define" $tmp/in >/dev/null; then + # If there are no defines, we may have an empty if/fi + : + cat >$tmp/defines.sed <$tmp/out + rm -f $tmp/in + mv $tmp/out $tmp/in + + cat >$tmp/defines.sed <$tmp/out + rm -f $tmp/in + mv $tmp/out $tmp/in + + fi # grep + + # Handle all the #undef templates + cat >$tmp/undefs.sed <$tmp/out + rm -f $tmp/in + mv $tmp/out $tmp/in + + cat >$tmp/undefs.sed <$tmp/out + rm -f $tmp/in + mv $tmp/out $tmp/in + + # Let's still pretend it is `configure' which instantiates (i.e., don't + # use $as_me), people would be surprised to read: + # /* config.h. Generated by config.status. */ + if test x"$ac_file" = x-; then + echo "/* Generated by configure. */" >$tmp/config.h + else + echo "/* $ac_file. Generated by configure. */" >$tmp/config.h + fi + cat $tmp/in >>$tmp/config.h + rm -f $tmp/in + if test x"$ac_file" != x-; then + if diff $ac_file $tmp/config.h >/dev/null 2>&1; then + { echo "$as_me:$LINENO: $ac_file is unchanged" >&5 +echo "$as_me: $ac_file is unchanged" >&6;} + else + ac_dir=`(dirname "$ac_file") 2>/dev/null || +$as_expr X"$ac_file" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$ac_file" : 'X\(//\)[^/]' \| \ + X"$ac_file" : 'X\(//\)$' \| \ + X"$ac_file" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$ac_file" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + { if $as_mkdir_p; then + mkdir -p "$ac_dir" + else + as_dir="$ac_dir" + as_dirs= + while test ! -d "$as_dir"; do + as_dirs="$as_dir $as_dirs" + as_dir=`(dirname "$as_dir") 2>/dev/null || +$as_expr X"$as_dir" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$as_dir" : 'X\(//\)[^/]' \| \ + X"$as_dir" : 'X\(//\)$' \| \ + X"$as_dir" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$as_dir" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + done + test ! -n "$as_dirs" || mkdir $as_dirs + fi || { { echo "$as_me:$LINENO: error: cannot create directory \"$ac_dir\"" >&5 +echo "$as_me: error: cannot create directory \"$ac_dir\"" >&2;} + { (exit 1); exit 1; }; }; } + + rm -f $ac_file + mv $tmp/config.h $ac_file + fi + else + cat $tmp/config.h + rm -f $tmp/config.h + fi +done + +{ (exit 0); exit 0; } diff --git a/libs/srtp/config.sub b/libs/srtp/config.sub new file mode 100755 index 0000000000..edb6b663ca --- /dev/null +++ b/libs/srtp/config.sub @@ -0,0 +1,1555 @@ +#! /bin/sh +# Configuration validation subroutine script. +# Copyright (C) 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999, +# 2000, 2001, 2002, 2003, 2004 Free Software Foundation, Inc. + +timestamp='2004-08-29' + +# This file is (in principle) common to ALL GNU software. +# The presence of a machine in this file suggests that SOME GNU software +# can handle that machine. It does not imply ALL GNU software can. +# +# This file is free software; you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation; either version 2 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program; if not, write to the Free Software +# Foundation, Inc., 59 Temple Place - Suite 330, +# Boston, MA 02111-1307, USA. + +# As a special exception to the GNU General Public License, if you +# distribute this file as part of a program that contains a +# configuration script generated by Autoconf, you may include it under +# the same distribution terms that you use for the rest of that program. + +# Please send patches to . Submit a context +# diff and a properly formatted ChangeLog entry. +# +# Configuration subroutine to validate and canonicalize a configuration type. +# Supply the specified configuration type as an argument. +# If it is invalid, we print an error message on stderr and exit with code 1. +# Otherwise, we print the canonical config type on stdout and succeed. + +# This file is supposed to be the same for all GNU packages +# and recognize all the CPU types, system types and aliases +# that are meaningful with *any* GNU software. +# Each package is responsible for reporting which valid configurations +# it does not support. The user should be able to distinguish +# a failure to support a valid configuration from a meaningless +# configuration. + +# The goal of this file is to map all the various variations of a given +# machine specification into a single specification in the form: +# CPU_TYPE-MANUFACTURER-OPERATING_SYSTEM +# or in some cases, the newer four-part form: +# CPU_TYPE-MANUFACTURER-KERNEL-OPERATING_SYSTEM +# It is wrong to echo any other type of specification. + +me=`echo "$0" | sed -e 's,.*/,,'` + +usage="\ +Usage: $0 [OPTION] CPU-MFR-OPSYS + $0 [OPTION] ALIAS + +Canonicalize a configuration name. + +Operation modes: + -h, --help print this help, then exit + -t, --time-stamp print date of last modification, then exit + -v, --version print version number, then exit + +Report bugs and patches to ." + +version="\ +GNU config.sub ($timestamp) + +Copyright (C) 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003, 2004 +Free Software Foundation, Inc. + +This is free software; see the source for copying conditions. There is NO +warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE." + +help=" +Try \`$me --help' for more information." + +# Parse command line +while test $# -gt 0 ; do + case $1 in + --time-stamp | --time* | -t ) + echo "$timestamp" ; exit 0 ;; + --version | -v ) + echo "$version" ; exit 0 ;; + --help | --h* | -h ) + echo "$usage"; exit 0 ;; + -- ) # Stop option processing + shift; break ;; + - ) # Use stdin as input. + break ;; + -* ) + echo "$me: invalid option $1$help" + exit 1 ;; + + *local*) + # First pass through any local machine types. + echo $1 + exit 0;; + + * ) + break ;; + esac +done + +case $# in + 0) echo "$me: missing argument$help" >&2 + exit 1;; + 1) ;; + *) echo "$me: too many arguments$help" >&2 + exit 1;; +esac + +# Separate what the user gave into CPU-COMPANY and OS or KERNEL-OS (if any). +# Here we must recognize all the valid KERNEL-OS combinations. +maybe_os=`echo $1 | sed 's/^\(.*\)-\([^-]*-[^-]*\)$/\2/'` +case $maybe_os in + nto-qnx* | linux-gnu* | linux-dietlibc | linux-uclibc* | uclinux-uclibc* | uclinux-gnu* | \ + kfreebsd*-gnu* | knetbsd*-gnu* | netbsd*-gnu* | storm-chaos* | os2-emx* | rtmk-nova*) + os=-$maybe_os + basic_machine=`echo $1 | sed 's/^\(.*\)-\([^-]*-[^-]*\)$/\1/'` + ;; + *) + basic_machine=`echo $1 | sed 's/-[^-]*$//'` + if [ $basic_machine != $1 ] + then os=`echo $1 | sed 's/.*-/-/'` + else os=; fi + ;; +esac + +### Let's recognize common machines as not being operating systems so +### that things like config.sub decstation-3100 work. We also +### recognize some manufacturers as not being operating systems, so we +### can provide default operating systems below. +case $os in + -sun*os*) + # Prevent following clause from handling this invalid input. + ;; + -dec* | -mips* | -sequent* | -encore* | -pc532* | -sgi* | -sony* | \ + -att* | -7300* | -3300* | -delta* | -motorola* | -sun[234]* | \ + -unicom* | -ibm* | -next | -hp | -isi* | -apollo | -altos* | \ + -convergent* | -ncr* | -news | -32* | -3600* | -3100* | -hitachi* |\ + -c[123]* | -convex* | -sun | -crds | -omron* | -dg | -ultra | -tti* | \ + -harris | -dolphin | -highlevel | -gould | -cbm | -ns | -masscomp | \ + -apple | -axis | -knuth | -cray) + os= + basic_machine=$1 + ;; + -sim | -cisco | -oki | -wec | -winbond) + os= + basic_machine=$1 + ;; + -scout) + ;; + -wrs) + os=-vxworks + basic_machine=$1 + ;; + -chorusos*) + os=-chorusos + basic_machine=$1 + ;; + -chorusrdb) + os=-chorusrdb + basic_machine=$1 + ;; + -hiux*) + os=-hiuxwe2 + ;; + -sco5) + os=-sco3.2v5 + basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'` + ;; + -sco4) + os=-sco3.2v4 + basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'` + ;; + -sco3.2.[4-9]*) + os=`echo $os | sed -e 's/sco3.2./sco3.2v/'` + basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'` + ;; + -sco3.2v[4-9]*) + # Don't forget version if it is 3.2v4 or newer. + basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'` + ;; + -sco*) + os=-sco3.2v2 + basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'` + ;; + -udk*) + basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'` + ;; + -isc) + os=-isc2.2 + basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'` + ;; + -clix*) + basic_machine=clipper-intergraph + ;; + -isc*) + basic_machine=`echo $1 | sed -e 's/86-.*/86-pc/'` + ;; + -lynx*) + os=-lynxos + ;; + -ptx*) + basic_machine=`echo $1 | sed -e 's/86-.*/86-sequent/'` + ;; + -windowsnt*) + os=`echo $os | sed -e 's/windowsnt/winnt/'` + ;; + -psos*) + os=-psos + ;; + -mint | -mint[0-9]*) + basic_machine=m68k-atari + os=-mint + ;; +esac + +# Decode aliases for certain CPU-COMPANY combinations. +case $basic_machine in + # Recognize the basic CPU types without company name. + # Some are omitted here because they have special meanings below. + 1750a | 580 \ + | a29k \ + | alpha | alphaev[4-8] | alphaev56 | alphaev6[78] | alphapca5[67] \ + | alpha64 | alpha64ev[4-8] | alpha64ev56 | alpha64ev6[78] | alpha64pca5[67] \ + | am33_2.0 \ + | arc | arm | arm[bl]e | arme[lb] | armv[2345] | armv[345][lb] | avr \ + | c4x | clipper \ + | d10v | d30v | dlx | dsp16xx \ + | fr30 | frv \ + | h8300 | h8500 | hppa | hppa1.[01] | hppa2.0 | hppa2.0[nw] | hppa64 \ + | i370 | i860 | i960 | ia64 \ + | ip2k | iq2000 \ + | m32r | m32rle | m68000 | m68k | m88k | mcore \ + | mips | mipsbe | mipseb | mipsel | mipsle \ + | mips16 \ + | mips64 | mips64el \ + | mips64vr | mips64vrel \ + | mips64orion | mips64orionel \ + | mips64vr4100 | mips64vr4100el \ + | mips64vr4300 | mips64vr4300el \ + | mips64vr5000 | mips64vr5000el \ + | mipsisa32 | mipsisa32el \ + | mipsisa32r2 | mipsisa32r2el \ + | mipsisa64 | mipsisa64el \ + | mipsisa64r2 | mipsisa64r2el \ + | mipsisa64sb1 | mipsisa64sb1el \ + | mipsisa64sr71k | mipsisa64sr71kel \ + | mipstx39 | mipstx39el \ + | mn10200 | mn10300 \ + | msp430 \ + | ns16k | ns32k \ + | openrisc | or32 \ + | pdp10 | pdp11 | pj | pjl \ + | powerpc | powerpc64 | powerpc64le | powerpcle | ppcbe \ + | pyramid \ + | sh | sh[1234] | sh[23]e | sh[34]eb | shbe | shle | sh[1234]le | sh3ele \ + | sh64 | sh64le \ + | sparc | sparc64 | sparc86x | sparclet | sparclite | sparcv8 | sparcv9 | sparcv9b \ + | strongarm \ + | tahoe | thumb | tic4x | tic80 | tron \ + | v850 | v850e \ + | we32k \ + | x86 | xscale | xstormy16 | xtensa \ + | z8k) + basic_machine=$basic_machine-unknown + ;; + m6811 | m68hc11 | m6812 | m68hc12) + # Motorola 68HC11/12. + basic_machine=$basic_machine-unknown + os=-none + ;; + m88110 | m680[12346]0 | m683?2 | m68360 | m5200 | v70 | w65 | z8k) + ;; + + # We use `pc' rather than `unknown' + # because (1) that's what they normally are, and + # (2) the word "unknown" tends to confuse beginning users. + i*86 | x86_64) + basic_machine=$basic_machine-pc + ;; + # Object if more than one company name word. + *-*-*) + echo Invalid configuration \`$1\': machine \`$basic_machine\' not recognized 1>&2 + exit 1 + ;; + # Recognize the basic CPU types with company name. + 580-* \ + | a29k-* \ + | alpha-* | alphaev[4-8]-* | alphaev56-* | alphaev6[78]-* \ + | alpha64-* | alpha64ev[4-8]-* | alpha64ev56-* | alpha64ev6[78]-* \ + | alphapca5[67]-* | alpha64pca5[67]-* | arc-* \ + | arm-* | armbe-* | armle-* | armeb-* | armv*-* \ + | avr-* \ + | bs2000-* \ + | c[123]* | c30-* | [cjt]90-* | c4x-* | c54x-* | c55x-* | c6x-* \ + | clipper-* | craynv-* | cydra-* \ + | d10v-* | d30v-* | dlx-* \ + | elxsi-* \ + | f30[01]-* | f700-* | fr30-* | frv-* | fx80-* \ + | h8300-* | h8500-* \ + | hppa-* | hppa1.[01]-* | hppa2.0-* | hppa2.0[nw]-* | hppa64-* \ + | i*86-* | i860-* | i960-* | ia64-* \ + | ip2k-* | iq2000-* \ + | m32r-* | m32rle-* \ + | m68000-* | m680[012346]0-* | m68360-* | m683?2-* | m68k-* \ + | m88110-* | m88k-* | mcore-* \ + | mips-* | mipsbe-* | mipseb-* | mipsel-* | mipsle-* \ + | mips16-* \ + | mips64-* | mips64el-* \ + | mips64vr-* | mips64vrel-* \ + | mips64orion-* | mips64orionel-* \ + | mips64vr4100-* | mips64vr4100el-* \ + | mips64vr4300-* | mips64vr4300el-* \ + | mips64vr5000-* | mips64vr5000el-* \ + | mipsisa32-* | mipsisa32el-* \ + | mipsisa32r2-* | mipsisa32r2el-* \ + | mipsisa64-* | mipsisa64el-* \ + | mipsisa64r2-* | mipsisa64r2el-* \ + | mipsisa64sb1-* | mipsisa64sb1el-* \ + | mipsisa64sr71k-* | mipsisa64sr71kel-* \ + | mipstx39-* | mipstx39el-* \ + | mmix-* \ + | msp430-* \ + | none-* | np1-* | ns16k-* | ns32k-* \ + | orion-* \ + | pdp10-* | pdp11-* | pj-* | pjl-* | pn-* | power-* \ + | powerpc-* | powerpc64-* | powerpc64le-* | powerpcle-* | ppcbe-* \ + | pyramid-* \ + | romp-* | rs6000-* \ + | sh-* | sh[1234]-* | sh[23]e-* | sh[34]eb-* | shbe-* \ + | shle-* | sh[1234]le-* | sh3ele-* | sh64-* | sh64le-* \ + | sparc-* | sparc64-* | sparc86x-* | sparclet-* | sparclite-* \ + | sparcv8-* | sparcv9-* | sparcv9b-* | strongarm-* | sv1-* | sx?-* \ + | tahoe-* | thumb-* \ + | tic30-* | tic4x-* | tic54x-* | tic55x-* | tic6x-* | tic80-* \ + | tron-* \ + | v850-* | v850e-* | vax-* \ + | we32k-* \ + | x86-* | x86_64-* | xps100-* | xscale-* | xstormy16-* \ + | xtensa-* \ + | ymp-* \ + | z8k-*) + ;; + # Recognize the various machine names and aliases which stand + # for a CPU type and a company and sometimes even an OS. + 386bsd) + basic_machine=i386-unknown + os=-bsd + ;; + 3b1 | 7300 | 7300-att | att-7300 | pc7300 | safari | unixpc) + basic_machine=m68000-att + ;; + 3b*) + basic_machine=we32k-att + ;; + a29khif) + basic_machine=a29k-amd + os=-udi + ;; + abacus) + basic_machine=abacus-unknown + ;; + adobe68k) + basic_machine=m68010-adobe + os=-scout + ;; + alliant | fx80) + basic_machine=fx80-alliant + ;; + altos | altos3068) + basic_machine=m68k-altos + ;; + am29k) + basic_machine=a29k-none + os=-bsd + ;; + amd64) + basic_machine=x86_64-pc + ;; + amd64-*) + basic_machine=x86_64-`echo $basic_machine | sed 's/^[^-]*-//'` + ;; + amdahl) + basic_machine=580-amdahl + os=-sysv + ;; + amiga | amiga-*) + basic_machine=m68k-unknown + ;; + amigaos | amigados) + basic_machine=m68k-unknown + os=-amigaos + ;; + amigaunix | amix) + basic_machine=m68k-unknown + os=-sysv4 + ;; + apollo68) + basic_machine=m68k-apollo + os=-sysv + ;; + apollo68bsd) + basic_machine=m68k-apollo + os=-bsd + ;; + aux) + basic_machine=m68k-apple + os=-aux + ;; + balance) + basic_machine=ns32k-sequent + os=-dynix + ;; + c90) + basic_machine=c90-cray + os=-unicos + ;; + convex-c1) + basic_machine=c1-convex + os=-bsd + ;; + convex-c2) + basic_machine=c2-convex + os=-bsd + ;; + convex-c32) + basic_machine=c32-convex + os=-bsd + ;; + convex-c34) + basic_machine=c34-convex + os=-bsd + ;; + convex-c38) + basic_machine=c38-convex + os=-bsd + ;; + cray | j90) + basic_machine=j90-cray + os=-unicos + ;; + craynv) + basic_machine=craynv-cray + os=-unicosmp + ;; + cr16c) + basic_machine=cr16c-unknown + os=-elf + ;; + crds | unos) + basic_machine=m68k-crds + ;; + crisv32 | crisv32-* | etraxfs*) + basic_machine=crisv32-axis + ;; + cris | cris-* | etrax*) + basic_machine=cris-axis + ;; + crx) + basic_machine=crx-unknown + os=-elf + ;; + da30 | da30-*) + basic_machine=m68k-da30 + ;; + decstation | decstation-3100 | pmax | pmax-* | pmin | dec3100 | decstatn) + basic_machine=mips-dec + ;; + decsystem10* | dec10*) + basic_machine=pdp10-dec + os=-tops10 + ;; + decsystem20* | dec20*) + basic_machine=pdp10-dec + os=-tops20 + ;; + delta | 3300 | motorola-3300 | motorola-delta \ + | 3300-motorola | delta-motorola) + basic_machine=m68k-motorola + ;; + delta88) + basic_machine=m88k-motorola + os=-sysv3 + ;; + dpx20 | dpx20-*) + basic_machine=rs6000-bull + os=-bosx + ;; + dpx2* | dpx2*-bull) + basic_machine=m68k-bull + os=-sysv3 + ;; + ebmon29k) + basic_machine=a29k-amd + os=-ebmon + ;; + elxsi) + basic_machine=elxsi-elxsi + os=-bsd + ;; + encore | umax | mmax) + basic_machine=ns32k-encore + ;; + es1800 | OSE68k | ose68k | ose | OSE) + basic_machine=m68k-ericsson + os=-ose + ;; + fx2800) + basic_machine=i860-alliant + ;; + genix) + basic_machine=ns32k-ns + ;; + gmicro) + basic_machine=tron-gmicro + os=-sysv + ;; + go32) + basic_machine=i386-pc + os=-go32 + ;; + h3050r* | hiux*) + basic_machine=hppa1.1-hitachi + os=-hiuxwe2 + ;; + h8300hms) + basic_machine=h8300-hitachi + os=-hms + ;; + h8300xray) + basic_machine=h8300-hitachi + os=-xray + ;; + h8500hms) + basic_machine=h8500-hitachi + os=-hms + ;; + harris) + basic_machine=m88k-harris + os=-sysv3 + ;; + hp300-*) + basic_machine=m68k-hp + ;; + hp300bsd) + basic_machine=m68k-hp + os=-bsd + ;; + hp300hpux) + basic_machine=m68k-hp + os=-hpux + ;; + hp3k9[0-9][0-9] | hp9[0-9][0-9]) + basic_machine=hppa1.0-hp + ;; + hp9k2[0-9][0-9] | hp9k31[0-9]) + basic_machine=m68000-hp + ;; + hp9k3[2-9][0-9]) + basic_machine=m68k-hp + ;; + hp9k6[0-9][0-9] | hp6[0-9][0-9]) + basic_machine=hppa1.0-hp + ;; + hp9k7[0-79][0-9] | hp7[0-79][0-9]) + basic_machine=hppa1.1-hp + ;; + hp9k78[0-9] | hp78[0-9]) + # FIXME: really hppa2.0-hp + basic_machine=hppa1.1-hp + ;; + hp9k8[67]1 | hp8[67]1 | hp9k80[24] | hp80[24] | hp9k8[78]9 | hp8[78]9 | hp9k893 | hp893) + # FIXME: really hppa2.0-hp + basic_machine=hppa1.1-hp + ;; + hp9k8[0-9][13679] | hp8[0-9][13679]) + basic_machine=hppa1.1-hp + ;; + hp9k8[0-9][0-9] | hp8[0-9][0-9]) + basic_machine=hppa1.0-hp + ;; + hppa-next) + os=-nextstep3 + ;; + hppaosf) + basic_machine=hppa1.1-hp + os=-osf + ;; + hppro) + basic_machine=hppa1.1-hp + os=-proelf + ;; + i370-ibm* | ibm*) + basic_machine=i370-ibm + ;; +# I'm not sure what "Sysv32" means. Should this be sysv3.2? + i*86v32) + basic_machine=`echo $1 | sed -e 's/86.*/86-pc/'` + os=-sysv32 + ;; + i*86v4*) + basic_machine=`echo $1 | sed -e 's/86.*/86-pc/'` + os=-sysv4 + ;; + i*86v) + basic_machine=`echo $1 | sed -e 's/86.*/86-pc/'` + os=-sysv + ;; + i*86sol2) + basic_machine=`echo $1 | sed -e 's/86.*/86-pc/'` + os=-solaris2 + ;; + i386mach) + basic_machine=i386-mach + os=-mach + ;; + i386-vsta | vsta) + basic_machine=i386-unknown + os=-vsta + ;; + iris | iris4d) + basic_machine=mips-sgi + case $os in + -irix*) + ;; + *) + os=-irix4 + ;; + esac + ;; + isi68 | isi) + basic_machine=m68k-isi + os=-sysv + ;; + m88k-omron*) + basic_machine=m88k-omron + ;; + magnum | m3230) + basic_machine=mips-mips + os=-sysv + ;; + merlin) + basic_machine=ns32k-utek + os=-sysv + ;; + mingw32) + basic_machine=i386-pc + os=-mingw32 + ;; + miniframe) + basic_machine=m68000-convergent + ;; + *mint | -mint[0-9]* | *MiNT | *MiNT[0-9]*) + basic_machine=m68k-atari + os=-mint + ;; + mips3*-*) + basic_machine=`echo $basic_machine | sed -e 's/mips3/mips64/'` + ;; + mips3*) + basic_machine=`echo $basic_machine | sed -e 's/mips3/mips64/'`-unknown + ;; + monitor) + basic_machine=m68k-rom68k + os=-coff + ;; + morphos) + basic_machine=powerpc-unknown + os=-morphos + ;; + msdos) + basic_machine=i386-pc + os=-msdos + ;; + mvs) + basic_machine=i370-ibm + os=-mvs + ;; + ncr3000) + basic_machine=i486-ncr + os=-sysv4 + ;; + netbsd386) + basic_machine=i386-unknown + os=-netbsd + ;; + netwinder) + basic_machine=armv4l-rebel + os=-linux + ;; + news | news700 | news800 | news900) + basic_machine=m68k-sony + os=-newsos + ;; + news1000) + basic_machine=m68030-sony + os=-newsos + ;; + news-3600 | risc-news) + basic_machine=mips-sony + os=-newsos + ;; + necv70) + basic_machine=v70-nec + os=-sysv + ;; + next | m*-next ) + basic_machine=m68k-next + case $os in + -nextstep* ) + ;; + -ns2*) + os=-nextstep2 + ;; + *) + os=-nextstep3 + ;; + esac + ;; + nh3000) + basic_machine=m68k-harris + os=-cxux + ;; + nh[45]000) + basic_machine=m88k-harris + os=-cxux + ;; + nindy960) + basic_machine=i960-intel + os=-nindy + ;; + mon960) + basic_machine=i960-intel + os=-mon960 + ;; + nonstopux) + basic_machine=mips-compaq + os=-nonstopux + ;; + np1) + basic_machine=np1-gould + ;; + nsr-tandem) + basic_machine=nsr-tandem + ;; + op50n-* | op60c-*) + basic_machine=hppa1.1-oki + os=-proelf + ;; + or32 | or32-*) + basic_machine=or32-unknown + os=-coff + ;; + os400) + basic_machine=powerpc-ibm + os=-os400 + ;; + OSE68000 | ose68000) + basic_machine=m68000-ericsson + os=-ose + ;; + os68k) + basic_machine=m68k-none + os=-os68k + ;; + pa-hitachi) + basic_machine=hppa1.1-hitachi + os=-hiuxwe2 + ;; + paragon) + basic_machine=i860-intel + os=-osf + ;; + pbd) + basic_machine=sparc-tti + ;; + pbb) + basic_machine=m68k-tti + ;; + pc532 | pc532-*) + basic_machine=ns32k-pc532 + ;; + pentium | p5 | k5 | k6 | nexgen | viac3) + basic_machine=i586-pc + ;; + pentiumpro | p6 | 6x86 | athlon | athlon_*) + basic_machine=i686-pc + ;; + pentiumii | pentium2 | pentiumiii | pentium3) + basic_machine=i686-pc + ;; + pentium4) + basic_machine=i786-pc + ;; + pentium-* | p5-* | k5-* | k6-* | nexgen-* | viac3-*) + basic_machine=i586-`echo $basic_machine | sed 's/^[^-]*-//'` + ;; + pentiumpro-* | p6-* | 6x86-* | athlon-*) + basic_machine=i686-`echo $basic_machine | sed 's/^[^-]*-//'` + ;; + pentiumii-* | pentium2-* | pentiumiii-* | pentium3-*) + basic_machine=i686-`echo $basic_machine | sed 's/^[^-]*-//'` + ;; + pentium4-*) + basic_machine=i786-`echo $basic_machine | sed 's/^[^-]*-//'` + ;; + pn) + basic_machine=pn-gould + ;; + power) basic_machine=power-ibm + ;; + ppc) basic_machine=powerpc-unknown + ;; + ppc-*) basic_machine=powerpc-`echo $basic_machine | sed 's/^[^-]*-//'` + ;; + ppcle | powerpclittle | ppc-le | powerpc-little) + basic_machine=powerpcle-unknown + ;; + ppcle-* | powerpclittle-*) + basic_machine=powerpcle-`echo $basic_machine | sed 's/^[^-]*-//'` + ;; + ppc64) basic_machine=powerpc64-unknown + ;; + ppc64-*) basic_machine=powerpc64-`echo $basic_machine | sed 's/^[^-]*-//'` + ;; + ppc64le | powerpc64little | ppc64-le | powerpc64-little) + basic_machine=powerpc64le-unknown + ;; + ppc64le-* | powerpc64little-*) + basic_machine=powerpc64le-`echo $basic_machine | sed 's/^[^-]*-//'` + ;; + ps2) + basic_machine=i386-ibm + ;; + pw32) + basic_machine=i586-unknown + os=-pw32 + ;; + rom68k) + basic_machine=m68k-rom68k + os=-coff + ;; + rm[46]00) + basic_machine=mips-siemens + ;; + rtpc | rtpc-*) + basic_machine=romp-ibm + ;; + s390 | s390-*) + basic_machine=s390-ibm + ;; + s390x | s390x-*) + basic_machine=s390x-ibm + ;; + sa29200) + basic_machine=a29k-amd + os=-udi + ;; + sb1) + basic_machine=mipsisa64sb1-unknown + ;; + sb1el) + basic_machine=mipsisa64sb1el-unknown + ;; + sei) + basic_machine=mips-sei + os=-seiux + ;; + sequent) + basic_machine=i386-sequent + ;; + sh) + basic_machine=sh-hitachi + os=-hms + ;; + sh64) + basic_machine=sh64-unknown + ;; + sparclite-wrs | simso-wrs) + basic_machine=sparclite-wrs + os=-vxworks + ;; + sps7) + basic_machine=m68k-bull + os=-sysv2 + ;; + spur) + basic_machine=spur-unknown + ;; + st2000) + basic_machine=m68k-tandem + ;; + stratus) + basic_machine=i860-stratus + os=-sysv4 + ;; + sun2) + basic_machine=m68000-sun + ;; + sun2os3) + basic_machine=m68000-sun + os=-sunos3 + ;; + sun2os4) + basic_machine=m68000-sun + os=-sunos4 + ;; + sun3os3) + basic_machine=m68k-sun + os=-sunos3 + ;; + sun3os4) + basic_machine=m68k-sun + os=-sunos4 + ;; + sun4os3) + basic_machine=sparc-sun + os=-sunos3 + ;; + sun4os4) + basic_machine=sparc-sun + os=-sunos4 + ;; + sun4sol2) + basic_machine=sparc-sun + os=-solaris2 + ;; + sun3 | sun3-*) + basic_machine=m68k-sun + ;; + sun4) + basic_machine=sparc-sun + ;; + sun386 | sun386i | roadrunner) + basic_machine=i386-sun + ;; + sv1) + basic_machine=sv1-cray + os=-unicos + ;; + symmetry) + basic_machine=i386-sequent + os=-dynix + ;; + t3e) + basic_machine=alphaev5-cray + os=-unicos + ;; + t90) + basic_machine=t90-cray + os=-unicos + ;; + tic54x | c54x*) + basic_machine=tic54x-unknown + os=-coff + ;; + tic55x | c55x*) + basic_machine=tic55x-unknown + os=-coff + ;; + tic6x | c6x*) + basic_machine=tic6x-unknown + os=-coff + ;; + tx39) + basic_machine=mipstx39-unknown + ;; + tx39el) + basic_machine=mipstx39el-unknown + ;; + toad1) + basic_machine=pdp10-xkl + os=-tops20 + ;; + tower | tower-32) + basic_machine=m68k-ncr + ;; + tpf) + basic_machine=s390x-ibm + os=-tpf + ;; + udi29k) + basic_machine=a29k-amd + os=-udi + ;; + ultra3) + basic_machine=a29k-nyu + os=-sym1 + ;; + v810 | necv810) + basic_machine=v810-nec + os=-none + ;; + vaxv) + basic_machine=vax-dec + os=-sysv + ;; + vms) + basic_machine=vax-dec + os=-vms + ;; + vpp*|vx|vx-*) + basic_machine=f301-fujitsu + ;; + vxworks960) + basic_machine=i960-wrs + os=-vxworks + ;; + vxworks68) + basic_machine=m68k-wrs + os=-vxworks + ;; + vxworks29k) + basic_machine=a29k-wrs + os=-vxworks + ;; + w65*) + basic_machine=w65-wdc + os=-none + ;; + w89k-*) + basic_machine=hppa1.1-winbond + os=-proelf + ;; + xps | xps100) + basic_machine=xps100-honeywell + ;; + ymp) + basic_machine=ymp-cray + os=-unicos + ;; + z8k-*-coff) + basic_machine=z8k-unknown + os=-sim + ;; + none) + basic_machine=none-none + os=-none + ;; + +# Here we handle the default manufacturer of certain CPU types. It is in +# some cases the only manufacturer, in others, it is the most popular. + w89k) + basic_machine=hppa1.1-winbond + ;; + op50n) + basic_machine=hppa1.1-oki + ;; + op60c) + basic_machine=hppa1.1-oki + ;; + romp) + basic_machine=romp-ibm + ;; + mmix) + basic_machine=mmix-knuth + ;; + rs6000) + basic_machine=rs6000-ibm + ;; + vax) + basic_machine=vax-dec + ;; + pdp10) + # there are many clones, so DEC is not a safe bet + basic_machine=pdp10-unknown + ;; + pdp11) + basic_machine=pdp11-dec + ;; + we32k) + basic_machine=we32k-att + ;; + sh3 | sh4 | sh[34]eb | sh[1234]le | sh[23]ele) + basic_machine=sh-unknown + ;; + sh64) + basic_machine=sh64-unknown + ;; + sparc | sparcv8 | sparcv9 | sparcv9b) + basic_machine=sparc-sun + ;; + cydra) + basic_machine=cydra-cydrome + ;; + orion) + basic_machine=orion-highlevel + ;; + orion105) + basic_machine=clipper-highlevel + ;; + mac | mpw | mac-mpw) + basic_machine=m68k-apple + ;; + pmac | pmac-mpw) + basic_machine=powerpc-apple + ;; + *-unknown) + # Make sure to match an already-canonicalized machine name. + ;; + *) + echo Invalid configuration \`$1\': machine \`$basic_machine\' not recognized 1>&2 + exit 1 + ;; +esac + +# Here we canonicalize certain aliases for manufacturers. +case $basic_machine in + *-digital*) + basic_machine=`echo $basic_machine | sed 's/digital.*/dec/'` + ;; + *-commodore*) + basic_machine=`echo $basic_machine | sed 's/commodore.*/cbm/'` + ;; + *) + ;; +esac + +# Decode manufacturer-specific aliases for certain operating systems. + +if [ x"$os" != x"" ] +then +case $os in + # First match some system type aliases + # that might get confused with valid system types. + # -solaris* is a basic system type, with this one exception. + -solaris1 | -solaris1.*) + os=`echo $os | sed -e 's|solaris1|sunos4|'` + ;; + -solaris) + os=-solaris2 + ;; + -svr4*) + os=-sysv4 + ;; + -unixware*) + os=-sysv4.2uw + ;; + -gnu/linux*) + os=`echo $os | sed -e 's|gnu/linux|linux-gnu|'` + ;; + # First accept the basic system types. + # The portable systems comes first. + # Each alternative MUST END IN A *, to match a version number. + # -sysv* is not here because it comes later, after sysvr4. + -gnu* | -bsd* | -mach* | -minix* | -genix* | -ultrix* | -irix* \ + | -*vms* | -sco* | -esix* | -isc* | -aix* | -sunos | -sunos[34]*\ + | -hpux* | -unos* | -osf* | -luna* | -dgux* | -solaris* | -sym* \ + | -amigaos* | -amigados* | -msdos* | -newsos* | -unicos* | -aof* \ + | -aos* \ + | -nindy* | -vxsim* | -vxworks* | -ebmon* | -hms* | -mvs* \ + | -clix* | -riscos* | -uniplus* | -iris* | -rtu* | -xenix* \ + | -hiux* | -386bsd* | -knetbsd* | -mirbsd* | -netbsd* | -openbsd* \ + | -ekkobsd* | -kfreebsd* | -freebsd* | -riscix* | -lynxos* \ + | -bosx* | -nextstep* | -cxux* | -aout* | -elf* | -oabi* \ + | -ptx* | -coff* | -ecoff* | -winnt* | -domain* | -vsta* \ + | -udi* | -eabi* | -lites* | -ieee* | -go32* | -aux* \ + | -chorusos* | -chorusrdb* \ + | -cygwin* | -pe* | -psos* | -moss* | -proelf* | -rtems* \ + | -mingw32* | -linux-gnu* | -linux-uclibc* | -uxpv* | -beos* | -mpeix* | -udk* \ + | -interix* | -uwin* | -mks* | -rhapsody* | -darwin* | -opened* \ + | -openstep* | -oskit* | -conix* | -pw32* | -nonstopux* \ + | -storm-chaos* | -tops10* | -tenex* | -tops20* | -its* \ + | -os2* | -vos* | -palmos* | -uclinux* | -nucleus* \ + | -morphos* | -superux* | -rtmk* | -rtmk-nova* | -windiss* \ + | -powermax* | -dnix* | -nx6 | -nx7 | -sei* | -dragonfly*) + # Remember, each alternative MUST END IN *, to match a version number. + ;; + -qnx*) + case $basic_machine in + x86-* | i*86-*) + ;; + *) + os=-nto$os + ;; + esac + ;; + -nto-qnx*) + ;; + -nto*) + os=`echo $os | sed -e 's|nto|nto-qnx|'` + ;; + -sim | -es1800* | -hms* | -xray | -os68k* | -none* | -v88r* \ + | -windows* | -osx | -abug | -netware* | -os9* | -beos* \ + | -macos* | -mpw* | -magic* | -mmixware* | -mon960* | -lnews*) + ;; + -mac*) + os=`echo $os | sed -e 's|mac|macos|'` + ;; + -linux-dietlibc) + os=-linux-dietlibc + ;; + -linux*) + os=`echo $os | sed -e 's|linux|linux-gnu|'` + ;; + -sunos5*) + os=`echo $os | sed -e 's|sunos5|solaris2|'` + ;; + -sunos6*) + os=`echo $os | sed -e 's|sunos6|solaris3|'` + ;; + -opened*) + os=-openedition + ;; + -os400*) + os=-os400 + ;; + -wince*) + os=-wince + ;; + -osfrose*) + os=-osfrose + ;; + -osf*) + os=-osf + ;; + -utek*) + os=-bsd + ;; + -dynix*) + os=-bsd + ;; + -acis*) + os=-aos + ;; + -atheos*) + os=-atheos + ;; + -syllable*) + os=-syllable + ;; + -386bsd) + os=-bsd + ;; + -ctix* | -uts*) + os=-sysv + ;; + -nova*) + os=-rtmk-nova + ;; + -ns2 ) + os=-nextstep2 + ;; + -nsk*) + os=-nsk + ;; + # Preserve the version number of sinix5. + -sinix5.*) + os=`echo $os | sed -e 's|sinix|sysv|'` + ;; + -sinix*) + os=-sysv4 + ;; + -tpf*) + os=-tpf + ;; + -triton*) + os=-sysv3 + ;; + -oss*) + os=-sysv3 + ;; + -svr4) + os=-sysv4 + ;; + -svr3) + os=-sysv3 + ;; + -sysvr4) + os=-sysv4 + ;; + # This must come after -sysvr4. + -sysv*) + ;; + -ose*) + os=-ose + ;; + -es1800*) + os=-ose + ;; + -xenix) + os=-xenix + ;; + -*mint | -mint[0-9]* | -*MiNT | -MiNT[0-9]*) + os=-mint + ;; + -aros*) + os=-aros + ;; + -kaos*) + os=-kaos + ;; + -none) + ;; + *) + # Get rid of the `-' at the beginning of $os. + os=`echo $os | sed 's/[^-]*-//'` + echo Invalid configuration \`$1\': system \`$os\' not recognized 1>&2 + exit 1 + ;; +esac +else + +# Here we handle the default operating systems that come with various machines. +# The value should be what the vendor currently ships out the door with their +# machine or put another way, the most popular os provided with the machine. + +# Note that if you're going to try to match "-MANUFACTURER" here (say, +# "-sun"), then you have to tell the case statement up towards the top +# that MANUFACTURER isn't an operating system. Otherwise, code above +# will signal an error saying that MANUFACTURER isn't an operating +# system, and we'll never get to this point. + +case $basic_machine in + *-acorn) + os=-riscix1.2 + ;; + arm*-rebel) + os=-linux + ;; + arm*-semi) + os=-aout + ;; + c4x-* | tic4x-*) + os=-coff + ;; + # This must come before the *-dec entry. + pdp10-*) + os=-tops20 + ;; + pdp11-*) + os=-none + ;; + *-dec | vax-*) + os=-ultrix4.2 + ;; + m68*-apollo) + os=-domain + ;; + i386-sun) + os=-sunos4.0.2 + ;; + m68000-sun) + os=-sunos3 + # This also exists in the configure program, but was not the + # default. + # os=-sunos4 + ;; + m68*-cisco) + os=-aout + ;; + mips*-cisco) + os=-elf + ;; + mips*-*) + os=-elf + ;; + or32-*) + os=-coff + ;; + *-tti) # must be before sparc entry or we get the wrong os. + os=-sysv3 + ;; + sparc-* | *-sun) + os=-sunos4.1.1 + ;; + *-be) + os=-beos + ;; + *-ibm) + os=-aix + ;; + *-knuth) + os=-mmixware + ;; + *-wec) + os=-proelf + ;; + *-winbond) + os=-proelf + ;; + *-oki) + os=-proelf + ;; + *-hp) + os=-hpux + ;; + *-hitachi) + os=-hiux + ;; + i860-* | *-att | *-ncr | *-altos | *-motorola | *-convergent) + os=-sysv + ;; + *-cbm) + os=-amigaos + ;; + *-dg) + os=-dgux + ;; + *-dolphin) + os=-sysv3 + ;; + m68k-ccur) + os=-rtu + ;; + m88k-omron*) + os=-luna + ;; + *-next ) + os=-nextstep + ;; + *-sequent) + os=-ptx + ;; + *-crds) + os=-unos + ;; + *-ns) + os=-genix + ;; + i370-*) + os=-mvs + ;; + *-next) + os=-nextstep3 + ;; + *-gould) + os=-sysv + ;; + *-highlevel) + os=-bsd + ;; + *-encore) + os=-bsd + ;; + *-sgi) + os=-irix + ;; + *-siemens) + os=-sysv4 + ;; + *-masscomp) + os=-rtu + ;; + f30[01]-fujitsu | f700-fujitsu) + os=-uxpv + ;; + *-rom68k) + os=-coff + ;; + *-*bug) + os=-coff + ;; + *-apple) + os=-macos + ;; + *-atari*) + os=-mint + ;; + *) + os=-none + ;; +esac +fi + +# Here we handle the case where we know the os, and the CPU type, but not the +# manufacturer. We pick the logical manufacturer. +vendor=unknown +case $basic_machine in + *-unknown) + case $os in + -riscix*) + vendor=acorn + ;; + -sunos*) + vendor=sun + ;; + -aix*) + vendor=ibm + ;; + -beos*) + vendor=be + ;; + -hpux*) + vendor=hp + ;; + -mpeix*) + vendor=hp + ;; + -hiux*) + vendor=hitachi + ;; + -unos*) + vendor=crds + ;; + -dgux*) + vendor=dg + ;; + -luna*) + vendor=omron + ;; + -genix*) + vendor=ns + ;; + -mvs* | -opened*) + vendor=ibm + ;; + -os400*) + vendor=ibm + ;; + -ptx*) + vendor=sequent + ;; + -tpf*) + vendor=ibm + ;; + -vxsim* | -vxworks* | -windiss*) + vendor=wrs + ;; + -aux*) + vendor=apple + ;; + -hms*) + vendor=hitachi + ;; + -mpw* | -macos*) + vendor=apple + ;; + -*mint | -mint[0-9]* | -*MiNT | -MiNT[0-9]*) + vendor=atari + ;; + -vos*) + vendor=stratus + ;; + esac + basic_machine=`echo $basic_machine | sed "s/unknown/$vendor/"` + ;; +esac + +echo $basic_machine$os +exit 0 + +# Local variables: +# eval: (add-hook 'write-file-hooks 'time-stamp) +# time-stamp-start: "timestamp='" +# time-stamp-format: "%:y-%02m-%02d" +# time-stamp-end: "'" +# End: diff --git a/libs/srtp/config_in.h b/libs/srtp/config_in.h new file mode 100644 index 0000000000..5951861e08 --- /dev/null +++ b/libs/srtp/config_in.h @@ -0,0 +1,170 @@ +/* config_in.h. Generated from configure.in by autoheader. */ + +/* Define if building for a CISC machine (e.g. Intel). */ +#undef CPU_CISC + +/* Define if building for a RISC machine (assume slow byte access). */ +#undef CPU_RISC + +/* Path to random device */ +#undef DEV_URANDOM + +/* Define to compile in dynamic debugging system. */ +#undef ENABLE_DEBUGGING + +/* Report errors to this file. */ +#undef ERR_REPORTING_FILE + +/* Define to use logging to stdout. */ +#undef ERR_REPORTING_STDOUT + +/* Define this to use ISMAcryp code. */ +#undef GENERIC_AESICM + +/* Define to 1 if you have the header file. */ +#undef HAVE_ARPA_INET_H + +/* Define to 1 if you have the header file. */ +#undef HAVE_BYTESWAP_H + +/* Define to 1 if you have the `inet_aton' function. */ +#undef HAVE_INET_ATON + +/* Define to 1 if the system has the type `int16_t'. */ +#undef HAVE_INT16_T + +/* Define to 1 if the system has the type `int32_t'. */ +#undef HAVE_INT32_T + +/* Define to 1 if the system has the type `int8_t'. */ +#undef HAVE_INT8_T + +/* Define to 1 if you have the header file. */ +#undef HAVE_INTTYPES_H + +/* Define to 1 if you have the `socket' library (-lsocket). */ +#undef HAVE_LIBSOCKET + +/* Define to 1 if you have the header file. */ +#undef HAVE_MACHINE_TYPES_H + +/* Define to 1 if you have the header file. */ +#undef HAVE_MEMORY_H + +/* Define to 1 if you have the header file. */ +#undef HAVE_NETINET_IN_H + +/* Define to 1 if you have the `socket' function. */ +#undef HAVE_SOCKET + +/* Define to 1 if you have the header file. */ +#undef HAVE_STDINT_H + +/* Define to 1 if you have the header file. */ +#undef HAVE_STDLIB_H + +/* Define to 1 if you have the header file. */ +#undef HAVE_STRINGS_H + +/* Define to 1 if you have the header file. */ +#undef HAVE_STRING_H + +/* Define to 1 if you have the header file. */ +#undef HAVE_SYSLOG_H + +/* Define to 1 if you have the header file. */ +#undef HAVE_SYS_INT_TYPES_H + +/* Define to 1 if you have the header file. */ +#undef HAVE_SYS_SOCKET_H + +/* Define to 1 if you have the header file. */ +#undef HAVE_SYS_STAT_H + +/* Define to 1 if you have the header file. */ +#undef HAVE_SYS_TYPES_H + +/* Define to 1 if you have the header file. */ +#undef HAVE_SYS_UIO_H + +/* Define to 1 if the system has the type `uint16_t'. */ +#undef HAVE_UINT16_T + +/* Define to 1 if the system has the type `uint32_t'. */ +#undef HAVE_UINT32_T + +/* Define to 1 if the system has the type `uint64_t'. */ +#undef HAVE_UINT64_T + +/* Define to 1 if the system has the type `uint8_t'. */ +#undef HAVE_UINT8_T + +/* Define to 1 if you have the header file. */ +#undef HAVE_UNISTD_H + +/* Define to 1 if you have the `usleep' function. */ +#undef HAVE_USLEEP + +/* Define to 1 if you have the header file. */ +#undef HAVE_WINDOWS_H + +/* Define to 1 if you have the header file. */ +#undef HAVE_WINSOCK2_H + +/* Define to use X86 inlined assembly code */ +#undef HAVE_X86 + +/* Define to the address where bug reports for this package should be sent. */ +#undef PACKAGE_BUGREPORT + +/* Define to the full name of this package. */ +#undef PACKAGE_NAME + +/* Define to the full name and version of this package. */ +#undef PACKAGE_STRING + +/* Define to the one symbol short name of this package. */ +#undef PACKAGE_TARNAME + +/* Define to the version of this package. */ +#undef PACKAGE_VERSION + +/* The size of a `unsigned long', as computed by sizeof. */ +#undef SIZEOF_UNSIGNED_LONG + +/* The size of a `unsigned long long', as computed by sizeof. */ +#undef SIZEOF_UNSIGNED_LONG_LONG + +/* Define to use GDOI. */ +#undef SRTP_GDOI + +/* Define to compile for kernel contexts. */ +#undef SRTP_KERNEL + +/* Define to compile for Linux kernel context. */ +#undef SRTP_KERNEL_LINUX + +/* Define to 1 if you have the ANSI C header files. */ +#undef STDC_HEADERS + +/* Write errors to this file */ +#undef USE_ERR_REPORTING_FILE + +/* Define to use syslog logging. */ +#undef USE_SYSLOG + +/* Define to 1 if your processor stores words with the most significant byte + first (like Motorola and SPARC, unlike Intel and VAX). */ +#undef WORDS_BIGENDIAN + +/* Define to empty if `const' does not conform to ANSI C. */ +#undef const + +/* Define to `__inline__' or `__inline' if that's what the C compiler + calls it, or to nothing if 'inline' is not supported under any name. */ +#ifndef __cplusplus +#undef inline +#endif + +/* Define to `unsigned' if does not define. */ +#undef size_t diff --git a/libs/srtp/configure b/libs/srtp/configure new file mode 100755 index 0000000000..aee93e1012 --- /dev/null +++ b/libs/srtp/configure @@ -0,0 +1,8609 @@ +#! /bin/sh +# Guess values for system-dependent variables and create Makefiles. +# Generated by GNU Autoconf 2.59. +# +# Copyright (C) 2003 Free Software Foundation, Inc. +# This configure script is free software; the Free Software Foundation +# gives unlimited permission to copy, distribute and modify it. +## --------------------- ## +## M4sh Initialization. ## +## --------------------- ## + +# Be Bourne compatible +if test -n "${ZSH_VERSION+set}" && (emulate sh) >/dev/null 2>&1; then + emulate sh + NULLCMD=: + # Zsh 3.x and 4.x performs word splitting on ${1+"$@"}, which + # is contrary to our usage. Disable this feature. + alias -g '${1+"$@"}'='"$@"' +elif test -n "${BASH_VERSION+set}" && (set -o posix) >/dev/null 2>&1; then + set -o posix +fi +DUALCASE=1; export DUALCASE # for MKS sh + +# Support unset when possible. +if ( (MAIL=60; unset MAIL) || exit) >/dev/null 2>&1; then + as_unset=unset +else + as_unset=false +fi + + +# Work around bugs in pre-3.0 UWIN ksh. +$as_unset ENV MAIL MAILPATH +PS1='$ ' +PS2='> ' +PS4='+ ' + +# NLS nuisances. +for as_var in \ + LANG LANGUAGE LC_ADDRESS LC_ALL LC_COLLATE LC_CTYPE LC_IDENTIFICATION \ + LC_MEASUREMENT LC_MESSAGES LC_MONETARY LC_NAME LC_NUMERIC LC_PAPER \ + LC_TELEPHONE LC_TIME +do + if (set +x; test -z "`(eval $as_var=C; export $as_var) 2>&1`"); then + eval $as_var=C; export $as_var + else + $as_unset $as_var + fi +done + +# Required to use basename. +if expr a : '\(a\)' >/dev/null 2>&1; then + as_expr=expr +else + as_expr=false +fi + +if (basename /) >/dev/null 2>&1 && test "X`basename / 2>&1`" = "X/"; then + as_basename=basename +else + as_basename=false +fi + + +# Name of the executable. +as_me=`$as_basename "$0" || +$as_expr X/"$0" : '.*/\([^/][^/]*\)/*$' \| \ + X"$0" : 'X\(//\)$' \| \ + X"$0" : 'X\(/\)$' \| \ + . : '\(.\)' 2>/dev/null || +echo X/"$0" | + sed '/^.*\/\([^/][^/]*\)\/*$/{ s//\1/; q; } + /^X\/\(\/\/\)$/{ s//\1/; q; } + /^X\/\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + + +# PATH needs CR, and LINENO needs CR and PATH. +# Avoid depending upon Character Ranges. +as_cr_letters='abcdefghijklmnopqrstuvwxyz' +as_cr_LETTERS='ABCDEFGHIJKLMNOPQRSTUVWXYZ' +as_cr_Letters=$as_cr_letters$as_cr_LETTERS +as_cr_digits='0123456789' +as_cr_alnum=$as_cr_Letters$as_cr_digits + +# The user is always right. +if test "${PATH_SEPARATOR+set}" != set; then + echo "#! /bin/sh" >conf$$.sh + echo "exit 0" >>conf$$.sh + chmod +x conf$$.sh + if (PATH="/nonexistent;."; conf$$.sh) >/dev/null 2>&1; then + PATH_SEPARATOR=';' + else + PATH_SEPARATOR=: + fi + rm -f conf$$.sh +fi + + + as_lineno_1=$LINENO + as_lineno_2=$LINENO + as_lineno_3=`(expr $as_lineno_1 + 1) 2>/dev/null` + test "x$as_lineno_1" != "x$as_lineno_2" && + test "x$as_lineno_3" = "x$as_lineno_2" || { + # Find who we are. Look in the path if we contain no path at all + # relative or not. + case $0 in + *[\\/]* ) as_myself=$0 ;; + *) as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + test -r "$as_dir/$0" && as_myself=$as_dir/$0 && break +done + + ;; + esac + # We did not find ourselves, most probably we were run as `sh COMMAND' + # in which case we are not to be found in the path. + if test "x$as_myself" = x; then + as_myself=$0 + fi + if test ! -f "$as_myself"; then + { echo "$as_me: error: cannot find myself; rerun with an absolute path" >&2 + { (exit 1); exit 1; }; } + fi + case $CONFIG_SHELL in + '') + as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in /bin$PATH_SEPARATOR/usr/bin$PATH_SEPARATOR$PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + for as_base in sh bash ksh sh5; do + case $as_dir in + /*) + if ("$as_dir/$as_base" -c ' + as_lineno_1=$LINENO + as_lineno_2=$LINENO + as_lineno_3=`(expr $as_lineno_1 + 1) 2>/dev/null` + test "x$as_lineno_1" != "x$as_lineno_2" && + test "x$as_lineno_3" = "x$as_lineno_2" ') 2>/dev/null; then + $as_unset BASH_ENV || test "${BASH_ENV+set}" != set || { BASH_ENV=; export BASH_ENV; } + $as_unset ENV || test "${ENV+set}" != set || { ENV=; export ENV; } + CONFIG_SHELL=$as_dir/$as_base + export CONFIG_SHELL + exec "$CONFIG_SHELL" "$0" ${1+"$@"} + fi;; + esac + done +done +;; + esac + + # Create $as_me.lineno as a copy of $as_myself, but with $LINENO + # uniformly replaced by the line number. The first 'sed' inserts a + # line-number line before each line; the second 'sed' does the real + # work. The second script uses 'N' to pair each line-number line + # with the numbered line, and appends trailing '-' during + # substitution so that $LINENO is not a special case at line end. + # (Raja R Harinath suggested sed '=', and Paul Eggert wrote the + # second 'sed' script. Blame Lee E. McMahon for sed's syntax. :-) + sed '=' <$as_myself | + sed ' + N + s,$,-, + : loop + s,^\(['$as_cr_digits']*\)\(.*\)[$]LINENO\([^'$as_cr_alnum'_]\),\1\2\1\3, + t loop + s,-$,, + s,^['$as_cr_digits']*\n,, + ' >$as_me.lineno && + chmod +x $as_me.lineno || + { echo "$as_me: error: cannot create $as_me.lineno; rerun with a POSIX shell" >&2 + { (exit 1); exit 1; }; } + + # Don't try to exec as it changes $[0], causing all sort of problems + # (the dirname of $[0] is not the place where we might find the + # original and so on. Autoconf is especially sensible to this). + . ./$as_me.lineno + # Exit status is that of the last command. + exit +} + + +case `echo "testing\c"; echo 1,2,3`,`echo -n testing; echo 1,2,3` in + *c*,-n*) ECHO_N= ECHO_C=' +' ECHO_T=' ' ;; + *c*,* ) ECHO_N=-n ECHO_C= ECHO_T= ;; + *) ECHO_N= ECHO_C='\c' ECHO_T= ;; +esac + +if expr a : '\(a\)' >/dev/null 2>&1; then + as_expr=expr +else + as_expr=false +fi + +rm -f conf$$ conf$$.exe conf$$.file +echo >conf$$.file +if ln -s conf$$.file conf$$ 2>/dev/null; then + # We could just check for DJGPP; but this test a) works b) is more generic + # and c) will remain valid once DJGPP supports symlinks (DJGPP 2.04). + if test -f conf$$.exe; then + # Don't use ln at all; we don't have any links + as_ln_s='cp -p' + else + as_ln_s='ln -s' + fi +elif ln conf$$.file conf$$ 2>/dev/null; then + as_ln_s=ln +else + as_ln_s='cp -p' +fi +rm -f conf$$ conf$$.exe conf$$.file + +if mkdir -p . 2>/dev/null; then + as_mkdir_p=: +else + test -d ./-p && rmdir ./-p + as_mkdir_p=false +fi + +as_executable_p="test -f" + +# Sed expression to map a string onto a valid CPP name. +as_tr_cpp="eval sed 'y%*$as_cr_letters%P$as_cr_LETTERS%;s%[^_$as_cr_alnum]%_%g'" + +# Sed expression to map a string onto a valid variable name. +as_tr_sh="eval sed 'y%*+%pp%;s%[^_$as_cr_alnum]%_%g'" + + +# IFS +# We need space, tab and new line, in precisely that order. +as_nl=' +' +IFS=" $as_nl" + +# CDPATH. +$as_unset CDPATH + + +# Name of the host. +# hostname on some systems (SVR3.2, Linux) returns a bogus exit status, +# so uname gets run too. +ac_hostname=`(hostname || uname -n) 2>/dev/null | sed 1q` + +exec 6>&1 + +# +# Initializations. +# +ac_default_prefix=/usr/local +ac_config_libobj_dir=. +cross_compiling=no +subdirs= +MFLAGS= +MAKEFLAGS= +SHELL=${CONFIG_SHELL-/bin/sh} + +# Maximum number of lines to put in a shell here document. +# This variable seems obsolete. It should probably be removed, and +# only ac_max_sed_lines should be used. +: ${ac_max_here_lines=38} + +# Identity of this package. +PACKAGE_NAME= +PACKAGE_TARNAME= +PACKAGE_VERSION= +PACKAGE_STRING= +PACKAGE_BUGREPORT= + +ac_unique_file="srtp" +# Factoring default headers for most tests. +ac_includes_default="\ +#include +#if HAVE_SYS_TYPES_H +# include +#endif +#if HAVE_SYS_STAT_H +# include +#endif +#if STDC_HEADERS +# include +# include +#else +# if HAVE_STDLIB_H +# include +# endif +#endif +#if HAVE_STRING_H +# if !STDC_HEADERS && HAVE_MEMORY_H +# include +# endif +# include +#endif +#if HAVE_STRINGS_H +# include +#endif +#if HAVE_INTTYPES_H +# include +#else +# if HAVE_STDINT_H +# include +# endif +#endif +#if HAVE_UNISTD_H +# include +#endif" + +ac_subst_vars='SHELL PATH_SEPARATOR PACKAGE_NAME PACKAGE_TARNAME PACKAGE_VERSION PACKAGE_STRING PACKAGE_BUGREPORT exec_prefix prefix program_transform_name bindir sbindir libexecdir datadir sysconfdir sharedstatedir localstatedir libdir includedir oldincludedir infodir mandir build_alias host_alias target_alias DEFS ECHO_C ECHO_N ECHO_T LIBS RANLIB ac_ct_RANLIB CC CFLAGS LDFLAGS CPPFLAGS ac_ct_CC EXEEXT OBJEXT INSTALL_PROGRAM INSTALL_SCRIPT INSTALL_DATA RNG_OBJS CPP EGREP build build_cpu build_vendor build_os host host_cpu host_vendor host_os EXE GDOI_OBJS LIBOBJS LTLIBOBJS' +ac_subst_files='' + +# Initialize some variables set by options. +ac_init_help= +ac_init_version=false +# The variables have the same names as the options, with +# dashes changed to underlines. +cache_file=/dev/null +exec_prefix=NONE +no_create= +no_recursion= +prefix=NONE +program_prefix=NONE +program_suffix=NONE +program_transform_name=s,x,x, +silent= +site= +srcdir= +verbose= +x_includes=NONE +x_libraries=NONE + +# Installation directory options. +# These are left unexpanded so users can "make install exec_prefix=/foo" +# and all the variables that are supposed to be based on exec_prefix +# by default will actually change. +# Use braces instead of parens because sh, perl, etc. also accept them. +bindir='${exec_prefix}/bin' +sbindir='${exec_prefix}/sbin' +libexecdir='${exec_prefix}/libexec' +datadir='${prefix}/share' +sysconfdir='${prefix}/etc' +sharedstatedir='${prefix}/com' +localstatedir='${prefix}/var' +libdir='${exec_prefix}/lib' +includedir='${prefix}/include' +oldincludedir='/usr/include' +infodir='${prefix}/info' +mandir='${prefix}/man' + +ac_prev= +for ac_option +do + # If the previous option needs an argument, assign it. + if test -n "$ac_prev"; then + eval "$ac_prev=\$ac_option" + ac_prev= + continue + fi + + ac_optarg=`expr "x$ac_option" : 'x[^=]*=\(.*\)'` + + # Accept the important Cygnus configure options, so we can diagnose typos. + + case $ac_option in + + -bindir | --bindir | --bindi | --bind | --bin | --bi) + ac_prev=bindir ;; + -bindir=* | --bindir=* | --bindi=* | --bind=* | --bin=* | --bi=*) + bindir=$ac_optarg ;; + + -build | --build | --buil | --bui | --bu) + ac_prev=build_alias ;; + -build=* | --build=* | --buil=* | --bui=* | --bu=*) + build_alias=$ac_optarg ;; + + -cache-file | --cache-file | --cache-fil | --cache-fi \ + | --cache-f | --cache- | --cache | --cach | --cac | --ca | --c) + ac_prev=cache_file ;; + -cache-file=* | --cache-file=* | --cache-fil=* | --cache-fi=* \ + | --cache-f=* | --cache-=* | --cache=* | --cach=* | --cac=* | --ca=* | --c=*) + cache_file=$ac_optarg ;; + + --config-cache | -C) + cache_file=config.cache ;; + + -datadir | --datadir | --datadi | --datad | --data | --dat | --da) + ac_prev=datadir ;; + -datadir=* | --datadir=* | --datadi=* | --datad=* | --data=* | --dat=* \ + | --da=*) + datadir=$ac_optarg ;; + + -disable-* | --disable-*) + ac_feature=`expr "x$ac_option" : 'x-*disable-\(.*\)'` + # Reject names that are not valid shell variable names. + expr "x$ac_feature" : ".*[^-_$as_cr_alnum]" >/dev/null && + { echo "$as_me: error: invalid feature name: $ac_feature" >&2 + { (exit 1); exit 1; }; } + ac_feature=`echo $ac_feature | sed 's/-/_/g'` + eval "enable_$ac_feature=no" ;; + + -enable-* | --enable-*) + ac_feature=`expr "x$ac_option" : 'x-*enable-\([^=]*\)'` + # Reject names that are not valid shell variable names. + expr "x$ac_feature" : ".*[^-_$as_cr_alnum]" >/dev/null && + { echo "$as_me: error: invalid feature name: $ac_feature" >&2 + { (exit 1); exit 1; }; } + ac_feature=`echo $ac_feature | sed 's/-/_/g'` + case $ac_option in + *=*) ac_optarg=`echo "$ac_optarg" | sed "s/'/'\\\\\\\\''/g"`;; + *) ac_optarg=yes ;; + esac + eval "enable_$ac_feature='$ac_optarg'" ;; + + -exec-prefix | --exec_prefix | --exec-prefix | --exec-prefi \ + | --exec-pref | --exec-pre | --exec-pr | --exec-p | --exec- \ + | --exec | --exe | --ex) + ac_prev=exec_prefix ;; + -exec-prefix=* | --exec_prefix=* | --exec-prefix=* | --exec-prefi=* \ + | --exec-pref=* | --exec-pre=* | --exec-pr=* | --exec-p=* | --exec-=* \ + | --exec=* | --exe=* | --ex=*) + exec_prefix=$ac_optarg ;; + + -gas | --gas | --ga | --g) + # Obsolete; use --with-gas. + with_gas=yes ;; + + -help | --help | --hel | --he | -h) + ac_init_help=long ;; + -help=r* | --help=r* | --hel=r* | --he=r* | -hr*) + ac_init_help=recursive ;; + -help=s* | --help=s* | --hel=s* | --he=s* | -hs*) + ac_init_help=short ;; + + -host | --host | --hos | --ho) + ac_prev=host_alias ;; + -host=* | --host=* | --hos=* | --ho=*) + host_alias=$ac_optarg ;; + + -includedir | --includedir | --includedi | --included | --include \ + | --includ | --inclu | --incl | --inc) + ac_prev=includedir ;; + -includedir=* | --includedir=* | --includedi=* | --included=* | --include=* \ + | --includ=* | --inclu=* | --incl=* | --inc=*) + includedir=$ac_optarg ;; + + -infodir | --infodir | --infodi | --infod | --info | --inf) + ac_prev=infodir ;; + -infodir=* | --infodir=* | --infodi=* | --infod=* | --info=* | --inf=*) + infodir=$ac_optarg ;; + + -libdir | --libdir | --libdi | --libd) + ac_prev=libdir ;; + -libdir=* | --libdir=* | --libdi=* | --libd=*) + libdir=$ac_optarg ;; + + -libexecdir | --libexecdir | --libexecdi | --libexecd | --libexec \ + | --libexe | --libex | --libe) + ac_prev=libexecdir ;; + -libexecdir=* | --libexecdir=* | --libexecdi=* | --libexecd=* | --libexec=* \ + | --libexe=* | --libex=* | --libe=*) + libexecdir=$ac_optarg ;; + + -localstatedir | --localstatedir | --localstatedi | --localstated \ + | --localstate | --localstat | --localsta | --localst \ + | --locals | --local | --loca | --loc | --lo) + ac_prev=localstatedir ;; + -localstatedir=* | --localstatedir=* | --localstatedi=* | --localstated=* \ + | --localstate=* | --localstat=* | --localsta=* | --localst=* \ + | --locals=* | --local=* | --loca=* | --loc=* | --lo=*) + localstatedir=$ac_optarg ;; + + -mandir | --mandir | --mandi | --mand | --man | --ma | --m) + ac_prev=mandir ;; + -mandir=* | --mandir=* | --mandi=* | --mand=* | --man=* | --ma=* | --m=*) + mandir=$ac_optarg ;; + + -nfp | --nfp | --nf) + # Obsolete; use --without-fp. + with_fp=no ;; + + -no-create | --no-create | --no-creat | --no-crea | --no-cre \ + | --no-cr | --no-c | -n) + no_create=yes ;; + + -no-recursion | --no-recursion | --no-recursio | --no-recursi \ + | --no-recurs | --no-recur | --no-recu | --no-rec | --no-re | --no-r) + no_recursion=yes ;; + + -oldincludedir | --oldincludedir | --oldincludedi | --oldincluded \ + | --oldinclude | --oldinclud | --oldinclu | --oldincl | --oldinc \ + | --oldin | --oldi | --old | --ol | --o) + ac_prev=oldincludedir ;; + -oldincludedir=* | --oldincludedir=* | --oldincludedi=* | --oldincluded=* \ + | --oldinclude=* | --oldinclud=* | --oldinclu=* | --oldincl=* | --oldinc=* \ + | --oldin=* | --oldi=* | --old=* | --ol=* | --o=*) + oldincludedir=$ac_optarg ;; + + -prefix | --prefix | --prefi | --pref | --pre | --pr | --p) + ac_prev=prefix ;; + -prefix=* | --prefix=* | --prefi=* | --pref=* | --pre=* | --pr=* | --p=*) + prefix=$ac_optarg ;; + + -program-prefix | --program-prefix | --program-prefi | --program-pref \ + | --program-pre | --program-pr | --program-p) + ac_prev=program_prefix ;; + -program-prefix=* | --program-prefix=* | --program-prefi=* \ + | --program-pref=* | --program-pre=* | --program-pr=* | --program-p=*) + program_prefix=$ac_optarg ;; + + -program-suffix | --program-suffix | --program-suffi | --program-suff \ + | --program-suf | --program-su | --program-s) + ac_prev=program_suffix ;; + -program-suffix=* | --program-suffix=* | --program-suffi=* \ + | --program-suff=* | --program-suf=* | --program-su=* | --program-s=*) + program_suffix=$ac_optarg ;; + + -program-transform-name | --program-transform-name \ + | --program-transform-nam | --program-transform-na \ + | --program-transform-n | --program-transform- \ + | --program-transform | --program-transfor \ + | --program-transfo | --program-transf \ + | --program-trans | --program-tran \ + | --progr-tra | --program-tr | --program-t) + ac_prev=program_transform_name ;; + -program-transform-name=* | --program-transform-name=* \ + | --program-transform-nam=* | --program-transform-na=* \ + | --program-transform-n=* | --program-transform-=* \ + | --program-transform=* | --program-transfor=* \ + | --program-transfo=* | --program-transf=* \ + | --program-trans=* | --program-tran=* \ + | --progr-tra=* | --program-tr=* | --program-t=*) + program_transform_name=$ac_optarg ;; + + -q | -quiet | --quiet | --quie | --qui | --qu | --q \ + | -silent | --silent | --silen | --sile | --sil) + silent=yes ;; + + -sbindir | --sbindir | --sbindi | --sbind | --sbin | --sbi | --sb) + ac_prev=sbindir ;; + -sbindir=* | --sbindir=* | --sbindi=* | --sbind=* | --sbin=* \ + | --sbi=* | --sb=*) + sbindir=$ac_optarg ;; + + -sharedstatedir | --sharedstatedir | --sharedstatedi \ + | --sharedstated | --sharedstate | --sharedstat | --sharedsta \ + | --sharedst | --shareds | --shared | --share | --shar \ + | --sha | --sh) + ac_prev=sharedstatedir ;; + -sharedstatedir=* | --sharedstatedir=* | --sharedstatedi=* \ + | --sharedstated=* | --sharedstate=* | --sharedstat=* | --sharedsta=* \ + | --sharedst=* | --shareds=* | --shared=* | --share=* | --shar=* \ + | --sha=* | --sh=*) + sharedstatedir=$ac_optarg ;; + + -site | --site | --sit) + ac_prev=site ;; + -site=* | --site=* | --sit=*) + site=$ac_optarg ;; + + -srcdir | --srcdir | --srcdi | --srcd | --src | --sr) + ac_prev=srcdir ;; + -srcdir=* | --srcdir=* | --srcdi=* | --srcd=* | --src=* | --sr=*) + srcdir=$ac_optarg ;; + + -sysconfdir | --sysconfdir | --sysconfdi | --sysconfd | --sysconf \ + | --syscon | --sysco | --sysc | --sys | --sy) + ac_prev=sysconfdir ;; + -sysconfdir=* | --sysconfdir=* | --sysconfdi=* | --sysconfd=* | --sysconf=* \ + | --syscon=* | --sysco=* | --sysc=* | --sys=* | --sy=*) + sysconfdir=$ac_optarg ;; + + -target | --target | --targe | --targ | --tar | --ta | --t) + ac_prev=target_alias ;; + -target=* | --target=* | --targe=* | --targ=* | --tar=* | --ta=* | --t=*) + target_alias=$ac_optarg ;; + + -v | -verbose | --verbose | --verbos | --verbo | --verb) + verbose=yes ;; + + -version | --version | --versio | --versi | --vers | -V) + ac_init_version=: ;; + + -with-* | --with-*) + ac_package=`expr "x$ac_option" : 'x-*with-\([^=]*\)'` + # Reject names that are not valid shell variable names. + expr "x$ac_package" : ".*[^-_$as_cr_alnum]" >/dev/null && + { echo "$as_me: error: invalid package name: $ac_package" >&2 + { (exit 1); exit 1; }; } + ac_package=`echo $ac_package| sed 's/-/_/g'` + case $ac_option in + *=*) ac_optarg=`echo "$ac_optarg" | sed "s/'/'\\\\\\\\''/g"`;; + *) ac_optarg=yes ;; + esac + eval "with_$ac_package='$ac_optarg'" ;; + + -without-* | --without-*) + ac_package=`expr "x$ac_option" : 'x-*without-\(.*\)'` + # Reject names that are not valid shell variable names. + expr "x$ac_package" : ".*[^-_$as_cr_alnum]" >/dev/null && + { echo "$as_me: error: invalid package name: $ac_package" >&2 + { (exit 1); exit 1; }; } + ac_package=`echo $ac_package | sed 's/-/_/g'` + eval "with_$ac_package=no" ;; + + --x) + # Obsolete; use --with-x. + with_x=yes ;; + + -x-includes | --x-includes | --x-include | --x-includ | --x-inclu \ + | --x-incl | --x-inc | --x-in | --x-i) + ac_prev=x_includes ;; + -x-includes=* | --x-includes=* | --x-include=* | --x-includ=* | --x-inclu=* \ + | --x-incl=* | --x-inc=* | --x-in=* | --x-i=*) + x_includes=$ac_optarg ;; + + -x-libraries | --x-libraries | --x-librarie | --x-librari \ + | --x-librar | --x-libra | --x-libr | --x-lib | --x-li | --x-l) + ac_prev=x_libraries ;; + -x-libraries=* | --x-libraries=* | --x-librarie=* | --x-librari=* \ + | --x-librar=* | --x-libra=* | --x-libr=* | --x-lib=* | --x-li=* | --x-l=*) + x_libraries=$ac_optarg ;; + + -*) { echo "$as_me: error: unrecognized option: $ac_option +Try \`$0 --help' for more information." >&2 + { (exit 1); exit 1; }; } + ;; + + *=*) + ac_envvar=`expr "x$ac_option" : 'x\([^=]*\)='` + # Reject names that are not valid shell variable names. + expr "x$ac_envvar" : ".*[^_$as_cr_alnum]" >/dev/null && + { echo "$as_me: error: invalid variable name: $ac_envvar" >&2 + { (exit 1); exit 1; }; } + ac_optarg=`echo "$ac_optarg" | sed "s/'/'\\\\\\\\''/g"` + eval "$ac_envvar='$ac_optarg'" + export $ac_envvar ;; + + *) + # FIXME: should be removed in autoconf 3.0. + echo "$as_me: WARNING: you should use --build, --host, --target" >&2 + expr "x$ac_option" : ".*[^-._$as_cr_alnum]" >/dev/null && + echo "$as_me: WARNING: invalid host type: $ac_option" >&2 + : ${build_alias=$ac_option} ${host_alias=$ac_option} ${target_alias=$ac_option} + ;; + + esac +done + +if test -n "$ac_prev"; then + ac_option=--`echo $ac_prev | sed 's/_/-/g'` + { echo "$as_me: error: missing argument to $ac_option" >&2 + { (exit 1); exit 1; }; } +fi + +# Be sure to have absolute paths. +for ac_var in exec_prefix prefix +do + eval ac_val=$`echo $ac_var` + case $ac_val in + [\\/$]* | ?:[\\/]* | NONE | '' ) ;; + *) { echo "$as_me: error: expected an absolute directory name for --$ac_var: $ac_val" >&2 + { (exit 1); exit 1; }; };; + esac +done + +# Be sure to have absolute paths. +for ac_var in bindir sbindir libexecdir datadir sysconfdir sharedstatedir \ + localstatedir libdir includedir oldincludedir infodir mandir +do + eval ac_val=$`echo $ac_var` + case $ac_val in + [\\/$]* | ?:[\\/]* ) ;; + *) { echo "$as_me: error: expected an absolute directory name for --$ac_var: $ac_val" >&2 + { (exit 1); exit 1; }; };; + esac +done + +# There might be people who depend on the old broken behavior: `$host' +# used to hold the argument of --host etc. +# FIXME: To remove some day. +build=$build_alias +host=$host_alias +target=$target_alias + +# FIXME: To remove some day. +if test "x$host_alias" != x; then + if test "x$build_alias" = x; then + cross_compiling=maybe + echo "$as_me: WARNING: If you wanted to set the --build type, don't use --host. + If a cross compiler is detected then cross compile mode will be used." >&2 + elif test "x$build_alias" != "x$host_alias"; then + cross_compiling=yes + fi +fi + +ac_tool_prefix= +test -n "$host_alias" && ac_tool_prefix=$host_alias- + +test "$silent" = yes && exec 6>/dev/null + + +# Find the source files, if location was not specified. +if test -z "$srcdir"; then + ac_srcdir_defaulted=yes + # Try the directory containing this script, then its parent. + ac_confdir=`(dirname "$0") 2>/dev/null || +$as_expr X"$0" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$0" : 'X\(//\)[^/]' \| \ + X"$0" : 'X\(//\)$' \| \ + X"$0" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$0" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + srcdir=$ac_confdir + if test ! -r $srcdir/$ac_unique_file; then + srcdir=.. + fi +else + ac_srcdir_defaulted=no +fi +if test ! -r $srcdir/$ac_unique_file; then + if test "$ac_srcdir_defaulted" = yes; then + { echo "$as_me: error: cannot find sources ($ac_unique_file) in $ac_confdir or .." >&2 + { (exit 1); exit 1; }; } + else + { echo "$as_me: error: cannot find sources ($ac_unique_file) in $srcdir" >&2 + { (exit 1); exit 1; }; } + fi +fi +(cd $srcdir && test -r ./$ac_unique_file) 2>/dev/null || + { echo "$as_me: error: sources are in $srcdir, but \`cd $srcdir' does not work" >&2 + { (exit 1); exit 1; }; } +srcdir=`echo "$srcdir" | sed 's%\([^\\/]\)[\\/]*$%\1%'` +ac_env_build_alias_set=${build_alias+set} +ac_env_build_alias_value=$build_alias +ac_cv_env_build_alias_set=${build_alias+set} +ac_cv_env_build_alias_value=$build_alias +ac_env_host_alias_set=${host_alias+set} +ac_env_host_alias_value=$host_alias +ac_cv_env_host_alias_set=${host_alias+set} +ac_cv_env_host_alias_value=$host_alias +ac_env_target_alias_set=${target_alias+set} +ac_env_target_alias_value=$target_alias +ac_cv_env_target_alias_set=${target_alias+set} +ac_cv_env_target_alias_value=$target_alias +ac_env_CC_set=${CC+set} +ac_env_CC_value=$CC +ac_cv_env_CC_set=${CC+set} +ac_cv_env_CC_value=$CC +ac_env_CFLAGS_set=${CFLAGS+set} +ac_env_CFLAGS_value=$CFLAGS +ac_cv_env_CFLAGS_set=${CFLAGS+set} +ac_cv_env_CFLAGS_value=$CFLAGS +ac_env_LDFLAGS_set=${LDFLAGS+set} +ac_env_LDFLAGS_value=$LDFLAGS +ac_cv_env_LDFLAGS_set=${LDFLAGS+set} +ac_cv_env_LDFLAGS_value=$LDFLAGS +ac_env_CPPFLAGS_set=${CPPFLAGS+set} +ac_env_CPPFLAGS_value=$CPPFLAGS +ac_cv_env_CPPFLAGS_set=${CPPFLAGS+set} +ac_cv_env_CPPFLAGS_value=$CPPFLAGS +ac_env_CPP_set=${CPP+set} +ac_env_CPP_value=$CPP +ac_cv_env_CPP_set=${CPP+set} +ac_cv_env_CPP_value=$CPP + +# +# Report the --help message. +# +if test "$ac_init_help" = "long"; then + # Omit some internal or obsolete options to make the list less imposing. + # This message is too long to be a string in the A/UX 3.1 sh. + cat <<_ACEOF +\`configure' configures this package to adapt to many kinds of systems. + +Usage: $0 [OPTION]... [VAR=VALUE]... + +To assign environment variables (e.g., CC, CFLAGS...), specify them as +VAR=VALUE. See below for descriptions of some of the useful variables. + +Defaults for the options are specified in brackets. + +Configuration: + -h, --help display this help and exit + --help=short display options specific to this package + --help=recursive display the short help of all the included packages + -V, --version display version information and exit + -q, --quiet, --silent do not print \`checking...' messages + --cache-file=FILE cache test results in FILE [disabled] + -C, --config-cache alias for \`--cache-file=config.cache' + -n, --no-create do not create output files + --srcdir=DIR find the sources in DIR [configure dir or \`..'] + +_ACEOF + + cat <<_ACEOF +Installation directories: + --prefix=PREFIX install architecture-independent files in PREFIX + [$ac_default_prefix] + --exec-prefix=EPREFIX install architecture-dependent files in EPREFIX + [PREFIX] + +By default, \`make install' will install all the files in +\`$ac_default_prefix/bin', \`$ac_default_prefix/lib' etc. You can specify +an installation prefix other than \`$ac_default_prefix' using \`--prefix', +for instance \`--prefix=\$HOME'. + +For better control, use the options below. + +Fine tuning of the installation directories: + --bindir=DIR user executables [EPREFIX/bin] + --sbindir=DIR system admin executables [EPREFIX/sbin] + --libexecdir=DIR program executables [EPREFIX/libexec] + --datadir=DIR read-only architecture-independent data [PREFIX/share] + --sysconfdir=DIR read-only single-machine data [PREFIX/etc] + --sharedstatedir=DIR modifiable architecture-independent data [PREFIX/com] + --localstatedir=DIR modifiable single-machine data [PREFIX/var] + --libdir=DIR object code libraries [EPREFIX/lib] + --includedir=DIR C header files [PREFIX/include] + --oldincludedir=DIR C header files for non-gcc [/usr/include] + --infodir=DIR info documentation [PREFIX/info] + --mandir=DIR man documentation [PREFIX/man] +_ACEOF + + cat <<\_ACEOF + +System types: + --build=BUILD configure for building on BUILD [guessed] + --host=HOST cross-compile to build programs to run on HOST [BUILD] +_ACEOF +fi + +if test -n "$ac_init_help"; then + + cat <<\_ACEOF + +Optional Features: + --disable-FEATURE do not include FEATURE (same as --enable-FEATURE=no) + --enable-FEATURE[=ARG] include FEATURE [ARG=yes] + --enable-pic build with PIC + --enable-kernel-linux build library to run in Linux kernel context + --disable-debug do not compile in dynamic debugging system + --enable-generic-aesicm compile in changes for ISMAcryp + --enable-syslog use syslog for error reporting + --disable-stdout don't use stdout for error reporting + --enable-console use /dev/console for error reporting + --enable-gdoi enable GDOI key management + +Some influential environment variables: + CC C compiler command + CFLAGS C compiler flags + LDFLAGS linker flags, e.g. -L if you have libraries in a + nonstandard directory + CPPFLAGS C/C++ preprocessor flags, e.g. -I if you have + headers in a nonstandard directory + CPP C preprocessor + +Use these variables to override the choices made by `configure' or to help +it to find libraries and programs with nonstandard names/locations. + +_ACEOF +fi + +if test "$ac_init_help" = "recursive"; then + # If there are subdirs, report their specific --help. + ac_popdir=`pwd` + for ac_dir in : $ac_subdirs_all; do test "x$ac_dir" = x: && continue + test -d $ac_dir || continue + ac_builddir=. + +if test "$ac_dir" != .; then + ac_dir_suffix=/`echo "$ac_dir" | sed 's,^\.[\\/],,'` + # A "../" for each directory in $ac_dir_suffix. + ac_top_builddir=`echo "$ac_dir_suffix" | sed 's,/[^\\/]*,../,g'` +else + ac_dir_suffix= ac_top_builddir= +fi + +case $srcdir in + .) # No --srcdir option. We are building in place. + ac_srcdir=. + if test -z "$ac_top_builddir"; then + ac_top_srcdir=. + else + ac_top_srcdir=`echo $ac_top_builddir | sed 's,/$,,'` + fi ;; + [\\/]* | ?:[\\/]* ) # Absolute path. + ac_srcdir=$srcdir$ac_dir_suffix; + ac_top_srcdir=$srcdir ;; + *) # Relative path. + ac_srcdir=$ac_top_builddir$srcdir$ac_dir_suffix + ac_top_srcdir=$ac_top_builddir$srcdir ;; +esac + +# Do not use `cd foo && pwd` to compute absolute paths, because +# the directories may not exist. +case `pwd` in +.) ac_abs_builddir="$ac_dir";; +*) + case "$ac_dir" in + .) ac_abs_builddir=`pwd`;; + [\\/]* | ?:[\\/]* ) ac_abs_builddir="$ac_dir";; + *) ac_abs_builddir=`pwd`/"$ac_dir";; + esac;; +esac +case $ac_abs_builddir in +.) ac_abs_top_builddir=${ac_top_builddir}.;; +*) + case ${ac_top_builddir}. in + .) ac_abs_top_builddir=$ac_abs_builddir;; + [\\/]* | ?:[\\/]* ) ac_abs_top_builddir=${ac_top_builddir}.;; + *) ac_abs_top_builddir=$ac_abs_builddir/${ac_top_builddir}.;; + esac;; +esac +case $ac_abs_builddir in +.) ac_abs_srcdir=$ac_srcdir;; +*) + case $ac_srcdir in + .) ac_abs_srcdir=$ac_abs_builddir;; + [\\/]* | ?:[\\/]* ) ac_abs_srcdir=$ac_srcdir;; + *) ac_abs_srcdir=$ac_abs_builddir/$ac_srcdir;; + esac;; +esac +case $ac_abs_builddir in +.) ac_abs_top_srcdir=$ac_top_srcdir;; +*) + case $ac_top_srcdir in + .) ac_abs_top_srcdir=$ac_abs_builddir;; + [\\/]* | ?:[\\/]* ) ac_abs_top_srcdir=$ac_top_srcdir;; + *) ac_abs_top_srcdir=$ac_abs_builddir/$ac_top_srcdir;; + esac;; +esac + + cd $ac_dir + # Check for guested configure; otherwise get Cygnus style configure. + if test -f $ac_srcdir/configure.gnu; then + echo + $SHELL $ac_srcdir/configure.gnu --help=recursive + elif test -f $ac_srcdir/configure; then + echo + $SHELL $ac_srcdir/configure --help=recursive + elif test -f $ac_srcdir/configure.ac || + test -f $ac_srcdir/configure.in; then + echo + $ac_configure --help + else + echo "$as_me: WARNING: no configuration information is in $ac_dir" >&2 + fi + cd $ac_popdir + done +fi + +test -n "$ac_init_help" && exit 0 +if $ac_init_version; then + cat <<\_ACEOF + +Copyright (C) 2003 Free Software Foundation, Inc. +This configure script is free software; the Free Software Foundation +gives unlimited permission to copy, distribute and modify it. +_ACEOF + exit 0 +fi +exec 5>config.log +cat >&5 <<_ACEOF +This file contains any messages produced by compilers while +running configure, to aid debugging if configure makes a mistake. + +It was created by $as_me, which was +generated by GNU Autoconf 2.59. Invocation command line was + + $ $0 $@ + +_ACEOF +{ +cat <<_ASUNAME +## --------- ## +## Platform. ## +## --------- ## + +hostname = `(hostname || uname -n) 2>/dev/null | sed 1q` +uname -m = `(uname -m) 2>/dev/null || echo unknown` +uname -r = `(uname -r) 2>/dev/null || echo unknown` +uname -s = `(uname -s) 2>/dev/null || echo unknown` +uname -v = `(uname -v) 2>/dev/null || echo unknown` + +/usr/bin/uname -p = `(/usr/bin/uname -p) 2>/dev/null || echo unknown` +/bin/uname -X = `(/bin/uname -X) 2>/dev/null || echo unknown` + +/bin/arch = `(/bin/arch) 2>/dev/null || echo unknown` +/usr/bin/arch -k = `(/usr/bin/arch -k) 2>/dev/null || echo unknown` +/usr/convex/getsysinfo = `(/usr/convex/getsysinfo) 2>/dev/null || echo unknown` +hostinfo = `(hostinfo) 2>/dev/null || echo unknown` +/bin/machine = `(/bin/machine) 2>/dev/null || echo unknown` +/usr/bin/oslevel = `(/usr/bin/oslevel) 2>/dev/null || echo unknown` +/bin/universe = `(/bin/universe) 2>/dev/null || echo unknown` + +_ASUNAME + +as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + echo "PATH: $as_dir" +done + +} >&5 + +cat >&5 <<_ACEOF + + +## ----------- ## +## Core tests. ## +## ----------- ## + +_ACEOF + + +# Keep a trace of the command line. +# Strip out --no-create and --no-recursion so they do not pile up. +# Strip out --silent because we don't want to record it for future runs. +# Also quote any args containing shell meta-characters. +# Make two passes to allow for proper duplicate-argument suppression. +ac_configure_args= +ac_configure_args0= +ac_configure_args1= +ac_sep= +ac_must_keep_next=false +for ac_pass in 1 2 +do + for ac_arg + do + case $ac_arg in + -no-create | --no-c* | -n | -no-recursion | --no-r*) continue ;; + -q | -quiet | --quiet | --quie | --qui | --qu | --q \ + | -silent | --silent | --silen | --sile | --sil) + continue ;; + *" "*|*" "*|*[\[\]\~\#\$\^\&\*\(\)\{\}\\\|\;\<\>\?\"\']*) + ac_arg=`echo "$ac_arg" | sed "s/'/'\\\\\\\\''/g"` ;; + esac + case $ac_pass in + 1) ac_configure_args0="$ac_configure_args0 '$ac_arg'" ;; + 2) + ac_configure_args1="$ac_configure_args1 '$ac_arg'" + if test $ac_must_keep_next = true; then + ac_must_keep_next=false # Got value, back to normal. + else + case $ac_arg in + *=* | --config-cache | -C | -disable-* | --disable-* \ + | -enable-* | --enable-* | -gas | --g* | -nfp | --nf* \ + | -q | -quiet | --q* | -silent | --sil* | -v | -verb* \ + | -with-* | --with-* | -without-* | --without-* | --x) + case "$ac_configure_args0 " in + "$ac_configure_args1"*" '$ac_arg' "* ) continue ;; + esac + ;; + -* ) ac_must_keep_next=true ;; + esac + fi + ac_configure_args="$ac_configure_args$ac_sep'$ac_arg'" + # Get rid of the leading space. + ac_sep=" " + ;; + esac + done +done +$as_unset ac_configure_args0 || test "${ac_configure_args0+set}" != set || { ac_configure_args0=; export ac_configure_args0; } +$as_unset ac_configure_args1 || test "${ac_configure_args1+set}" != set || { ac_configure_args1=; export ac_configure_args1; } + +# When interrupted or exit'd, cleanup temporary files, and complete +# config.log. We remove comments because anyway the quotes in there +# would cause problems or look ugly. +# WARNING: Be sure not to use single quotes in there, as some shells, +# such as our DU 5.0 friend, will then `close' the trap. +trap 'exit_status=$? + # Save into config.log some information that might help in debugging. + { + echo + + cat <<\_ASBOX +## ---------------- ## +## Cache variables. ## +## ---------------- ## +_ASBOX + echo + # The following way of writing the cache mishandles newlines in values, +{ + (set) 2>&1 | + case `(ac_space='"'"' '"'"'; set | grep ac_space) 2>&1` in + *ac_space=\ *) + sed -n \ + "s/'"'"'/'"'"'\\\\'"'"''"'"'/g; + s/^\\([_$as_cr_alnum]*_cv_[_$as_cr_alnum]*\\)=\\(.*\\)/\\1='"'"'\\2'"'"'/p" + ;; + *) + sed -n \ + "s/^\\([_$as_cr_alnum]*_cv_[_$as_cr_alnum]*\\)=\\(.*\\)/\\1=\\2/p" + ;; + esac; +} + echo + + cat <<\_ASBOX +## ----------------- ## +## Output variables. ## +## ----------------- ## +_ASBOX + echo + for ac_var in $ac_subst_vars + do + eval ac_val=$`echo $ac_var` + echo "$ac_var='"'"'$ac_val'"'"'" + done | sort + echo + + if test -n "$ac_subst_files"; then + cat <<\_ASBOX +## ------------- ## +## Output files. ## +## ------------- ## +_ASBOX + echo + for ac_var in $ac_subst_files + do + eval ac_val=$`echo $ac_var` + echo "$ac_var='"'"'$ac_val'"'"'" + done | sort + echo + fi + + if test -s confdefs.h; then + cat <<\_ASBOX +## ----------- ## +## confdefs.h. ## +## ----------- ## +_ASBOX + echo + sed "/^$/d" confdefs.h | sort + echo + fi + test "$ac_signal" != 0 && + echo "$as_me: caught signal $ac_signal" + echo "$as_me: exit $exit_status" + } >&5 + rm -f core *.core && + rm -rf conftest* confdefs* conf$$* $ac_clean_files && + exit $exit_status + ' 0 +for ac_signal in 1 2 13 15; do + trap 'ac_signal='$ac_signal'; { (exit 1); exit 1; }' $ac_signal +done +ac_signal=0 + +# confdefs.h avoids OS command line length limits that DEFS can exceed. +rm -rf conftest* confdefs.h +# AIX cpp loses on an empty file, so make sure it contains at least a newline. +echo >confdefs.h + +# Predefined preprocessor variables. + +cat >>confdefs.h <<_ACEOF +#define PACKAGE_NAME "$PACKAGE_NAME" +_ACEOF + + +cat >>confdefs.h <<_ACEOF +#define PACKAGE_TARNAME "$PACKAGE_TARNAME" +_ACEOF + + +cat >>confdefs.h <<_ACEOF +#define PACKAGE_VERSION "$PACKAGE_VERSION" +_ACEOF + + +cat >>confdefs.h <<_ACEOF +#define PACKAGE_STRING "$PACKAGE_STRING" +_ACEOF + + +cat >>confdefs.h <<_ACEOF +#define PACKAGE_BUGREPORT "$PACKAGE_BUGREPORT" +_ACEOF + + +# Let the site file select an alternate cache file if it wants to. +# Prefer explicitly selected file to automatically selected ones. +if test -z "$CONFIG_SITE"; then + if test "x$prefix" != xNONE; then + CONFIG_SITE="$prefix/share/config.site $prefix/etc/config.site" + else + CONFIG_SITE="$ac_default_prefix/share/config.site $ac_default_prefix/etc/config.site" + fi +fi +for ac_site_file in $CONFIG_SITE; do + if test -r "$ac_site_file"; then + { echo "$as_me:$LINENO: loading site script $ac_site_file" >&5 +echo "$as_me: loading site script $ac_site_file" >&6;} + sed 's/^/| /' "$ac_site_file" >&5 + . "$ac_site_file" + fi +done + +if test -r "$cache_file"; then + # Some versions of bash will fail to source /dev/null (special + # files actually), so we avoid doing that. + if test -f "$cache_file"; then + { echo "$as_me:$LINENO: loading cache $cache_file" >&5 +echo "$as_me: loading cache $cache_file" >&6;} + case $cache_file in + [\\/]* | ?:[\\/]* ) . $cache_file;; + *) . ./$cache_file;; + esac + fi +else + { echo "$as_me:$LINENO: creating cache $cache_file" >&5 +echo "$as_me: creating cache $cache_file" >&6;} + >$cache_file +fi + +# Check that the precious variables saved in the cache have kept the same +# value. +ac_cache_corrupted=false +for ac_var in `(set) 2>&1 | + sed -n 's/^ac_env_\([a-zA-Z_0-9]*\)_set=.*/\1/p'`; do + eval ac_old_set=\$ac_cv_env_${ac_var}_set + eval ac_new_set=\$ac_env_${ac_var}_set + eval ac_old_val="\$ac_cv_env_${ac_var}_value" + eval ac_new_val="\$ac_env_${ac_var}_value" + case $ac_old_set,$ac_new_set in + set,) + { echo "$as_me:$LINENO: error: \`$ac_var' was set to \`$ac_old_val' in the previous run" >&5 +echo "$as_me: error: \`$ac_var' was set to \`$ac_old_val' in the previous run" >&2;} + ac_cache_corrupted=: ;; + ,set) + { echo "$as_me:$LINENO: error: \`$ac_var' was not set in the previous run" >&5 +echo "$as_me: error: \`$ac_var' was not set in the previous run" >&2;} + ac_cache_corrupted=: ;; + ,);; + *) + if test "x$ac_old_val" != "x$ac_new_val"; then + { echo "$as_me:$LINENO: error: \`$ac_var' has changed since the previous run:" >&5 +echo "$as_me: error: \`$ac_var' has changed since the previous run:" >&2;} + { echo "$as_me:$LINENO: former value: $ac_old_val" >&5 +echo "$as_me: former value: $ac_old_val" >&2;} + { echo "$as_me:$LINENO: current value: $ac_new_val" >&5 +echo "$as_me: current value: $ac_new_val" >&2;} + ac_cache_corrupted=: + fi;; + esac + # Pass precious variables to config.status. + if test "$ac_new_set" = set; then + case $ac_new_val in + *" "*|*" "*|*[\[\]\~\#\$\^\&\*\(\)\{\}\\\|\;\<\>\?\"\']*) + ac_arg=$ac_var=`echo "$ac_new_val" | sed "s/'/'\\\\\\\\''/g"` ;; + *) ac_arg=$ac_var=$ac_new_val ;; + esac + case " $ac_configure_args " in + *" '$ac_arg' "*) ;; # Avoid dups. Use of quotes ensures accuracy. + *) ac_configure_args="$ac_configure_args '$ac_arg'" ;; + esac + fi +done +if $ac_cache_corrupted; then + { echo "$as_me:$LINENO: error: changes in the environment can compromise the build" >&5 +echo "$as_me: error: changes in the environment can compromise the build" >&2;} + { { echo "$as_me:$LINENO: error: run \`make distclean' and/or \`rm $cache_file' and start over" >&5 +echo "$as_me: error: run \`make distclean' and/or \`rm $cache_file' and start over" >&2;} + { (exit 1); exit 1; }; } +fi + +ac_ext=c +ac_cpp='$CPP $CPPFLAGS' +ac_compile='$CC -c $CFLAGS $CPPFLAGS conftest.$ac_ext >&5' +ac_link='$CC -o conftest$ac_exeext $CFLAGS $CPPFLAGS $LDFLAGS conftest.$ac_ext $LIBS >&5' +ac_compiler_gnu=$ac_cv_c_compiler_gnu + + + + + + + + + + + + + + + + + + + + +if test -z "$CFLAGS"; then + CFLAGS="-Wall -O4 -fexpensive-optimizations -funroll-loops" +fi + +if test -n "$ac_tool_prefix"; then + # Extract the first word of "${ac_tool_prefix}ranlib", so it can be a program name with args. +set dummy ${ac_tool_prefix}ranlib; ac_word=$2 +echo "$as_me:$LINENO: checking for $ac_word" >&5 +echo $ECHO_N "checking for $ac_word... $ECHO_C" >&6 +if test "${ac_cv_prog_RANLIB+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + if test -n "$RANLIB"; then + ac_cv_prog_RANLIB="$RANLIB" # Let the user override the test. +else +as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + for ac_exec_ext in '' $ac_executable_extensions; do + if $as_executable_p "$as_dir/$ac_word$ac_exec_ext"; then + ac_cv_prog_RANLIB="${ac_tool_prefix}ranlib" + echo "$as_me:$LINENO: found $as_dir/$ac_word$ac_exec_ext" >&5 + break 2 + fi +done +done + +fi +fi +RANLIB=$ac_cv_prog_RANLIB +if test -n "$RANLIB"; then + echo "$as_me:$LINENO: result: $RANLIB" >&5 +echo "${ECHO_T}$RANLIB" >&6 +else + echo "$as_me:$LINENO: result: no" >&5 +echo "${ECHO_T}no" >&6 +fi + +fi +if test -z "$ac_cv_prog_RANLIB"; then + ac_ct_RANLIB=$RANLIB + # Extract the first word of "ranlib", so it can be a program name with args. +set dummy ranlib; ac_word=$2 +echo "$as_me:$LINENO: checking for $ac_word" >&5 +echo $ECHO_N "checking for $ac_word... $ECHO_C" >&6 +if test "${ac_cv_prog_ac_ct_RANLIB+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + if test -n "$ac_ct_RANLIB"; then + ac_cv_prog_ac_ct_RANLIB="$ac_ct_RANLIB" # Let the user override the test. +else +as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + for ac_exec_ext in '' $ac_executable_extensions; do + if $as_executable_p "$as_dir/$ac_word$ac_exec_ext"; then + ac_cv_prog_ac_ct_RANLIB="ranlib" + echo "$as_me:$LINENO: found $as_dir/$ac_word$ac_exec_ext" >&5 + break 2 + fi +done +done + + test -z "$ac_cv_prog_ac_ct_RANLIB" && ac_cv_prog_ac_ct_RANLIB=":" +fi +fi +ac_ct_RANLIB=$ac_cv_prog_ac_ct_RANLIB +if test -n "$ac_ct_RANLIB"; then + echo "$as_me:$LINENO: result: $ac_ct_RANLIB" >&5 +echo "${ECHO_T}$ac_ct_RANLIB" >&6 +else + echo "$as_me:$LINENO: result: no" >&5 +echo "${ECHO_T}no" >&6 +fi + + RANLIB=$ac_ct_RANLIB +else + RANLIB="$ac_cv_prog_RANLIB" +fi + +ac_ext=c +ac_cpp='$CPP $CPPFLAGS' +ac_compile='$CC -c $CFLAGS $CPPFLAGS conftest.$ac_ext >&5' +ac_link='$CC -o conftest$ac_exeext $CFLAGS $CPPFLAGS $LDFLAGS conftest.$ac_ext $LIBS >&5' +ac_compiler_gnu=$ac_cv_c_compiler_gnu +if test -n "$ac_tool_prefix"; then + # Extract the first word of "${ac_tool_prefix}gcc", so it can be a program name with args. +set dummy ${ac_tool_prefix}gcc; ac_word=$2 +echo "$as_me:$LINENO: checking for $ac_word" >&5 +echo $ECHO_N "checking for $ac_word... $ECHO_C" >&6 +if test "${ac_cv_prog_CC+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + if test -n "$CC"; then + ac_cv_prog_CC="$CC" # Let the user override the test. +else +as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + for ac_exec_ext in '' $ac_executable_extensions; do + if $as_executable_p "$as_dir/$ac_word$ac_exec_ext"; then + ac_cv_prog_CC="${ac_tool_prefix}gcc" + echo "$as_me:$LINENO: found $as_dir/$ac_word$ac_exec_ext" >&5 + break 2 + fi +done +done + +fi +fi +CC=$ac_cv_prog_CC +if test -n "$CC"; then + echo "$as_me:$LINENO: result: $CC" >&5 +echo "${ECHO_T}$CC" >&6 +else + echo "$as_me:$LINENO: result: no" >&5 +echo "${ECHO_T}no" >&6 +fi + +fi +if test -z "$ac_cv_prog_CC"; then + ac_ct_CC=$CC + # Extract the first word of "gcc", so it can be a program name with args. +set dummy gcc; ac_word=$2 +echo "$as_me:$LINENO: checking for $ac_word" >&5 +echo $ECHO_N "checking for $ac_word... $ECHO_C" >&6 +if test "${ac_cv_prog_ac_ct_CC+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + if test -n "$ac_ct_CC"; then + ac_cv_prog_ac_ct_CC="$ac_ct_CC" # Let the user override the test. +else +as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + for ac_exec_ext in '' $ac_executable_extensions; do + if $as_executable_p "$as_dir/$ac_word$ac_exec_ext"; then + ac_cv_prog_ac_ct_CC="gcc" + echo "$as_me:$LINENO: found $as_dir/$ac_word$ac_exec_ext" >&5 + break 2 + fi +done +done + +fi +fi +ac_ct_CC=$ac_cv_prog_ac_ct_CC +if test -n "$ac_ct_CC"; then + echo "$as_me:$LINENO: result: $ac_ct_CC" >&5 +echo "${ECHO_T}$ac_ct_CC" >&6 +else + echo "$as_me:$LINENO: result: no" >&5 +echo "${ECHO_T}no" >&6 +fi + + CC=$ac_ct_CC +else + CC="$ac_cv_prog_CC" +fi + +if test -z "$CC"; then + if test -n "$ac_tool_prefix"; then + # Extract the first word of "${ac_tool_prefix}cc", so it can be a program name with args. +set dummy ${ac_tool_prefix}cc; ac_word=$2 +echo "$as_me:$LINENO: checking for $ac_word" >&5 +echo $ECHO_N "checking for $ac_word... $ECHO_C" >&6 +if test "${ac_cv_prog_CC+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + if test -n "$CC"; then + ac_cv_prog_CC="$CC" # Let the user override the test. +else +as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + for ac_exec_ext in '' $ac_executable_extensions; do + if $as_executable_p "$as_dir/$ac_word$ac_exec_ext"; then + ac_cv_prog_CC="${ac_tool_prefix}cc" + echo "$as_me:$LINENO: found $as_dir/$ac_word$ac_exec_ext" >&5 + break 2 + fi +done +done + +fi +fi +CC=$ac_cv_prog_CC +if test -n "$CC"; then + echo "$as_me:$LINENO: result: $CC" >&5 +echo "${ECHO_T}$CC" >&6 +else + echo "$as_me:$LINENO: result: no" >&5 +echo "${ECHO_T}no" >&6 +fi + +fi +if test -z "$ac_cv_prog_CC"; then + ac_ct_CC=$CC + # Extract the first word of "cc", so it can be a program name with args. +set dummy cc; ac_word=$2 +echo "$as_me:$LINENO: checking for $ac_word" >&5 +echo $ECHO_N "checking for $ac_word... $ECHO_C" >&6 +if test "${ac_cv_prog_ac_ct_CC+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + if test -n "$ac_ct_CC"; then + ac_cv_prog_ac_ct_CC="$ac_ct_CC" # Let the user override the test. +else +as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + for ac_exec_ext in '' $ac_executable_extensions; do + if $as_executable_p "$as_dir/$ac_word$ac_exec_ext"; then + ac_cv_prog_ac_ct_CC="cc" + echo "$as_me:$LINENO: found $as_dir/$ac_word$ac_exec_ext" >&5 + break 2 + fi +done +done + +fi +fi +ac_ct_CC=$ac_cv_prog_ac_ct_CC +if test -n "$ac_ct_CC"; then + echo "$as_me:$LINENO: result: $ac_ct_CC" >&5 +echo "${ECHO_T}$ac_ct_CC" >&6 +else + echo "$as_me:$LINENO: result: no" >&5 +echo "${ECHO_T}no" >&6 +fi + + CC=$ac_ct_CC +else + CC="$ac_cv_prog_CC" +fi + +fi +if test -z "$CC"; then + # Extract the first word of "cc", so it can be a program name with args. +set dummy cc; ac_word=$2 +echo "$as_me:$LINENO: checking for $ac_word" >&5 +echo $ECHO_N "checking for $ac_word... $ECHO_C" >&6 +if test "${ac_cv_prog_CC+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + if test -n "$CC"; then + ac_cv_prog_CC="$CC" # Let the user override the test. +else + ac_prog_rejected=no +as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + for ac_exec_ext in '' $ac_executable_extensions; do + if $as_executable_p "$as_dir/$ac_word$ac_exec_ext"; then + if test "$as_dir/$ac_word$ac_exec_ext" = "/usr/ucb/cc"; then + ac_prog_rejected=yes + continue + fi + ac_cv_prog_CC="cc" + echo "$as_me:$LINENO: found $as_dir/$ac_word$ac_exec_ext" >&5 + break 2 + fi +done +done + +if test $ac_prog_rejected = yes; then + # We found a bogon in the path, so make sure we never use it. + set dummy $ac_cv_prog_CC + shift + if test $# != 0; then + # We chose a different compiler from the bogus one. + # However, it has the same basename, so the bogon will be chosen + # first if we set CC to just the basename; use the full file name. + shift + ac_cv_prog_CC="$as_dir/$ac_word${1+' '}$@" + fi +fi +fi +fi +CC=$ac_cv_prog_CC +if test -n "$CC"; then + echo "$as_me:$LINENO: result: $CC" >&5 +echo "${ECHO_T}$CC" >&6 +else + echo "$as_me:$LINENO: result: no" >&5 +echo "${ECHO_T}no" >&6 +fi + +fi +if test -z "$CC"; then + if test -n "$ac_tool_prefix"; then + for ac_prog in cl + do + # Extract the first word of "$ac_tool_prefix$ac_prog", so it can be a program name with args. +set dummy $ac_tool_prefix$ac_prog; ac_word=$2 +echo "$as_me:$LINENO: checking for $ac_word" >&5 +echo $ECHO_N "checking for $ac_word... $ECHO_C" >&6 +if test "${ac_cv_prog_CC+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + if test -n "$CC"; then + ac_cv_prog_CC="$CC" # Let the user override the test. +else +as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + for ac_exec_ext in '' $ac_executable_extensions; do + if $as_executable_p "$as_dir/$ac_word$ac_exec_ext"; then + ac_cv_prog_CC="$ac_tool_prefix$ac_prog" + echo "$as_me:$LINENO: found $as_dir/$ac_word$ac_exec_ext" >&5 + break 2 + fi +done +done + +fi +fi +CC=$ac_cv_prog_CC +if test -n "$CC"; then + echo "$as_me:$LINENO: result: $CC" >&5 +echo "${ECHO_T}$CC" >&6 +else + echo "$as_me:$LINENO: result: no" >&5 +echo "${ECHO_T}no" >&6 +fi + + test -n "$CC" && break + done +fi +if test -z "$CC"; then + ac_ct_CC=$CC + for ac_prog in cl +do + # Extract the first word of "$ac_prog", so it can be a program name with args. +set dummy $ac_prog; ac_word=$2 +echo "$as_me:$LINENO: checking for $ac_word" >&5 +echo $ECHO_N "checking for $ac_word... $ECHO_C" >&6 +if test "${ac_cv_prog_ac_ct_CC+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + if test -n "$ac_ct_CC"; then + ac_cv_prog_ac_ct_CC="$ac_ct_CC" # Let the user override the test. +else +as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + for ac_exec_ext in '' $ac_executable_extensions; do + if $as_executable_p "$as_dir/$ac_word$ac_exec_ext"; then + ac_cv_prog_ac_ct_CC="$ac_prog" + echo "$as_me:$LINENO: found $as_dir/$ac_word$ac_exec_ext" >&5 + break 2 + fi +done +done + +fi +fi +ac_ct_CC=$ac_cv_prog_ac_ct_CC +if test -n "$ac_ct_CC"; then + echo "$as_me:$LINENO: result: $ac_ct_CC" >&5 +echo "${ECHO_T}$ac_ct_CC" >&6 +else + echo "$as_me:$LINENO: result: no" >&5 +echo "${ECHO_T}no" >&6 +fi + + test -n "$ac_ct_CC" && break +done + + CC=$ac_ct_CC +fi + +fi + + +test -z "$CC" && { { echo "$as_me:$LINENO: error: no acceptable C compiler found in \$PATH +See \`config.log' for more details." >&5 +echo "$as_me: error: no acceptable C compiler found in \$PATH +See \`config.log' for more details." >&2;} + { (exit 1); exit 1; }; } + +# Provide some information about the compiler. +echo "$as_me:$LINENO:" \ + "checking for C compiler version" >&5 +ac_compiler=`set X $ac_compile; echo $2` +{ (eval echo "$as_me:$LINENO: \"$ac_compiler --version &5\"") >&5 + (eval $ac_compiler --version &5) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } +{ (eval echo "$as_me:$LINENO: \"$ac_compiler -v &5\"") >&5 + (eval $ac_compiler -v &5) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } +{ (eval echo "$as_me:$LINENO: \"$ac_compiler -V &5\"") >&5 + (eval $ac_compiler -V &5) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } + +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ + +int +main () +{ + + ; + return 0; +} +_ACEOF +ac_clean_files_save=$ac_clean_files +ac_clean_files="$ac_clean_files a.out a.exe b.out" +# Try to create an executable without -o first, disregard a.out. +# It will help us diagnose broken compilers, and finding out an intuition +# of exeext. +echo "$as_me:$LINENO: checking for C compiler default output file name" >&5 +echo $ECHO_N "checking for C compiler default output file name... $ECHO_C" >&6 +ac_link_default=`echo "$ac_link" | sed 's/ -o *conftest[^ ]*//'` +if { (eval echo "$as_me:$LINENO: \"$ac_link_default\"") >&5 + (eval $ac_link_default) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; then + # Find the output, starting from the most likely. This scheme is +# not robust to junk in `.', hence go to wildcards (a.*) only as a last +# resort. + +# Be careful to initialize this variable, since it used to be cached. +# Otherwise an old cache value of `no' led to `EXEEXT = no' in a Makefile. +ac_cv_exeext= +# b.out is created by i960 compilers. +for ac_file in a_out.exe a.exe conftest.exe a.out conftest a.* conftest.* b.out +do + test -f "$ac_file" || continue + case $ac_file in + *.$ac_ext | *.xcoff | *.tds | *.d | *.pdb | *.xSYM | *.bb | *.bbg | *.o | *.obj ) + ;; + conftest.$ac_ext ) + # This is the source file. + ;; + [ab].out ) + # We found the default executable, but exeext='' is most + # certainly right. + break;; + *.* ) + ac_cv_exeext=`expr "$ac_file" : '[^.]*\(\..*\)'` + # FIXME: I believe we export ac_cv_exeext for Libtool, + # but it would be cool to find out if it's true. Does anybody + # maintain Libtool? --akim. + export ac_cv_exeext + break;; + * ) + break;; + esac +done +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +{ { echo "$as_me:$LINENO: error: C compiler cannot create executables +See \`config.log' for more details." >&5 +echo "$as_me: error: C compiler cannot create executables +See \`config.log' for more details." >&2;} + { (exit 77); exit 77; }; } +fi + +ac_exeext=$ac_cv_exeext +echo "$as_me:$LINENO: result: $ac_file" >&5 +echo "${ECHO_T}$ac_file" >&6 + +# Check the compiler produces executables we can run. If not, either +# the compiler is broken, or we cross compile. +echo "$as_me:$LINENO: checking whether the C compiler works" >&5 +echo $ECHO_N "checking whether the C compiler works... $ECHO_C" >&6 +# FIXME: These cross compiler hacks should be removed for Autoconf 3.0 +# If not cross compiling, check that we can run a simple program. +if test "$cross_compiling" != yes; then + if { ac_try='./$ac_file' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + cross_compiling=no + else + if test "$cross_compiling" = maybe; then + cross_compiling=yes + else + { { echo "$as_me:$LINENO: error: cannot run C compiled programs. +If you meant to cross compile, use \`--host'. +See \`config.log' for more details." >&5 +echo "$as_me: error: cannot run C compiled programs. +If you meant to cross compile, use \`--host'. +See \`config.log' for more details." >&2;} + { (exit 1); exit 1; }; } + fi + fi +fi +echo "$as_me:$LINENO: result: yes" >&5 +echo "${ECHO_T}yes" >&6 + +rm -f a.out a.exe conftest$ac_cv_exeext b.out +ac_clean_files=$ac_clean_files_save +# Check the compiler produces executables we can run. If not, either +# the compiler is broken, or we cross compile. +echo "$as_me:$LINENO: checking whether we are cross compiling" >&5 +echo $ECHO_N "checking whether we are cross compiling... $ECHO_C" >&6 +echo "$as_me:$LINENO: result: $cross_compiling" >&5 +echo "${ECHO_T}$cross_compiling" >&6 + +echo "$as_me:$LINENO: checking for suffix of executables" >&5 +echo $ECHO_N "checking for suffix of executables... $ECHO_C" >&6 +if { (eval echo "$as_me:$LINENO: \"$ac_link\"") >&5 + (eval $ac_link) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; then + # If both `conftest.exe' and `conftest' are `present' (well, observable) +# catch `conftest.exe'. For instance with Cygwin, `ls conftest' will +# work properly (i.e., refer to `conftest.exe'), while it won't with +# `rm'. +for ac_file in conftest.exe conftest conftest.*; do + test -f "$ac_file" || continue + case $ac_file in + *.$ac_ext | *.xcoff | *.tds | *.d | *.pdb | *.xSYM | *.bb | *.bbg | *.o | *.obj ) ;; + *.* ) ac_cv_exeext=`expr "$ac_file" : '[^.]*\(\..*\)'` + export ac_cv_exeext + break;; + * ) break;; + esac +done +else + { { echo "$as_me:$LINENO: error: cannot compute suffix of executables: cannot compile and link +See \`config.log' for more details." >&5 +echo "$as_me: error: cannot compute suffix of executables: cannot compile and link +See \`config.log' for more details." >&2;} + { (exit 1); exit 1; }; } +fi + +rm -f conftest$ac_cv_exeext +echo "$as_me:$LINENO: result: $ac_cv_exeext" >&5 +echo "${ECHO_T}$ac_cv_exeext" >&6 + +rm -f conftest.$ac_ext +EXEEXT=$ac_cv_exeext +ac_exeext=$EXEEXT +echo "$as_me:$LINENO: checking for suffix of object files" >&5 +echo $ECHO_N "checking for suffix of object files... $ECHO_C" >&6 +if test "${ac_cv_objext+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ + +int +main () +{ + + ; + return 0; +} +_ACEOF +rm -f conftest.o conftest.obj +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; then + for ac_file in `(ls conftest.o conftest.obj; ls conftest.*) 2>/dev/null`; do + case $ac_file in + *.$ac_ext | *.xcoff | *.tds | *.d | *.pdb | *.xSYM | *.bb | *.bbg ) ;; + *) ac_cv_objext=`expr "$ac_file" : '.*\.\(.*\)'` + break;; + esac +done +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +{ { echo "$as_me:$LINENO: error: cannot compute suffix of object files: cannot compile +See \`config.log' for more details." >&5 +echo "$as_me: error: cannot compute suffix of object files: cannot compile +See \`config.log' for more details." >&2;} + { (exit 1); exit 1; }; } +fi + +rm -f conftest.$ac_cv_objext conftest.$ac_ext +fi +echo "$as_me:$LINENO: result: $ac_cv_objext" >&5 +echo "${ECHO_T}$ac_cv_objext" >&6 +OBJEXT=$ac_cv_objext +ac_objext=$OBJEXT +echo "$as_me:$LINENO: checking whether we are using the GNU C compiler" >&5 +echo $ECHO_N "checking whether we are using the GNU C compiler... $ECHO_C" >&6 +if test "${ac_cv_c_compiler_gnu+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ + +int +main () +{ +#ifndef __GNUC__ + choke me +#endif + + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_compiler_gnu=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_compiler_gnu=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +ac_cv_c_compiler_gnu=$ac_compiler_gnu + +fi +echo "$as_me:$LINENO: result: $ac_cv_c_compiler_gnu" >&5 +echo "${ECHO_T}$ac_cv_c_compiler_gnu" >&6 +GCC=`test $ac_compiler_gnu = yes && echo yes` +ac_test_CFLAGS=${CFLAGS+set} +ac_save_CFLAGS=$CFLAGS +CFLAGS="-g" +echo "$as_me:$LINENO: checking whether $CC accepts -g" >&5 +echo $ECHO_N "checking whether $CC accepts -g... $ECHO_C" >&6 +if test "${ac_cv_prog_cc_g+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ + +int +main () +{ + + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_prog_cc_g=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_cv_prog_cc_g=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +fi +echo "$as_me:$LINENO: result: $ac_cv_prog_cc_g" >&5 +echo "${ECHO_T}$ac_cv_prog_cc_g" >&6 +if test "$ac_test_CFLAGS" = set; then + CFLAGS=$ac_save_CFLAGS +elif test $ac_cv_prog_cc_g = yes; then + if test "$GCC" = yes; then + CFLAGS="-g -O2" + else + CFLAGS="-g" + fi +else + if test "$GCC" = yes; then + CFLAGS="-O2" + else + CFLAGS= + fi +fi +echo "$as_me:$LINENO: checking for $CC option to accept ANSI C" >&5 +echo $ECHO_N "checking for $CC option to accept ANSI C... $ECHO_C" >&6 +if test "${ac_cv_prog_cc_stdc+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + ac_cv_prog_cc_stdc=no +ac_save_CC=$CC +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include +#include +#include +#include +/* Most of the following tests are stolen from RCS 5.7's src/conf.sh. */ +struct buf { int x; }; +FILE * (*rcsopen) (struct buf *, struct stat *, int); +static char *e (p, i) + char **p; + int i; +{ + return p[i]; +} +static char *f (char * (*g) (char **, int), char **p, ...) +{ + char *s; + va_list v; + va_start (v,p); + s = g (p, va_arg (v,int)); + va_end (v); + return s; +} + +/* OSF 4.0 Compaq cc is some sort of almost-ANSI by default. It has + function prototypes and stuff, but not '\xHH' hex character constants. + These don't provoke an error unfortunately, instead are silently treated + as 'x'. The following induces an error, until -std1 is added to get + proper ANSI mode. Curiously '\x00'!='x' always comes out true, for an + array size at least. It's necessary to write '\x00'==0 to get something + that's true only with -std1. */ +int osf4_cc_array ['\x00' == 0 ? 1 : -1]; + +int test (int i, double x); +struct s1 {int (*f) (int a);}; +struct s2 {int (*f) (double a);}; +int pairnames (int, char **, FILE *(*)(struct buf *, struct stat *, int), int, int); +int argc; +char **argv; +int +main () +{ +return f (e, argv, 0) != argv[0] || f (e, argv, 1) != argv[1]; + ; + return 0; +} +_ACEOF +# Don't try gcc -ansi; that turns off useful extensions and +# breaks some systems' header files. +# AIX -qlanglvl=ansi +# Ultrix and OSF/1 -std1 +# HP-UX 10.20 and later -Ae +# HP-UX older versions -Aa -D_HPUX_SOURCE +# SVR4 -Xc -D__EXTENSIONS__ +for ac_arg in "" -qlanglvl=ansi -std1 -Ae "-Aa -D_HPUX_SOURCE" "-Xc -D__EXTENSIONS__" +do + CC="$ac_save_CC $ac_arg" + rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_prog_cc_stdc=$ac_arg +break +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +fi +rm -f conftest.err conftest.$ac_objext +done +rm -f conftest.$ac_ext conftest.$ac_objext +CC=$ac_save_CC + +fi + +case "x$ac_cv_prog_cc_stdc" in + x|xno) + echo "$as_me:$LINENO: result: none needed" >&5 +echo "${ECHO_T}none needed" >&6 ;; + *) + echo "$as_me:$LINENO: result: $ac_cv_prog_cc_stdc" >&5 +echo "${ECHO_T}$ac_cv_prog_cc_stdc" >&6 + CC="$CC $ac_cv_prog_cc_stdc" ;; +esac + +# Some people use a C++ compiler to compile C. Since we use `exit', +# in C++ we need to declare it. In case someone uses the same compiler +# for both compiling C and C++ we need to have the C++ compiler decide +# the declaration of exit, since it's the most demanding environment. +cat >conftest.$ac_ext <<_ACEOF +#ifndef __cplusplus + choke me +#endif +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + for ac_declaration in \ + '' \ + 'extern "C" void std::exit (int) throw (); using std::exit;' \ + 'extern "C" void std::exit (int); using std::exit;' \ + 'extern "C" void exit (int) throw ();' \ + 'extern "C" void exit (int);' \ + 'void exit (int);' +do + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_declaration +#include +int +main () +{ +exit (42); + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + : +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +continue +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_declaration +int +main () +{ +exit (42); + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + break +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +done +rm -f conftest* +if test -n "$ac_declaration"; then + echo '#ifdef __cplusplus' >>confdefs.h + echo $ac_declaration >>confdefs.h + echo '#endif' >>confdefs.h +fi + +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +ac_ext=c +ac_cpp='$CPP $CPPFLAGS' +ac_compile='$CC -c $CFLAGS $CPPFLAGS conftest.$ac_ext >&5' +ac_link='$CC -o conftest$ac_exeext $CFLAGS $CPPFLAGS $LDFLAGS conftest.$ac_ext $LIBS >&5' +ac_compiler_gnu=$ac_cv_c_compiler_gnu + +ac_aux_dir= +for ac_dir in $srcdir $srcdir/.. $srcdir/../..; do + if test -f $ac_dir/install-sh; then + ac_aux_dir=$ac_dir + ac_install_sh="$ac_aux_dir/install-sh -c" + break + elif test -f $ac_dir/install.sh; then + ac_aux_dir=$ac_dir + ac_install_sh="$ac_aux_dir/install.sh -c" + break + elif test -f $ac_dir/shtool; then + ac_aux_dir=$ac_dir + ac_install_sh="$ac_aux_dir/shtool install -c" + break + fi +done +if test -z "$ac_aux_dir"; then + { { echo "$as_me:$LINENO: error: cannot find install-sh or install.sh in $srcdir $srcdir/.. $srcdir/../.." >&5 +echo "$as_me: error: cannot find install-sh or install.sh in $srcdir $srcdir/.. $srcdir/../.." >&2;} + { (exit 1); exit 1; }; } +fi +ac_config_guess="$SHELL $ac_aux_dir/config.guess" +ac_config_sub="$SHELL $ac_aux_dir/config.sub" +ac_configure="$SHELL $ac_aux_dir/configure" # This should be Cygnus configure. + +# Find a good install program. We prefer a C program (faster), +# so one script is as good as another. But avoid the broken or +# incompatible versions: +# SysV /etc/install, /usr/sbin/install +# SunOS /usr/etc/install +# IRIX /sbin/install +# AIX /bin/install +# AmigaOS /C/install, which installs bootblocks on floppy discs +# AIX 4 /usr/bin/installbsd, which doesn't work without a -g flag +# AFS /usr/afsws/bin/install, which mishandles nonexistent args +# SVR4 /usr/ucb/install, which tries to use the nonexistent group "staff" +# OS/2's system install, which has a completely different semantic +# ./install, which can be erroneously created by make from ./install.sh. +echo "$as_me:$LINENO: checking for a BSD-compatible install" >&5 +echo $ECHO_N "checking for a BSD-compatible install... $ECHO_C" >&6 +if test -z "$INSTALL"; then +if test "${ac_cv_path_install+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + # Account for people who put trailing slashes in PATH elements. +case $as_dir/ in + ./ | .// | /cC/* | \ + /etc/* | /usr/sbin/* | /usr/etc/* | /sbin/* | /usr/afsws/bin/* | \ + ?:\\/os2\\/install\\/* | ?:\\/OS2\\/INSTALL\\/* | \ + /usr/ucb/* ) ;; + *) + # OSF1 and SCO ODT 3.0 have their own names for install. + # Don't use installbsd from OSF since it installs stuff as root + # by default. + for ac_prog in ginstall scoinst install; do + for ac_exec_ext in '' $ac_executable_extensions; do + if $as_executable_p "$as_dir/$ac_prog$ac_exec_ext"; then + if test $ac_prog = install && + grep dspmsg "$as_dir/$ac_prog$ac_exec_ext" >/dev/null 2>&1; then + # AIX install. It has an incompatible calling convention. + : + elif test $ac_prog = install && + grep pwplus "$as_dir/$ac_prog$ac_exec_ext" >/dev/null 2>&1; then + # program-specific install script used by HP pwplus--don't use. + : + else + ac_cv_path_install="$as_dir/$ac_prog$ac_exec_ext -c" + break 3 + fi + fi + done + done + ;; +esac +done + + +fi + if test "${ac_cv_path_install+set}" = set; then + INSTALL=$ac_cv_path_install + else + # As a last resort, use the slow shell script. We don't cache a + # path for INSTALL within a source directory, because that will + # break other packages using the cache if that directory is + # removed, or if the path is relative. + INSTALL=$ac_install_sh + fi +fi +echo "$as_me:$LINENO: result: $INSTALL" >&5 +echo "${ECHO_T}$INSTALL" >&6 + +# Use test -z because SunOS4 sh mishandles braces in ${var-val}. +# It thinks the first close brace ends the variable substitution. +test -z "$INSTALL_PROGRAM" && INSTALL_PROGRAM='${INSTALL}' + +test -z "$INSTALL_SCRIPT" && INSTALL_SCRIPT='${INSTALL}' + +test -z "$INSTALL_DATA" && INSTALL_DATA='${INSTALL} -m 644' + + +# Check whether --enable-pic or --disable-pic was given. +if test "${enable_pic+set}" = set; then + enableval="$enable_pic" + CFLAGS="-fPIC $CFLAGS" +fi; + +# Check whether --enable-kernel-linux or --disable-kernel-linux was given. +if test "${enable_kernel_linux+set}" = set; then + enableval="$enable_kernel_linux" + +else + enable_kernel_linux=no +fi; +echo "$as_me:$LINENO: checking whether to build for Linux kernel context" >&5 +echo $ECHO_N "checking whether to build for Linux kernel context... $ECHO_C" >&6 +if test "$enable_kernel_linux" = "yes"; then + +cat >>confdefs.h <<\_ACEOF +#define SRTP_KERNEL 1 +_ACEOF + + +cat >>confdefs.h <<\_ACEOF +#define SRTP_KERNEL_LINUX 1 +_ACEOF + +fi +echo "$as_me:$LINENO: result: $enable_kernel_linux" >&5 +echo "${ECHO_T}$enable_kernel_linux" >&6 + +if test "$cross_compiling" != yes; then + echo "$as_me:$LINENO: checking for /dev/urandom" >&5 +echo $ECHO_N "checking for /dev/urandom... $ECHO_C" >&6 +if test "${ac_cv_file__dev_urandom+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + test "$cross_compiling" = yes && + { { echo "$as_me:$LINENO: error: cannot check for file existence when cross compiling" >&5 +echo "$as_me: error: cannot check for file existence when cross compiling" >&2;} + { (exit 1); exit 1; }; } +if test -r "/dev/urandom"; then + ac_cv_file__dev_urandom=yes +else + ac_cv_file__dev_urandom=no +fi +fi +echo "$as_me:$LINENO: result: $ac_cv_file__dev_urandom" >&5 +echo "${ECHO_T}$ac_cv_file__dev_urandom" >&6 +if test $ac_cv_file__dev_urandom = yes; then + DEV_URANDOM=/dev/urandom +else + echo "$as_me:$LINENO: checking for /dev/random" >&5 +echo $ECHO_N "checking for /dev/random... $ECHO_C" >&6 +if test "${ac_cv_file__dev_random+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + test "$cross_compiling" = yes && + { { echo "$as_me:$LINENO: error: cannot check for file existence when cross compiling" >&5 +echo "$as_me: error: cannot check for file existence when cross compiling" >&2;} + { (exit 1); exit 1; }; } +if test -r "/dev/random"; then + ac_cv_file__dev_random=yes +else + ac_cv_file__dev_random=no +fi +fi +echo "$as_me:$LINENO: result: $ac_cv_file__dev_random" >&5 +echo "${ECHO_T}$ac_cv_file__dev_random" >&6 +if test $ac_cv_file__dev_random = yes; then + DEV_URANDOM=/dev/random +fi + +fi + +fi + +echo "$as_me:$LINENO: checking which random device to use" >&5 +echo $ECHO_N "checking which random device to use... $ECHO_C" >&6 +if test "$enable_kernel_linux" = "yes"; then + RNG_OBJS=rand_linux_kernel.o + echo "$as_me:$LINENO: result: Linux kernel builtin" >&5 +echo "${ECHO_T}Linux kernel builtin" >&6 +else + RNG_OBJS=rand_source.o + if test -n "$DEV_URANDOM"; then + +cat >>confdefs.h <<_ACEOF +#define DEV_URANDOM "$DEV_URANDOM" +_ACEOF + + echo "$as_me:$LINENO: result: $DEV_URANDOM" >&5 +echo "${ECHO_T}$DEV_URANDOM" >&6 + else + echo "$as_me:$LINENO: result: standard rand() function..." >&5 +echo "${ECHO_T}standard rand() function..." >&6 + fi +fi + + + + +ac_ext=c +ac_cpp='$CPP $CPPFLAGS' +ac_compile='$CC -c $CFLAGS $CPPFLAGS conftest.$ac_ext >&5' +ac_link='$CC -o conftest$ac_exeext $CFLAGS $CPPFLAGS $LDFLAGS conftest.$ac_ext $LIBS >&5' +ac_compiler_gnu=$ac_cv_c_compiler_gnu +echo "$as_me:$LINENO: checking how to run the C preprocessor" >&5 +echo $ECHO_N "checking how to run the C preprocessor... $ECHO_C" >&6 +# On Suns, sometimes $CPP names a directory. +if test -n "$CPP" && test -d "$CPP"; then + CPP= +fi +if test -z "$CPP"; then + if test "${ac_cv_prog_CPP+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + # Double quotes because CPP needs to be expanded + for CPP in "$CC -E" "$CC -E -traditional-cpp" "/lib/cpp" + do + ac_preproc_ok=false +for ac_c_preproc_warn_flag in '' yes +do + # Use a header file that comes with gcc, so configuring glibc + # with a fresh cross-compiler works. + # Prefer to if __STDC__ is defined, since + # exists even on freestanding compilers. + # On the NeXT, cc -E runs the code through the compiler's parser, + # not just through cpp. "Syntax error" is here to catch this case. + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#ifdef __STDC__ +# include +#else +# include +#endif + Syntax error +_ACEOF +if { (eval echo "$as_me:$LINENO: \"$ac_cpp conftest.$ac_ext\"") >&5 + (eval $ac_cpp conftest.$ac_ext) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } >/dev/null; then + if test -s conftest.err; then + ac_cpp_err=$ac_c_preproc_warn_flag + ac_cpp_err=$ac_cpp_err$ac_c_werror_flag + else + ac_cpp_err= + fi +else + ac_cpp_err=yes +fi +if test -z "$ac_cpp_err"; then + : +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + + # Broken: fails on valid input. +continue +fi +rm -f conftest.err conftest.$ac_ext + + # OK, works on sane cases. Now check whether non-existent headers + # can be detected and how. + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include +_ACEOF +if { (eval echo "$as_me:$LINENO: \"$ac_cpp conftest.$ac_ext\"") >&5 + (eval $ac_cpp conftest.$ac_ext) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } >/dev/null; then + if test -s conftest.err; then + ac_cpp_err=$ac_c_preproc_warn_flag + ac_cpp_err=$ac_cpp_err$ac_c_werror_flag + else + ac_cpp_err= + fi +else + ac_cpp_err=yes +fi +if test -z "$ac_cpp_err"; then + # Broken: success on invalid input. +continue +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + + # Passes both tests. +ac_preproc_ok=: +break +fi +rm -f conftest.err conftest.$ac_ext + +done +# Because of `break', _AC_PREPROC_IFELSE's cleaning code was skipped. +rm -f conftest.err conftest.$ac_ext +if $ac_preproc_ok; then + break +fi + + done + ac_cv_prog_CPP=$CPP + +fi + CPP=$ac_cv_prog_CPP +else + ac_cv_prog_CPP=$CPP +fi +echo "$as_me:$LINENO: result: $CPP" >&5 +echo "${ECHO_T}$CPP" >&6 +ac_preproc_ok=false +for ac_c_preproc_warn_flag in '' yes +do + # Use a header file that comes with gcc, so configuring glibc + # with a fresh cross-compiler works. + # Prefer to if __STDC__ is defined, since + # exists even on freestanding compilers. + # On the NeXT, cc -E runs the code through the compiler's parser, + # not just through cpp. "Syntax error" is here to catch this case. + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#ifdef __STDC__ +# include +#else +# include +#endif + Syntax error +_ACEOF +if { (eval echo "$as_me:$LINENO: \"$ac_cpp conftest.$ac_ext\"") >&5 + (eval $ac_cpp conftest.$ac_ext) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } >/dev/null; then + if test -s conftest.err; then + ac_cpp_err=$ac_c_preproc_warn_flag + ac_cpp_err=$ac_cpp_err$ac_c_werror_flag + else + ac_cpp_err= + fi +else + ac_cpp_err=yes +fi +if test -z "$ac_cpp_err"; then + : +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + + # Broken: fails on valid input. +continue +fi +rm -f conftest.err conftest.$ac_ext + + # OK, works on sane cases. Now check whether non-existent headers + # can be detected and how. + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include +_ACEOF +if { (eval echo "$as_me:$LINENO: \"$ac_cpp conftest.$ac_ext\"") >&5 + (eval $ac_cpp conftest.$ac_ext) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } >/dev/null; then + if test -s conftest.err; then + ac_cpp_err=$ac_c_preproc_warn_flag + ac_cpp_err=$ac_cpp_err$ac_c_werror_flag + else + ac_cpp_err= + fi +else + ac_cpp_err=yes +fi +if test -z "$ac_cpp_err"; then + # Broken: success on invalid input. +continue +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + + # Passes both tests. +ac_preproc_ok=: +break +fi +rm -f conftest.err conftest.$ac_ext + +done +# Because of `break', _AC_PREPROC_IFELSE's cleaning code was skipped. +rm -f conftest.err conftest.$ac_ext +if $ac_preproc_ok; then + : +else + { { echo "$as_me:$LINENO: error: C preprocessor \"$CPP\" fails sanity check +See \`config.log' for more details." >&5 +echo "$as_me: error: C preprocessor \"$CPP\" fails sanity check +See \`config.log' for more details." >&2;} + { (exit 1); exit 1; }; } +fi + +ac_ext=c +ac_cpp='$CPP $CPPFLAGS' +ac_compile='$CC -c $CFLAGS $CPPFLAGS conftest.$ac_ext >&5' +ac_link='$CC -o conftest$ac_exeext $CFLAGS $CPPFLAGS $LDFLAGS conftest.$ac_ext $LIBS >&5' +ac_compiler_gnu=$ac_cv_c_compiler_gnu + + +echo "$as_me:$LINENO: checking for egrep" >&5 +echo $ECHO_N "checking for egrep... $ECHO_C" >&6 +if test "${ac_cv_prog_egrep+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + if echo a | (grep -E '(a|b)') >/dev/null 2>&1 + then ac_cv_prog_egrep='grep -E' + else ac_cv_prog_egrep='egrep' + fi +fi +echo "$as_me:$LINENO: result: $ac_cv_prog_egrep" >&5 +echo "${ECHO_T}$ac_cv_prog_egrep" >&6 + EGREP=$ac_cv_prog_egrep + + +echo "$as_me:$LINENO: checking for ANSI C header files" >&5 +echo $ECHO_N "checking for ANSI C header files... $ECHO_C" >&6 +if test "${ac_cv_header_stdc+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include +#include +#include +#include + +int +main () +{ + + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_header_stdc=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_cv_header_stdc=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext + +if test $ac_cv_header_stdc = yes; then + # SunOS 4.x string.h does not declare mem*, contrary to ANSI. + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include + +_ACEOF +if (eval "$ac_cpp conftest.$ac_ext") 2>&5 | + $EGREP "memchr" >/dev/null 2>&1; then + : +else + ac_cv_header_stdc=no +fi +rm -f conftest* + +fi + +if test $ac_cv_header_stdc = yes; then + # ISC 2.0.2 stdlib.h does not declare free, contrary to ANSI. + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include + +_ACEOF +if (eval "$ac_cpp conftest.$ac_ext") 2>&5 | + $EGREP "free" >/dev/null 2>&1; then + : +else + ac_cv_header_stdc=no +fi +rm -f conftest* + +fi + +if test $ac_cv_header_stdc = yes; then + # /bin/cc in Irix-4.0.5 gets non-ANSI ctype macros unless using -ansi. + if test "$cross_compiling" = yes; then + : +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include +#if ((' ' & 0x0FF) == 0x020) +# define ISLOWER(c) ('a' <= (c) && (c) <= 'z') +# define TOUPPER(c) (ISLOWER(c) ? 'A' + ((c) - 'a') : (c)) +#else +# define ISLOWER(c) \ + (('a' <= (c) && (c) <= 'i') \ + || ('j' <= (c) && (c) <= 'r') \ + || ('s' <= (c) && (c) <= 'z')) +# define TOUPPER(c) (ISLOWER(c) ? ((c) | 0x40) : (c)) +#endif + +#define XOR(e, f) (((e) && !(f)) || (!(e) && (f))) +int +main () +{ + int i; + for (i = 0; i < 256; i++) + if (XOR (islower (i), ISLOWER (i)) + || toupper (i) != TOUPPER (i)) + exit(2); + exit (0); +} +_ACEOF +rm -f conftest$ac_exeext +if { (eval echo "$as_me:$LINENO: \"$ac_link\"") >&5 + (eval $ac_link) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && { ac_try='./conftest$ac_exeext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + : +else + echo "$as_me: program exited with status $ac_status" >&5 +echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +( exit $ac_status ) +ac_cv_header_stdc=no +fi +rm -f core *.core gmon.out bb.out conftest$ac_exeext conftest.$ac_objext conftest.$ac_ext +fi +fi +fi +echo "$as_me:$LINENO: result: $ac_cv_header_stdc" >&5 +echo "${ECHO_T}$ac_cv_header_stdc" >&6 +if test $ac_cv_header_stdc = yes; then + +cat >>confdefs.h <<\_ACEOF +#define STDC_HEADERS 1 +_ACEOF + +fi + +# On IRIX 5.3, sys/types and inttypes.h are conflicting. + + + + + + + + + +for ac_header in sys/types.h sys/stat.h stdlib.h string.h memory.h strings.h \ + inttypes.h stdint.h unistd.h +do +as_ac_Header=`echo "ac_cv_header_$ac_header" | $as_tr_sh` +echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default + +#include <$ac_header> +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + eval "$as_ac_Header=yes" +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +eval "$as_ac_Header=no" +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 +if test `eval echo '${'$as_ac_Header'}'` = yes; then + cat >>confdefs.h <<_ACEOF +#define `echo "HAVE_$ac_header" | $as_tr_cpp` 1 +_ACEOF + +fi + +done + + + +for ac_header in stdlib.h +do +as_ac_Header=`echo "ac_cv_header_$ac_header" | $as_tr_sh` +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 +else + # Is the header compilable? +echo "$as_me:$LINENO: checking $ac_header usability" >&5 +echo $ECHO_N "checking $ac_header usability... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +#include <$ac_header> +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_header_compiler=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_header_compiler=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_compiler" >&5 +echo "${ECHO_T}$ac_header_compiler" >&6 + +# Is the header present? +echo "$as_me:$LINENO: checking $ac_header presence" >&5 +echo $ECHO_N "checking $ac_header presence... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include <$ac_header> +_ACEOF +if { (eval echo "$as_me:$LINENO: \"$ac_cpp conftest.$ac_ext\"") >&5 + (eval $ac_cpp conftest.$ac_ext) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } >/dev/null; then + if test -s conftest.err; then + ac_cpp_err=$ac_c_preproc_warn_flag + ac_cpp_err=$ac_cpp_err$ac_c_werror_flag + else + ac_cpp_err= + fi +else + ac_cpp_err=yes +fi +if test -z "$ac_cpp_err"; then + ac_header_preproc=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + + ac_header_preproc=no +fi +rm -f conftest.err conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_preproc" >&5 +echo "${ECHO_T}$ac_header_preproc" >&6 + +# So? What about this header? +case $ac_header_compiler:$ac_header_preproc:$ac_c_preproc_warn_flag in + yes:no: ) + { echo "$as_me:$LINENO: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&5 +echo "$as_me: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the compiler's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the compiler's result" >&2;} + ac_header_preproc=yes + ;; + no:yes:* ) + { echo "$as_me:$LINENO: WARNING: $ac_header: present but cannot be compiled" >&5 +echo "$as_me: WARNING: $ac_header: present but cannot be compiled" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: check for missing prerequisite headers?" >&5 +echo "$as_me: WARNING: $ac_header: check for missing prerequisite headers?" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: see the Autoconf documentation" >&5 +echo "$as_me: WARNING: $ac_header: see the Autoconf documentation" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&5 +echo "$as_me: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the preprocessor's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the preprocessor's result" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: in the future, the compiler will take precedence" >&5 +echo "$as_me: WARNING: $ac_header: in the future, the compiler will take precedence" >&2;} + ( + cat <<\_ASBOX +## ------------------------------------------ ## +## Report this to the AC_PACKAGE_NAME lists. ## +## ------------------------------------------ ## +_ASBOX + ) | + sed "s/^/$as_me: WARNING: /" >&2 + ;; +esac +echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + eval "$as_ac_Header=\$ac_header_preproc" +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 + +fi +if test `eval echo '${'$as_ac_Header'}'` = yes; then + cat >>confdefs.h <<_ACEOF +#define `echo "HAVE_$ac_header" | $as_tr_cpp` 1 +_ACEOF + +fi + +done + + +for ac_header in unistd.h +do +as_ac_Header=`echo "ac_cv_header_$ac_header" | $as_tr_sh` +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 +else + # Is the header compilable? +echo "$as_me:$LINENO: checking $ac_header usability" >&5 +echo $ECHO_N "checking $ac_header usability... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +#include <$ac_header> +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_header_compiler=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_header_compiler=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_compiler" >&5 +echo "${ECHO_T}$ac_header_compiler" >&6 + +# Is the header present? +echo "$as_me:$LINENO: checking $ac_header presence" >&5 +echo $ECHO_N "checking $ac_header presence... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include <$ac_header> +_ACEOF +if { (eval echo "$as_me:$LINENO: \"$ac_cpp conftest.$ac_ext\"") >&5 + (eval $ac_cpp conftest.$ac_ext) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } >/dev/null; then + if test -s conftest.err; then + ac_cpp_err=$ac_c_preproc_warn_flag + ac_cpp_err=$ac_cpp_err$ac_c_werror_flag + else + ac_cpp_err= + fi +else + ac_cpp_err=yes +fi +if test -z "$ac_cpp_err"; then + ac_header_preproc=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + + ac_header_preproc=no +fi +rm -f conftest.err conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_preproc" >&5 +echo "${ECHO_T}$ac_header_preproc" >&6 + +# So? What about this header? +case $ac_header_compiler:$ac_header_preproc:$ac_c_preproc_warn_flag in + yes:no: ) + { echo "$as_me:$LINENO: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&5 +echo "$as_me: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the compiler's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the compiler's result" >&2;} + ac_header_preproc=yes + ;; + no:yes:* ) + { echo "$as_me:$LINENO: WARNING: $ac_header: present but cannot be compiled" >&5 +echo "$as_me: WARNING: $ac_header: present but cannot be compiled" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: check for missing prerequisite headers?" >&5 +echo "$as_me: WARNING: $ac_header: check for missing prerequisite headers?" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: see the Autoconf documentation" >&5 +echo "$as_me: WARNING: $ac_header: see the Autoconf documentation" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&5 +echo "$as_me: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the preprocessor's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the preprocessor's result" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: in the future, the compiler will take precedence" >&5 +echo "$as_me: WARNING: $ac_header: in the future, the compiler will take precedence" >&2;} + ( + cat <<\_ASBOX +## ------------------------------------------ ## +## Report this to the AC_PACKAGE_NAME lists. ## +## ------------------------------------------ ## +_ASBOX + ) | + sed "s/^/$as_me: WARNING: /" >&2 + ;; +esac +echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + eval "$as_ac_Header=\$ac_header_preproc" +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 + +fi +if test `eval echo '${'$as_ac_Header'}'` = yes; then + cat >>confdefs.h <<_ACEOF +#define `echo "HAVE_$ac_header" | $as_tr_cpp` 1 +_ACEOF + +fi + +done + + +for ac_header in byteswap.h +do +as_ac_Header=`echo "ac_cv_header_$ac_header" | $as_tr_sh` +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 +else + # Is the header compilable? +echo "$as_me:$LINENO: checking $ac_header usability" >&5 +echo $ECHO_N "checking $ac_header usability... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +#include <$ac_header> +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_header_compiler=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_header_compiler=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_compiler" >&5 +echo "${ECHO_T}$ac_header_compiler" >&6 + +# Is the header present? +echo "$as_me:$LINENO: checking $ac_header presence" >&5 +echo $ECHO_N "checking $ac_header presence... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include <$ac_header> +_ACEOF +if { (eval echo "$as_me:$LINENO: \"$ac_cpp conftest.$ac_ext\"") >&5 + (eval $ac_cpp conftest.$ac_ext) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } >/dev/null; then + if test -s conftest.err; then + ac_cpp_err=$ac_c_preproc_warn_flag + ac_cpp_err=$ac_cpp_err$ac_c_werror_flag + else + ac_cpp_err= + fi +else + ac_cpp_err=yes +fi +if test -z "$ac_cpp_err"; then + ac_header_preproc=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + + ac_header_preproc=no +fi +rm -f conftest.err conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_preproc" >&5 +echo "${ECHO_T}$ac_header_preproc" >&6 + +# So? What about this header? +case $ac_header_compiler:$ac_header_preproc:$ac_c_preproc_warn_flag in + yes:no: ) + { echo "$as_me:$LINENO: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&5 +echo "$as_me: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the compiler's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the compiler's result" >&2;} + ac_header_preproc=yes + ;; + no:yes:* ) + { echo "$as_me:$LINENO: WARNING: $ac_header: present but cannot be compiled" >&5 +echo "$as_me: WARNING: $ac_header: present but cannot be compiled" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: check for missing prerequisite headers?" >&5 +echo "$as_me: WARNING: $ac_header: check for missing prerequisite headers?" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: see the Autoconf documentation" >&5 +echo "$as_me: WARNING: $ac_header: see the Autoconf documentation" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&5 +echo "$as_me: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the preprocessor's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the preprocessor's result" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: in the future, the compiler will take precedence" >&5 +echo "$as_me: WARNING: $ac_header: in the future, the compiler will take precedence" >&2;} + ( + cat <<\_ASBOX +## ------------------------------------------ ## +## Report this to the AC_PACKAGE_NAME lists. ## +## ------------------------------------------ ## +_ASBOX + ) | + sed "s/^/$as_me: WARNING: /" >&2 + ;; +esac +echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + eval "$as_ac_Header=\$ac_header_preproc" +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 + +fi +if test `eval echo '${'$as_ac_Header'}'` = yes; then + cat >>confdefs.h <<_ACEOF +#define `echo "HAVE_$ac_header" | $as_tr_cpp` 1 +_ACEOF + +fi + +done + + +for ac_header in stdint.h +do +as_ac_Header=`echo "ac_cv_header_$ac_header" | $as_tr_sh` +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 +else + # Is the header compilable? +echo "$as_me:$LINENO: checking $ac_header usability" >&5 +echo $ECHO_N "checking $ac_header usability... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +#include <$ac_header> +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_header_compiler=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_header_compiler=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_compiler" >&5 +echo "${ECHO_T}$ac_header_compiler" >&6 + +# Is the header present? +echo "$as_me:$LINENO: checking $ac_header presence" >&5 +echo $ECHO_N "checking $ac_header presence... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include <$ac_header> +_ACEOF +if { (eval echo "$as_me:$LINENO: \"$ac_cpp conftest.$ac_ext\"") >&5 + (eval $ac_cpp conftest.$ac_ext) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } >/dev/null; then + if test -s conftest.err; then + ac_cpp_err=$ac_c_preproc_warn_flag + ac_cpp_err=$ac_cpp_err$ac_c_werror_flag + else + ac_cpp_err= + fi +else + ac_cpp_err=yes +fi +if test -z "$ac_cpp_err"; then + ac_header_preproc=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + + ac_header_preproc=no +fi +rm -f conftest.err conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_preproc" >&5 +echo "${ECHO_T}$ac_header_preproc" >&6 + +# So? What about this header? +case $ac_header_compiler:$ac_header_preproc:$ac_c_preproc_warn_flag in + yes:no: ) + { echo "$as_me:$LINENO: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&5 +echo "$as_me: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the compiler's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the compiler's result" >&2;} + ac_header_preproc=yes + ;; + no:yes:* ) + { echo "$as_me:$LINENO: WARNING: $ac_header: present but cannot be compiled" >&5 +echo "$as_me: WARNING: $ac_header: present but cannot be compiled" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: check for missing prerequisite headers?" >&5 +echo "$as_me: WARNING: $ac_header: check for missing prerequisite headers?" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: see the Autoconf documentation" >&5 +echo "$as_me: WARNING: $ac_header: see the Autoconf documentation" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&5 +echo "$as_me: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the preprocessor's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the preprocessor's result" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: in the future, the compiler will take precedence" >&5 +echo "$as_me: WARNING: $ac_header: in the future, the compiler will take precedence" >&2;} + ( + cat <<\_ASBOX +## ------------------------------------------ ## +## Report this to the AC_PACKAGE_NAME lists. ## +## ------------------------------------------ ## +_ASBOX + ) | + sed "s/^/$as_me: WARNING: /" >&2 + ;; +esac +echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + eval "$as_ac_Header=\$ac_header_preproc" +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 + +fi +if test `eval echo '${'$as_ac_Header'}'` = yes; then + cat >>confdefs.h <<_ACEOF +#define `echo "HAVE_$ac_header" | $as_tr_cpp` 1 +_ACEOF + +fi + +done + + +for ac_header in sys/uio.h +do +as_ac_Header=`echo "ac_cv_header_$ac_header" | $as_tr_sh` +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 +else + # Is the header compilable? +echo "$as_me:$LINENO: checking $ac_header usability" >&5 +echo $ECHO_N "checking $ac_header usability... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +#include <$ac_header> +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_header_compiler=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_header_compiler=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_compiler" >&5 +echo "${ECHO_T}$ac_header_compiler" >&6 + +# Is the header present? +echo "$as_me:$LINENO: checking $ac_header presence" >&5 +echo $ECHO_N "checking $ac_header presence... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include <$ac_header> +_ACEOF +if { (eval echo "$as_me:$LINENO: \"$ac_cpp conftest.$ac_ext\"") >&5 + (eval $ac_cpp conftest.$ac_ext) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } >/dev/null; then + if test -s conftest.err; then + ac_cpp_err=$ac_c_preproc_warn_flag + ac_cpp_err=$ac_cpp_err$ac_c_werror_flag + else + ac_cpp_err= + fi +else + ac_cpp_err=yes +fi +if test -z "$ac_cpp_err"; then + ac_header_preproc=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + + ac_header_preproc=no +fi +rm -f conftest.err conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_preproc" >&5 +echo "${ECHO_T}$ac_header_preproc" >&6 + +# So? What about this header? +case $ac_header_compiler:$ac_header_preproc:$ac_c_preproc_warn_flag in + yes:no: ) + { echo "$as_me:$LINENO: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&5 +echo "$as_me: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the compiler's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the compiler's result" >&2;} + ac_header_preproc=yes + ;; + no:yes:* ) + { echo "$as_me:$LINENO: WARNING: $ac_header: present but cannot be compiled" >&5 +echo "$as_me: WARNING: $ac_header: present but cannot be compiled" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: check for missing prerequisite headers?" >&5 +echo "$as_me: WARNING: $ac_header: check for missing prerequisite headers?" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: see the Autoconf documentation" >&5 +echo "$as_me: WARNING: $ac_header: see the Autoconf documentation" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&5 +echo "$as_me: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the preprocessor's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the preprocessor's result" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: in the future, the compiler will take precedence" >&5 +echo "$as_me: WARNING: $ac_header: in the future, the compiler will take precedence" >&2;} + ( + cat <<\_ASBOX +## ------------------------------------------ ## +## Report this to the AC_PACKAGE_NAME lists. ## +## ------------------------------------------ ## +_ASBOX + ) | + sed "s/^/$as_me: WARNING: /" >&2 + ;; +esac +echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + eval "$as_ac_Header=\$ac_header_preproc" +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 + +fi +if test `eval echo '${'$as_ac_Header'}'` = yes; then + cat >>confdefs.h <<_ACEOF +#define `echo "HAVE_$ac_header" | $as_tr_cpp` 1 +_ACEOF + +fi + +done + + +for ac_header in inttypes.h +do +as_ac_Header=`echo "ac_cv_header_$ac_header" | $as_tr_sh` +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 +else + # Is the header compilable? +echo "$as_me:$LINENO: checking $ac_header usability" >&5 +echo $ECHO_N "checking $ac_header usability... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +#include <$ac_header> +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_header_compiler=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_header_compiler=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_compiler" >&5 +echo "${ECHO_T}$ac_header_compiler" >&6 + +# Is the header present? +echo "$as_me:$LINENO: checking $ac_header presence" >&5 +echo $ECHO_N "checking $ac_header presence... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include <$ac_header> +_ACEOF +if { (eval echo "$as_me:$LINENO: \"$ac_cpp conftest.$ac_ext\"") >&5 + (eval $ac_cpp conftest.$ac_ext) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } >/dev/null; then + if test -s conftest.err; then + ac_cpp_err=$ac_c_preproc_warn_flag + ac_cpp_err=$ac_cpp_err$ac_c_werror_flag + else + ac_cpp_err= + fi +else + ac_cpp_err=yes +fi +if test -z "$ac_cpp_err"; then + ac_header_preproc=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + + ac_header_preproc=no +fi +rm -f conftest.err conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_preproc" >&5 +echo "${ECHO_T}$ac_header_preproc" >&6 + +# So? What about this header? +case $ac_header_compiler:$ac_header_preproc:$ac_c_preproc_warn_flag in + yes:no: ) + { echo "$as_me:$LINENO: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&5 +echo "$as_me: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the compiler's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the compiler's result" >&2;} + ac_header_preproc=yes + ;; + no:yes:* ) + { echo "$as_me:$LINENO: WARNING: $ac_header: present but cannot be compiled" >&5 +echo "$as_me: WARNING: $ac_header: present but cannot be compiled" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: check for missing prerequisite headers?" >&5 +echo "$as_me: WARNING: $ac_header: check for missing prerequisite headers?" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: see the Autoconf documentation" >&5 +echo "$as_me: WARNING: $ac_header: see the Autoconf documentation" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&5 +echo "$as_me: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the preprocessor's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the preprocessor's result" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: in the future, the compiler will take precedence" >&5 +echo "$as_me: WARNING: $ac_header: in the future, the compiler will take precedence" >&2;} + ( + cat <<\_ASBOX +## ------------------------------------------ ## +## Report this to the AC_PACKAGE_NAME lists. ## +## ------------------------------------------ ## +_ASBOX + ) | + sed "s/^/$as_me: WARNING: /" >&2 + ;; +esac +echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + eval "$as_ac_Header=\$ac_header_preproc" +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 + +fi +if test `eval echo '${'$as_ac_Header'}'` = yes; then + cat >>confdefs.h <<_ACEOF +#define `echo "HAVE_$ac_header" | $as_tr_cpp` 1 +_ACEOF + +fi + +done + + +for ac_header in sys/types.h +do +as_ac_Header=`echo "ac_cv_header_$ac_header" | $as_tr_sh` +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 +else + # Is the header compilable? +echo "$as_me:$LINENO: checking $ac_header usability" >&5 +echo $ECHO_N "checking $ac_header usability... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +#include <$ac_header> +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_header_compiler=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_header_compiler=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_compiler" >&5 +echo "${ECHO_T}$ac_header_compiler" >&6 + +# Is the header present? +echo "$as_me:$LINENO: checking $ac_header presence" >&5 +echo $ECHO_N "checking $ac_header presence... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include <$ac_header> +_ACEOF +if { (eval echo "$as_me:$LINENO: \"$ac_cpp conftest.$ac_ext\"") >&5 + (eval $ac_cpp conftest.$ac_ext) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } >/dev/null; then + if test -s conftest.err; then + ac_cpp_err=$ac_c_preproc_warn_flag + ac_cpp_err=$ac_cpp_err$ac_c_werror_flag + else + ac_cpp_err= + fi +else + ac_cpp_err=yes +fi +if test -z "$ac_cpp_err"; then + ac_header_preproc=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + + ac_header_preproc=no +fi +rm -f conftest.err conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_preproc" >&5 +echo "${ECHO_T}$ac_header_preproc" >&6 + +# So? What about this header? +case $ac_header_compiler:$ac_header_preproc:$ac_c_preproc_warn_flag in + yes:no: ) + { echo "$as_me:$LINENO: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&5 +echo "$as_me: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the compiler's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the compiler's result" >&2;} + ac_header_preproc=yes + ;; + no:yes:* ) + { echo "$as_me:$LINENO: WARNING: $ac_header: present but cannot be compiled" >&5 +echo "$as_me: WARNING: $ac_header: present but cannot be compiled" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: check for missing prerequisite headers?" >&5 +echo "$as_me: WARNING: $ac_header: check for missing prerequisite headers?" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: see the Autoconf documentation" >&5 +echo "$as_me: WARNING: $ac_header: see the Autoconf documentation" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&5 +echo "$as_me: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the preprocessor's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the preprocessor's result" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: in the future, the compiler will take precedence" >&5 +echo "$as_me: WARNING: $ac_header: in the future, the compiler will take precedence" >&2;} + ( + cat <<\_ASBOX +## ------------------------------------------ ## +## Report this to the AC_PACKAGE_NAME lists. ## +## ------------------------------------------ ## +_ASBOX + ) | + sed "s/^/$as_me: WARNING: /" >&2 + ;; +esac +echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + eval "$as_ac_Header=\$ac_header_preproc" +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 + +fi +if test `eval echo '${'$as_ac_Header'}'` = yes; then + cat >>confdefs.h <<_ACEOF +#define `echo "HAVE_$ac_header" | $as_tr_cpp` 1 +_ACEOF + +fi + +done + + +for ac_header in machine/types.h +do +as_ac_Header=`echo "ac_cv_header_$ac_header" | $as_tr_sh` +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 +else + # Is the header compilable? +echo "$as_me:$LINENO: checking $ac_header usability" >&5 +echo $ECHO_N "checking $ac_header usability... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +#include <$ac_header> +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_header_compiler=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_header_compiler=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_compiler" >&5 +echo "${ECHO_T}$ac_header_compiler" >&6 + +# Is the header present? +echo "$as_me:$LINENO: checking $ac_header presence" >&5 +echo $ECHO_N "checking $ac_header presence... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include <$ac_header> +_ACEOF +if { (eval echo "$as_me:$LINENO: \"$ac_cpp conftest.$ac_ext\"") >&5 + (eval $ac_cpp conftest.$ac_ext) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } >/dev/null; then + if test -s conftest.err; then + ac_cpp_err=$ac_c_preproc_warn_flag + ac_cpp_err=$ac_cpp_err$ac_c_werror_flag + else + ac_cpp_err= + fi +else + ac_cpp_err=yes +fi +if test -z "$ac_cpp_err"; then + ac_header_preproc=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + + ac_header_preproc=no +fi +rm -f conftest.err conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_preproc" >&5 +echo "${ECHO_T}$ac_header_preproc" >&6 + +# So? What about this header? +case $ac_header_compiler:$ac_header_preproc:$ac_c_preproc_warn_flag in + yes:no: ) + { echo "$as_me:$LINENO: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&5 +echo "$as_me: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the compiler's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the compiler's result" >&2;} + ac_header_preproc=yes + ;; + no:yes:* ) + { echo "$as_me:$LINENO: WARNING: $ac_header: present but cannot be compiled" >&5 +echo "$as_me: WARNING: $ac_header: present but cannot be compiled" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: check for missing prerequisite headers?" >&5 +echo "$as_me: WARNING: $ac_header: check for missing prerequisite headers?" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: see the Autoconf documentation" >&5 +echo "$as_me: WARNING: $ac_header: see the Autoconf documentation" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&5 +echo "$as_me: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the preprocessor's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the preprocessor's result" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: in the future, the compiler will take precedence" >&5 +echo "$as_me: WARNING: $ac_header: in the future, the compiler will take precedence" >&2;} + ( + cat <<\_ASBOX +## ------------------------------------------ ## +## Report this to the AC_PACKAGE_NAME lists. ## +## ------------------------------------------ ## +_ASBOX + ) | + sed "s/^/$as_me: WARNING: /" >&2 + ;; +esac +echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + eval "$as_ac_Header=\$ac_header_preproc" +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 + +fi +if test `eval echo '${'$as_ac_Header'}'` = yes; then + cat >>confdefs.h <<_ACEOF +#define `echo "HAVE_$ac_header" | $as_tr_cpp` 1 +_ACEOF + +fi + +done + + +for ac_header in sys/int_types.h +do +as_ac_Header=`echo "ac_cv_header_$ac_header" | $as_tr_sh` +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 +else + # Is the header compilable? +echo "$as_me:$LINENO: checking $ac_header usability" >&5 +echo $ECHO_N "checking $ac_header usability... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +#include <$ac_header> +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_header_compiler=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_header_compiler=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_compiler" >&5 +echo "${ECHO_T}$ac_header_compiler" >&6 + +# Is the header present? +echo "$as_me:$LINENO: checking $ac_header presence" >&5 +echo $ECHO_N "checking $ac_header presence... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include <$ac_header> +_ACEOF +if { (eval echo "$as_me:$LINENO: \"$ac_cpp conftest.$ac_ext\"") >&5 + (eval $ac_cpp conftest.$ac_ext) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } >/dev/null; then + if test -s conftest.err; then + ac_cpp_err=$ac_c_preproc_warn_flag + ac_cpp_err=$ac_cpp_err$ac_c_werror_flag + else + ac_cpp_err= + fi +else + ac_cpp_err=yes +fi +if test -z "$ac_cpp_err"; then + ac_header_preproc=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + + ac_header_preproc=no +fi +rm -f conftest.err conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_preproc" >&5 +echo "${ECHO_T}$ac_header_preproc" >&6 + +# So? What about this header? +case $ac_header_compiler:$ac_header_preproc:$ac_c_preproc_warn_flag in + yes:no: ) + { echo "$as_me:$LINENO: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&5 +echo "$as_me: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the compiler's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the compiler's result" >&2;} + ac_header_preproc=yes + ;; + no:yes:* ) + { echo "$as_me:$LINENO: WARNING: $ac_header: present but cannot be compiled" >&5 +echo "$as_me: WARNING: $ac_header: present but cannot be compiled" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: check for missing prerequisite headers?" >&5 +echo "$as_me: WARNING: $ac_header: check for missing prerequisite headers?" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: see the Autoconf documentation" >&5 +echo "$as_me: WARNING: $ac_header: see the Autoconf documentation" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&5 +echo "$as_me: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the preprocessor's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the preprocessor's result" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: in the future, the compiler will take precedence" >&5 +echo "$as_me: WARNING: $ac_header: in the future, the compiler will take precedence" >&2;} + ( + cat <<\_ASBOX +## ------------------------------------------ ## +## Report this to the AC_PACKAGE_NAME lists. ## +## ------------------------------------------ ## +_ASBOX + ) | + sed "s/^/$as_me: WARNING: /" >&2 + ;; +esac +echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + eval "$as_ac_Header=\$ac_header_preproc" +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 + +fi +if test `eval echo '${'$as_ac_Header'}'` = yes; then + cat >>confdefs.h <<_ACEOF +#define `echo "HAVE_$ac_header" | $as_tr_cpp` 1 +_ACEOF + +fi + +done + + + + + +for ac_header in sys/socket.h netinet/in.h arpa/inet.h +do +as_ac_Header=`echo "ac_cv_header_$ac_header" | $as_tr_sh` +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 +else + # Is the header compilable? +echo "$as_me:$LINENO: checking $ac_header usability" >&5 +echo $ECHO_N "checking $ac_header usability... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +#include <$ac_header> +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_header_compiler=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_header_compiler=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_compiler" >&5 +echo "${ECHO_T}$ac_header_compiler" >&6 + +# Is the header present? +echo "$as_me:$LINENO: checking $ac_header presence" >&5 +echo $ECHO_N "checking $ac_header presence... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include <$ac_header> +_ACEOF +if { (eval echo "$as_me:$LINENO: \"$ac_cpp conftest.$ac_ext\"") >&5 + (eval $ac_cpp conftest.$ac_ext) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } >/dev/null; then + if test -s conftest.err; then + ac_cpp_err=$ac_c_preproc_warn_flag + ac_cpp_err=$ac_cpp_err$ac_c_werror_flag + else + ac_cpp_err= + fi +else + ac_cpp_err=yes +fi +if test -z "$ac_cpp_err"; then + ac_header_preproc=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + + ac_header_preproc=no +fi +rm -f conftest.err conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_preproc" >&5 +echo "${ECHO_T}$ac_header_preproc" >&6 + +# So? What about this header? +case $ac_header_compiler:$ac_header_preproc:$ac_c_preproc_warn_flag in + yes:no: ) + { echo "$as_me:$LINENO: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&5 +echo "$as_me: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the compiler's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the compiler's result" >&2;} + ac_header_preproc=yes + ;; + no:yes:* ) + { echo "$as_me:$LINENO: WARNING: $ac_header: present but cannot be compiled" >&5 +echo "$as_me: WARNING: $ac_header: present but cannot be compiled" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: check for missing prerequisite headers?" >&5 +echo "$as_me: WARNING: $ac_header: check for missing prerequisite headers?" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: see the Autoconf documentation" >&5 +echo "$as_me: WARNING: $ac_header: see the Autoconf documentation" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&5 +echo "$as_me: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the preprocessor's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the preprocessor's result" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: in the future, the compiler will take precedence" >&5 +echo "$as_me: WARNING: $ac_header: in the future, the compiler will take precedence" >&2;} + ( + cat <<\_ASBOX +## ------------------------------------------ ## +## Report this to the AC_PACKAGE_NAME lists. ## +## ------------------------------------------ ## +_ASBOX + ) | + sed "s/^/$as_me: WARNING: /" >&2 + ;; +esac +echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + eval "$as_ac_Header=\$ac_header_preproc" +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 + +fi +if test `eval echo '${'$as_ac_Header'}'` = yes; then + cat >>confdefs.h <<_ACEOF +#define `echo "HAVE_$ac_header" | $as_tr_cpp` 1 +_ACEOF + +fi + +done + + +for ac_header in windows.h +do +as_ac_Header=`echo "ac_cv_header_$ac_header" | $as_tr_sh` +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 +else + # Is the header compilable? +echo "$as_me:$LINENO: checking $ac_header usability" >&5 +echo $ECHO_N "checking $ac_header usability... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +#include <$ac_header> +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_header_compiler=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_header_compiler=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_compiler" >&5 +echo "${ECHO_T}$ac_header_compiler" >&6 + +# Is the header present? +echo "$as_me:$LINENO: checking $ac_header presence" >&5 +echo $ECHO_N "checking $ac_header presence... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include <$ac_header> +_ACEOF +if { (eval echo "$as_me:$LINENO: \"$ac_cpp conftest.$ac_ext\"") >&5 + (eval $ac_cpp conftest.$ac_ext) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } >/dev/null; then + if test -s conftest.err; then + ac_cpp_err=$ac_c_preproc_warn_flag + ac_cpp_err=$ac_cpp_err$ac_c_werror_flag + else + ac_cpp_err= + fi +else + ac_cpp_err=yes +fi +if test -z "$ac_cpp_err"; then + ac_header_preproc=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + + ac_header_preproc=no +fi +rm -f conftest.err conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_preproc" >&5 +echo "${ECHO_T}$ac_header_preproc" >&6 + +# So? What about this header? +case $ac_header_compiler:$ac_header_preproc:$ac_c_preproc_warn_flag in + yes:no: ) + { echo "$as_me:$LINENO: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&5 +echo "$as_me: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the compiler's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the compiler's result" >&2;} + ac_header_preproc=yes + ;; + no:yes:* ) + { echo "$as_me:$LINENO: WARNING: $ac_header: present but cannot be compiled" >&5 +echo "$as_me: WARNING: $ac_header: present but cannot be compiled" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: check for missing prerequisite headers?" >&5 +echo "$as_me: WARNING: $ac_header: check for missing prerequisite headers?" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: see the Autoconf documentation" >&5 +echo "$as_me: WARNING: $ac_header: see the Autoconf documentation" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&5 +echo "$as_me: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the preprocessor's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the preprocessor's result" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: in the future, the compiler will take precedence" >&5 +echo "$as_me: WARNING: $ac_header: in the future, the compiler will take precedence" >&2;} + ( + cat <<\_ASBOX +## ------------------------------------------ ## +## Report this to the AC_PACKAGE_NAME lists. ## +## ------------------------------------------ ## +_ASBOX + ) | + sed "s/^/$as_me: WARNING: /" >&2 + ;; +esac +echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + eval "$as_ac_Header=\$ac_header_preproc" +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 + +fi +if test `eval echo '${'$as_ac_Header'}'` = yes; then + cat >>confdefs.h <<_ACEOF +#define `echo "HAVE_$ac_header" | $as_tr_cpp` 1 +_ACEOF + +for ac_header in winsock2.h +do +as_ac_Header=`echo "ac_cv_header_$ac_header" | $as_tr_sh` +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 +else + # Is the header compilable? +echo "$as_me:$LINENO: checking $ac_header usability" >&5 +echo $ECHO_N "checking $ac_header usability... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +#include <$ac_header> +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_header_compiler=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_header_compiler=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_compiler" >&5 +echo "${ECHO_T}$ac_header_compiler" >&6 + +# Is the header present? +echo "$as_me:$LINENO: checking $ac_header presence" >&5 +echo $ECHO_N "checking $ac_header presence... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include <$ac_header> +_ACEOF +if { (eval echo "$as_me:$LINENO: \"$ac_cpp conftest.$ac_ext\"") >&5 + (eval $ac_cpp conftest.$ac_ext) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } >/dev/null; then + if test -s conftest.err; then + ac_cpp_err=$ac_c_preproc_warn_flag + ac_cpp_err=$ac_cpp_err$ac_c_werror_flag + else + ac_cpp_err= + fi +else + ac_cpp_err=yes +fi +if test -z "$ac_cpp_err"; then + ac_header_preproc=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + + ac_header_preproc=no +fi +rm -f conftest.err conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_preproc" >&5 +echo "${ECHO_T}$ac_header_preproc" >&6 + +# So? What about this header? +case $ac_header_compiler:$ac_header_preproc:$ac_c_preproc_warn_flag in + yes:no: ) + { echo "$as_me:$LINENO: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&5 +echo "$as_me: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the compiler's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the compiler's result" >&2;} + ac_header_preproc=yes + ;; + no:yes:* ) + { echo "$as_me:$LINENO: WARNING: $ac_header: present but cannot be compiled" >&5 +echo "$as_me: WARNING: $ac_header: present but cannot be compiled" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: check for missing prerequisite headers?" >&5 +echo "$as_me: WARNING: $ac_header: check for missing prerequisite headers?" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: see the Autoconf documentation" >&5 +echo "$as_me: WARNING: $ac_header: see the Autoconf documentation" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&5 +echo "$as_me: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the preprocessor's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the preprocessor's result" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: in the future, the compiler will take precedence" >&5 +echo "$as_me: WARNING: $ac_header: in the future, the compiler will take precedence" >&2;} + ( + cat <<\_ASBOX +## ------------------------------------------ ## +## Report this to the AC_PACKAGE_NAME lists. ## +## ------------------------------------------ ## +_ASBOX + ) | + sed "s/^/$as_me: WARNING: /" >&2 + ;; +esac +echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + eval "$as_ac_Header=\$ac_header_preproc" +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 + +fi +if test `eval echo '${'$as_ac_Header'}'` = yes; then + cat >>confdefs.h <<_ACEOF +#define `echo "HAVE_$ac_header" | $as_tr_cpp` 1 +_ACEOF + +fi + +done + +fi + +done + + + +for ac_header in syslog.h +do +as_ac_Header=`echo "ac_cv_header_$ac_header" | $as_tr_sh` +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 +else + # Is the header compilable? +echo "$as_me:$LINENO: checking $ac_header usability" >&5 +echo $ECHO_N "checking $ac_header usability... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +#include <$ac_header> +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_header_compiler=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_header_compiler=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_compiler" >&5 +echo "${ECHO_T}$ac_header_compiler" >&6 + +# Is the header present? +echo "$as_me:$LINENO: checking $ac_header presence" >&5 +echo $ECHO_N "checking $ac_header presence... $ECHO_C" >&6 +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include <$ac_header> +_ACEOF +if { (eval echo "$as_me:$LINENO: \"$ac_cpp conftest.$ac_ext\"") >&5 + (eval $ac_cpp conftest.$ac_ext) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } >/dev/null; then + if test -s conftest.err; then + ac_cpp_err=$ac_c_preproc_warn_flag + ac_cpp_err=$ac_cpp_err$ac_c_werror_flag + else + ac_cpp_err= + fi +else + ac_cpp_err=yes +fi +if test -z "$ac_cpp_err"; then + ac_header_preproc=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + + ac_header_preproc=no +fi +rm -f conftest.err conftest.$ac_ext +echo "$as_me:$LINENO: result: $ac_header_preproc" >&5 +echo "${ECHO_T}$ac_header_preproc" >&6 + +# So? What about this header? +case $ac_header_compiler:$ac_header_preproc:$ac_c_preproc_warn_flag in + yes:no: ) + { echo "$as_me:$LINENO: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&5 +echo "$as_me: WARNING: $ac_header: accepted by the compiler, rejected by the preprocessor!" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the compiler's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the compiler's result" >&2;} + ac_header_preproc=yes + ;; + no:yes:* ) + { echo "$as_me:$LINENO: WARNING: $ac_header: present but cannot be compiled" >&5 +echo "$as_me: WARNING: $ac_header: present but cannot be compiled" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: check for missing prerequisite headers?" >&5 +echo "$as_me: WARNING: $ac_header: check for missing prerequisite headers?" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: see the Autoconf documentation" >&5 +echo "$as_me: WARNING: $ac_header: see the Autoconf documentation" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&5 +echo "$as_me: WARNING: $ac_header: section \"Present But Cannot Be Compiled\"" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: proceeding with the preprocessor's result" >&5 +echo "$as_me: WARNING: $ac_header: proceeding with the preprocessor's result" >&2;} + { echo "$as_me:$LINENO: WARNING: $ac_header: in the future, the compiler will take precedence" >&5 +echo "$as_me: WARNING: $ac_header: in the future, the compiler will take precedence" >&2;} + ( + cat <<\_ASBOX +## ------------------------------------------ ## +## Report this to the AC_PACKAGE_NAME lists. ## +## ------------------------------------------ ## +_ASBOX + ) | + sed "s/^/$as_me: WARNING: /" >&2 + ;; +esac +echo "$as_me:$LINENO: checking for $ac_header" >&5 +echo $ECHO_N "checking for $ac_header... $ECHO_C" >&6 +if eval "test \"\${$as_ac_Header+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + eval "$as_ac_Header=\$ac_header_preproc" +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_Header'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_Header'}'`" >&6 + +fi +if test `eval echo '${'$as_ac_Header'}'` = yes; then + cat >>confdefs.h <<_ACEOF +#define `echo "HAVE_$ac_header" | $as_tr_cpp` 1 +_ACEOF + +fi + +done + + +echo "$as_me:$LINENO: checking for int8_t" >&5 +echo $ECHO_N "checking for int8_t... $ECHO_C" >&6 +if test "${ac_cv_type_int8_t+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +if ((int8_t *) 0) + return 0; +if (sizeof (int8_t)) + return 0; + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_type_int8_t=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_cv_type_int8_t=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +fi +echo "$as_me:$LINENO: result: $ac_cv_type_int8_t" >&5 +echo "${ECHO_T}$ac_cv_type_int8_t" >&6 +if test $ac_cv_type_int8_t = yes; then + +cat >>confdefs.h <<_ACEOF +#define HAVE_INT8_T 1 +_ACEOF + + +fi +echo "$as_me:$LINENO: checking for uint8_t" >&5 +echo $ECHO_N "checking for uint8_t... $ECHO_C" >&6 +if test "${ac_cv_type_uint8_t+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +if ((uint8_t *) 0) + return 0; +if (sizeof (uint8_t)) + return 0; + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_type_uint8_t=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_cv_type_uint8_t=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +fi +echo "$as_me:$LINENO: result: $ac_cv_type_uint8_t" >&5 +echo "${ECHO_T}$ac_cv_type_uint8_t" >&6 +if test $ac_cv_type_uint8_t = yes; then + +cat >>confdefs.h <<_ACEOF +#define HAVE_UINT8_T 1 +_ACEOF + + +fi +echo "$as_me:$LINENO: checking for int16_t" >&5 +echo $ECHO_N "checking for int16_t... $ECHO_C" >&6 +if test "${ac_cv_type_int16_t+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +if ((int16_t *) 0) + return 0; +if (sizeof (int16_t)) + return 0; + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_type_int16_t=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_cv_type_int16_t=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +fi +echo "$as_me:$LINENO: result: $ac_cv_type_int16_t" >&5 +echo "${ECHO_T}$ac_cv_type_int16_t" >&6 +if test $ac_cv_type_int16_t = yes; then + +cat >>confdefs.h <<_ACEOF +#define HAVE_INT16_T 1 +_ACEOF + + +fi +echo "$as_me:$LINENO: checking for uint16_t" >&5 +echo $ECHO_N "checking for uint16_t... $ECHO_C" >&6 +if test "${ac_cv_type_uint16_t+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +if ((uint16_t *) 0) + return 0; +if (sizeof (uint16_t)) + return 0; + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_type_uint16_t=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_cv_type_uint16_t=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +fi +echo "$as_me:$LINENO: result: $ac_cv_type_uint16_t" >&5 +echo "${ECHO_T}$ac_cv_type_uint16_t" >&6 +if test $ac_cv_type_uint16_t = yes; then + +cat >>confdefs.h <<_ACEOF +#define HAVE_UINT16_T 1 +_ACEOF + + +fi +echo "$as_me:$LINENO: checking for int32_t" >&5 +echo $ECHO_N "checking for int32_t... $ECHO_C" >&6 +if test "${ac_cv_type_int32_t+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +if ((int32_t *) 0) + return 0; +if (sizeof (int32_t)) + return 0; + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_type_int32_t=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_cv_type_int32_t=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +fi +echo "$as_me:$LINENO: result: $ac_cv_type_int32_t" >&5 +echo "${ECHO_T}$ac_cv_type_int32_t" >&6 +if test $ac_cv_type_int32_t = yes; then + +cat >>confdefs.h <<_ACEOF +#define HAVE_INT32_T 1 +_ACEOF + + +fi +echo "$as_me:$LINENO: checking for uint32_t" >&5 +echo $ECHO_N "checking for uint32_t... $ECHO_C" >&6 +if test "${ac_cv_type_uint32_t+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +if ((uint32_t *) 0) + return 0; +if (sizeof (uint32_t)) + return 0; + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_type_uint32_t=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_cv_type_uint32_t=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +fi +echo "$as_me:$LINENO: result: $ac_cv_type_uint32_t" >&5 +echo "${ECHO_T}$ac_cv_type_uint32_t" >&6 +if test $ac_cv_type_uint32_t = yes; then + +cat >>confdefs.h <<_ACEOF +#define HAVE_UINT32_T 1 +_ACEOF + + +fi +echo "$as_me:$LINENO: checking for uint64_t" >&5 +echo $ECHO_N "checking for uint64_t... $ECHO_C" >&6 +if test "${ac_cv_type_uint64_t+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +if ((uint64_t *) 0) + return 0; +if (sizeof (uint64_t)) + return 0; + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_type_uint64_t=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_cv_type_uint64_t=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +fi +echo "$as_me:$LINENO: result: $ac_cv_type_uint64_t" >&5 +echo "${ECHO_T}$ac_cv_type_uint64_t" >&6 +if test $ac_cv_type_uint64_t = yes; then + +cat >>confdefs.h <<_ACEOF +#define HAVE_UINT64_T 1 +_ACEOF + + +fi + +echo "$as_me:$LINENO: checking for unsigned long" >&5 +echo $ECHO_N "checking for unsigned long... $ECHO_C" >&6 +if test "${ac_cv_type_unsigned_long+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +if ((unsigned long *) 0) + return 0; +if (sizeof (unsigned long)) + return 0; + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_type_unsigned_long=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_cv_type_unsigned_long=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +fi +echo "$as_me:$LINENO: result: $ac_cv_type_unsigned_long" >&5 +echo "${ECHO_T}$ac_cv_type_unsigned_long" >&6 + +echo "$as_me:$LINENO: checking size of unsigned long" >&5 +echo $ECHO_N "checking size of unsigned long... $ECHO_C" >&6 +if test "${ac_cv_sizeof_unsigned_long+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + if test "$ac_cv_type_unsigned_long" = yes; then + # The cast to unsigned long works around a bug in the HP C Compiler + # version HP92453-01 B.11.11.23709.GP, which incorrectly rejects + # declarations like `int a3[[(sizeof (unsigned char)) >= 0]];'. + # This bug is HP SR number 8606223364. + if test "$cross_compiling" = yes; then + # Depending upon the size, compute the lo and hi bounds. +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +static int test_array [1 - 2 * !(((long) (sizeof (unsigned long))) >= 0)]; +test_array [0] = 0 + + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_lo=0 ac_mid=0 + while :; do + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +static int test_array [1 - 2 * !(((long) (sizeof (unsigned long))) <= $ac_mid)]; +test_array [0] = 0 + + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_hi=$ac_mid; break +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_lo=`expr $ac_mid + 1` + if test $ac_lo -le $ac_mid; then + ac_lo= ac_hi= + break + fi + ac_mid=`expr 2 '*' $ac_mid + 1` +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext + done +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +static int test_array [1 - 2 * !(((long) (sizeof (unsigned long))) < 0)]; +test_array [0] = 0 + + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_hi=-1 ac_mid=-1 + while :; do + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +static int test_array [1 - 2 * !(((long) (sizeof (unsigned long))) >= $ac_mid)]; +test_array [0] = 0 + + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_lo=$ac_mid; break +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_hi=`expr '(' $ac_mid ')' - 1` + if test $ac_mid -le $ac_hi; then + ac_lo= ac_hi= + break + fi + ac_mid=`expr 2 '*' $ac_mid` +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext + done +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_lo= ac_hi= +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +# Binary search between lo and hi bounds. +while test "x$ac_lo" != "x$ac_hi"; do + ac_mid=`expr '(' $ac_hi - $ac_lo ')' / 2 + $ac_lo` + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +static int test_array [1 - 2 * !(((long) (sizeof (unsigned long))) <= $ac_mid)]; +test_array [0] = 0 + + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_hi=$ac_mid +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_lo=`expr '(' $ac_mid ')' + 1` +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +done +case $ac_lo in +?*) ac_cv_sizeof_unsigned_long=$ac_lo;; +'') { { echo "$as_me:$LINENO: error: cannot compute sizeof (unsigned long), 77 +See \`config.log' for more details." >&5 +echo "$as_me: error: cannot compute sizeof (unsigned long), 77 +See \`config.log' for more details." >&2;} + { (exit 1); exit 1; }; } ;; +esac +else + if test "$cross_compiling" = yes; then + { { echo "$as_me:$LINENO: error: cannot run test program while cross compiling +See \`config.log' for more details." >&5 +echo "$as_me: error: cannot run test program while cross compiling +See \`config.log' for more details." >&2;} + { (exit 1); exit 1; }; } +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +long longval () { return (long) (sizeof (unsigned long)); } +unsigned long ulongval () { return (long) (sizeof (unsigned long)); } +#include +#include +int +main () +{ + + FILE *f = fopen ("conftest.val", "w"); + if (! f) + exit (1); + if (((long) (sizeof (unsigned long))) < 0) + { + long i = longval (); + if (i != ((long) (sizeof (unsigned long)))) + exit (1); + fprintf (f, "%ld\n", i); + } + else + { + unsigned long i = ulongval (); + if (i != ((long) (sizeof (unsigned long)))) + exit (1); + fprintf (f, "%lu\n", i); + } + exit (ferror (f) || fclose (f) != 0); + + ; + return 0; +} +_ACEOF +rm -f conftest$ac_exeext +if { (eval echo "$as_me:$LINENO: \"$ac_link\"") >&5 + (eval $ac_link) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && { ac_try='./conftest$ac_exeext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_sizeof_unsigned_long=`cat conftest.val` +else + echo "$as_me: program exited with status $ac_status" >&5 +echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +( exit $ac_status ) +{ { echo "$as_me:$LINENO: error: cannot compute sizeof (unsigned long), 77 +See \`config.log' for more details." >&5 +echo "$as_me: error: cannot compute sizeof (unsigned long), 77 +See \`config.log' for more details." >&2;} + { (exit 1); exit 1; }; } +fi +rm -f core *.core gmon.out bb.out conftest$ac_exeext conftest.$ac_objext conftest.$ac_ext +fi +fi +rm -f conftest.val +else + ac_cv_sizeof_unsigned_long=0 +fi +fi +echo "$as_me:$LINENO: result: $ac_cv_sizeof_unsigned_long" >&5 +echo "${ECHO_T}$ac_cv_sizeof_unsigned_long" >&6 +cat >>confdefs.h <<_ACEOF +#define SIZEOF_UNSIGNED_LONG $ac_cv_sizeof_unsigned_long +_ACEOF + + +echo "$as_me:$LINENO: checking for unsigned long long" >&5 +echo $ECHO_N "checking for unsigned long long... $ECHO_C" >&6 +if test "${ac_cv_type_unsigned_long_long+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +if ((unsigned long long *) 0) + return 0; +if (sizeof (unsigned long long)) + return 0; + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_type_unsigned_long_long=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_cv_type_unsigned_long_long=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +fi +echo "$as_me:$LINENO: result: $ac_cv_type_unsigned_long_long" >&5 +echo "${ECHO_T}$ac_cv_type_unsigned_long_long" >&6 + +echo "$as_me:$LINENO: checking size of unsigned long long" >&5 +echo $ECHO_N "checking size of unsigned long long... $ECHO_C" >&6 +if test "${ac_cv_sizeof_unsigned_long_long+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + if test "$ac_cv_type_unsigned_long_long" = yes; then + # The cast to unsigned long works around a bug in the HP C Compiler + # version HP92453-01 B.11.11.23709.GP, which incorrectly rejects + # declarations like `int a3[[(sizeof (unsigned char)) >= 0]];'. + # This bug is HP SR number 8606223364. + if test "$cross_compiling" = yes; then + # Depending upon the size, compute the lo and hi bounds. +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +static int test_array [1 - 2 * !(((long) (sizeof (unsigned long long))) >= 0)]; +test_array [0] = 0 + + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_lo=0 ac_mid=0 + while :; do + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +static int test_array [1 - 2 * !(((long) (sizeof (unsigned long long))) <= $ac_mid)]; +test_array [0] = 0 + + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_hi=$ac_mid; break +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_lo=`expr $ac_mid + 1` + if test $ac_lo -le $ac_mid; then + ac_lo= ac_hi= + break + fi + ac_mid=`expr 2 '*' $ac_mid + 1` +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext + done +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +static int test_array [1 - 2 * !(((long) (sizeof (unsigned long long))) < 0)]; +test_array [0] = 0 + + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_hi=-1 ac_mid=-1 + while :; do + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +static int test_array [1 - 2 * !(((long) (sizeof (unsigned long long))) >= $ac_mid)]; +test_array [0] = 0 + + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_lo=$ac_mid; break +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_hi=`expr '(' $ac_mid ')' - 1` + if test $ac_mid -le $ac_hi; then + ac_lo= ac_hi= + break + fi + ac_mid=`expr 2 '*' $ac_mid` +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext + done +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_lo= ac_hi= +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +# Binary search between lo and hi bounds. +while test "x$ac_lo" != "x$ac_hi"; do + ac_mid=`expr '(' $ac_hi - $ac_lo ')' / 2 + $ac_lo` + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +static int test_array [1 - 2 * !(((long) (sizeof (unsigned long long))) <= $ac_mid)]; +test_array [0] = 0 + + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_hi=$ac_mid +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_lo=`expr '(' $ac_mid ')' + 1` +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +done +case $ac_lo in +?*) ac_cv_sizeof_unsigned_long_long=$ac_lo;; +'') { { echo "$as_me:$LINENO: error: cannot compute sizeof (unsigned long long), 77 +See \`config.log' for more details." >&5 +echo "$as_me: error: cannot compute sizeof (unsigned long long), 77 +See \`config.log' for more details." >&2;} + { (exit 1); exit 1; }; } ;; +esac +else + if test "$cross_compiling" = yes; then + { { echo "$as_me:$LINENO: error: cannot run test program while cross compiling +See \`config.log' for more details." >&5 +echo "$as_me: error: cannot run test program while cross compiling +See \`config.log' for more details." >&2;} + { (exit 1); exit 1; }; } +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +long longval () { return (long) (sizeof (unsigned long long)); } +unsigned long ulongval () { return (long) (sizeof (unsigned long long)); } +#include +#include +int +main () +{ + + FILE *f = fopen ("conftest.val", "w"); + if (! f) + exit (1); + if (((long) (sizeof (unsigned long long))) < 0) + { + long i = longval (); + if (i != ((long) (sizeof (unsigned long long)))) + exit (1); + fprintf (f, "%ld\n", i); + } + else + { + unsigned long i = ulongval (); + if (i != ((long) (sizeof (unsigned long long)))) + exit (1); + fprintf (f, "%lu\n", i); + } + exit (ferror (f) || fclose (f) != 0); + + ; + return 0; +} +_ACEOF +rm -f conftest$ac_exeext +if { (eval echo "$as_me:$LINENO: \"$ac_link\"") >&5 + (eval $ac_link) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && { ac_try='./conftest$ac_exeext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_sizeof_unsigned_long_long=`cat conftest.val` +else + echo "$as_me: program exited with status $ac_status" >&5 +echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +( exit $ac_status ) +{ { echo "$as_me:$LINENO: error: cannot compute sizeof (unsigned long long), 77 +See \`config.log' for more details." >&5 +echo "$as_me: error: cannot compute sizeof (unsigned long long), 77 +See \`config.log' for more details." >&2;} + { (exit 1); exit 1; }; } +fi +rm -f core *.core gmon.out bb.out conftest$ac_exeext conftest.$ac_objext conftest.$ac_ext +fi +fi +rm -f conftest.val +else + ac_cv_sizeof_unsigned_long_long=0 +fi +fi +echo "$as_me:$LINENO: result: $ac_cv_sizeof_unsigned_long_long" >&5 +echo "${ECHO_T}$ac_cv_sizeof_unsigned_long_long" >&6 +cat >>confdefs.h <<_ACEOF +#define SIZEOF_UNSIGNED_LONG_LONG $ac_cv_sizeof_unsigned_long_long +_ACEOF + + + +echo "$as_me:$LINENO: checking for an ANSI C-conforming const" >&5 +echo $ECHO_N "checking for an ANSI C-conforming const... $ECHO_C" >&6 +if test "${ac_cv_c_const+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ + +int +main () +{ +/* FIXME: Include the comments suggested by Paul. */ +#ifndef __cplusplus + /* Ultrix mips cc rejects this. */ + typedef int charset[2]; + const charset x; + /* SunOS 4.1.1 cc rejects this. */ + char const *const *ccp; + char **p; + /* NEC SVR4.0.2 mips cc rejects this. */ + struct point {int x, y;}; + static struct point const zero = {0,0}; + /* AIX XL C 1.02.0.0 rejects this. + It does not let you subtract one const X* pointer from another in + an arm of an if-expression whose if-part is not a constant + expression */ + const char *g = "string"; + ccp = &g + (g ? g-g : 0); + /* HPUX 7.0 cc rejects these. */ + ++ccp; + p = (char**) ccp; + ccp = (char const *const *) p; + { /* SCO 3.2v4 cc rejects this. */ + char *t; + char const *s = 0 ? (char *) 0 : (char const *) 0; + + *t++ = 0; + } + { /* Someone thinks the Sun supposedly-ANSI compiler will reject this. */ + int x[] = {25, 17}; + const int *foo = &x[0]; + ++foo; + } + { /* Sun SC1.0 ANSI compiler rejects this -- but not the above. */ + typedef const int *iptr; + iptr p = 0; + ++p; + } + { /* AIX XL C 1.02.0.0 rejects this saying + "k.c", line 2.27: 1506-025 (S) Operand must be a modifiable lvalue. */ + struct s { int j; const int *ap[3]; }; + struct s *b; b->j = 5; + } + { /* ULTRIX-32 V3.1 (Rev 9) vcc rejects this */ + const int foo = 10; + } +#endif + + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_c_const=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_cv_c_const=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +fi +echo "$as_me:$LINENO: result: $ac_cv_c_const" >&5 +echo "${ECHO_T}$ac_cv_c_const" >&6 +if test $ac_cv_c_const = no; then + +cat >>confdefs.h <<\_ACEOF +#define const +_ACEOF + +fi + +echo "$as_me:$LINENO: checking for inline" >&5 +echo $ECHO_N "checking for inline... $ECHO_C" >&6 +if test "${ac_cv_c_inline+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + ac_cv_c_inline=no +for ac_kw in inline __inline__ __inline; do + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#ifndef __cplusplus +typedef int foo_t; +static $ac_kw foo_t static_foo () {return 0; } +$ac_kw foo_t foo () {return 0; } +#endif + +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_c_inline=$ac_kw; break +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +done + +fi +echo "$as_me:$LINENO: result: $ac_cv_c_inline" >&5 +echo "${ECHO_T}$ac_cv_c_inline" >&6 + + +case $ac_cv_c_inline in + inline | yes) ;; + *) + case $ac_cv_c_inline in + no) ac_val=;; + *) ac_val=$ac_cv_c_inline;; + esac + cat >>confdefs.h <<_ACEOF +#ifndef __cplusplus +#define inline $ac_val +#endif +_ACEOF + ;; +esac + +echo "$as_me:$LINENO: checking for size_t" >&5 +echo $ECHO_N "checking for size_t... $ECHO_C" >&6 +if test "${ac_cv_type_size_t+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +$ac_includes_default +int +main () +{ +if ((size_t *) 0) + return 0; +if (sizeof (size_t)) + return 0; + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_type_size_t=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_cv_type_size_t=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +fi +echo "$as_me:$LINENO: result: $ac_cv_type_size_t" >&5 +echo "${ECHO_T}$ac_cv_type_size_t" >&6 +if test $ac_cv_type_size_t = yes; then + : +else + +cat >>confdefs.h <<_ACEOF +#define size_t unsigned +_ACEOF + +fi + + + + + +for ac_func in socket inet_aton usleep +do +as_ac_var=`echo "ac_cv_func_$ac_func" | $as_tr_sh` +echo "$as_me:$LINENO: checking for $ac_func" >&5 +echo $ECHO_N "checking for $ac_func... $ECHO_C" >&6 +if eval "test \"\${$as_ac_var+set}\" = set"; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +/* Define $ac_func to an innocuous variant, in case declares $ac_func. + For example, HP-UX 11i declares gettimeofday. */ +#define $ac_func innocuous_$ac_func + +/* System header to define __stub macros and hopefully few prototypes, + which can conflict with char $ac_func (); below. + Prefer to if __STDC__ is defined, since + exists even on freestanding compilers. */ + +#ifdef __STDC__ +# include +#else +# include +#endif + +#undef $ac_func + +/* Override any gcc2 internal prototype to avoid an error. */ +#ifdef __cplusplus +extern "C" +{ +#endif +/* We use char because int might match the return type of a gcc2 + builtin and then its argument prototype would still apply. */ +char $ac_func (); +/* The GNU C library defines this for functions which it implements + to always fail with ENOSYS. Some functions are actually named + something starting with __ and the normal name is an alias. */ +#if defined (__stub_$ac_func) || defined (__stub___$ac_func) +choke me +#else +char (*f) () = $ac_func; +#endif +#ifdef __cplusplus +} +#endif + +int +main () +{ +return f != $ac_func; + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext conftest$ac_exeext +if { (eval echo "$as_me:$LINENO: \"$ac_link\"") >&5 + (eval $ac_link) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest$ac_exeext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + eval "$as_ac_var=yes" +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +eval "$as_ac_var=no" +fi +rm -f conftest.err conftest.$ac_objext \ + conftest$ac_exeext conftest.$ac_ext +fi +echo "$as_me:$LINENO: result: `eval echo '${'$as_ac_var'}'`" >&5 +echo "${ECHO_T}`eval echo '${'$as_ac_var'}'`" >&6 +if test `eval echo '${'$as_ac_var'}'` = yes; then + cat >>confdefs.h <<_ACEOF +#define `echo "HAVE_$ac_func" | $as_tr_cpp` 1 +_ACEOF + +fi +done + + +if test "x$ac_cv_func_socket" = "xno"; then + +echo "$as_me:$LINENO: checking for socket in -lsocket" >&5 +echo $ECHO_N "checking for socket in -lsocket... $ECHO_C" >&6 +if test "${ac_cv_lib_socket_socket+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + ac_check_lib_save_LIBS=$LIBS +LIBS="-lsocket $LIBS" +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ + +/* Override any gcc2 internal prototype to avoid an error. */ +#ifdef __cplusplus +extern "C" +#endif +/* We use char because int might match the return type of a gcc2 + builtin and then its argument prototype would still apply. */ +char socket (); +int +main () +{ +socket (); + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext conftest$ac_exeext +if { (eval echo "$as_me:$LINENO: \"$ac_link\"") >&5 + (eval $ac_link) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest$ac_exeext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_lib_socket_socket=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_cv_lib_socket_socket=no +fi +rm -f conftest.err conftest.$ac_objext \ + conftest$ac_exeext conftest.$ac_ext +LIBS=$ac_check_lib_save_LIBS +fi +echo "$as_me:$LINENO: result: $ac_cv_lib_socket_socket" >&5 +echo "${ECHO_T}$ac_cv_lib_socket_socket" >&6 +if test $ac_cv_lib_socket_socket = yes; then + cat >>confdefs.h <<_ACEOF +#define HAVE_LIBSOCKET 1 +_ACEOF + + LIBS="-lsocket $LIBS" + +fi + + echo "$as_me:$LINENO: checking for socket in -lwsock32" >&5 +echo $ECHO_N "checking for socket in -lwsock32... $ECHO_C" >&6 + SAVELIBS="$LIBS" + LIBS="$LIBS -lwsock32" + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ + +#include + +int +main () +{ + +socket(0, 0, 0); + + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext conftest$ac_exeext +if { (eval echo "$as_me:$LINENO: \"$ac_link\"") >&5 + (eval $ac_link) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest$ac_exeext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_func_socket=yes + echo "$as_me:$LINENO: result: yes" >&5 +echo "${ECHO_T}yes" >&6 +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +LIBS="$SAVELIBS" + echo "$as_me:$LINENO: result: no" >&5 +echo "${ECHO_T}no" >&6 +fi +rm -f conftest.err conftest.$ac_objext \ + conftest$ac_exeext conftest.$ac_ext +fi + +echo "$as_me:$LINENO: checking whether byte ordering is bigendian" >&5 +echo $ECHO_N "checking whether byte ordering is bigendian... $ECHO_C" >&6 +if test "${ac_cv_c_bigendian+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + # See if sys/param.h defines the BYTE_ORDER macro. +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include +#include + +int +main () +{ +#if !BYTE_ORDER || !BIG_ENDIAN || !LITTLE_ENDIAN + bogus endian macros +#endif + + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + # It does; now see whether it defined to BIG_ENDIAN or not. +cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +#include +#include + +int +main () +{ +#if BYTE_ORDER != BIG_ENDIAN + not big endian +#endif + + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_c_bigendian=yes +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +ac_cv_c_bigendian=no +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +# It does not; compile a test program. +if test "$cross_compiling" = yes; then + # try to guess the endianness by grepping values into an object file + ac_cv_c_bigendian=unknown + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +short ascii_mm[] = { 0x4249, 0x4765, 0x6E44, 0x6961, 0x6E53, 0x7953, 0 }; +short ascii_ii[] = { 0x694C, 0x5454, 0x656C, 0x6E45, 0x6944, 0x6E61, 0 }; +void _ascii () { char *s = (char *) ascii_mm; s = (char *) ascii_ii; } +short ebcdic_ii[] = { 0x89D3, 0xE3E3, 0x8593, 0x95C5, 0x89C4, 0x9581, 0 }; +short ebcdic_mm[] = { 0xC2C9, 0xC785, 0x95C4, 0x8981, 0x95E2, 0xA8E2, 0 }; +void _ebcdic () { char *s = (char *) ebcdic_mm; s = (char *) ebcdic_ii; } +int +main () +{ + _ascii (); _ebcdic (); + ; + return 0; +} +_ACEOF +rm -f conftest.$ac_objext +if { (eval echo "$as_me:$LINENO: \"$ac_compile\"") >&5 + (eval $ac_compile) 2>conftest.er1 + ac_status=$? + grep -v '^ *+' conftest.er1 >conftest.err + rm -f conftest.er1 + cat conftest.err >&5 + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && + { ac_try='test -z "$ac_c_werror_flag" + || test ! -s conftest.err' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; } && + { ac_try='test -s conftest.$ac_objext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + if grep BIGenDianSyS conftest.$ac_objext >/dev/null ; then + ac_cv_c_bigendian=yes +fi +if grep LiTTleEnDian conftest.$ac_objext >/dev/null ; then + if test "$ac_cv_c_bigendian" = unknown; then + ac_cv_c_bigendian=no + else + # finding both strings is unlikely to happen, but who knows? + ac_cv_c_bigendian=unknown + fi +fi +else + echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +else + cat >conftest.$ac_ext <<_ACEOF +/* confdefs.h. */ +_ACEOF +cat confdefs.h >>conftest.$ac_ext +cat >>conftest.$ac_ext <<_ACEOF +/* end confdefs.h. */ +int +main () +{ + /* Are we little or big endian? From Harbison&Steele. */ + union + { + long l; + char c[sizeof (long)]; + } u; + u.l = 1; + exit (u.c[sizeof (long) - 1] == 1); +} +_ACEOF +rm -f conftest$ac_exeext +if { (eval echo "$as_me:$LINENO: \"$ac_link\"") >&5 + (eval $ac_link) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); } && { ac_try='./conftest$ac_exeext' + { (eval echo "$as_me:$LINENO: \"$ac_try\"") >&5 + (eval $ac_try) 2>&5 + ac_status=$? + echo "$as_me:$LINENO: \$? = $ac_status" >&5 + (exit $ac_status); }; }; then + ac_cv_c_bigendian=no +else + echo "$as_me: program exited with status $ac_status" >&5 +echo "$as_me: failed program was:" >&5 +sed 's/^/| /' conftest.$ac_ext >&5 + +( exit $ac_status ) +ac_cv_c_bigendian=yes +fi +rm -f core *.core gmon.out bb.out conftest$ac_exeext conftest.$ac_objext conftest.$ac_ext +fi +fi +rm -f conftest.err conftest.$ac_objext conftest.$ac_ext +fi +echo "$as_me:$LINENO: result: $ac_cv_c_bigendian" >&5 +echo "${ECHO_T}$ac_cv_c_bigendian" >&6 +case $ac_cv_c_bigendian in + yes) + +cat >>confdefs.h <<\_ACEOF +#define WORDS_BIGENDIAN 1 +_ACEOF + ;; + no) + ;; + *) + { { echo "$as_me:$LINENO: error: unknown endianness +presetting ac_cv_c_bigendian=no (or yes) will help" >&5 +echo "$as_me: error: unknown endianness +presetting ac_cv_c_bigendian=no (or yes) will help" >&2;} + { (exit 1); exit 1; }; } ;; +esac + + +# Make sure we can run config.sub. +$ac_config_sub sun4 >/dev/null 2>&1 || + { { echo "$as_me:$LINENO: error: cannot run $ac_config_sub" >&5 +echo "$as_me: error: cannot run $ac_config_sub" >&2;} + { (exit 1); exit 1; }; } + +echo "$as_me:$LINENO: checking build system type" >&5 +echo $ECHO_N "checking build system type... $ECHO_C" >&6 +if test "${ac_cv_build+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + ac_cv_build_alias=$build_alias +test -z "$ac_cv_build_alias" && + ac_cv_build_alias=`$ac_config_guess` +test -z "$ac_cv_build_alias" && + { { echo "$as_me:$LINENO: error: cannot guess build type; you must specify one" >&5 +echo "$as_me: error: cannot guess build type; you must specify one" >&2;} + { (exit 1); exit 1; }; } +ac_cv_build=`$ac_config_sub $ac_cv_build_alias` || + { { echo "$as_me:$LINENO: error: $ac_config_sub $ac_cv_build_alias failed" >&5 +echo "$as_me: error: $ac_config_sub $ac_cv_build_alias failed" >&2;} + { (exit 1); exit 1; }; } + +fi +echo "$as_me:$LINENO: result: $ac_cv_build" >&5 +echo "${ECHO_T}$ac_cv_build" >&6 +build=$ac_cv_build +build_cpu=`echo $ac_cv_build | sed 's/^\([^-]*\)-\([^-]*\)-\(.*\)$/\1/'` +build_vendor=`echo $ac_cv_build | sed 's/^\([^-]*\)-\([^-]*\)-\(.*\)$/\2/'` +build_os=`echo $ac_cv_build | sed 's/^\([^-]*\)-\([^-]*\)-\(.*\)$/\3/'` + + +echo "$as_me:$LINENO: checking host system type" >&5 +echo $ECHO_N "checking host system type... $ECHO_C" >&6 +if test "${ac_cv_host+set}" = set; then + echo $ECHO_N "(cached) $ECHO_C" >&6 +else + ac_cv_host_alias=$host_alias +test -z "$ac_cv_host_alias" && + ac_cv_host_alias=$ac_cv_build_alias +ac_cv_host=`$ac_config_sub $ac_cv_host_alias` || + { { echo "$as_me:$LINENO: error: $ac_config_sub $ac_cv_host_alias failed" >&5 +echo "$as_me: error: $ac_config_sub $ac_cv_host_alias failed" >&2;} + { (exit 1); exit 1; }; } + +fi +echo "$as_me:$LINENO: result: $ac_cv_host" >&5 +echo "${ECHO_T}$ac_cv_host" >&6 +host=$ac_cv_host +host_cpu=`echo $ac_cv_host | sed 's/^\([^-]*\)-\([^-]*\)-\(.*\)$/\1/'` +host_vendor=`echo $ac_cv_host | sed 's/^\([^-]*\)-\([^-]*\)-\(.*\)$/\2/'` +host_os=`echo $ac_cv_host | sed 's/^\([^-]*\)-\([^-]*\)-\(.*\)$/\3/'` + + + +case $host_cpu in + i*86 ) + +cat >>confdefs.h <<\_ACEOF +#define CPU_CISC 1 +_ACEOF + + +cat >>confdefs.h <<\_ACEOF +#define HAVE_X86 1 +_ACEOF +;; + * ) + # CPU_RISC is only supported for big endian machines. + if test "$ac_cv_c_bigendian" = "yes"; then + +cat >>confdefs.h <<\_ACEOF +#define CPU_RISC 1 +_ACEOF + + else + cat >>confdefs.h <<\_ACEOF +#define CPU_CISC 1 +_ACEOF + + fi + ;; +esac + +case $host_os in + *cygwin*|*mingw* ) + EXE=.exe;; + * ) EXE="";; +esac + + # define executable suffix; this is needed for `make clean' + +echo "$as_me:$LINENO: checking whether to compile in debugging" >&5 +echo $ECHO_N "checking whether to compile in debugging... $ECHO_C" >&6 +# Check whether --enable-debug or --disable-debug was given. +if test "${enable_debug+set}" = set; then + enableval="$enable_debug" + +else + enable_debug=yes +fi; +if test "$enable_debug" = "yes"; then + +cat >>confdefs.h <<\_ACEOF +#define ENABLE_DEBUGGING 1 +_ACEOF + +fi +echo "$as_me:$LINENO: result: $enable_debug" >&5 +echo "${ECHO_T}$enable_debug" >&6 + +echo "$as_me:$LINENO: checking whether to use ISMAcryp code" >&5 +echo $ECHO_N "checking whether to use ISMAcryp code... $ECHO_C" >&6 +# Check whether --enable-generic-aesicm or --disable-generic-aesicm was given. +if test "${enable_generic_aesicm+set}" = set; then + enableval="$enable_generic_aesicm" + +else + enable_generic_aesicm=no +fi; +if test "$enable_generic_aesicm" = "yes"; then + +cat >>confdefs.h <<\_ACEOF +#define GENERIC_AESICM 1 +_ACEOF + +fi +echo "$as_me:$LINENO: result: $enable_generic_aesicm" >&5 +echo "${ECHO_T}$enable_generic_aesicm" >&6 + +echo "$as_me:$LINENO: checking whether to use syslog for error reporting" >&5 +echo $ECHO_N "checking whether to use syslog for error reporting... $ECHO_C" >&6 +# Check whether --enable-syslog or --disable-syslog was given. +if test "${enable_syslog+set}" = set; then + enableval="$enable_syslog" + +else + enable_syslog=no +fi; +if test "$enable_syslog" = "yes"; then + +cat >>confdefs.h <<\_ACEOF +#define USE_SYSLOG 1 +_ACEOF + +fi +echo "$as_me:$LINENO: result: $enable_syslog" >&5 +echo "${ECHO_T}$enable_syslog" >&6 + +echo "$as_me:$LINENO: checking whether to use stdout for error reporting" >&5 +echo $ECHO_N "checking whether to use stdout for error reporting... $ECHO_C" >&6 +# Check whether --enable-stdout or --disable-stdout was given. +if test "${enable_stdout+set}" = set; then + enableval="$enable_stdout" + +else + enable_stdout=yes +fi; +if test "$enable_stdout" = "yes"; then + +cat >>confdefs.h <<\_ACEOF +#define ERR_REPORTING_STDOUT 1 +_ACEOF + +fi +echo "$as_me:$LINENO: result: $enable_stdout" >&5 +echo "${ECHO_T}$enable_stdout" >&6 + +echo "$as_me:$LINENO: checking whether to use /dev/console for error reporting" >&5 +echo $ECHO_N "checking whether to use /dev/console for error reporting... $ECHO_C" >&6 +# Check whether --enable-console or --disable-console was given. +if test "${enable_console+set}" = set; then + enableval="$enable_console" + +else + enable_console=no +fi; +if test "$enable_console" = "yes"; then + +cat >>confdefs.h <<\_ACEOF +#define USE_ERR_REPORTING_FILE 1 +_ACEOF + + +cat >>confdefs.h <<\_ACEOF +#define ERR_REPORTING_FILE "/dev/console" +_ACEOF + +fi +echo "$as_me:$LINENO: result: $enable_console" >&5 +echo "${ECHO_T}$enable_console" >&6 + +echo "$as_me:$LINENO: checking whether to use GDOI key management" >&5 +echo $ECHO_N "checking whether to use GDOI key management... $ECHO_C" >&6 +# Check whether --enable-gdoi or --disable-gdoi was given. +if test "${enable_gdoi+set}" = set; then + enableval="$enable_gdoi" + +else + enable_gdoi=no +fi; +if test "$enable_gdoi" = "yes"; then + +cat >>confdefs.h <<\_ACEOF +#define SRTP_GDOI 1 +_ACEOF + + GDOI_OBJS=gdoi/srtp+gdoi.o + +fi +echo "$as_me:$LINENO: result: $enable_gdoi" >&5 +echo "${ECHO_T}$enable_gdoi" >&6 + + ac_config_headers="$ac_config_headers crypto/include/config.h:config_in.h" + + + ac_config_files="$ac_config_files Makefile crypto/Makefile doc/Makefile" +cat >confcache <<\_ACEOF +# This file is a shell script that caches the results of configure +# tests run on this system so they can be shared between configure +# scripts and configure runs, see configure's option --config-cache. +# It is not useful on other systems. If it contains results you don't +# want to keep, you may remove or edit it. +# +# config.status only pays attention to the cache file if you give it +# the --recheck option to rerun configure. +# +# `ac_cv_env_foo' variables (set or unset) will be overridden when +# loading this file, other *unset* `ac_cv_foo' will be assigned the +# following values. + +_ACEOF + +# The following way of writing the cache mishandles newlines in values, +# but we know of no workaround that is simple, portable, and efficient. +# So, don't put newlines in cache variables' values. +# Ultrix sh set writes to stderr and can't be redirected directly, +# and sets the high bit in the cache file unless we assign to the vars. +{ + (set) 2>&1 | + case `(ac_space=' '; set | grep ac_space) 2>&1` in + *ac_space=\ *) + # `set' does not quote correctly, so add quotes (double-quote + # substitution turns \\\\ into \\, and sed turns \\ into \). + sed -n \ + "s/'/'\\\\''/g; + s/^\\([_$as_cr_alnum]*_cv_[_$as_cr_alnum]*\\)=\\(.*\\)/\\1='\\2'/p" + ;; + *) + # `set' quotes correctly as required by POSIX, so do not add quotes. + sed -n \ + "s/^\\([_$as_cr_alnum]*_cv_[_$as_cr_alnum]*\\)=\\(.*\\)/\\1=\\2/p" + ;; + esac; +} | + sed ' + t clear + : clear + s/^\([^=]*\)=\(.*[{}].*\)$/test "${\1+set}" = set || &/ + t end + /^ac_cv_env/!s/^\([^=]*\)=\(.*\)$/\1=${\1=\2}/ + : end' >>confcache +if diff $cache_file confcache >/dev/null 2>&1; then :; else + if test -w $cache_file; then + test "x$cache_file" != "x/dev/null" && echo "updating cache $cache_file" + cat confcache >$cache_file + else + echo "not updating unwritable cache $cache_file" + fi +fi +rm -f confcache + +test "x$prefix" = xNONE && prefix=$ac_default_prefix +# Let make expand exec_prefix. +test "x$exec_prefix" = xNONE && exec_prefix='${prefix}' + +# VPATH may cause trouble with some makes, so we remove $(srcdir), +# ${srcdir} and @srcdir@ from VPATH if srcdir is ".", strip leading and +# trailing colons and then remove the whole line if VPATH becomes empty +# (actually we leave an empty line to preserve line numbers). +if test "x$srcdir" = x.; then + ac_vpsub='/^[ ]*VPATH[ ]*=/{ +s/:*\$(srcdir):*/:/; +s/:*\${srcdir}:*/:/; +s/:*@srcdir@:*/:/; +s/^\([^=]*=[ ]*\):*/\1/; +s/:*$//; +s/^[^=]*=[ ]*$//; +}' +fi + +DEFS=-DHAVE_CONFIG_H + +ac_libobjs= +ac_ltlibobjs= +for ac_i in : $LIBOBJS; do test "x$ac_i" = x: && continue + # 1. Remove the extension, and $U if already installed. + ac_i=`echo "$ac_i" | + sed 's/\$U\././;s/\.o$//;s/\.obj$//'` + # 2. Add them. + ac_libobjs="$ac_libobjs $ac_i\$U.$ac_objext" + ac_ltlibobjs="$ac_ltlibobjs $ac_i"'$U.lo' +done +LIBOBJS=$ac_libobjs + +LTLIBOBJS=$ac_ltlibobjs + + + +: ${CONFIG_STATUS=./config.status} +ac_clean_files_save=$ac_clean_files +ac_clean_files="$ac_clean_files $CONFIG_STATUS" +{ echo "$as_me:$LINENO: creating $CONFIG_STATUS" >&5 +echo "$as_me: creating $CONFIG_STATUS" >&6;} +cat >$CONFIG_STATUS <<_ACEOF +#! $SHELL +# Generated by $as_me. +# Run this file to recreate the current configuration. +# Compiler output produced by configure, useful for debugging +# configure, is in config.log if it exists. + +debug=false +ac_cs_recheck=false +ac_cs_silent=false +SHELL=\${CONFIG_SHELL-$SHELL} +_ACEOF + +cat >>$CONFIG_STATUS <<\_ACEOF +## --------------------- ## +## M4sh Initialization. ## +## --------------------- ## + +# Be Bourne compatible +if test -n "${ZSH_VERSION+set}" && (emulate sh) >/dev/null 2>&1; then + emulate sh + NULLCMD=: + # Zsh 3.x and 4.x performs word splitting on ${1+"$@"}, which + # is contrary to our usage. Disable this feature. + alias -g '${1+"$@"}'='"$@"' +elif test -n "${BASH_VERSION+set}" && (set -o posix) >/dev/null 2>&1; then + set -o posix +fi +DUALCASE=1; export DUALCASE # for MKS sh + +# Support unset when possible. +if ( (MAIL=60; unset MAIL) || exit) >/dev/null 2>&1; then + as_unset=unset +else + as_unset=false +fi + + +# Work around bugs in pre-3.0 UWIN ksh. +$as_unset ENV MAIL MAILPATH +PS1='$ ' +PS2='> ' +PS4='+ ' + +# NLS nuisances. +for as_var in \ + LANG LANGUAGE LC_ADDRESS LC_ALL LC_COLLATE LC_CTYPE LC_IDENTIFICATION \ + LC_MEASUREMENT LC_MESSAGES LC_MONETARY LC_NAME LC_NUMERIC LC_PAPER \ + LC_TELEPHONE LC_TIME +do + if (set +x; test -z "`(eval $as_var=C; export $as_var) 2>&1`"); then + eval $as_var=C; export $as_var + else + $as_unset $as_var + fi +done + +# Required to use basename. +if expr a : '\(a\)' >/dev/null 2>&1; then + as_expr=expr +else + as_expr=false +fi + +if (basename /) >/dev/null 2>&1 && test "X`basename / 2>&1`" = "X/"; then + as_basename=basename +else + as_basename=false +fi + + +# Name of the executable. +as_me=`$as_basename "$0" || +$as_expr X/"$0" : '.*/\([^/][^/]*\)/*$' \| \ + X"$0" : 'X\(//\)$' \| \ + X"$0" : 'X\(/\)$' \| \ + . : '\(.\)' 2>/dev/null || +echo X/"$0" | + sed '/^.*\/\([^/][^/]*\)\/*$/{ s//\1/; q; } + /^X\/\(\/\/\)$/{ s//\1/; q; } + /^X\/\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + + +# PATH needs CR, and LINENO needs CR and PATH. +# Avoid depending upon Character Ranges. +as_cr_letters='abcdefghijklmnopqrstuvwxyz' +as_cr_LETTERS='ABCDEFGHIJKLMNOPQRSTUVWXYZ' +as_cr_Letters=$as_cr_letters$as_cr_LETTERS +as_cr_digits='0123456789' +as_cr_alnum=$as_cr_Letters$as_cr_digits + +# The user is always right. +if test "${PATH_SEPARATOR+set}" != set; then + echo "#! /bin/sh" >conf$$.sh + echo "exit 0" >>conf$$.sh + chmod +x conf$$.sh + if (PATH="/nonexistent;."; conf$$.sh) >/dev/null 2>&1; then + PATH_SEPARATOR=';' + else + PATH_SEPARATOR=: + fi + rm -f conf$$.sh +fi + + + as_lineno_1=$LINENO + as_lineno_2=$LINENO + as_lineno_3=`(expr $as_lineno_1 + 1) 2>/dev/null` + test "x$as_lineno_1" != "x$as_lineno_2" && + test "x$as_lineno_3" = "x$as_lineno_2" || { + # Find who we are. Look in the path if we contain no path at all + # relative or not. + case $0 in + *[\\/]* ) as_myself=$0 ;; + *) as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in $PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + test -r "$as_dir/$0" && as_myself=$as_dir/$0 && break +done + + ;; + esac + # We did not find ourselves, most probably we were run as `sh COMMAND' + # in which case we are not to be found in the path. + if test "x$as_myself" = x; then + as_myself=$0 + fi + if test ! -f "$as_myself"; then + { { echo "$as_me:$LINENO: error: cannot find myself; rerun with an absolute path" >&5 +echo "$as_me: error: cannot find myself; rerun with an absolute path" >&2;} + { (exit 1); exit 1; }; } + fi + case $CONFIG_SHELL in + '') + as_save_IFS=$IFS; IFS=$PATH_SEPARATOR +for as_dir in /bin$PATH_SEPARATOR/usr/bin$PATH_SEPARATOR$PATH +do + IFS=$as_save_IFS + test -z "$as_dir" && as_dir=. + for as_base in sh bash ksh sh5; do + case $as_dir in + /*) + if ("$as_dir/$as_base" -c ' + as_lineno_1=$LINENO + as_lineno_2=$LINENO + as_lineno_3=`(expr $as_lineno_1 + 1) 2>/dev/null` + test "x$as_lineno_1" != "x$as_lineno_2" && + test "x$as_lineno_3" = "x$as_lineno_2" ') 2>/dev/null; then + $as_unset BASH_ENV || test "${BASH_ENV+set}" != set || { BASH_ENV=; export BASH_ENV; } + $as_unset ENV || test "${ENV+set}" != set || { ENV=; export ENV; } + CONFIG_SHELL=$as_dir/$as_base + export CONFIG_SHELL + exec "$CONFIG_SHELL" "$0" ${1+"$@"} + fi;; + esac + done +done +;; + esac + + # Create $as_me.lineno as a copy of $as_myself, but with $LINENO + # uniformly replaced by the line number. The first 'sed' inserts a + # line-number line before each line; the second 'sed' does the real + # work. The second script uses 'N' to pair each line-number line + # with the numbered line, and appends trailing '-' during + # substitution so that $LINENO is not a special case at line end. + # (Raja R Harinath suggested sed '=', and Paul Eggert wrote the + # second 'sed' script. Blame Lee E. McMahon for sed's syntax. :-) + sed '=' <$as_myself | + sed ' + N + s,$,-, + : loop + s,^\(['$as_cr_digits']*\)\(.*\)[$]LINENO\([^'$as_cr_alnum'_]\),\1\2\1\3, + t loop + s,-$,, + s,^['$as_cr_digits']*\n,, + ' >$as_me.lineno && + chmod +x $as_me.lineno || + { { echo "$as_me:$LINENO: error: cannot create $as_me.lineno; rerun with a POSIX shell" >&5 +echo "$as_me: error: cannot create $as_me.lineno; rerun with a POSIX shell" >&2;} + { (exit 1); exit 1; }; } + + # Don't try to exec as it changes $[0], causing all sort of problems + # (the dirname of $[0] is not the place where we might find the + # original and so on. Autoconf is especially sensible to this). + . ./$as_me.lineno + # Exit status is that of the last command. + exit +} + + +case `echo "testing\c"; echo 1,2,3`,`echo -n testing; echo 1,2,3` in + *c*,-n*) ECHO_N= ECHO_C=' +' ECHO_T=' ' ;; + *c*,* ) ECHO_N=-n ECHO_C= ECHO_T= ;; + *) ECHO_N= ECHO_C='\c' ECHO_T= ;; +esac + +if expr a : '\(a\)' >/dev/null 2>&1; then + as_expr=expr +else + as_expr=false +fi + +rm -f conf$$ conf$$.exe conf$$.file +echo >conf$$.file +if ln -s conf$$.file conf$$ 2>/dev/null; then + # We could just check for DJGPP; but this test a) works b) is more generic + # and c) will remain valid once DJGPP supports symlinks (DJGPP 2.04). + if test -f conf$$.exe; then + # Don't use ln at all; we don't have any links + as_ln_s='cp -p' + else + as_ln_s='ln -s' + fi +elif ln conf$$.file conf$$ 2>/dev/null; then + as_ln_s=ln +else + as_ln_s='cp -p' +fi +rm -f conf$$ conf$$.exe conf$$.file + +if mkdir -p . 2>/dev/null; then + as_mkdir_p=: +else + test -d ./-p && rmdir ./-p + as_mkdir_p=false +fi + +as_executable_p="test -f" + +# Sed expression to map a string onto a valid CPP name. +as_tr_cpp="eval sed 'y%*$as_cr_letters%P$as_cr_LETTERS%;s%[^_$as_cr_alnum]%_%g'" + +# Sed expression to map a string onto a valid variable name. +as_tr_sh="eval sed 'y%*+%pp%;s%[^_$as_cr_alnum]%_%g'" + + +# IFS +# We need space, tab and new line, in precisely that order. +as_nl=' +' +IFS=" $as_nl" + +# CDPATH. +$as_unset CDPATH + +exec 6>&1 + +# Open the log real soon, to keep \$[0] and so on meaningful, and to +# report actual input values of CONFIG_FILES etc. instead of their +# values after options handling. Logging --version etc. is OK. +exec 5>>config.log +{ + echo + sed 'h;s/./-/g;s/^.../## /;s/...$/ ##/;p;x;p;x' <<_ASBOX +## Running $as_me. ## +_ASBOX +} >&5 +cat >&5 <<_CSEOF + +This file was extended by $as_me, which was +generated by GNU Autoconf 2.59. Invocation command line was + + CONFIG_FILES = $CONFIG_FILES + CONFIG_HEADERS = $CONFIG_HEADERS + CONFIG_LINKS = $CONFIG_LINKS + CONFIG_COMMANDS = $CONFIG_COMMANDS + $ $0 $@ + +_CSEOF +echo "on `(hostname || uname -n) 2>/dev/null | sed 1q`" >&5 +echo >&5 +_ACEOF + +# Files that config.status was made for. +if test -n "$ac_config_files"; then + echo "config_files=\"$ac_config_files\"" >>$CONFIG_STATUS +fi + +if test -n "$ac_config_headers"; then + echo "config_headers=\"$ac_config_headers\"" >>$CONFIG_STATUS +fi + +if test -n "$ac_config_links"; then + echo "config_links=\"$ac_config_links\"" >>$CONFIG_STATUS +fi + +if test -n "$ac_config_commands"; then + echo "config_commands=\"$ac_config_commands\"" >>$CONFIG_STATUS +fi + +cat >>$CONFIG_STATUS <<\_ACEOF + +ac_cs_usage="\ +\`$as_me' instantiates files from templates according to the +current configuration. + +Usage: $0 [OPTIONS] [FILE]... + + -h, --help print this help, then exit + -V, --version print version number, then exit + -q, --quiet do not print progress messages + -d, --debug don't remove temporary files + --recheck update $as_me by reconfiguring in the same conditions + --file=FILE[:TEMPLATE] + instantiate the configuration file FILE + --header=FILE[:TEMPLATE] + instantiate the configuration header FILE + +Configuration files: +$config_files + +Configuration headers: +$config_headers + +Report bugs to ." +_ACEOF + +cat >>$CONFIG_STATUS <<_ACEOF +ac_cs_version="\\ +config.status +configured by $0, generated by GNU Autoconf 2.59, + with options \\"`echo "$ac_configure_args" | sed 's/[\\""\`\$]/\\\\&/g'`\\" + +Copyright (C) 2003 Free Software Foundation, Inc. +This config.status script is free software; the Free Software Foundation +gives unlimited permission to copy, distribute and modify it." +srcdir=$srcdir +INSTALL="$INSTALL" +_ACEOF + +cat >>$CONFIG_STATUS <<\_ACEOF +# If no file are specified by the user, then we need to provide default +# value. By we need to know if files were specified by the user. +ac_need_defaults=: +while test $# != 0 +do + case $1 in + --*=*) + ac_option=`expr "x$1" : 'x\([^=]*\)='` + ac_optarg=`expr "x$1" : 'x[^=]*=\(.*\)'` + ac_shift=: + ;; + -*) + ac_option=$1 + ac_optarg=$2 + ac_shift=shift + ;; + *) # This is not an option, so the user has probably given explicit + # arguments. + ac_option=$1 + ac_need_defaults=false;; + esac + + case $ac_option in + # Handling of the options. +_ACEOF +cat >>$CONFIG_STATUS <<\_ACEOF + -recheck | --recheck | --rechec | --reche | --rech | --rec | --re | --r) + ac_cs_recheck=: ;; + --version | --vers* | -V ) + echo "$ac_cs_version"; exit 0 ;; + --he | --h) + # Conflict between --help and --header + { { echo "$as_me:$LINENO: error: ambiguous option: $1 +Try \`$0 --help' for more information." >&5 +echo "$as_me: error: ambiguous option: $1 +Try \`$0 --help' for more information." >&2;} + { (exit 1); exit 1; }; };; + --help | --hel | -h ) + echo "$ac_cs_usage"; exit 0 ;; + --debug | --d* | -d ) + debug=: ;; + --file | --fil | --fi | --f ) + $ac_shift + CONFIG_FILES="$CONFIG_FILES $ac_optarg" + ac_need_defaults=false;; + --header | --heade | --head | --hea ) + $ac_shift + CONFIG_HEADERS="$CONFIG_HEADERS $ac_optarg" + ac_need_defaults=false;; + -q | -quiet | --quiet | --quie | --qui | --qu | --q \ + | -silent | --silent | --silen | --sile | --sil | --si | --s) + ac_cs_silent=: ;; + + # This is an error. + -*) { { echo "$as_me:$LINENO: error: unrecognized option: $1 +Try \`$0 --help' for more information." >&5 +echo "$as_me: error: unrecognized option: $1 +Try \`$0 --help' for more information." >&2;} + { (exit 1); exit 1; }; } ;; + + *) ac_config_targets="$ac_config_targets $1" ;; + + esac + shift +done + +ac_configure_extra_args= + +if $ac_cs_silent; then + exec 6>/dev/null + ac_configure_extra_args="$ac_configure_extra_args --silent" +fi + +_ACEOF +cat >>$CONFIG_STATUS <<_ACEOF +if \$ac_cs_recheck; then + echo "running $SHELL $0 " $ac_configure_args \$ac_configure_extra_args " --no-create --no-recursion" >&6 + exec $SHELL $0 $ac_configure_args \$ac_configure_extra_args --no-create --no-recursion +fi + +_ACEOF + + + + + +cat >>$CONFIG_STATUS <<\_ACEOF +for ac_config_target in $ac_config_targets +do + case "$ac_config_target" in + # Handling of arguments. + "Makefile" ) CONFIG_FILES="$CONFIG_FILES Makefile" ;; + "crypto/Makefile" ) CONFIG_FILES="$CONFIG_FILES crypto/Makefile" ;; + "doc/Makefile" ) CONFIG_FILES="$CONFIG_FILES doc/Makefile" ;; + "crypto/include/config.h" ) CONFIG_HEADERS="$CONFIG_HEADERS crypto/include/config.h:config_in.h" ;; + *) { { echo "$as_me:$LINENO: error: invalid argument: $ac_config_target" >&5 +echo "$as_me: error: invalid argument: $ac_config_target" >&2;} + { (exit 1); exit 1; }; };; + esac +done + +# If the user did not use the arguments to specify the items to instantiate, +# then the envvar interface is used. Set only those that are not. +# We use the long form for the default assignment because of an extremely +# bizarre bug on SunOS 4.1.3. +if $ac_need_defaults; then + test "${CONFIG_FILES+set}" = set || CONFIG_FILES=$config_files + test "${CONFIG_HEADERS+set}" = set || CONFIG_HEADERS=$config_headers +fi + +# Have a temporary directory for convenience. Make it in the build tree +# simply because there is no reason to put it here, and in addition, +# creating and moving files from /tmp can sometimes cause problems. +# Create a temporary directory, and hook for its removal unless debugging. +$debug || +{ + trap 'exit_status=$?; rm -rf $tmp && exit $exit_status' 0 + trap '{ (exit 1); exit 1; }' 1 2 13 15 +} + +# Create a (secure) tmp directory for tmp files. + +{ + tmp=`(umask 077 && mktemp -d -q "./confstatXXXXXX") 2>/dev/null` && + test -n "$tmp" && test -d "$tmp" +} || +{ + tmp=./confstat$$-$RANDOM + (umask 077 && mkdir $tmp) +} || +{ + echo "$me: cannot create a temporary directory in ." >&2 + { (exit 1); exit 1; } +} + +_ACEOF + +cat >>$CONFIG_STATUS <<_ACEOF + +# +# CONFIG_FILES section. +# + +# No need to generate the scripts if there are no CONFIG_FILES. +# This happens for instance when ./config.status config.h +if test -n "\$CONFIG_FILES"; then + # Protect against being on the right side of a sed subst in config.status. + sed 's/,@/@@/; s/@,/@@/; s/,;t t\$/@;t t/; /@;t t\$/s/[\\\\&,]/\\\\&/g; + s/@@/,@/; s/@@/@,/; s/@;t t\$/,;t t/' >\$tmp/subs.sed <<\\CEOF +s,@SHELL@,$SHELL,;t t +s,@PATH_SEPARATOR@,$PATH_SEPARATOR,;t t +s,@PACKAGE_NAME@,$PACKAGE_NAME,;t t +s,@PACKAGE_TARNAME@,$PACKAGE_TARNAME,;t t +s,@PACKAGE_VERSION@,$PACKAGE_VERSION,;t t +s,@PACKAGE_STRING@,$PACKAGE_STRING,;t t +s,@PACKAGE_BUGREPORT@,$PACKAGE_BUGREPORT,;t t +s,@exec_prefix@,$exec_prefix,;t t +s,@prefix@,$prefix,;t t +s,@program_transform_name@,$program_transform_name,;t t +s,@bindir@,$bindir,;t t +s,@sbindir@,$sbindir,;t t +s,@libexecdir@,$libexecdir,;t t +s,@datadir@,$datadir,;t t +s,@sysconfdir@,$sysconfdir,;t t +s,@sharedstatedir@,$sharedstatedir,;t t +s,@localstatedir@,$localstatedir,;t t +s,@libdir@,$libdir,;t t +s,@includedir@,$includedir,;t t +s,@oldincludedir@,$oldincludedir,;t t +s,@infodir@,$infodir,;t t +s,@mandir@,$mandir,;t t +s,@build_alias@,$build_alias,;t t +s,@host_alias@,$host_alias,;t t +s,@target_alias@,$target_alias,;t t +s,@DEFS@,$DEFS,;t t +s,@ECHO_C@,$ECHO_C,;t t +s,@ECHO_N@,$ECHO_N,;t t +s,@ECHO_T@,$ECHO_T,;t t +s,@LIBS@,$LIBS,;t t +s,@RANLIB@,$RANLIB,;t t +s,@ac_ct_RANLIB@,$ac_ct_RANLIB,;t t +s,@CC@,$CC,;t t +s,@CFLAGS@,$CFLAGS,;t t +s,@LDFLAGS@,$LDFLAGS,;t t +s,@CPPFLAGS@,$CPPFLAGS,;t t +s,@ac_ct_CC@,$ac_ct_CC,;t t +s,@EXEEXT@,$EXEEXT,;t t +s,@OBJEXT@,$OBJEXT,;t t +s,@INSTALL_PROGRAM@,$INSTALL_PROGRAM,;t t +s,@INSTALL_SCRIPT@,$INSTALL_SCRIPT,;t t +s,@INSTALL_DATA@,$INSTALL_DATA,;t t +s,@RNG_OBJS@,$RNG_OBJS,;t t +s,@CPP@,$CPP,;t t +s,@EGREP@,$EGREP,;t t +s,@build@,$build,;t t +s,@build_cpu@,$build_cpu,;t t +s,@build_vendor@,$build_vendor,;t t +s,@build_os@,$build_os,;t t +s,@host@,$host,;t t +s,@host_cpu@,$host_cpu,;t t +s,@host_vendor@,$host_vendor,;t t +s,@host_os@,$host_os,;t t +s,@EXE@,$EXE,;t t +s,@GDOI_OBJS@,$GDOI_OBJS,;t t +s,@LIBOBJS@,$LIBOBJS,;t t +s,@LTLIBOBJS@,$LTLIBOBJS,;t t +CEOF + +_ACEOF + + cat >>$CONFIG_STATUS <<\_ACEOF + # Split the substitutions into bite-sized pieces for seds with + # small command number limits, like on Digital OSF/1 and HP-UX. + ac_max_sed_lines=48 + ac_sed_frag=1 # Number of current file. + ac_beg=1 # First line for current file. + ac_end=$ac_max_sed_lines # Line after last line for current file. + ac_more_lines=: + ac_sed_cmds= + while $ac_more_lines; do + if test $ac_beg -gt 1; then + sed "1,${ac_beg}d; ${ac_end}q" $tmp/subs.sed >$tmp/subs.frag + else + sed "${ac_end}q" $tmp/subs.sed >$tmp/subs.frag + fi + if test ! -s $tmp/subs.frag; then + ac_more_lines=false + else + # The purpose of the label and of the branching condition is to + # speed up the sed processing (if there are no `@' at all, there + # is no need to browse any of the substitutions). + # These are the two extra sed commands mentioned above. + (echo ':t + /@[a-zA-Z_][a-zA-Z_0-9]*@/!b' && cat $tmp/subs.frag) >$tmp/subs-$ac_sed_frag.sed + if test -z "$ac_sed_cmds"; then + ac_sed_cmds="sed -f $tmp/subs-$ac_sed_frag.sed" + else + ac_sed_cmds="$ac_sed_cmds | sed -f $tmp/subs-$ac_sed_frag.sed" + fi + ac_sed_frag=`expr $ac_sed_frag + 1` + ac_beg=$ac_end + ac_end=`expr $ac_end + $ac_max_sed_lines` + fi + done + if test -z "$ac_sed_cmds"; then + ac_sed_cmds=cat + fi +fi # test -n "$CONFIG_FILES" + +_ACEOF +cat >>$CONFIG_STATUS <<\_ACEOF +for ac_file in : $CONFIG_FILES; do test "x$ac_file" = x: && continue + # Support "outfile[:infile[:infile...]]", defaulting infile="outfile.in". + case $ac_file in + - | *:- | *:-:* ) # input from stdin + cat >$tmp/stdin + ac_file_in=`echo "$ac_file" | sed 's,[^:]*:,,'` + ac_file=`echo "$ac_file" | sed 's,:.*,,'` ;; + *:* ) ac_file_in=`echo "$ac_file" | sed 's,[^:]*:,,'` + ac_file=`echo "$ac_file" | sed 's,:.*,,'` ;; + * ) ac_file_in=$ac_file.in ;; + esac + + # Compute @srcdir@, @top_srcdir@, and @INSTALL@ for subdirectories. + ac_dir=`(dirname "$ac_file") 2>/dev/null || +$as_expr X"$ac_file" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$ac_file" : 'X\(//\)[^/]' \| \ + X"$ac_file" : 'X\(//\)$' \| \ + X"$ac_file" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$ac_file" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + { if $as_mkdir_p; then + mkdir -p "$ac_dir" + else + as_dir="$ac_dir" + as_dirs= + while test ! -d "$as_dir"; do + as_dirs="$as_dir $as_dirs" + as_dir=`(dirname "$as_dir") 2>/dev/null || +$as_expr X"$as_dir" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$as_dir" : 'X\(//\)[^/]' \| \ + X"$as_dir" : 'X\(//\)$' \| \ + X"$as_dir" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$as_dir" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + done + test ! -n "$as_dirs" || mkdir $as_dirs + fi || { { echo "$as_me:$LINENO: error: cannot create directory \"$ac_dir\"" >&5 +echo "$as_me: error: cannot create directory \"$ac_dir\"" >&2;} + { (exit 1); exit 1; }; }; } + + ac_builddir=. + +if test "$ac_dir" != .; then + ac_dir_suffix=/`echo "$ac_dir" | sed 's,^\.[\\/],,'` + # A "../" for each directory in $ac_dir_suffix. + ac_top_builddir=`echo "$ac_dir_suffix" | sed 's,/[^\\/]*,../,g'` +else + ac_dir_suffix= ac_top_builddir= +fi + +case $srcdir in + .) # No --srcdir option. We are building in place. + ac_srcdir=. + if test -z "$ac_top_builddir"; then + ac_top_srcdir=. + else + ac_top_srcdir=`echo $ac_top_builddir | sed 's,/$,,'` + fi ;; + [\\/]* | ?:[\\/]* ) # Absolute path. + ac_srcdir=$srcdir$ac_dir_suffix; + ac_top_srcdir=$srcdir ;; + *) # Relative path. + ac_srcdir=$ac_top_builddir$srcdir$ac_dir_suffix + ac_top_srcdir=$ac_top_builddir$srcdir ;; +esac + +# Do not use `cd foo && pwd` to compute absolute paths, because +# the directories may not exist. +case `pwd` in +.) ac_abs_builddir="$ac_dir";; +*) + case "$ac_dir" in + .) ac_abs_builddir=`pwd`;; + [\\/]* | ?:[\\/]* ) ac_abs_builddir="$ac_dir";; + *) ac_abs_builddir=`pwd`/"$ac_dir";; + esac;; +esac +case $ac_abs_builddir in +.) ac_abs_top_builddir=${ac_top_builddir}.;; +*) + case ${ac_top_builddir}. in + .) ac_abs_top_builddir=$ac_abs_builddir;; + [\\/]* | ?:[\\/]* ) ac_abs_top_builddir=${ac_top_builddir}.;; + *) ac_abs_top_builddir=$ac_abs_builddir/${ac_top_builddir}.;; + esac;; +esac +case $ac_abs_builddir in +.) ac_abs_srcdir=$ac_srcdir;; +*) + case $ac_srcdir in + .) ac_abs_srcdir=$ac_abs_builddir;; + [\\/]* | ?:[\\/]* ) ac_abs_srcdir=$ac_srcdir;; + *) ac_abs_srcdir=$ac_abs_builddir/$ac_srcdir;; + esac;; +esac +case $ac_abs_builddir in +.) ac_abs_top_srcdir=$ac_top_srcdir;; +*) + case $ac_top_srcdir in + .) ac_abs_top_srcdir=$ac_abs_builddir;; + [\\/]* | ?:[\\/]* ) ac_abs_top_srcdir=$ac_top_srcdir;; + *) ac_abs_top_srcdir=$ac_abs_builddir/$ac_top_srcdir;; + esac;; +esac + + + case $INSTALL in + [\\/$]* | ?:[\\/]* ) ac_INSTALL=$INSTALL ;; + *) ac_INSTALL=$ac_top_builddir$INSTALL ;; + esac + + if test x"$ac_file" != x-; then + { echo "$as_me:$LINENO: creating $ac_file" >&5 +echo "$as_me: creating $ac_file" >&6;} + rm -f "$ac_file" + fi + # Let's still pretend it is `configure' which instantiates (i.e., don't + # use $as_me), people would be surprised to read: + # /* config.h. Generated by config.status. */ + if test x"$ac_file" = x-; then + configure_input= + else + configure_input="$ac_file. " + fi + configure_input=$configure_input"Generated from `echo $ac_file_in | + sed 's,.*/,,'` by configure." + + # First look for the input files in the build tree, otherwise in the + # src tree. + ac_file_inputs=`IFS=: + for f in $ac_file_in; do + case $f in + -) echo $tmp/stdin ;; + [\\/$]*) + # Absolute (can't be DOS-style, as IFS=:) + test -f "$f" || { { echo "$as_me:$LINENO: error: cannot find input file: $f" >&5 +echo "$as_me: error: cannot find input file: $f" >&2;} + { (exit 1); exit 1; }; } + echo "$f";; + *) # Relative + if test -f "$f"; then + # Build tree + echo "$f" + elif test -f "$srcdir/$f"; then + # Source tree + echo "$srcdir/$f" + else + # /dev/null tree + { { echo "$as_me:$LINENO: error: cannot find input file: $f" >&5 +echo "$as_me: error: cannot find input file: $f" >&2;} + { (exit 1); exit 1; }; } + fi;; + esac + done` || { (exit 1); exit 1; } +_ACEOF +cat >>$CONFIG_STATUS <<_ACEOF + sed "$ac_vpsub +$extrasub +_ACEOF +cat >>$CONFIG_STATUS <<\_ACEOF +:t +/@[a-zA-Z_][a-zA-Z_0-9]*@/!b +s,@configure_input@,$configure_input,;t t +s,@srcdir@,$ac_srcdir,;t t +s,@abs_srcdir@,$ac_abs_srcdir,;t t +s,@top_srcdir@,$ac_top_srcdir,;t t +s,@abs_top_srcdir@,$ac_abs_top_srcdir,;t t +s,@builddir@,$ac_builddir,;t t +s,@abs_builddir@,$ac_abs_builddir,;t t +s,@top_builddir@,$ac_top_builddir,;t t +s,@abs_top_builddir@,$ac_abs_top_builddir,;t t +s,@INSTALL@,$ac_INSTALL,;t t +" $ac_file_inputs | (eval "$ac_sed_cmds") >$tmp/out + rm -f $tmp/stdin + if test x"$ac_file" != x-; then + mv $tmp/out $ac_file + else + cat $tmp/out + rm -f $tmp/out + fi + +done +_ACEOF +cat >>$CONFIG_STATUS <<\_ACEOF + +# +# CONFIG_HEADER section. +# + +# These sed commands are passed to sed as "A NAME B NAME C VALUE D", where +# NAME is the cpp macro being defined and VALUE is the value it is being given. +# +# ac_d sets the value in "#define NAME VALUE" lines. +ac_dA='s,^\([ ]*\)#\([ ]*define[ ][ ]*\)' +ac_dB='[ ].*$,\1#\2' +ac_dC=' ' +ac_dD=',;t' +# ac_u turns "#undef NAME" without trailing blanks into "#define NAME VALUE". +ac_uA='s,^\([ ]*\)#\([ ]*\)undef\([ ][ ]*\)' +ac_uB='$,\1#\2define\3' +ac_uC=' ' +ac_uD=',;t' + +for ac_file in : $CONFIG_HEADERS; do test "x$ac_file" = x: && continue + # Support "outfile[:infile[:infile...]]", defaulting infile="outfile.in". + case $ac_file in + - | *:- | *:-:* ) # input from stdin + cat >$tmp/stdin + ac_file_in=`echo "$ac_file" | sed 's,[^:]*:,,'` + ac_file=`echo "$ac_file" | sed 's,:.*,,'` ;; + *:* ) ac_file_in=`echo "$ac_file" | sed 's,[^:]*:,,'` + ac_file=`echo "$ac_file" | sed 's,:.*,,'` ;; + * ) ac_file_in=$ac_file.in ;; + esac + + test x"$ac_file" != x- && { echo "$as_me:$LINENO: creating $ac_file" >&5 +echo "$as_me: creating $ac_file" >&6;} + + # First look for the input files in the build tree, otherwise in the + # src tree. + ac_file_inputs=`IFS=: + for f in $ac_file_in; do + case $f in + -) echo $tmp/stdin ;; + [\\/$]*) + # Absolute (can't be DOS-style, as IFS=:) + test -f "$f" || { { echo "$as_me:$LINENO: error: cannot find input file: $f" >&5 +echo "$as_me: error: cannot find input file: $f" >&2;} + { (exit 1); exit 1; }; } + # Do quote $f, to prevent DOS paths from being IFS'd. + echo "$f";; + *) # Relative + if test -f "$f"; then + # Build tree + echo "$f" + elif test -f "$srcdir/$f"; then + # Source tree + echo "$srcdir/$f" + else + # /dev/null tree + { { echo "$as_me:$LINENO: error: cannot find input file: $f" >&5 +echo "$as_me: error: cannot find input file: $f" >&2;} + { (exit 1); exit 1; }; } + fi;; + esac + done` || { (exit 1); exit 1; } + # Remove the trailing spaces. + sed 's/[ ]*$//' $ac_file_inputs >$tmp/in + +_ACEOF + +# Transform confdefs.h into two sed scripts, `conftest.defines' and +# `conftest.undefs', that substitutes the proper values into +# config.h.in to produce config.h. The first handles `#define' +# templates, and the second `#undef' templates. +# And first: Protect against being on the right side of a sed subst in +# config.status. Protect against being in an unquoted here document +# in config.status. +rm -f conftest.defines conftest.undefs +# Using a here document instead of a string reduces the quoting nightmare. +# Putting comments in sed scripts is not portable. +# +# `end' is used to avoid that the second main sed command (meant for +# 0-ary CPP macros) applies to n-ary macro definitions. +# See the Autoconf documentation for `clear'. +cat >confdef2sed.sed <<\_ACEOF +s/[\\&,]/\\&/g +s,[\\$`],\\&,g +t clear +: clear +s,^[ ]*#[ ]*define[ ][ ]*\([^ (][^ (]*\)\(([^)]*)\)[ ]*\(.*\)$,${ac_dA}\1${ac_dB}\1\2${ac_dC}\3${ac_dD},gp +t end +s,^[ ]*#[ ]*define[ ][ ]*\([^ ][^ ]*\)[ ]*\(.*\)$,${ac_dA}\1${ac_dB}\1${ac_dC}\2${ac_dD},gp +: end +_ACEOF +# If some macros were called several times there might be several times +# the same #defines, which is useless. Nevertheless, we may not want to +# sort them, since we want the *last* AC-DEFINE to be honored. +uniq confdefs.h | sed -n -f confdef2sed.sed >conftest.defines +sed 's/ac_d/ac_u/g' conftest.defines >conftest.undefs +rm -f confdef2sed.sed + +# This sed command replaces #undef with comments. This is necessary, for +# example, in the case of _POSIX_SOURCE, which is predefined and required +# on some systems where configure will not decide to define it. +cat >>conftest.undefs <<\_ACEOF +s,^[ ]*#[ ]*undef[ ][ ]*[a-zA-Z_][a-zA-Z_0-9]*,/* & */, +_ACEOF + +# Break up conftest.defines because some shells have a limit on the size +# of here documents, and old seds have small limits too (100 cmds). +echo ' # Handle all the #define templates only if necessary.' >>$CONFIG_STATUS +echo ' if grep "^[ ]*#[ ]*define" $tmp/in >/dev/null; then' >>$CONFIG_STATUS +echo ' # If there are no defines, we may have an empty if/fi' >>$CONFIG_STATUS +echo ' :' >>$CONFIG_STATUS +rm -f conftest.tail +while grep . conftest.defines >/dev/null +do + # Write a limited-size here document to $tmp/defines.sed. + echo ' cat >$tmp/defines.sed <>$CONFIG_STATUS + # Speed up: don't consider the non `#define' lines. + echo '/^[ ]*#[ ]*define/!b' >>$CONFIG_STATUS + # Work around the forget-to-reset-the-flag bug. + echo 't clr' >>$CONFIG_STATUS + echo ': clr' >>$CONFIG_STATUS + sed ${ac_max_here_lines}q conftest.defines >>$CONFIG_STATUS + echo 'CEOF + sed -f $tmp/defines.sed $tmp/in >$tmp/out + rm -f $tmp/in + mv $tmp/out $tmp/in +' >>$CONFIG_STATUS + sed 1,${ac_max_here_lines}d conftest.defines >conftest.tail + rm -f conftest.defines + mv conftest.tail conftest.defines +done +rm -f conftest.defines +echo ' fi # grep' >>$CONFIG_STATUS +echo >>$CONFIG_STATUS + +# Break up conftest.undefs because some shells have a limit on the size +# of here documents, and old seds have small limits too (100 cmds). +echo ' # Handle all the #undef templates' >>$CONFIG_STATUS +rm -f conftest.tail +while grep . conftest.undefs >/dev/null +do + # Write a limited-size here document to $tmp/undefs.sed. + echo ' cat >$tmp/undefs.sed <>$CONFIG_STATUS + # Speed up: don't consider the non `#undef' + echo '/^[ ]*#[ ]*undef/!b' >>$CONFIG_STATUS + # Work around the forget-to-reset-the-flag bug. + echo 't clr' >>$CONFIG_STATUS + echo ': clr' >>$CONFIG_STATUS + sed ${ac_max_here_lines}q conftest.undefs >>$CONFIG_STATUS + echo 'CEOF + sed -f $tmp/undefs.sed $tmp/in >$tmp/out + rm -f $tmp/in + mv $tmp/out $tmp/in +' >>$CONFIG_STATUS + sed 1,${ac_max_here_lines}d conftest.undefs >conftest.tail + rm -f conftest.undefs + mv conftest.tail conftest.undefs +done +rm -f conftest.undefs + +cat >>$CONFIG_STATUS <<\_ACEOF + # Let's still pretend it is `configure' which instantiates (i.e., don't + # use $as_me), people would be surprised to read: + # /* config.h. Generated by config.status. */ + if test x"$ac_file" = x-; then + echo "/* Generated by configure. */" >$tmp/config.h + else + echo "/* $ac_file. Generated by configure. */" >$tmp/config.h + fi + cat $tmp/in >>$tmp/config.h + rm -f $tmp/in + if test x"$ac_file" != x-; then + if diff $ac_file $tmp/config.h >/dev/null 2>&1; then + { echo "$as_me:$LINENO: $ac_file is unchanged" >&5 +echo "$as_me: $ac_file is unchanged" >&6;} + else + ac_dir=`(dirname "$ac_file") 2>/dev/null || +$as_expr X"$ac_file" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$ac_file" : 'X\(//\)[^/]' \| \ + X"$ac_file" : 'X\(//\)$' \| \ + X"$ac_file" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$ac_file" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + { if $as_mkdir_p; then + mkdir -p "$ac_dir" + else + as_dir="$ac_dir" + as_dirs= + while test ! -d "$as_dir"; do + as_dirs="$as_dir $as_dirs" + as_dir=`(dirname "$as_dir") 2>/dev/null || +$as_expr X"$as_dir" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$as_dir" : 'X\(//\)[^/]' \| \ + X"$as_dir" : 'X\(//\)$' \| \ + X"$as_dir" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$as_dir" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + done + test ! -n "$as_dirs" || mkdir $as_dirs + fi || { { echo "$as_me:$LINENO: error: cannot create directory \"$ac_dir\"" >&5 +echo "$as_me: error: cannot create directory \"$ac_dir\"" >&2;} + { (exit 1); exit 1; }; }; } + + rm -f $ac_file + mv $tmp/config.h $ac_file + fi + else + cat $tmp/config.h + rm -f $tmp/config.h + fi +done +_ACEOF + +cat >>$CONFIG_STATUS <<\_ACEOF + +{ (exit 0); exit 0; } +_ACEOF +chmod +x $CONFIG_STATUS +ac_clean_files=$ac_clean_files_save + + +# configure is writing to config.log, and then calls config.status. +# config.status does its own redirection, appending to config.log. +# Unfortunately, on DOS this fails, as config.log is still kept open +# by configure, so config.status won't be able to write to it; its +# output is simply discarded. So we exec the FD to /dev/null, +# effectively closing config.log, so it can be properly (re)opened and +# appended to by config.status. When coming back to configure, we +# need to make the FD available again. +if test "$no_create" != yes; then + ac_cs_success=: + ac_config_status_args= + test "$silent" = yes && + ac_config_status_args="$ac_config_status_args --quiet" + exec 5>/dev/null + $SHELL $CONFIG_STATUS $ac_config_status_args || ac_cs_success=false + exec 5>>config.log + # Use ||, not &&, to avoid exiting from the if with $? = 1, which + # would make configure fail if this is the last instruction. + $ac_cs_success || { (exit 1); exit 1; } +fi + + +# This is needed when building outside the source dir. +{ if $as_mkdir_p; then + mkdir -p crypto/ae_xfm + else + as_dir=crypto/ae_xfm + as_dirs= + while test ! -d "$as_dir"; do + as_dirs="$as_dir $as_dirs" + as_dir=`(dirname "$as_dir") 2>/dev/null || +$as_expr X"$as_dir" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$as_dir" : 'X\(//\)[^/]' \| \ + X"$as_dir" : 'X\(//\)$' \| \ + X"$as_dir" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$as_dir" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + done + test ! -n "$as_dirs" || mkdir $as_dirs + fi || { { echo "$as_me:$LINENO: error: cannot create directory crypto/ae_xfm" >&5 +echo "$as_me: error: cannot create directory crypto/ae_xfm" >&2;} + { (exit 1); exit 1; }; }; } + +{ if $as_mkdir_p; then + mkdir -p crypto/cipher + else + as_dir=crypto/cipher + as_dirs= + while test ! -d "$as_dir"; do + as_dirs="$as_dir $as_dirs" + as_dir=`(dirname "$as_dir") 2>/dev/null || +$as_expr X"$as_dir" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$as_dir" : 'X\(//\)[^/]' \| \ + X"$as_dir" : 'X\(//\)$' \| \ + X"$as_dir" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$as_dir" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + done + test ! -n "$as_dirs" || mkdir $as_dirs + fi || { { echo "$as_me:$LINENO: error: cannot create directory crypto/cipher" >&5 +echo "$as_me: error: cannot create directory crypto/cipher" >&2;} + { (exit 1); exit 1; }; }; } + +{ if $as_mkdir_p; then + mkdir -p crypto/hash + else + as_dir=crypto/hash + as_dirs= + while test ! -d "$as_dir"; do + as_dirs="$as_dir $as_dirs" + as_dir=`(dirname "$as_dir") 2>/dev/null || +$as_expr X"$as_dir" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$as_dir" : 'X\(//\)[^/]' \| \ + X"$as_dir" : 'X\(//\)$' \| \ + X"$as_dir" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$as_dir" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + done + test ! -n "$as_dirs" || mkdir $as_dirs + fi || { { echo "$as_me:$LINENO: error: cannot create directory crypto/hash" >&5 +echo "$as_me: error: cannot create directory crypto/hash" >&2;} + { (exit 1); exit 1; }; }; } + +{ if $as_mkdir_p; then + mkdir -p crypto/kernel + else + as_dir=crypto/kernel + as_dirs= + while test ! -d "$as_dir"; do + as_dirs="$as_dir $as_dirs" + as_dir=`(dirname "$as_dir") 2>/dev/null || +$as_expr X"$as_dir" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$as_dir" : 'X\(//\)[^/]' \| \ + X"$as_dir" : 'X\(//\)$' \| \ + X"$as_dir" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$as_dir" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + done + test ! -n "$as_dirs" || mkdir $as_dirs + fi || { { echo "$as_me:$LINENO: error: cannot create directory crypto/kernel" >&5 +echo "$as_me: error: cannot create directory crypto/kernel" >&2;} + { (exit 1); exit 1; }; }; } + +{ if $as_mkdir_p; then + mkdir -p crypto/math + else + as_dir=crypto/math + as_dirs= + while test ! -d "$as_dir"; do + as_dirs="$as_dir $as_dirs" + as_dir=`(dirname "$as_dir") 2>/dev/null || +$as_expr X"$as_dir" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$as_dir" : 'X\(//\)[^/]' \| \ + X"$as_dir" : 'X\(//\)$' \| \ + X"$as_dir" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$as_dir" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + done + test ! -n "$as_dirs" || mkdir $as_dirs + fi || { { echo "$as_me:$LINENO: error: cannot create directory crypto/math" >&5 +echo "$as_me: error: cannot create directory crypto/math" >&2;} + { (exit 1); exit 1; }; }; } + +{ if $as_mkdir_p; then + mkdir -p crypto/replay + else + as_dir=crypto/replay + as_dirs= + while test ! -d "$as_dir"; do + as_dirs="$as_dir $as_dirs" + as_dir=`(dirname "$as_dir") 2>/dev/null || +$as_expr X"$as_dir" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$as_dir" : 'X\(//\)[^/]' \| \ + X"$as_dir" : 'X\(//\)$' \| \ + X"$as_dir" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$as_dir" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + done + test ! -n "$as_dirs" || mkdir $as_dirs + fi || { { echo "$as_me:$LINENO: error: cannot create directory crypto/replay" >&5 +echo "$as_me: error: cannot create directory crypto/replay" >&2;} + { (exit 1); exit 1; }; }; } + +{ if $as_mkdir_p; then + mkdir -p crypto/rng + else + as_dir=crypto/rng + as_dirs= + while test ! -d "$as_dir"; do + as_dirs="$as_dir $as_dirs" + as_dir=`(dirname "$as_dir") 2>/dev/null || +$as_expr X"$as_dir" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$as_dir" : 'X\(//\)[^/]' \| \ + X"$as_dir" : 'X\(//\)$' \| \ + X"$as_dir" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$as_dir" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + done + test ! -n "$as_dirs" || mkdir $as_dirs + fi || { { echo "$as_me:$LINENO: error: cannot create directory crypto/rng" >&5 +echo "$as_me: error: cannot create directory crypto/rng" >&2;} + { (exit 1); exit 1; }; }; } + +{ if $as_mkdir_p; then + mkdir -p crypto/test + else + as_dir=crypto/test + as_dirs= + while test ! -d "$as_dir"; do + as_dirs="$as_dir $as_dirs" + as_dir=`(dirname "$as_dir") 2>/dev/null || +$as_expr X"$as_dir" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$as_dir" : 'X\(//\)[^/]' \| \ + X"$as_dir" : 'X\(//\)$' \| \ + X"$as_dir" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$as_dir" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + done + test ! -n "$as_dirs" || mkdir $as_dirs + fi || { { echo "$as_me:$LINENO: error: cannot create directory crypto/test" >&5 +echo "$as_me: error: cannot create directory crypto/test" >&2;} + { (exit 1); exit 1; }; }; } + +{ if $as_mkdir_p; then + mkdir -p doc + else + as_dir=doc + as_dirs= + while test ! -d "$as_dir"; do + as_dirs="$as_dir $as_dirs" + as_dir=`(dirname "$as_dir") 2>/dev/null || +$as_expr X"$as_dir" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$as_dir" : 'X\(//\)[^/]' \| \ + X"$as_dir" : 'X\(//\)$' \| \ + X"$as_dir" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$as_dir" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + done + test ! -n "$as_dirs" || mkdir $as_dirs + fi || { { echo "$as_me:$LINENO: error: cannot create directory doc" >&5 +echo "$as_me: error: cannot create directory doc" >&2;} + { (exit 1); exit 1; }; }; } + +{ if $as_mkdir_p; then + mkdir -p srtp + else + as_dir=srtp + as_dirs= + while test ! -d "$as_dir"; do + as_dirs="$as_dir $as_dirs" + as_dir=`(dirname "$as_dir") 2>/dev/null || +$as_expr X"$as_dir" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$as_dir" : 'X\(//\)[^/]' \| \ + X"$as_dir" : 'X\(//\)$' \| \ + X"$as_dir" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$as_dir" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + done + test ! -n "$as_dirs" || mkdir $as_dirs + fi || { { echo "$as_me:$LINENO: error: cannot create directory srtp" >&5 +echo "$as_me: error: cannot create directory srtp" >&2;} + { (exit 1); exit 1; }; }; } + +{ if $as_mkdir_p; then + mkdir -p tables + else + as_dir=tables + as_dirs= + while test ! -d "$as_dir"; do + as_dirs="$as_dir $as_dirs" + as_dir=`(dirname "$as_dir") 2>/dev/null || +$as_expr X"$as_dir" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$as_dir" : 'X\(//\)[^/]' \| \ + X"$as_dir" : 'X\(//\)$' \| \ + X"$as_dir" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$as_dir" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + done + test ! -n "$as_dirs" || mkdir $as_dirs + fi || { { echo "$as_me:$LINENO: error: cannot create directory tables" >&5 +echo "$as_me: error: cannot create directory tables" >&2;} + { (exit 1); exit 1; }; }; } + +{ if $as_mkdir_p; then + mkdir -p test + else + as_dir=test + as_dirs= + while test ! -d "$as_dir"; do + as_dirs="$as_dir $as_dirs" + as_dir=`(dirname "$as_dir") 2>/dev/null || +$as_expr X"$as_dir" : 'X\(.*[^/]\)//*[^/][^/]*/*$' \| \ + X"$as_dir" : 'X\(//\)[^/]' \| \ + X"$as_dir" : 'X\(//\)$' \| \ + X"$as_dir" : 'X\(/\)' \| \ + . : '\(.\)' 2>/dev/null || +echo X"$as_dir" | + sed '/^X\(.*[^/]\)\/\/*[^/][^/]*\/*$/{ s//\1/; q; } + /^X\(\/\/\)[^/].*/{ s//\1/; q; } + /^X\(\/\/\)$/{ s//\1/; q; } + /^X\(\/\).*/{ s//\1/; q; } + s/.*/./; q'` + done + test ! -n "$as_dirs" || mkdir $as_dirs + fi || { { echo "$as_me:$LINENO: error: cannot create directory test" >&5 +echo "$as_me: error: cannot create directory test" >&2;} + { (exit 1); exit 1; }; }; } + diff --git a/libs/srtp/configure.in b/libs/srtp/configure.in new file mode 100644 index 0000000000..ca7ab90644 --- /dev/null +++ b/libs/srtp/configure.in @@ -0,0 +1,207 @@ +dnl Process this file with autoconf to produce a configure script. +AC_INIT(srtp) + +dnl Must come before AC_PROG_CC +if test -z "$CFLAGS"; then + dnl Default value for CFLAGS if not specified. + CFLAGS="-Wall -O4 -fexpensive-optimizations -funroll-loops" +fi + +dnl Checks for programs. +AC_PROG_RANLIB +AC_PROG_CC +AC_PROG_INSTALL + +AC_ARG_ENABLE(pic, [AS_HELP_STRING([--enable-pic],[build with PIC])],[CFLAGS="-fPIC $CFLAGS"]) + +AC_ARG_ENABLE(kernel-linux, + [AS_HELP_STRING([--enable-kernel-linux], + [build library to run in Linux kernel context])], + [], enable_kernel_linux=no) +AC_MSG_CHECKING(whether to build for Linux kernel context) +if test "$enable_kernel_linux" = "yes"; then + AC_DEFINE(SRTP_KERNEL, 1, + [Define to compile for kernel contexts.]) + AC_DEFINE(SRTP_KERNEL_LINUX, 1, + [Define to compile for Linux kernel context.]) +fi +AC_MSG_RESULT($enable_kernel_linux) + +if test "$cross_compiling" != yes; then + dnl Check for /dev/urandom + AC_CHECK_FILE(/dev/urandom, DEV_URANDOM=/dev/urandom, + [AC_CHECK_FILE(/dev/random, DEV_URANDOM=/dev/random)]) +fi + +AC_MSG_CHECKING(which random device to use) +if test "$enable_kernel_linux" = "yes"; then + RNG_OBJS=rand_linux_kernel.o + AC_MSG_RESULT([Linux kernel builtin]) +else + RNG_OBJS=rand_source.o + if test -n "$DEV_URANDOM"; then + AC_DEFINE_UNQUOTED(DEV_URANDOM, "$DEV_URANDOM",[Path to random device]) + AC_MSG_RESULT([$DEV_URANDOM]) + else + AC_MSG_RESULT([standard rand() function...]) + fi +fi +AC_SUBST(RNG_OBJS) + + +dnl Checks for header files. +AC_HEADER_STDC +AC_CHECK_HEADERS(stdlib.h) +AC_CHECK_HEADERS(unistd.h) +AC_CHECK_HEADERS(byteswap.h) +AC_CHECK_HEADERS(stdint.h) +AC_CHECK_HEADERS(sys/uio.h) +AC_CHECK_HEADERS(inttypes.h) +AC_CHECK_HEADERS(sys/types.h) +AC_CHECK_HEADERS(machine/types.h) +AC_CHECK_HEADERS(sys/int_types.h) + +dnl socket() and friends +AC_CHECK_HEADERS(sys/socket.h netinet/in.h arpa/inet.h) +AC_CHECK_HEADERS(windows.h, [AC_CHECK_HEADERS(winsock2.h)]) + +AC_CHECK_HEADERS(syslog.h) + +AC_CHECK_TYPES([int8_t,uint8_t,int16_t,uint16_t,int32_t,uint32_t,uint64_t]) +AC_CHECK_SIZEOF(unsigned long) +AC_CHECK_SIZEOF(unsigned long long) + +dnl Checks for typedefs, structures, and compiler characteristics. +AC_C_CONST +AC_C_INLINE +AC_TYPE_SIZE_T + +dnl Checks for library functions. +AC_CHECK_FUNCS(socket inet_aton usleep) + +dnl Find socket function if not found yet. +if test "x$ac_cv_func_socket" = "xno"; then + AC_CHECK_LIB(socket, socket) + AC_MSG_CHECKING([for socket in -lwsock32]) + SAVELIBS="$LIBS" + LIBS="$LIBS -lwsock32" + AC_TRY_LINK([ +#include +],[ +socket(0, 0, 0); +], + ac_cv_func_socket=yes + AC_MSG_RESULT(yes), + LIBS="$SAVELIBS" + AC_MSG_RESULT(no)) +fi + +dnl Check the byte order +AC_C_BIGENDIAN + +AC_CANONICAL_HOST + +dnl check host_cpu type, set defines appropriately +case $host_cpu in + i*86 ) + AC_DEFINE(CPU_CISC, 1, + [Define if building for a CISC machine (e.g. Intel).]) + AC_DEFINE(HAVE_X86, 1, + [Define to use X86 inlined assembly code]);; + * ) + # CPU_RISC is only supported for big endian machines. + if test "$ac_cv_c_bigendian" = "yes"; then + AC_DEFINE(CPU_RISC, 1, + [Define if building for a RISC machine (assume slow byte access).]) + else + AC_DEFINE(CPU_CISC, 1) + fi + ;; +esac + +dnl Check if we're on a Windows platform. +case $host_os in + *cygwin*|*mingw* ) + EXE=.exe;; + * ) EXE="";; +esac + +AC_SUBST(EXE) # define executable suffix; this is needed for `make clean' + +AC_MSG_CHECKING(whether to compile in debugging) +AC_ARG_ENABLE(debug, + [AS_HELP_STRING([--disable-debug], + [do not compile in dynamic debugging system])], + [], enable_debug=yes) +if test "$enable_debug" = "yes"; then + AC_DEFINE(ENABLE_DEBUGGING, 1, + [Define to compile in dynamic debugging system.]) +fi +AC_MSG_RESULT($enable_debug) + +AC_MSG_CHECKING(whether to use ISMAcryp code) +AC_ARG_ENABLE(generic-aesicm, + [AS_HELP_STRING([--enable-generic-aesicm], + [compile in changes for ISMAcryp])], + [], enable_generic_aesicm=no) +if test "$enable_generic_aesicm" = "yes"; then + AC_DEFINE(GENERIC_AESICM, 1, [Define this to use ISMAcryp code.]) +fi +AC_MSG_RESULT($enable_generic_aesicm) + +AC_MSG_CHECKING(whether to use syslog for error reporting) +AC_ARG_ENABLE(syslog, + [AS_HELP_STRING([--enable-syslog], [use syslog for error reporting])], + [], enable_syslog=no) +if test "$enable_syslog" = "yes"; then + AC_DEFINE(USE_SYSLOG, 1, [Define to use syslog logging.]) +fi +AC_MSG_RESULT($enable_syslog) + +AC_MSG_CHECKING(whether to use stdout for error reporting) +AC_ARG_ENABLE(stdout, + [AS_HELP_STRING([--disable-stdout], [don't use stdout for error reporting])], + [], enable_stdout=yes) +if test "$enable_stdout" = "yes"; then + AC_DEFINE(ERR_REPORTING_STDOUT, 1, [Define to use logging to stdout.]) +fi +AC_MSG_RESULT($enable_stdout) + +AC_MSG_CHECKING(whether to use /dev/console for error reporting) +AC_ARG_ENABLE(console, + [AS_HELP_STRING([--enable-console], [use /dev/console for error reporting])], + [], enable_console=no) +if test "$enable_console" = "yes"; then + AC_DEFINE(USE_ERR_REPORTING_FILE, 1, [Write errors to this file]) + AC_DEFINE(ERR_REPORTING_FILE, "/dev/console", [Report errors to this file.]) +fi +AC_MSG_RESULT($enable_console) + +AC_MSG_CHECKING(whether to use GDOI key management) +AC_ARG_ENABLE(gdoi, + [AS_HELP_STRING([--enable-gdoi], [enable GDOI key management])], + [], enable_gdoi=no) +if test "$enable_gdoi" = "yes"; then + AC_DEFINE(SRTP_GDOI, 1, [Define to use GDOI.]) + GDOI_OBJS=gdoi/srtp+gdoi.o + AC_SUBST(GDOI_OBJS) +fi +AC_MSG_RESULT($enable_gdoi) + +AC_CONFIG_HEADER(crypto/include/config.h:config_in.h) + +AC_OUTPUT(Makefile crypto/Makefile doc/Makefile) + +# This is needed when building outside the source dir. +AS_MKDIR_P(crypto/ae_xfm) +AS_MKDIR_P(crypto/cipher) +AS_MKDIR_P(crypto/hash) +AS_MKDIR_P(crypto/kernel) +AS_MKDIR_P(crypto/math) +AS_MKDIR_P(crypto/replay) +AS_MKDIR_P(crypto/rng) +AS_MKDIR_P(crypto/test) +AS_MKDIR_P(doc) +AS_MKDIR_P(srtp) +AS_MKDIR_P(tables) +AS_MKDIR_P(test) diff --git a/libs/srtp/crypto/.cvsignore b/libs/srtp/crypto/.cvsignore new file mode 100644 index 0000000000..f3c7a7c5da --- /dev/null +++ b/libs/srtp/crypto/.cvsignore @@ -0,0 +1 @@ +Makefile diff --git a/libs/srtp/crypto/CVS/Entries b/libs/srtp/crypto/CVS/Entries new file mode 100644 index 0000000000..42d879bcc0 --- /dev/null +++ b/libs/srtp/crypto/CVS/Entries @@ -0,0 +1,12 @@ +/.cvsignore/1.1/Thu Sep 29 13:27:59 2005// +/Makefile.in/1.5/Mon Oct 3 15:16:37 2005// +/VERSION/1.1.1.1/Wed Sep 21 22:51:38 2005// +D/ae_xfm//// +D/cipher//// +D/hash//// +D/include//// +D/kernel//// +D/math//// +D/replay//// +D/rng//// +D/test//// diff --git a/libs/srtp/crypto/CVS/Repository b/libs/srtp/crypto/CVS/Repository new file mode 100644 index 0000000000..3abe7a39c4 --- /dev/null +++ b/libs/srtp/crypto/CVS/Repository @@ -0,0 +1 @@ +srtp/crypto diff --git a/libs/srtp/crypto/CVS/Root b/libs/srtp/crypto/CVS/Root new file mode 100644 index 0000000000..b4b74237d9 --- /dev/null +++ b/libs/srtp/crypto/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@cvs.sourceforge.net:/cvsroot/srtp diff --git a/libs/srtp/crypto/Makefile b/libs/srtp/crypto/Makefile new file mode 100644 index 0000000000..fe5ae96f33 --- /dev/null +++ b/libs/srtp/crypto/Makefile @@ -0,0 +1,130 @@ +# Makefile for libcryptomodule.a +# +# David A. McGrew +# Cisco Systems, Inc. + +srcdir = . +top_srcdir = .. +top_builddir = ../ + + +CC = gcc +INCDIR = -Iinclude -I$(srcdir)/include +DEFS = -DHAVE_CONFIG_H +CPPFLAGS= +CFLAGS = -fPIC -Wall -O4 -fexpensive-optimizations -funroll-loops +LIBS = +LDFLAGS = -L. +COMPILE = $(CC) $(DEFS) $(INCDIR) $(CPPFLAGS) $(CFLAGS) +CRYPTOLIB = -lcryptomodule + +RANLIB = ranlib + +# EXE defines the suffix on executables - it's .exe for cygwin, and +# null on linux, bsd, and OS X and other OSes. we define this so that +# `make clean` will work on the cygwin platform +EXE = +# Random source. +RNG_OBJS = rand_source.o + +ifdef ARCH + DEFS += -D$(ARCH)=1 +endif + +ifdef sysname + DEFS += -D$(sysname)=1 +endif + +.PHONY: dummy all runtest clean superclean + +dummy : all runtest + +# test applications + +testapp = test/cipher_driver$(EXE) test/datatypes_driver$(EXE) \ + test/stat_driver$(EXE) test/sha1_driver$(EXE) \ + test/kernel_driver$(EXE) test/aes_calc$(EXE) test/rand_gen$(EXE) \ + test/env$(EXE) + +# data values used to test the aes_calc application + +k=000102030405060708090a0b0c0d0e0f +p=00112233445566778899aabbccddeeff +c=69c4e0d86a7b0430d8cdb78070b4c55a + +runtest: libcryptomodule.a $(testapp) + test/env$(EXE) # print out information on the build environment + @echo "running libcryptomodule test applications..." + test `test/aes_calc $k $p` = $c + test/cipher_driver$(EXE) -v >/dev/null + test/datatypes_driver$(EXE) -v >/dev/null + test/stat_driver$(EXE) >/dev/null + test/sha1_driver$(EXE) -v >/dev/null + test/kernel_driver$(EXE) -v >/dev/null + test/rand_gen$(EXE) -n 256 >/dev/null + @echo "libcryptomodule test applications passed." + +# libcryptomodule.a (the crypto engine) + +ciphers = cipher/cipher.o cipher/null_cipher.o \ + cipher/aes.o cipher/aes_icm.o \ + cipher/aes_cbc.o + +hashes = hash/null_auth.o hash/sha1.o \ + hash/hmac.o hash/auth.o + +math = math/datatypes.o math/stat.o + +rng = rng/$(RNG_OBJS) rng/rand_source.o rng/prng.o rng/ctr_prng.o + +err = kernel/err.o + +kernel = kernel/crypto_kernel.o kernel/alloc.o \ + kernel/key.o $(rng) $(err) + +xfm = ae_xfm/xfm.o + +cryptobj = $(ciphers) $(hashes) $(math) $(stat) $(kernel) $(xfm) + +# the rule for making object files and test apps + +%.o: %.c + $(COMPILE) -c $< -o $@ + +%$(EXE): %.c libcryptomodule.a + $(COMPILE) $(LDFLAGS) $< -o $@ $(CRYPTOLIB) $(LIBS) + +ifndef AR + AR=ar +endif + +# and the crypto module library itself + +libcryptomodule.a: $(cryptobj) + $(AR) cr libcryptomodule.a $(cryptobj) + $(RANLIB) libcryptomodule.a + +all: libcryptomodule.a $(testapp) + +# housekeeping functions + +clean: + rm -f libcryptomodule.a + rm -f $(testapp) *.o */*.o + for a in * .* */*; do if [ -f "$$a~" ] ; then rm $$a~; fi; done; + rm -f `find . -name "*.[ch]~*~"` + rm -rf latex + +superclean: clean + rm -f *core TAGS ktrace.out + + +# the target 'package' builds a compressed tar archive of the source code + +distname = crypto-$(shell cat VERSION) + +package: superclean + cd ..; tar cvzf $(distname).tgz crypto/ + + +# EOF diff --git a/libs/srtp/crypto/Makefile.in b/libs/srtp/crypto/Makefile.in new file mode 100644 index 0000000000..c14dba50c5 --- /dev/null +++ b/libs/srtp/crypto/Makefile.in @@ -0,0 +1,130 @@ +# Makefile for libcryptomodule.a +# +# David A. McGrew +# Cisco Systems, Inc. + +srcdir = @srcdir@ +top_srcdir = @top_srcdir@ +top_builddir = @top_builddir@ +VPATH = @srcdir@ + +CC = @CC@ +INCDIR = -Iinclude -I$(srcdir)/include +DEFS = @DEFS@ +CPPFLAGS= @CPPFLAGS@ +CFLAGS = @CFLAGS@ +LIBS = @LIBS@ +LDFLAGS = @LDFLAGS@ -L. +COMPILE = $(CC) $(DEFS) $(INCDIR) $(CPPFLAGS) $(CFLAGS) +CRYPTOLIB = -lcryptomodule + +RANLIB = @RANLIB@ + +# EXE defines the suffix on executables - it's .exe for cygwin, and +# null on linux, bsd, and OS X and other OSes. we define this so that +# `make clean` will work on the cygwin platform +EXE = @EXE@ +# Random source. +RNG_OBJS = @RNG_OBJS@ + +ifdef ARCH + DEFS += -D$(ARCH)=1 +endif + +ifdef sysname + DEFS += -D$(sysname)=1 +endif + +.PHONY: dummy all runtest clean superclean + +dummy : all runtest + +# test applications + +testapp = test/cipher_driver$(EXE) test/datatypes_driver$(EXE) \ + test/stat_driver$(EXE) test/sha1_driver$(EXE) \ + test/kernel_driver$(EXE) test/aes_calc$(EXE) test/rand_gen$(EXE) \ + test/env$(EXE) + +# data values used to test the aes_calc application + +k=000102030405060708090a0b0c0d0e0f +p=00112233445566778899aabbccddeeff +c=69c4e0d86a7b0430d8cdb78070b4c55a + +runtest: libcryptomodule.a $(testapp) + test/env$(EXE) # print out information on the build environment + @echo "running libcryptomodule test applications..." + test `test/aes_calc $k $p` = $c + test/cipher_driver$(EXE) -v >/dev/null + test/datatypes_driver$(EXE) -v >/dev/null + test/stat_driver$(EXE) >/dev/null + test/sha1_driver$(EXE) -v >/dev/null + test/kernel_driver$(EXE) -v >/dev/null + test/rand_gen$(EXE) -n 256 >/dev/null + @echo "libcryptomodule test applications passed." + +# libcryptomodule.a (the crypto engine) + +ciphers = cipher/cipher.o cipher/null_cipher.o \ + cipher/aes.o cipher/aes_icm.o \ + cipher/aes_cbc.o + +hashes = hash/null_auth.o hash/sha1.o \ + hash/hmac.o hash/auth.o + +math = math/datatypes.o math/stat.o + +rng = rng/$(RNG_OBJS) rng/rand_source.o rng/prng.o rng/ctr_prng.o + +err = kernel/err.o + +kernel = kernel/crypto_kernel.o kernel/alloc.o \ + kernel/key.o $(rng) $(err) + +xfm = ae_xfm/xfm.o + +cryptobj = $(ciphers) $(hashes) $(math) $(stat) $(kernel) $(xfm) + +# the rule for making object files and test apps + +%.o: %.c + $(COMPILE) -c $< -o $@ + +%$(EXE): %.c libcryptomodule.a + $(COMPILE) $(LDFLAGS) $< -o $@ $(CRYPTOLIB) $(LIBS) + +ifndef AR + AR=ar +endif + +# and the crypto module library itself + +libcryptomodule.a: $(cryptobj) + $(AR) cr libcryptomodule.a $(cryptobj) + $(RANLIB) libcryptomodule.a + +all: libcryptomodule.a $(testapp) + +# housekeeping functions + +clean: + rm -f libcryptomodule.a + rm -f $(testapp) *.o */*.o + for a in * .* */*; do if [ -f "$$a~" ] ; then rm $$a~; fi; done; + rm -f `find . -name "*.[ch]~*~"` + rm -rf latex + +superclean: clean + rm -f *core TAGS ktrace.out + + +# the target 'package' builds a compressed tar archive of the source code + +distname = crypto-$(shell cat VERSION) + +package: superclean + cd ..; tar cvzf $(distname).tgz crypto/ + + +# EOF diff --git a/libs/srtp/crypto/VERSION b/libs/srtp/crypto/VERSION new file mode 100644 index 0000000000..3eefcb9dd5 --- /dev/null +++ b/libs/srtp/crypto/VERSION @@ -0,0 +1 @@ +1.0.0 diff --git a/libs/srtp/crypto/ae_xfm/CVS/Entries b/libs/srtp/crypto/ae_xfm/CVS/Entries new file mode 100644 index 0000000000..fbd615e89c --- /dev/null +++ b/libs/srtp/crypto/ae_xfm/CVS/Entries @@ -0,0 +1,2 @@ +/xfm.c/1.2/Sun Oct 2 20:23:23 2005// +D diff --git a/libs/srtp/crypto/ae_xfm/CVS/Repository b/libs/srtp/crypto/ae_xfm/CVS/Repository new file mode 100644 index 0000000000..116e5b3af8 --- /dev/null +++ b/libs/srtp/crypto/ae_xfm/CVS/Repository @@ -0,0 +1 @@ +srtp/crypto/ae_xfm diff --git a/libs/srtp/crypto/ae_xfm/CVS/Root b/libs/srtp/crypto/ae_xfm/CVS/Root new file mode 100644 index 0000000000..b4b74237d9 --- /dev/null +++ b/libs/srtp/crypto/ae_xfm/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@cvs.sourceforge.net:/cvsroot/srtp diff --git a/libs/srtp/crypto/ae_xfm/xfm.c b/libs/srtp/crypto/ae_xfm/xfm.c new file mode 100644 index 0000000000..05ff710f90 --- /dev/null +++ b/libs/srtp/crypto/ae_xfm/xfm.c @@ -0,0 +1,570 @@ +/* + * xfm.c + * + * Crypto transform implementation + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +#include "cryptoalg.h" +#include "aes_cbc.h" +#include "hmac.h" +#include "crypto_kernel.h" /* for crypto_get_random() */ + +#define KEY_LEN 16 +#define ENC_KEY_LEN 16 +#define MAC_KEY_LEN 16 +#define IV_LEN 16 +#define TAG_LEN 12 +#define MAX_EXPAND 27 + +err_status_t +aes_128_cbc_hmac_sha1_96_func(void *key, + void *clear, + unsigned clear_len, + void *iv, + void *opaque, + unsigned *opaque_len, + void *auth_tag) { + aes_cbc_ctx_t aes_ctx; + hmac_ctx_t hmac_ctx; + unsigned char enc_key[ENC_KEY_LEN]; + unsigned char mac_key[MAC_KEY_LEN]; + err_status_t status; + + /* check if we're doing authentication only */ + if ((iv == NULL) && (opaque == NULL) && (opaque_len == NULL)) { + + /* perform authentication only */ + + } else if ((iv == NULL) || (opaque == NULL) || (opaque_len == NULL)) { + + /* + * bad parameter - we expect either all three pointers to be NULL, + * or none of those pointers to be NULL + */ + return err_status_fail; + + } else { + + /* derive encryption and authentication keys from the input key */ + status = hmac_init(&hmac_ctx, key, KEY_LEN); + if (status) return status; + status = hmac_compute(&hmac_ctx, "ENC", 3, ENC_KEY_LEN, enc_key); + if (status) return status; + + status = hmac_init(&hmac_ctx, key, KEY_LEN); + if (status) return status; + status = hmac_compute(&hmac_ctx, "MAC", 3, MAC_KEY_LEN, mac_key); + if (status) return status; + + + /* perform encryption and authentication */ + + /* set aes key */ + status = aes_cbc_context_init(&aes_ctx, key, direction_encrypt); + if (status) return status; + + /* set iv */ + status = crypto_get_random(iv, IV_LEN); + if (status) return status; + status = aes_cbc_set_iv(&aes_ctx, iv); + + /* encrypt the opaque data */ + status = aes_cbc_nist_encrypt(&aes_ctx, opaque, opaque_len); + if (status) return status; + + /* authenticate clear and opaque data */ + status = hmac_init(&hmac_ctx, mac_key, MAC_KEY_LEN); + if (status) return status; + + status = hmac_start(&hmac_ctx); + if (status) return status; + + status = hmac_update(&hmac_ctx, clear, clear_len); + if (status) return status; + + status = hmac_compute(&hmac_ctx, opaque, *opaque_len, TAG_LEN, auth_tag); + if (status) return status; + + } + + return err_status_ok; +} + +err_status_t +aes_128_cbc_hmac_sha1_96_inv(void *key, + void *clear, + unsigned clear_len, + void *iv, + void *opaque, + unsigned *opaque_len, + void *auth_tag) { + aes_cbc_ctx_t aes_ctx; + hmac_ctx_t hmac_ctx; + unsigned char enc_key[ENC_KEY_LEN]; + unsigned char mac_key[MAC_KEY_LEN]; + unsigned char tmp_tag[TAG_LEN]; + unsigned char *tag = auth_tag; + err_status_t status; + int i; + + /* check if we're doing authentication only */ + if ((iv == NULL) && (opaque == NULL) && (opaque_len == NULL)) { + + /* perform authentication only */ + + } else if ((iv == NULL) || (opaque == NULL) || (opaque_len == NULL)) { + + /* + * bad parameter - we expect either all three pointers to be NULL, + * or none of those pointers to be NULL + */ + return err_status_fail; + + } else { + + /* derive encryption and authentication keys from the input key */ + status = hmac_init(&hmac_ctx, key, KEY_LEN); + if (status) return status; + status = hmac_compute(&hmac_ctx, "ENC", 3, ENC_KEY_LEN, enc_key); + if (status) return status; + + status = hmac_init(&hmac_ctx, key, KEY_LEN); + if (status) return status; + status = hmac_compute(&hmac_ctx, "MAC", 3, MAC_KEY_LEN, mac_key); + if (status) return status; + + /* perform encryption and authentication */ + + /* set aes key */ + status = aes_cbc_context_init(&aes_ctx, key, direction_decrypt); + if (status) return status; + + /* set iv */ + status = rand_source_get_octet_string(iv, IV_LEN); + if (status) return status; + status = aes_cbc_set_iv(&aes_ctx, iv); + + /* encrypt the opaque data */ + status = aes_cbc_nist_decrypt(&aes_ctx, opaque, opaque_len); + if (status) return status; + + /* authenticate clear and opaque data */ + status = hmac_init(&hmac_ctx, mac_key, MAC_KEY_LEN); + if (status) return status; + + status = hmac_start(&hmac_ctx); + if (status) return status; + + status = hmac_update(&hmac_ctx, clear, clear_len); + if (status) return status; + + status = hmac_compute(&hmac_ctx, opaque, *opaque_len, TAG_LEN, tmp_tag); + if (status) return status; + + /* compare the computed tag with the one provided as input */ + for (i=0; i < TAG_LEN; i++) + if (tmp_tag[i] != tag[i]) + return err_status_auth_fail; + + } + + return err_status_ok; +} + + +#define ENC 1 + +#define DEBUG 0 + +err_status_t +aes_128_cbc_hmac_sha1_96_enc(void *key, + const void *clear, + unsigned clear_len, + void *iv, + void *opaque, + unsigned *opaque_len) { + aes_cbc_ctx_t aes_ctx; + hmac_ctx_t hmac_ctx; + unsigned char enc_key[ENC_KEY_LEN]; + unsigned char mac_key[MAC_KEY_LEN]; + unsigned char *auth_tag; + err_status_t status; + + /* check if we're doing authentication only */ + if ((iv == NULL) && (opaque == NULL) && (opaque_len == NULL)) { + + /* perform authentication only */ + + } else if ((iv == NULL) || (opaque == NULL) || (opaque_len == NULL)) { + + /* + * bad parameter - we expect either all three pointers to be NULL, + * or none of those pointers to be NULL + */ + return err_status_fail; + + } else { + +#if DEBUG + printf("ENC using key %s\n", octet_string_hex_string(key, KEY_LEN)); +#endif + + /* derive encryption and authentication keys from the input key */ + status = hmac_init(&hmac_ctx, key, KEY_LEN); + if (status) return status; + status = hmac_compute(&hmac_ctx, "ENC", 3, ENC_KEY_LEN, enc_key); + if (status) return status; + + status = hmac_init(&hmac_ctx, key, KEY_LEN); + if (status) return status; + status = hmac_compute(&hmac_ctx, "MAC", 3, MAC_KEY_LEN, mac_key); + if (status) return status; + + + /* perform encryption and authentication */ + + /* set aes key */ + status = aes_cbc_context_init(&aes_ctx, key, direction_encrypt); + if (status) return status; + + /* set iv */ + status = rand_source_get_octet_string(iv, IV_LEN); + if (status) return status; + status = aes_cbc_set_iv(&aes_ctx, iv); + if (status) return status; + +#if DEBUG + printf("plaintext len: %d\n", *opaque_len); + printf("iv: %s\n", octet_string_hex_string(iv, IV_LEN)); + printf("plaintext: %s\n", octet_string_hex_string(opaque, *opaque_len)); +#endif + +#if ENC + /* encrypt the opaque data */ + status = aes_cbc_nist_encrypt(&aes_ctx, opaque, opaque_len); + if (status) return status; +#endif + +#if DEBUG + printf("ciphertext len: %d\n", *opaque_len); + printf("ciphertext: %s\n", octet_string_hex_string(opaque, *opaque_len)); +#endif + + /* + * authenticate clear and opaque data, then write the + * authentication tag to the location immediately following the + * ciphertext + */ + status = hmac_init(&hmac_ctx, mac_key, MAC_KEY_LEN); + if (status) return status; + + status = hmac_start(&hmac_ctx); + if (status) return status; + + status = hmac_update(&hmac_ctx, clear, clear_len); + if (status) return status; +#if DEBUG + printf("hmac input: %s\n", + octet_string_hex_string(clear, clear_len)); +#endif + auth_tag = (unsigned char *)opaque; + auth_tag += *opaque_len; + status = hmac_compute(&hmac_ctx, opaque, *opaque_len, TAG_LEN, auth_tag); + if (status) return status; +#if DEBUG + printf("hmac input: %s\n", + octet_string_hex_string(opaque, *opaque_len)); +#endif + /* bump up the opaque_len to reflect the authentication tag */ + *opaque_len += TAG_LEN; + +#if DEBUG + printf("prot data len: %d\n", *opaque_len); + printf("prot data: %s\n", octet_string_hex_string(opaque, *opaque_len)); +#endif + } + + return err_status_ok; +} + +err_status_t +aes_128_cbc_hmac_sha1_96_dec(void *key, + const void *clear, + unsigned clear_len, + void *iv, + void *opaque, + unsigned *opaque_len) { + aes_cbc_ctx_t aes_ctx; + hmac_ctx_t hmac_ctx; + unsigned char enc_key[ENC_KEY_LEN]; + unsigned char mac_key[MAC_KEY_LEN]; + unsigned char tmp_tag[TAG_LEN]; + unsigned char *auth_tag; + unsigned ciphertext_len; + err_status_t status; + int i; + + /* check if we're doing authentication only */ + if ((iv == NULL) && (opaque == NULL) && (opaque_len == NULL)) { + + /* perform authentication only */ + + } else if ((iv == NULL) || (opaque == NULL) || (opaque_len == NULL)) { + + /* + * bad parameter - we expect either all three pointers to be NULL, + * or none of those pointers to be NULL + */ + return err_status_fail; + + } else { +#if DEBUG + printf("DEC using key %s\n", octet_string_hex_string(key, KEY_LEN)); +#endif + + /* derive encryption and authentication keys from the input key */ + status = hmac_init(&hmac_ctx, key, KEY_LEN); + if (status) return status; + status = hmac_compute(&hmac_ctx, "ENC", 3, ENC_KEY_LEN, enc_key); + if (status) return status; + + status = hmac_init(&hmac_ctx, key, KEY_LEN); + if (status) return status; + status = hmac_compute(&hmac_ctx, "MAC", 3, MAC_KEY_LEN, mac_key); + if (status) return status; + +#if DEBUG + printf("prot data len: %d\n", *opaque_len); + printf("prot data: %s\n", octet_string_hex_string(opaque, *opaque_len)); +#endif + + /* + * set the protected data length to that of the ciphertext, by + * subtracting out the length of the authentication tag + */ + ciphertext_len = *opaque_len - TAG_LEN; + +#if DEBUG + printf("ciphertext len: %d\n", ciphertext_len); +#endif + /* verify the authentication tag */ + + /* + * compute the authentication tag for the clear and opaque data, + * and write it to a temporary location + */ + status = hmac_init(&hmac_ctx, mac_key, MAC_KEY_LEN); + if (status) return status; + + status = hmac_start(&hmac_ctx); + if (status) return status; + + status = hmac_update(&hmac_ctx, clear, clear_len); + if (status) return status; + +#if DEBUG + printf("hmac input: %s\n", + octet_string_hex_string(clear, clear_len)); +#endif + + status = hmac_compute(&hmac_ctx, opaque, ciphertext_len, TAG_LEN, tmp_tag); + if (status) return status; + +#if DEBUG + printf("hmac input: %s\n", + octet_string_hex_string(opaque, ciphertext_len)); +#endif + + /* + * compare the computed tag with the one provided as input (which + * immediately follows the ciphertext) + */ + auth_tag = (unsigned char *)opaque; + auth_tag += ciphertext_len; +#if DEBUG + printf("auth_tag: %s\n", octet_string_hex_string(auth_tag, TAG_LEN)); + printf("tmp_tag: %s\n", octet_string_hex_string(tmp_tag, TAG_LEN)); +#endif + for (i=0; i < TAG_LEN; i++) { + if (tmp_tag[i] != auth_tag[i]) + return err_status_auth_fail; + } + + /* bump down the opaque_len to reflect the authentication tag */ + *opaque_len -= TAG_LEN; + + /* decrypt the confidential data */ + status = aes_cbc_context_init(&aes_ctx, key, direction_decrypt); + if (status) return status; + status = aes_cbc_set_iv(&aes_ctx, iv); + if (status) return status; + +#if DEBUG + printf("ciphertext: %s\n", octet_string_hex_string(opaque, *opaque_len)); + printf("iv: %s\n", octet_string_hex_string(iv, IV_LEN)); +#endif + +#if ENC + status = aes_cbc_nist_decrypt(&aes_ctx, opaque, &ciphertext_len); + if (status) return status; +#endif + +#if DEBUG + printf("plaintext len: %d\n", ciphertext_len); + printf("plaintext: %s\n", + octet_string_hex_string(opaque, ciphertext_len)); +#endif + + /* indicate the length of the plaintext */ + *opaque_len = ciphertext_len; + } + + return err_status_ok; +} + +cryptoalg_ctx_t cryptoalg_ctx = { + aes_128_cbc_hmac_sha1_96_enc, + aes_128_cbc_hmac_sha1_96_dec, + KEY_LEN, + IV_LEN, + TAG_LEN, + MAX_EXPAND, +}; + +cryptoalg_t cryptoalg = &cryptoalg_ctx; + +#define NULL_TAG_LEN 12 + +err_status_t +null_enc(void *key, + const void *clear, + unsigned clear_len, + void *iv, + void *opaque, + unsigned *opaque_len) { + int i; + unsigned char *auth_tag; + unsigned char *init_vec = iv; + + /* check if we're doing authentication only */ + if ((iv == NULL) && (opaque == NULL) && (opaque_len == NULL)) { + + /* perform authentication only */ + + } else if ((iv == NULL) || (opaque == NULL) || (opaque_len == NULL)) { + + /* + * bad parameter - we expect either all three pointers to be NULL, + * or none of those pointers to be NULL + */ + return err_status_fail; + + } else { + +#if DEBUG + printf("NULL ENC using key %s\n", octet_string_hex_string(key, KEY_LEN)); + printf("NULL_TAG_LEN: %d\n", NULL_TAG_LEN); + printf("plaintext len: %d\n", *opaque_len); +#endif + for (i=0; i < IV_LEN; i++) + init_vec[i] = i + (i * 16); +#if DEBUG + printf("iv: %s\n", + octet_string_hex_string(iv, IV_LEN)); + printf("plaintext: %s\n", + octet_string_hex_string(opaque, *opaque_len)); +#endif + auth_tag = opaque; + auth_tag += *opaque_len; + for (i=0; i < NULL_TAG_LEN; i++) + auth_tag[i] = i + (i * 16); + *opaque_len += NULL_TAG_LEN; +#if DEBUG + printf("protected data len: %d\n", *opaque_len); + printf("protected data: %s\n", + octet_string_hex_string(opaque, *opaque_len)); +#endif + + } + + return err_status_ok; +} + +err_status_t +null_dec(void *key, + const void *clear, + unsigned clear_len, + void *iv, + void *opaque, + unsigned *opaque_len) { + unsigned char *auth_tag; + + /* check if we're doing authentication only */ + if ((iv == NULL) && (opaque == NULL) && (opaque_len == NULL)) { + + /* perform authentication only */ + + } else if ((iv == NULL) || (opaque == NULL) || (opaque_len == NULL)) { + + /* + * bad parameter - we expect either all three pointers to be NULL, + * or none of those pointers to be NULL + */ + return err_status_fail; + + } else { + +#if DEBUG + printf("NULL DEC using key %s\n", octet_string_hex_string(key, KEY_LEN)); + + printf("protected data len: %d\n", *opaque_len); + printf("protected data: %s\n", + octet_string_hex_string(opaque, *opaque_len)); +#endif + auth_tag = opaque; + auth_tag += (*opaque_len - NULL_TAG_LEN); +#if DEBUG + printf("iv: %s\n", octet_string_hex_string(iv, IV_LEN)); +#endif + *opaque_len -= NULL_TAG_LEN; +#if DEBUG + printf("plaintext len: %d\n", *opaque_len); + printf("plaintext: %s\n", + octet_string_hex_string(opaque, *opaque_len)); +#endif + } + + return err_status_ok; +} + +cryptoalg_ctx_t null_cryptoalg_ctx = { + null_enc, + null_dec, + KEY_LEN, + IV_LEN, + NULL_TAG_LEN, + MAX_EXPAND, +}; + +cryptoalg_t null_cryptoalg = &null_cryptoalg_ctx; + +int +cryptoalg_get_id(cryptoalg_t c) { + if (c == cryptoalg) + return 1; + return 0; +} + +cryptoalg_t +cryptoalg_find_by_id(int id) { + switch(id) { + case 1: + return cryptoalg; + default: + return 0; + } + return 0; +} diff --git a/libs/srtp/crypto/cipher/CVS/Entries b/libs/srtp/crypto/cipher/CVS/Entries new file mode 100644 index 0000000000..11e048d8af --- /dev/null +++ b/libs/srtp/crypto/cipher/CVS/Entries @@ -0,0 +1,6 @@ +/aes.c/1.8/Sat Oct 8 16:39:25 2005// +/aes_cbc.c/1.5/Sat Oct 8 16:38:06 2005// +/aes_icm.c/1.10/Thu Mar 16 17:11:29 2006// +/cipher.c/1.4/Mon Oct 3 15:27:53 2005// +/null_cipher.c/1.2/Thu Sep 29 12:36:43 2005// +D diff --git a/libs/srtp/crypto/cipher/CVS/Repository b/libs/srtp/crypto/cipher/CVS/Repository new file mode 100644 index 0000000000..06fe3bc833 --- /dev/null +++ b/libs/srtp/crypto/cipher/CVS/Repository @@ -0,0 +1 @@ +srtp/crypto/cipher diff --git a/libs/srtp/crypto/cipher/CVS/Root b/libs/srtp/crypto/cipher/CVS/Root new file mode 100644 index 0000000000..b4b74237d9 --- /dev/null +++ b/libs/srtp/crypto/cipher/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@cvs.sourceforge.net:/cvsroot/srtp diff --git a/libs/srtp/crypto/cipher/aes.c b/libs/srtp/crypto/cipher/aes.c new file mode 100644 index 0000000000..921de32838 --- /dev/null +++ b/libs/srtp/crypto/cipher/aes.c @@ -0,0 +1,1951 @@ +/* + * aes.c + * + * An implemnetation of the AES block cipher. + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#include "aes.h" +#include "err.h" + +/* + * we use the tables T0, T1, T2, T3, and T4 to compute AES, and + * the tables U0, U1, U2, and U4 to compute its inverse + * + * different tables are used on little-endian (Intel, VMS) and + * big-endian processors (everything else) + * + * these tables are computed using the program tables/aes_tables; use + * this program to generate different tables for porting or + * optimization on a different platform + */ + +#ifndef WORDS_BIGENDIAN + +static uint32_t T0[256] = { + 0xa56363c6, 0x847c7cf8, 0x997777ee, 0x8d7b7bf6, + 0xdf2f2ff, 0xbd6b6bd6, 0xb16f6fde, 0x54c5c591, + 0x50303060, 0x3010102, 0xa96767ce, 0x7d2b2b56, + 0x19fefee7, 0x62d7d7b5, 0xe6abab4d, 0x9a7676ec, + 0x45caca8f, 0x9d82821f, 0x40c9c989, 0x877d7dfa, + 0x15fafaef, 0xeb5959b2, 0xc947478e, 0xbf0f0fb, + 0xecadad41, 0x67d4d4b3, 0xfda2a25f, 0xeaafaf45, + 0xbf9c9c23, 0xf7a4a453, 0x967272e4, 0x5bc0c09b, + 0xc2b7b775, 0x1cfdfde1, 0xae93933d, 0x6a26264c, + 0x5a36366c, 0x413f3f7e, 0x2f7f7f5, 0x4fcccc83, + 0x5c343468, 0xf4a5a551, 0x34e5e5d1, 0x8f1f1f9, + 0x937171e2, 0x73d8d8ab, 0x53313162, 0x3f15152a, + 0xc040408, 0x52c7c795, 0x65232346, 0x5ec3c39d, + 0x28181830, 0xa1969637, 0xf05050a, 0xb59a9a2f, + 0x907070e, 0x36121224, 0x9b80801b, 0x3de2e2df, + 0x26ebebcd, 0x6927274e, 0xcdb2b27f, 0x9f7575ea, + 0x1b090912, 0x9e83831d, 0x742c2c58, 0x2e1a1a34, + 0x2d1b1b36, 0xb26e6edc, 0xee5a5ab4, 0xfba0a05b, + 0xf65252a4, 0x4d3b3b76, 0x61d6d6b7, 0xceb3b37d, + 0x7b292952, 0x3ee3e3dd, 0x712f2f5e, 0x97848413, + 0xf55353a6, 0x68d1d1b9, 0x0, 0x2cededc1, + 0x60202040, 0x1ffcfce3, 0xc8b1b179, 0xed5b5bb6, + 0xbe6a6ad4, 0x46cbcb8d, 0xd9bebe67, 0x4b393972, + 0xde4a4a94, 0xd44c4c98, 0xe85858b0, 0x4acfcf85, + 0x6bd0d0bb, 0x2aefefc5, 0xe5aaaa4f, 0x16fbfbed, + 0xc5434386, 0xd74d4d9a, 0x55333366, 0x94858511, + 0xcf45458a, 0x10f9f9e9, 0x6020204, 0x817f7ffe, + 0xf05050a0, 0x443c3c78, 0xba9f9f25, 0xe3a8a84b, + 0xf35151a2, 0xfea3a35d, 0xc0404080, 0x8a8f8f05, + 0xad92923f, 0xbc9d9d21, 0x48383870, 0x4f5f5f1, + 0xdfbcbc63, 0xc1b6b677, 0x75dadaaf, 0x63212142, + 0x30101020, 0x1affffe5, 0xef3f3fd, 0x6dd2d2bf, + 0x4ccdcd81, 0x140c0c18, 0x35131326, 0x2fececc3, + 0xe15f5fbe, 0xa2979735, 0xcc444488, 0x3917172e, + 0x57c4c493, 0xf2a7a755, 0x827e7efc, 0x473d3d7a, + 0xac6464c8, 0xe75d5dba, 0x2b191932, 0x957373e6, + 0xa06060c0, 0x98818119, 0xd14f4f9e, 0x7fdcdca3, + 0x66222244, 0x7e2a2a54, 0xab90903b, 0x8388880b, + 0xca46468c, 0x29eeeec7, 0xd3b8b86b, 0x3c141428, + 0x79dedea7, 0xe25e5ebc, 0x1d0b0b16, 0x76dbdbad, + 0x3be0e0db, 0x56323264, 0x4e3a3a74, 0x1e0a0a14, + 0xdb494992, 0xa06060c, 0x6c242448, 0xe45c5cb8, + 0x5dc2c29f, 0x6ed3d3bd, 0xefacac43, 0xa66262c4, + 0xa8919139, 0xa4959531, 0x37e4e4d3, 0x8b7979f2, + 0x32e7e7d5, 0x43c8c88b, 0x5937376e, 0xb76d6dda, + 0x8c8d8d01, 0x64d5d5b1, 0xd24e4e9c, 0xe0a9a949, + 0xb46c6cd8, 0xfa5656ac, 0x7f4f4f3, 0x25eaeacf, + 0xaf6565ca, 0x8e7a7af4, 0xe9aeae47, 0x18080810, + 0xd5baba6f, 0x887878f0, 0x6f25254a, 0x722e2e5c, + 0x241c1c38, 0xf1a6a657, 0xc7b4b473, 0x51c6c697, + 0x23e8e8cb, 0x7cdddda1, 0x9c7474e8, 0x211f1f3e, + 0xdd4b4b96, 0xdcbdbd61, 0x868b8b0d, 0x858a8a0f, + 0x907070e0, 0x423e3e7c, 0xc4b5b571, 0xaa6666cc, + 0xd8484890, 0x5030306, 0x1f6f6f7, 0x120e0e1c, + 0xa36161c2, 0x5f35356a, 0xf95757ae, 0xd0b9b969, + 0x91868617, 0x58c1c199, 0x271d1d3a, 0xb99e9e27, + 0x38e1e1d9, 0x13f8f8eb, 0xb398982b, 0x33111122, + 0xbb6969d2, 0x70d9d9a9, 0x898e8e07, 0xa7949433, + 0xb69b9b2d, 0x221e1e3c, 0x92878715, 0x20e9e9c9, + 0x49cece87, 0xff5555aa, 0x78282850, 0x7adfdfa5, + 0x8f8c8c03, 0xf8a1a159, 0x80898909, 0x170d0d1a, + 0xdabfbf65, 0x31e6e6d7, 0xc6424284, 0xb86868d0, + 0xc3414182, 0xb0999929, 0x772d2d5a, 0x110f0f1e, + 0xcbb0b07b, 0xfc5454a8, 0xd6bbbb6d, 0x3a16162c, +}; + +static uint32_t T1[256] = { + 0x6363c6a5, 0x7c7cf884, 0x7777ee99, 0x7b7bf68d, + 0xf2f2ff0d, 0x6b6bd6bd, 0x6f6fdeb1, 0xc5c59154, + 0x30306050, 0x1010203, 0x6767cea9, 0x2b2b567d, + 0xfefee719, 0xd7d7b562, 0xabab4de6, 0x7676ec9a, + 0xcaca8f45, 0x82821f9d, 0xc9c98940, 0x7d7dfa87, + 0xfafaef15, 0x5959b2eb, 0x47478ec9, 0xf0f0fb0b, + 0xadad41ec, 0xd4d4b367, 0xa2a25ffd, 0xafaf45ea, + 0x9c9c23bf, 0xa4a453f7, 0x7272e496, 0xc0c09b5b, + 0xb7b775c2, 0xfdfde11c, 0x93933dae, 0x26264c6a, + 0x36366c5a, 0x3f3f7e41, 0xf7f7f502, 0xcccc834f, + 0x3434685c, 0xa5a551f4, 0xe5e5d134, 0xf1f1f908, + 0x7171e293, 0xd8d8ab73, 0x31316253, 0x15152a3f, + 0x404080c, 0xc7c79552, 0x23234665, 0xc3c39d5e, + 0x18183028, 0x969637a1, 0x5050a0f, 0x9a9a2fb5, + 0x7070e09, 0x12122436, 0x80801b9b, 0xe2e2df3d, + 0xebebcd26, 0x27274e69, 0xb2b27fcd, 0x7575ea9f, + 0x909121b, 0x83831d9e, 0x2c2c5874, 0x1a1a342e, + 0x1b1b362d, 0x6e6edcb2, 0x5a5ab4ee, 0xa0a05bfb, + 0x5252a4f6, 0x3b3b764d, 0xd6d6b761, 0xb3b37dce, + 0x2929527b, 0xe3e3dd3e, 0x2f2f5e71, 0x84841397, + 0x5353a6f5, 0xd1d1b968, 0x00000000, 0xededc12c, + 0x20204060, 0xfcfce31f, 0xb1b179c8, 0x5b5bb6ed, + 0x6a6ad4be, 0xcbcb8d46, 0xbebe67d9, 0x3939724b, + 0x4a4a94de, 0x4c4c98d4, 0x5858b0e8, 0xcfcf854a, + 0xd0d0bb6b, 0xefefc52a, 0xaaaa4fe5, 0xfbfbed16, + 0x434386c5, 0x4d4d9ad7, 0x33336655, 0x85851194, + 0x45458acf, 0xf9f9e910, 0x2020406, 0x7f7ffe81, + 0x5050a0f0, 0x3c3c7844, 0x9f9f25ba, 0xa8a84be3, + 0x5151a2f3, 0xa3a35dfe, 0x404080c0, 0x8f8f058a, + 0x92923fad, 0x9d9d21bc, 0x38387048, 0xf5f5f104, + 0xbcbc63df, 0xb6b677c1, 0xdadaaf75, 0x21214263, + 0x10102030, 0xffffe51a, 0xf3f3fd0e, 0xd2d2bf6d, + 0xcdcd814c, 0xc0c1814, 0x13132635, 0xececc32f, + 0x5f5fbee1, 0x979735a2, 0x444488cc, 0x17172e39, + 0xc4c49357, 0xa7a755f2, 0x7e7efc82, 0x3d3d7a47, + 0x6464c8ac, 0x5d5dbae7, 0x1919322b, 0x7373e695, + 0x6060c0a0, 0x81811998, 0x4f4f9ed1, 0xdcdca37f, + 0x22224466, 0x2a2a547e, 0x90903bab, 0x88880b83, + 0x46468cca, 0xeeeec729, 0xb8b86bd3, 0x1414283c, + 0xdedea779, 0x5e5ebce2, 0xb0b161d, 0xdbdbad76, + 0xe0e0db3b, 0x32326456, 0x3a3a744e, 0xa0a141e, + 0x494992db, 0x6060c0a, 0x2424486c, 0x5c5cb8e4, + 0xc2c29f5d, 0xd3d3bd6e, 0xacac43ef, 0x6262c4a6, + 0x919139a8, 0x959531a4, 0xe4e4d337, 0x7979f28b, + 0xe7e7d532, 0xc8c88b43, 0x37376e59, 0x6d6ddab7, + 0x8d8d018c, 0xd5d5b164, 0x4e4e9cd2, 0xa9a949e0, + 0x6c6cd8b4, 0x5656acfa, 0xf4f4f307, 0xeaeacf25, + 0x6565caaf, 0x7a7af48e, 0xaeae47e9, 0x8081018, + 0xbaba6fd5, 0x7878f088, 0x25254a6f, 0x2e2e5c72, + 0x1c1c3824, 0xa6a657f1, 0xb4b473c7, 0xc6c69751, + 0xe8e8cb23, 0xdddda17c, 0x7474e89c, 0x1f1f3e21, + 0x4b4b96dd, 0xbdbd61dc, 0x8b8b0d86, 0x8a8a0f85, + 0x7070e090, 0x3e3e7c42, 0xb5b571c4, 0x6666ccaa, + 0x484890d8, 0x3030605, 0xf6f6f701, 0xe0e1c12, + 0x6161c2a3, 0x35356a5f, 0x5757aef9, 0xb9b969d0, + 0x86861791, 0xc1c19958, 0x1d1d3a27, 0x9e9e27b9, + 0xe1e1d938, 0xf8f8eb13, 0x98982bb3, 0x11112233, + 0x6969d2bb, 0xd9d9a970, 0x8e8e0789, 0x949433a7, + 0x9b9b2db6, 0x1e1e3c22, 0x87871592, 0xe9e9c920, + 0xcece8749, 0x5555aaff, 0x28285078, 0xdfdfa57a, + 0x8c8c038f, 0xa1a159f8, 0x89890980, 0xd0d1a17, + 0xbfbf65da, 0xe6e6d731, 0x424284c6, 0x6868d0b8, + 0x414182c3, 0x999929b0, 0x2d2d5a77, 0xf0f1e11, + 0xb0b07bcb, 0x5454a8fc, 0xbbbb6dd6, 0x16162c3a, +}; + +static uint32_t T2[256] = { + 0x63c6a563, 0x7cf8847c, 0x77ee9977, 0x7bf68d7b, + 0xf2ff0df2, 0x6bd6bd6b, 0x6fdeb16f, 0xc59154c5, + 0x30605030, 0x1020301, 0x67cea967, 0x2b567d2b, + 0xfee719fe, 0xd7b562d7, 0xab4de6ab, 0x76ec9a76, + 0xca8f45ca, 0x821f9d82, 0xc98940c9, 0x7dfa877d, + 0xfaef15fa, 0x59b2eb59, 0x478ec947, 0xf0fb0bf0, + 0xad41ecad, 0xd4b367d4, 0xa25ffda2, 0xaf45eaaf, + 0x9c23bf9c, 0xa453f7a4, 0x72e49672, 0xc09b5bc0, + 0xb775c2b7, 0xfde11cfd, 0x933dae93, 0x264c6a26, + 0x366c5a36, 0x3f7e413f, 0xf7f502f7, 0xcc834fcc, + 0x34685c34, 0xa551f4a5, 0xe5d134e5, 0xf1f908f1, + 0x71e29371, 0xd8ab73d8, 0x31625331, 0x152a3f15, + 0x4080c04, 0xc79552c7, 0x23466523, 0xc39d5ec3, + 0x18302818, 0x9637a196, 0x50a0f05, 0x9a2fb59a, + 0x70e0907, 0x12243612, 0x801b9b80, 0xe2df3de2, + 0xebcd26eb, 0x274e6927, 0xb27fcdb2, 0x75ea9f75, + 0x9121b09, 0x831d9e83, 0x2c58742c, 0x1a342e1a, + 0x1b362d1b, 0x6edcb26e, 0x5ab4ee5a, 0xa05bfba0, + 0x52a4f652, 0x3b764d3b, 0xd6b761d6, 0xb37dceb3, + 0x29527b29, 0xe3dd3ee3, 0x2f5e712f, 0x84139784, + 0x53a6f553, 0xd1b968d1, 0x0, 0xedc12ced, + 0x20406020, 0xfce31ffc, 0xb179c8b1, 0x5bb6ed5b, + 0x6ad4be6a, 0xcb8d46cb, 0xbe67d9be, 0x39724b39, + 0x4a94de4a, 0x4c98d44c, 0x58b0e858, 0xcf854acf, + 0xd0bb6bd0, 0xefc52aef, 0xaa4fe5aa, 0xfbed16fb, + 0x4386c543, 0x4d9ad74d, 0x33665533, 0x85119485, + 0x458acf45, 0xf9e910f9, 0x2040602, 0x7ffe817f, + 0x50a0f050, 0x3c78443c, 0x9f25ba9f, 0xa84be3a8, + 0x51a2f351, 0xa35dfea3, 0x4080c040, 0x8f058a8f, + 0x923fad92, 0x9d21bc9d, 0x38704838, 0xf5f104f5, + 0xbc63dfbc, 0xb677c1b6, 0xdaaf75da, 0x21426321, + 0x10203010, 0xffe51aff, 0xf3fd0ef3, 0xd2bf6dd2, + 0xcd814ccd, 0xc18140c, 0x13263513, 0xecc32fec, + 0x5fbee15f, 0x9735a297, 0x4488cc44, 0x172e3917, + 0xc49357c4, 0xa755f2a7, 0x7efc827e, 0x3d7a473d, + 0x64c8ac64, 0x5dbae75d, 0x19322b19, 0x73e69573, + 0x60c0a060, 0x81199881, 0x4f9ed14f, 0xdca37fdc, + 0x22446622, 0x2a547e2a, 0x903bab90, 0x880b8388, + 0x468cca46, 0xeec729ee, 0xb86bd3b8, 0x14283c14, + 0xdea779de, 0x5ebce25e, 0xb161d0b, 0xdbad76db, + 0xe0db3be0, 0x32645632, 0x3a744e3a, 0xa141e0a, + 0x4992db49, 0x60c0a06, 0x24486c24, 0x5cb8e45c, + 0xc29f5dc2, 0xd3bd6ed3, 0xac43efac, 0x62c4a662, + 0x9139a891, 0x9531a495, 0xe4d337e4, 0x79f28b79, + 0xe7d532e7, 0xc88b43c8, 0x376e5937, 0x6ddab76d, + 0x8d018c8d, 0xd5b164d5, 0x4e9cd24e, 0xa949e0a9, + 0x6cd8b46c, 0x56acfa56, 0xf4f307f4, 0xeacf25ea, + 0x65caaf65, 0x7af48e7a, 0xae47e9ae, 0x8101808, + 0xba6fd5ba, 0x78f08878, 0x254a6f25, 0x2e5c722e, + 0x1c38241c, 0xa657f1a6, 0xb473c7b4, 0xc69751c6, + 0xe8cb23e8, 0xdda17cdd, 0x74e89c74, 0x1f3e211f, + 0x4b96dd4b, 0xbd61dcbd, 0x8b0d868b, 0x8a0f858a, + 0x70e09070, 0x3e7c423e, 0xb571c4b5, 0x66ccaa66, + 0x4890d848, 0x3060503, 0xf6f701f6, 0xe1c120e, + 0x61c2a361, 0x356a5f35, 0x57aef957, 0xb969d0b9, + 0x86179186, 0xc19958c1, 0x1d3a271d, 0x9e27b99e, + 0xe1d938e1, 0xf8eb13f8, 0x982bb398, 0x11223311, + 0x69d2bb69, 0xd9a970d9, 0x8e07898e, 0x9433a794, + 0x9b2db69b, 0x1e3c221e, 0x87159287, 0xe9c920e9, + 0xce8749ce, 0x55aaff55, 0x28507828, 0xdfa57adf, + 0x8c038f8c, 0xa159f8a1, 0x89098089, 0xd1a170d, + 0xbf65dabf, 0xe6d731e6, 0x4284c642, 0x68d0b868, + 0x4182c341, 0x9929b099, 0x2d5a772d, 0xf1e110f, + 0xb07bcbb0, 0x54a8fc54, 0xbb6dd6bb, 0x162c3a16, +}; + +static uint32_t T3[256] = { + 0xc6a56363, 0xf8847c7c, 0xee997777, 0xf68d7b7b, + 0xff0df2f2, 0xd6bd6b6b, 0xdeb16f6f, 0x9154c5c5, + 0x60503030, 0x2030101, 0xcea96767, 0x567d2b2b, + 0xe719fefe, 0xb562d7d7, 0x4de6abab, 0xec9a7676, + 0x8f45caca, 0x1f9d8282, 0x8940c9c9, 0xfa877d7d, + 0xef15fafa, 0xb2eb5959, 0x8ec94747, 0xfb0bf0f0, + 0x41ecadad, 0xb367d4d4, 0x5ffda2a2, 0x45eaafaf, + 0x23bf9c9c, 0x53f7a4a4, 0xe4967272, 0x9b5bc0c0, + 0x75c2b7b7, 0xe11cfdfd, 0x3dae9393, 0x4c6a2626, + 0x6c5a3636, 0x7e413f3f, 0xf502f7f7, 0x834fcccc, + 0x685c3434, 0x51f4a5a5, 0xd134e5e5, 0xf908f1f1, + 0xe2937171, 0xab73d8d8, 0x62533131, 0x2a3f1515, + 0x80c0404, 0x9552c7c7, 0x46652323, 0x9d5ec3c3, + 0x30281818, 0x37a19696, 0xa0f0505, 0x2fb59a9a, + 0xe090707, 0x24361212, 0x1b9b8080, 0xdf3de2e2, + 0xcd26ebeb, 0x4e692727, 0x7fcdb2b2, 0xea9f7575, + 0x121b0909, 0x1d9e8383, 0x58742c2c, 0x342e1a1a, + 0x362d1b1b, 0xdcb26e6e, 0xb4ee5a5a, 0x5bfba0a0, + 0xa4f65252, 0x764d3b3b, 0xb761d6d6, 0x7dceb3b3, + 0x527b2929, 0xdd3ee3e3, 0x5e712f2f, 0x13978484, + 0xa6f55353, 0xb968d1d1, 0x0, 0xc12ceded, + 0x40602020, 0xe31ffcfc, 0x79c8b1b1, 0xb6ed5b5b, + 0xd4be6a6a, 0x8d46cbcb, 0x67d9bebe, 0x724b3939, + 0x94de4a4a, 0x98d44c4c, 0xb0e85858, 0x854acfcf, + 0xbb6bd0d0, 0xc52aefef, 0x4fe5aaaa, 0xed16fbfb, + 0x86c54343, 0x9ad74d4d, 0x66553333, 0x11948585, + 0x8acf4545, 0xe910f9f9, 0x4060202, 0xfe817f7f, + 0xa0f05050, 0x78443c3c, 0x25ba9f9f, 0x4be3a8a8, + 0xa2f35151, 0x5dfea3a3, 0x80c04040, 0x58a8f8f, + 0x3fad9292, 0x21bc9d9d, 0x70483838, 0xf104f5f5, + 0x63dfbcbc, 0x77c1b6b6, 0xaf75dada, 0x42632121, + 0x20301010, 0xe51affff, 0xfd0ef3f3, 0xbf6dd2d2, + 0x814ccdcd, 0x18140c0c, 0x26351313, 0xc32fecec, + 0xbee15f5f, 0x35a29797, 0x88cc4444, 0x2e391717, + 0x9357c4c4, 0x55f2a7a7, 0xfc827e7e, 0x7a473d3d, + 0xc8ac6464, 0xbae75d5d, 0x322b1919, 0xe6957373, + 0xc0a06060, 0x19988181, 0x9ed14f4f, 0xa37fdcdc, + 0x44662222, 0x547e2a2a, 0x3bab9090, 0xb838888, + 0x8cca4646, 0xc729eeee, 0x6bd3b8b8, 0x283c1414, + 0xa779dede, 0xbce25e5e, 0x161d0b0b, 0xad76dbdb, + 0xdb3be0e0, 0x64563232, 0x744e3a3a, 0x141e0a0a, + 0x92db4949, 0xc0a0606, 0x486c2424, 0xb8e45c5c, + 0x9f5dc2c2, 0xbd6ed3d3, 0x43efacac, 0xc4a66262, + 0x39a89191, 0x31a49595, 0xd337e4e4, 0xf28b7979, + 0xd532e7e7, 0x8b43c8c8, 0x6e593737, 0xdab76d6d, + 0x18c8d8d, 0xb164d5d5, 0x9cd24e4e, 0x49e0a9a9, + 0xd8b46c6c, 0xacfa5656, 0xf307f4f4, 0xcf25eaea, + 0xcaaf6565, 0xf48e7a7a, 0x47e9aeae, 0x10180808, + 0x6fd5baba, 0xf0887878, 0x4a6f2525, 0x5c722e2e, + 0x38241c1c, 0x57f1a6a6, 0x73c7b4b4, 0x9751c6c6, + 0xcb23e8e8, 0xa17cdddd, 0xe89c7474, 0x3e211f1f, + 0x96dd4b4b, 0x61dcbdbd, 0xd868b8b, 0xf858a8a, + 0xe0907070, 0x7c423e3e, 0x71c4b5b5, 0xccaa6666, + 0x90d84848, 0x6050303, 0xf701f6f6, 0x1c120e0e, + 0xc2a36161, 0x6a5f3535, 0xaef95757, 0x69d0b9b9, + 0x17918686, 0x9958c1c1, 0x3a271d1d, 0x27b99e9e, + 0xd938e1e1, 0xeb13f8f8, 0x2bb39898, 0x22331111, + 0xd2bb6969, 0xa970d9d9, 0x7898e8e, 0x33a79494, + 0x2db69b9b, 0x3c221e1e, 0x15928787, 0xc920e9e9, + 0x8749cece, 0xaaff5555, 0x50782828, 0xa57adfdf, + 0x38f8c8c, 0x59f8a1a1, 0x9808989, 0x1a170d0d, + 0x65dabfbf, 0xd731e6e6, 0x84c64242, 0xd0b86868, + 0x82c34141, 0x29b09999, 0x5a772d2d, 0x1e110f0f, + 0x7bcbb0b0, 0xa8fc5454, 0x6dd6bbbb, 0x2c3a1616, +}; + +static uint32_t U0[256] = { + 0x50a7f451, 0x5365417e, 0xc3a4171a, 0x965e273a, + 0xcb6bab3b, 0xf1459d1f, 0xab58faac, 0x9303e34b, + 0x55fa3020, 0xf66d76ad, 0x9176cc88, 0x254c02f5, + 0xfcd7e54f, 0xd7cb2ac5, 0x80443526, 0x8fa362b5, + 0x495ab1de, 0x671bba25, 0x980eea45, 0xe1c0fe5d, + 0x2752fc3, 0x12f04c81, 0xa397468d, 0xc6f9d36b, + 0xe75f8f03, 0x959c9215, 0xeb7a6dbf, 0xda595295, + 0x2d83bed4, 0xd3217458, 0x2969e049, 0x44c8c98e, + 0x6a89c275, 0x78798ef4, 0x6b3e5899, 0xdd71b927, + 0xb64fe1be, 0x17ad88f0, 0x66ac20c9, 0xb43ace7d, + 0x184adf63, 0x82311ae5, 0x60335197, 0x457f5362, + 0xe07764b1, 0x84ae6bbb, 0x1ca081fe, 0x942b08f9, + 0x58684870, 0x19fd458f, 0x876cde94, 0xb7f87b52, + 0x23d373ab, 0xe2024b72, 0x578f1fe3, 0x2aab5566, + 0x728ebb2, 0x3c2b52f, 0x9a7bc586, 0xa50837d3, + 0xf2872830, 0xb2a5bf23, 0xba6a0302, 0x5c8216ed, + 0x2b1ccf8a, 0x92b479a7, 0xf0f207f3, 0xa1e2694e, + 0xcdf4da65, 0xd5be0506, 0x1f6234d1, 0x8afea6c4, + 0x9d532e34, 0xa055f3a2, 0x32e18a05, 0x75ebf6a4, + 0x39ec830b, 0xaaef6040, 0x69f715e, 0x51106ebd, + 0xf98a213e, 0x3d06dd96, 0xae053edd, 0x46bde64d, + 0xb58d5491, 0x55dc471, 0x6fd40604, 0xff155060, + 0x24fb9819, 0x97e9bdd6, 0xcc434089, 0x779ed967, + 0xbd42e8b0, 0x888b8907, 0x385b19e7, 0xdbeec879, + 0x470a7ca1, 0xe90f427c, 0xc91e84f8, 0x0, + 0x83868009, 0x48ed2b32, 0xac70111e, 0x4e725a6c, + 0xfbff0efd, 0x5638850f, 0x1ed5ae3d, 0x27392d36, + 0x64d90f0a, 0x21a65c68, 0xd1545b9b, 0x3a2e3624, + 0xb1670a0c, 0xfe75793, 0xd296eeb4, 0x9e919b1b, + 0x4fc5c080, 0xa220dc61, 0x694b775a, 0x161a121c, + 0xaba93e2, 0xe52aa0c0, 0x43e0223c, 0x1d171b12, + 0xb0d090e, 0xadc78bf2, 0xb9a8b62d, 0xc8a91e14, + 0x8519f157, 0x4c0775af, 0xbbdd99ee, 0xfd607fa3, + 0x9f2601f7, 0xbcf5725c, 0xc53b6644, 0x347efb5b, + 0x7629438b, 0xdcc623cb, 0x68fcedb6, 0x63f1e4b8, + 0xcadc31d7, 0x10856342, 0x40229713, 0x2011c684, + 0x7d244a85, 0xf83dbbd2, 0x1132f9ae, 0x6da129c7, + 0x4b2f9e1d, 0xf330b2dc, 0xec52860d, 0xd0e3c177, + 0x6c16b32b, 0x99b970a9, 0xfa489411, 0x2264e947, + 0xc48cfca8, 0x1a3ff0a0, 0xd82c7d56, 0xef903322, + 0xc74e4987, 0xc1d138d9, 0xfea2ca8c, 0x360bd498, + 0xcf81f5a6, 0x28de7aa5, 0x268eb7da, 0xa4bfad3f, + 0xe49d3a2c, 0xd927850, 0x9bcc5f6a, 0x62467e54, + 0xc2138df6, 0xe8b8d890, 0x5ef7392e, 0xf5afc382, + 0xbe805d9f, 0x7c93d069, 0xa92dd56f, 0xb31225cf, + 0x3b99acc8, 0xa77d1810, 0x6e639ce8, 0x7bbb3bdb, + 0x97826cd, 0xf418596e, 0x1b79aec, 0xa89a4f83, + 0x656e95e6, 0x7ee6ffaa, 0x8cfbc21, 0xe6e815ef, + 0xd99be7ba, 0xce366f4a, 0xd4099fea, 0xd67cb029, + 0xafb2a431, 0x31233f2a, 0x3094a5c6, 0xc066a235, + 0x37bc4e74, 0xa6ca82fc, 0xb0d090e0, 0x15d8a733, + 0x4a9804f1, 0xf7daec41, 0xe50cd7f, 0x2ff69117, + 0x8dd64d76, 0x4db0ef43, 0x544daacc, 0xdf0496e4, + 0xe3b5d19e, 0x1b886a4c, 0xb81f2cc1, 0x7f516546, + 0x4ea5e9d, 0x5d358c01, 0x737487fa, 0x2e410bfb, + 0x5a1d67b3, 0x52d2db92, 0x335610e9, 0x1347d66d, + 0x8c61d79a, 0x7a0ca137, 0x8e14f859, 0x893c13eb, + 0xee27a9ce, 0x35c961b7, 0xede51ce1, 0x3cb1477a, + 0x59dfd29c, 0x3f73f255, 0x79ce1418, 0xbf37c773, + 0xeacdf753, 0x5baafd5f, 0x146f3ddf, 0x86db4478, + 0x81f3afca, 0x3ec468b9, 0x2c342438, 0x5f40a3c2, + 0x72c31d16, 0xc25e2bc, 0x8b493c28, 0x41950dff, + 0x7101a839, 0xdeb30c08, 0x9ce4b4d8, 0x90c15664, + 0x6184cb7b, 0x70b632d5, 0x745c6c48, 0x4257b8d0, +}; + +static uint32_t U1[256] = { + 0xa7f45150, 0x65417e53, 0xa4171ac3, 0x5e273a96, + 0x6bab3bcb, 0x459d1ff1, 0x58faacab, 0x3e34b93, + 0xfa302055, 0x6d76adf6, 0x76cc8891, 0x4c02f525, + 0xd7e54ffc, 0xcb2ac5d7, 0x44352680, 0xa362b58f, + 0x5ab1de49, 0x1bba2567, 0xeea4598, 0xc0fe5de1, + 0x752fc302, 0xf04c8112, 0x97468da3, 0xf9d36bc6, + 0x5f8f03e7, 0x9c921595, 0x7a6dbfeb, 0x595295da, + 0x83bed42d, 0x217458d3, 0x69e04929, 0xc8c98e44, + 0x89c2756a, 0x798ef478, 0x3e58996b, 0x71b927dd, + 0x4fe1beb6, 0xad88f017, 0xac20c966, 0x3ace7db4, + 0x4adf6318, 0x311ae582, 0x33519760, 0x7f536245, + 0x7764b1e0, 0xae6bbb84, 0xa081fe1c, 0x2b08f994, + 0x68487058, 0xfd458f19, 0x6cde9487, 0xf87b52b7, + 0xd373ab23, 0x24b72e2, 0x8f1fe357, 0xab55662a, + 0x28ebb207, 0xc2b52f03, 0x7bc5869a, 0x837d3a5, + 0x872830f2, 0xa5bf23b2, 0x6a0302ba, 0x8216ed5c, + 0x1ccf8a2b, 0xb479a792, 0xf207f3f0, 0xe2694ea1, + 0xf4da65cd, 0xbe0506d5, 0x6234d11f, 0xfea6c48a, + 0x532e349d, 0x55f3a2a0, 0xe18a0532, 0xebf6a475, + 0xec830b39, 0xef6040aa, 0x9f715e06, 0x106ebd51, + 0x8a213ef9, 0x6dd963d, 0x53eddae, 0xbde64d46, + 0x8d5491b5, 0x5dc47105, 0xd406046f, 0x155060ff, + 0xfb981924, 0xe9bdd697, 0x434089cc, 0x9ed96777, + 0x42e8b0bd, 0x8b890788, 0x5b19e738, 0xeec879db, + 0xa7ca147, 0xf427ce9, 0x1e84f8c9, 0x0, + 0x86800983, 0xed2b3248, 0x70111eac, 0x725a6c4e, + 0xff0efdfb, 0x38850f56, 0xd5ae3d1e, 0x392d3627, + 0xd90f0a64, 0xa65c6821, 0x545b9bd1, 0x2e36243a, + 0x670a0cb1, 0xe757930f, 0x96eeb4d2, 0x919b1b9e, + 0xc5c0804f, 0x20dc61a2, 0x4b775a69, 0x1a121c16, + 0xba93e20a, 0x2aa0c0e5, 0xe0223c43, 0x171b121d, + 0xd090e0b, 0xc78bf2ad, 0xa8b62db9, 0xa91e14c8, + 0x19f15785, 0x775af4c, 0xdd99eebb, 0x607fa3fd, + 0x2601f79f, 0xf5725cbc, 0x3b6644c5, 0x7efb5b34, + 0x29438b76, 0xc623cbdc, 0xfcedb668, 0xf1e4b863, + 0xdc31d7ca, 0x85634210, 0x22971340, 0x11c68420, + 0x244a857d, 0x3dbbd2f8, 0x32f9ae11, 0xa129c76d, + 0x2f9e1d4b, 0x30b2dcf3, 0x52860dec, 0xe3c177d0, + 0x16b32b6c, 0xb970a999, 0x489411fa, 0x64e94722, + 0x8cfca8c4, 0x3ff0a01a, 0x2c7d56d8, 0x903322ef, + 0x4e4987c7, 0xd138d9c1, 0xa2ca8cfe, 0xbd49836, + 0x81f5a6cf, 0xde7aa528, 0x8eb7da26, 0xbfad3fa4, + 0x9d3a2ce4, 0x9278500d, 0xcc5f6a9b, 0x467e5462, + 0x138df6c2, 0xb8d890e8, 0xf7392e5e, 0xafc382f5, + 0x805d9fbe, 0x93d0697c, 0x2dd56fa9, 0x1225cfb3, + 0x99acc83b, 0x7d1810a7, 0x639ce86e, 0xbb3bdb7b, + 0x7826cd09, 0x18596ef4, 0xb79aec01, 0x9a4f83a8, + 0x6e95e665, 0xe6ffaa7e, 0xcfbc2108, 0xe815efe6, + 0x9be7bad9, 0x366f4ace, 0x99fead4, 0x7cb029d6, + 0xb2a431af, 0x233f2a31, 0x94a5c630, 0x66a235c0, + 0xbc4e7437, 0xca82fca6, 0xd090e0b0, 0xd8a73315, + 0x9804f14a, 0xdaec41f7, 0x50cd7f0e, 0xf691172f, + 0xd64d768d, 0xb0ef434d, 0x4daacc54, 0x496e4df, + 0xb5d19ee3, 0x886a4c1b, 0x1f2cc1b8, 0x5165467f, + 0xea5e9d04, 0x358c015d, 0x7487fa73, 0x410bfb2e, + 0x1d67b35a, 0xd2db9252, 0x5610e933, 0x47d66d13, + 0x61d79a8c, 0xca1377a, 0x14f8598e, 0x3c13eb89, + 0x27a9ceee, 0xc961b735, 0xe51ce1ed, 0xb1477a3c, + 0xdfd29c59, 0x73f2553f, 0xce141879, 0x37c773bf, + 0xcdf753ea, 0xaafd5f5b, 0x6f3ddf14, 0xdb447886, + 0xf3afca81, 0xc468b93e, 0x3424382c, 0x40a3c25f, + 0xc31d1672, 0x25e2bc0c, 0x493c288b, 0x950dff41, + 0x1a83971, 0xb30c08de, 0xe4b4d89c, 0xc1566490, + 0x84cb7b61, 0xb632d570, 0x5c6c4874, 0x57b8d042, +}; + +static uint32_t U2[256] = { + 0xf45150a7, 0x417e5365, 0x171ac3a4, 0x273a965e, + 0xab3bcb6b, 0x9d1ff145, 0xfaacab58, 0xe34b9303, + 0x302055fa, 0x76adf66d, 0xcc889176, 0x2f5254c, + 0xe54ffcd7, 0x2ac5d7cb, 0x35268044, 0x62b58fa3, + 0xb1de495a, 0xba25671b, 0xea45980e, 0xfe5de1c0, + 0x2fc30275, 0x4c8112f0, 0x468da397, 0xd36bc6f9, + 0x8f03e75f, 0x9215959c, 0x6dbfeb7a, 0x5295da59, + 0xbed42d83, 0x7458d321, 0xe0492969, 0xc98e44c8, + 0xc2756a89, 0x8ef47879, 0x58996b3e, 0xb927dd71, + 0xe1beb64f, 0x88f017ad, 0x20c966ac, 0xce7db43a, + 0xdf63184a, 0x1ae58231, 0x51976033, 0x5362457f, + 0x64b1e077, 0x6bbb84ae, 0x81fe1ca0, 0x8f9942b, + 0x48705868, 0x458f19fd, 0xde94876c, 0x7b52b7f8, + 0x73ab23d3, 0x4b72e202, 0x1fe3578f, 0x55662aab, + 0xebb20728, 0xb52f03c2, 0xc5869a7b, 0x37d3a508, + 0x2830f287, 0xbf23b2a5, 0x302ba6a, 0x16ed5c82, + 0xcf8a2b1c, 0x79a792b4, 0x7f3f0f2, 0x694ea1e2, + 0xda65cdf4, 0x506d5be, 0x34d11f62, 0xa6c48afe, + 0x2e349d53, 0xf3a2a055, 0x8a0532e1, 0xf6a475eb, + 0x830b39ec, 0x6040aaef, 0x715e069f, 0x6ebd5110, + 0x213ef98a, 0xdd963d06, 0x3eddae05, 0xe64d46bd, + 0x5491b58d, 0xc471055d, 0x6046fd4, 0x5060ff15, + 0x981924fb, 0xbdd697e9, 0x4089cc43, 0xd967779e, + 0xe8b0bd42, 0x8907888b, 0x19e7385b, 0xc879dbee, + 0x7ca1470a, 0x427ce90f, 0x84f8c91e, 0x0, + 0x80098386, 0x2b3248ed, 0x111eac70, 0x5a6c4e72, + 0xefdfbff, 0x850f5638, 0xae3d1ed5, 0x2d362739, + 0xf0a64d9, 0x5c6821a6, 0x5b9bd154, 0x36243a2e, + 0xa0cb167, 0x57930fe7, 0xeeb4d296, 0x9b1b9e91, + 0xc0804fc5, 0xdc61a220, 0x775a694b, 0x121c161a, + 0x93e20aba, 0xa0c0e52a, 0x223c43e0, 0x1b121d17, + 0x90e0b0d, 0x8bf2adc7, 0xb62db9a8, 0x1e14c8a9, + 0xf1578519, 0x75af4c07, 0x99eebbdd, 0x7fa3fd60, + 0x1f79f26, 0x725cbcf5, 0x6644c53b, 0xfb5b347e, + 0x438b7629, 0x23cbdcc6, 0xedb668fc, 0xe4b863f1, + 0x31d7cadc, 0x63421085, 0x97134022, 0xc6842011, + 0x4a857d24, 0xbbd2f83d, 0xf9ae1132, 0x29c76da1, + 0x9e1d4b2f, 0xb2dcf330, 0x860dec52, 0xc177d0e3, + 0xb32b6c16, 0x70a999b9, 0x9411fa48, 0xe9472264, + 0xfca8c48c, 0xf0a01a3f, 0x7d56d82c, 0x3322ef90, + 0x4987c74e, 0x38d9c1d1, 0xca8cfea2, 0xd498360b, + 0xf5a6cf81, 0x7aa528de, 0xb7da268e, 0xad3fa4bf, + 0x3a2ce49d, 0x78500d92, 0x5f6a9bcc, 0x7e546246, + 0x8df6c213, 0xd890e8b8, 0x392e5ef7, 0xc382f5af, + 0x5d9fbe80, 0xd0697c93, 0xd56fa92d, 0x25cfb312, + 0xacc83b99, 0x1810a77d, 0x9ce86e63, 0x3bdb7bbb, + 0x26cd0978, 0x596ef418, 0x9aec01b7, 0x4f83a89a, + 0x95e6656e, 0xffaa7ee6, 0xbc2108cf, 0x15efe6e8, + 0xe7bad99b, 0x6f4ace36, 0x9fead409, 0xb029d67c, + 0xa431afb2, 0x3f2a3123, 0xa5c63094, 0xa235c066, + 0x4e7437bc, 0x82fca6ca, 0x90e0b0d0, 0xa73315d8, + 0x4f14a98, 0xec41f7da, 0xcd7f0e50, 0x91172ff6, + 0x4d768dd6, 0xef434db0, 0xaacc544d, 0x96e4df04, + 0xd19ee3b5, 0x6a4c1b88, 0x2cc1b81f, 0x65467f51, + 0x5e9d04ea, 0x8c015d35, 0x87fa7374, 0xbfb2e41, + 0x67b35a1d, 0xdb9252d2, 0x10e93356, 0xd66d1347, + 0xd79a8c61, 0xa1377a0c, 0xf8598e14, 0x13eb893c, + 0xa9ceee27, 0x61b735c9, 0x1ce1ede5, 0x477a3cb1, + 0xd29c59df, 0xf2553f73, 0x141879ce, 0xc773bf37, + 0xf753eacd, 0xfd5f5baa, 0x3ddf146f, 0x447886db, + 0xafca81f3, 0x68b93ec4, 0x24382c34, 0xa3c25f40, + 0x1d1672c3, 0xe2bc0c25, 0x3c288b49, 0xdff4195, + 0xa8397101, 0xc08deb3, 0xb4d89ce4, 0x566490c1, + 0xcb7b6184, 0x32d570b6, 0x6c48745c, 0xb8d04257, +}; + +static uint32_t U3[256] = { + 0x5150a7f4, 0x7e536541, 0x1ac3a417, 0x3a965e27, + 0x3bcb6bab, 0x1ff1459d, 0xacab58fa, 0x4b9303e3, + 0x2055fa30, 0xadf66d76, 0x889176cc, 0xf5254c02, + 0x4ffcd7e5, 0xc5d7cb2a, 0x26804435, 0xb58fa362, + 0xde495ab1, 0x25671bba, 0x45980eea, 0x5de1c0fe, + 0xc302752f, 0x8112f04c, 0x8da39746, 0x6bc6f9d3, + 0x3e75f8f, 0x15959c92, 0xbfeb7a6d, 0x95da5952, + 0xd42d83be, 0x58d32174, 0x492969e0, 0x8e44c8c9, + 0x756a89c2, 0xf478798e, 0x996b3e58, 0x27dd71b9, + 0xbeb64fe1, 0xf017ad88, 0xc966ac20, 0x7db43ace, + 0x63184adf, 0xe582311a, 0x97603351, 0x62457f53, + 0xb1e07764, 0xbb84ae6b, 0xfe1ca081, 0xf9942b08, + 0x70586848, 0x8f19fd45, 0x94876cde, 0x52b7f87b, + 0xab23d373, 0x72e2024b, 0xe3578f1f, 0x662aab55, + 0xb20728eb, 0x2f03c2b5, 0x869a7bc5, 0xd3a50837, + 0x30f28728, 0x23b2a5bf, 0x2ba6a03, 0xed5c8216, + 0x8a2b1ccf, 0xa792b479, 0xf3f0f207, 0x4ea1e269, + 0x65cdf4da, 0x6d5be05, 0xd11f6234, 0xc48afea6, + 0x349d532e, 0xa2a055f3, 0x532e18a, 0xa475ebf6, + 0xb39ec83, 0x40aaef60, 0x5e069f71, 0xbd51106e, + 0x3ef98a21, 0x963d06dd, 0xddae053e, 0x4d46bde6, + 0x91b58d54, 0x71055dc4, 0x46fd406, 0x60ff1550, + 0x1924fb98, 0xd697e9bd, 0x89cc4340, 0x67779ed9, + 0xb0bd42e8, 0x7888b89, 0xe7385b19, 0x79dbeec8, + 0xa1470a7c, 0x7ce90f42, 0xf8c91e84, 0x0, + 0x9838680, 0x3248ed2b, 0x1eac7011, 0x6c4e725a, + 0xfdfbff0e, 0xf563885, 0x3d1ed5ae, 0x3627392d, + 0xa64d90f, 0x6821a65c, 0x9bd1545b, 0x243a2e36, + 0xcb1670a, 0x930fe757, 0xb4d296ee, 0x1b9e919b, + 0x804fc5c0, 0x61a220dc, 0x5a694b77, 0x1c161a12, + 0xe20aba93, 0xc0e52aa0, 0x3c43e022, 0x121d171b, + 0xe0b0d09, 0xf2adc78b, 0x2db9a8b6, 0x14c8a91e, + 0x578519f1, 0xaf4c0775, 0xeebbdd99, 0xa3fd607f, + 0xf79f2601, 0x5cbcf572, 0x44c53b66, 0x5b347efb, + 0x8b762943, 0xcbdcc623, 0xb668fced, 0xb863f1e4, + 0xd7cadc31, 0x42108563, 0x13402297, 0x842011c6, + 0x857d244a, 0xd2f83dbb, 0xae1132f9, 0xc76da129, + 0x1d4b2f9e, 0xdcf330b2, 0xdec5286, 0x77d0e3c1, + 0x2b6c16b3, 0xa999b970, 0x11fa4894, 0x472264e9, + 0xa8c48cfc, 0xa01a3ff0, 0x56d82c7d, 0x22ef9033, + 0x87c74e49, 0xd9c1d138, 0x8cfea2ca, 0x98360bd4, + 0xa6cf81f5, 0xa528de7a, 0xda268eb7, 0x3fa4bfad, + 0x2ce49d3a, 0x500d9278, 0x6a9bcc5f, 0x5462467e, + 0xf6c2138d, 0x90e8b8d8, 0x2e5ef739, 0x82f5afc3, + 0x9fbe805d, 0x697c93d0, 0x6fa92dd5, 0xcfb31225, + 0xc83b99ac, 0x10a77d18, 0xe86e639c, 0xdb7bbb3b, + 0xcd097826, 0x6ef41859, 0xec01b79a, 0x83a89a4f, + 0xe6656e95, 0xaa7ee6ff, 0x2108cfbc, 0xefe6e815, + 0xbad99be7, 0x4ace366f, 0xead4099f, 0x29d67cb0, + 0x31afb2a4, 0x2a31233f, 0xc63094a5, 0x35c066a2, + 0x7437bc4e, 0xfca6ca82, 0xe0b0d090, 0x3315d8a7, + 0xf14a9804, 0x41f7daec, 0x7f0e50cd, 0x172ff691, + 0x768dd64d, 0x434db0ef, 0xcc544daa, 0xe4df0496, + 0x9ee3b5d1, 0x4c1b886a, 0xc1b81f2c, 0x467f5165, + 0x9d04ea5e, 0x15d358c, 0xfa737487, 0xfb2e410b, + 0xb35a1d67, 0x9252d2db, 0xe9335610, 0x6d1347d6, + 0x9a8c61d7, 0x377a0ca1, 0x598e14f8, 0xeb893c13, + 0xceee27a9, 0xb735c961, 0xe1ede51c, 0x7a3cb147, + 0x9c59dfd2, 0x553f73f2, 0x1879ce14, 0x73bf37c7, + 0x53eacdf7, 0x5f5baafd, 0xdf146f3d, 0x7886db44, + 0xca81f3af, 0xb93ec468, 0x382c3424, 0xc25f40a3, + 0x1672c31d, 0xbc0c25e2, 0x288b493c, 0xff41950d, + 0x397101a8, 0x8deb30c, 0xd89ce4b4, 0x6490c156, + 0x7b6184cb, 0xd570b632, 0x48745c6c, 0xd04257b8, +}; + +#else /* assume big endian */ + +static uint32_t T0[256] = { + 0xc66363a5, 0xf87c7c84, 0xee777799, 0xf67b7b8d, + 0xfff2f20d, 0xd66b6bbd, 0xde6f6fb1, 0x91c5c554, + 0x60303050, 0x2010103, 0xce6767a9, 0x562b2b7d, + 0xe7fefe19, 0xb5d7d762, 0x4dababe6, 0xec76769a, + 0x8fcaca45, 0x1f82829d, 0x89c9c940, 0xfa7d7d87, + 0xeffafa15, 0xb25959eb, 0x8e4747c9, 0xfbf0f00b, + 0x41adadec, 0xb3d4d467, 0x5fa2a2fd, 0x45afafea, + 0x239c9cbf, 0x53a4a4f7, 0xe4727296, 0x9bc0c05b, + 0x75b7b7c2, 0xe1fdfd1c, 0x3d9393ae, 0x4c26266a, + 0x6c36365a, 0x7e3f3f41, 0xf5f7f702, 0x83cccc4f, + 0x6834345c, 0x51a5a5f4, 0xd1e5e534, 0xf9f1f108, + 0xe2717193, 0xabd8d873, 0x62313153, 0x2a15153f, + 0x804040c, 0x95c7c752, 0x46232365, 0x9dc3c35e, + 0x30181828, 0x379696a1, 0xa05050f, 0x2f9a9ab5, + 0xe070709, 0x24121236, 0x1b80809b, 0xdfe2e23d, + 0xcdebeb26, 0x4e272769, 0x7fb2b2cd, 0xea75759f, + 0x1209091b, 0x1d83839e, 0x582c2c74, 0x341a1a2e, + 0x361b1b2d, 0xdc6e6eb2, 0xb45a5aee, 0x5ba0a0fb, + 0xa45252f6, 0x763b3b4d, 0xb7d6d661, 0x7db3b3ce, + 0x5229297b, 0xdde3e33e, 0x5e2f2f71, 0x13848497, + 0xa65353f5, 0xb9d1d168, 0x0, 0xc1eded2c, + 0x40202060, 0xe3fcfc1f, 0x79b1b1c8, 0xb65b5bed, + 0xd46a6abe, 0x8dcbcb46, 0x67bebed9, 0x7239394b, + 0x944a4ade, 0x984c4cd4, 0xb05858e8, 0x85cfcf4a, + 0xbbd0d06b, 0xc5efef2a, 0x4faaaae5, 0xedfbfb16, + 0x864343c5, 0x9a4d4dd7, 0x66333355, 0x11858594, + 0x8a4545cf, 0xe9f9f910, 0x4020206, 0xfe7f7f81, + 0xa05050f0, 0x783c3c44, 0x259f9fba, 0x4ba8a8e3, + 0xa25151f3, 0x5da3a3fe, 0x804040c0, 0x58f8f8a, + 0x3f9292ad, 0x219d9dbc, 0x70383848, 0xf1f5f504, + 0x63bcbcdf, 0x77b6b6c1, 0xafdada75, 0x42212163, + 0x20101030, 0xe5ffff1a, 0xfdf3f30e, 0xbfd2d26d, + 0x81cdcd4c, 0x180c0c14, 0x26131335, 0xc3ecec2f, + 0xbe5f5fe1, 0x359797a2, 0x884444cc, 0x2e171739, + 0x93c4c457, 0x55a7a7f2, 0xfc7e7e82, 0x7a3d3d47, + 0xc86464ac, 0xba5d5de7, 0x3219192b, 0xe6737395, + 0xc06060a0, 0x19818198, 0x9e4f4fd1, 0xa3dcdc7f, + 0x44222266, 0x542a2a7e, 0x3b9090ab, 0xb888883, + 0x8c4646ca, 0xc7eeee29, 0x6bb8b8d3, 0x2814143c, + 0xa7dede79, 0xbc5e5ee2, 0x160b0b1d, 0xaddbdb76, + 0xdbe0e03b, 0x64323256, 0x743a3a4e, 0x140a0a1e, + 0x924949db, 0xc06060a, 0x4824246c, 0xb85c5ce4, + 0x9fc2c25d, 0xbdd3d36e, 0x43acacef, 0xc46262a6, + 0x399191a8, 0x319595a4, 0xd3e4e437, 0xf279798b, + 0xd5e7e732, 0x8bc8c843, 0x6e373759, 0xda6d6db7, + 0x18d8d8c, 0xb1d5d564, 0x9c4e4ed2, 0x49a9a9e0, + 0xd86c6cb4, 0xac5656fa, 0xf3f4f407, 0xcfeaea25, + 0xca6565af, 0xf47a7a8e, 0x47aeaee9, 0x10080818, + 0x6fbabad5, 0xf0787888, 0x4a25256f, 0x5c2e2e72, + 0x381c1c24, 0x57a6a6f1, 0x73b4b4c7, 0x97c6c651, + 0xcbe8e823, 0xa1dddd7c, 0xe874749c, 0x3e1f1f21, + 0x964b4bdd, 0x61bdbddc, 0xd8b8b86, 0xf8a8a85, + 0xe0707090, 0x7c3e3e42, 0x71b5b5c4, 0xcc6666aa, + 0x904848d8, 0x6030305, 0xf7f6f601, 0x1c0e0e12, + 0xc26161a3, 0x6a35355f, 0xae5757f9, 0x69b9b9d0, + 0x17868691, 0x99c1c158, 0x3a1d1d27, 0x279e9eb9, + 0xd9e1e138, 0xebf8f813, 0x2b9898b3, 0x22111133, + 0xd26969bb, 0xa9d9d970, 0x78e8e89, 0x339494a7, + 0x2d9b9bb6, 0x3c1e1e22, 0x15878792, 0xc9e9e920, + 0x87cece49, 0xaa5555ff, 0x50282878, 0xa5dfdf7a, + 0x38c8c8f, 0x59a1a1f8, 0x9898980, 0x1a0d0d17, + 0x65bfbfda, 0xd7e6e631, 0x844242c6, 0xd06868b8, + 0x824141c3, 0x299999b0, 0x5a2d2d77, 0x1e0f0f11, + 0x7bb0b0cb, 0xa85454fc, 0x6dbbbbd6, 0x2c16163a, +}; + +static uint32_t T1[256] = { + 0xa5c66363, 0x84f87c7c, 0x99ee7777, 0x8df67b7b, + 0xdfff2f2, 0xbdd66b6b, 0xb1de6f6f, 0x5491c5c5, + 0x50603030, 0x3020101, 0xa9ce6767, 0x7d562b2b, + 0x19e7fefe, 0x62b5d7d7, 0xe64dabab, 0x9aec7676, + 0x458fcaca, 0x9d1f8282, 0x4089c9c9, 0x87fa7d7d, + 0x15effafa, 0xebb25959, 0xc98e4747, 0xbfbf0f0, + 0xec41adad, 0x67b3d4d4, 0xfd5fa2a2, 0xea45afaf, + 0xbf239c9c, 0xf753a4a4, 0x96e47272, 0x5b9bc0c0, + 0xc275b7b7, 0x1ce1fdfd, 0xae3d9393, 0x6a4c2626, + 0x5a6c3636, 0x417e3f3f, 0x2f5f7f7, 0x4f83cccc, + 0x5c683434, 0xf451a5a5, 0x34d1e5e5, 0x8f9f1f1, + 0x93e27171, 0x73abd8d8, 0x53623131, 0x3f2a1515, + 0xc080404, 0x5295c7c7, 0x65462323, 0x5e9dc3c3, + 0x28301818, 0xa1379696, 0xf0a0505, 0xb52f9a9a, + 0x90e0707, 0x36241212, 0x9b1b8080, 0x3ddfe2e2, + 0x26cdebeb, 0x694e2727, 0xcd7fb2b2, 0x9fea7575, + 0x1b120909, 0x9e1d8383, 0x74582c2c, 0x2e341a1a, + 0x2d361b1b, 0xb2dc6e6e, 0xeeb45a5a, 0xfb5ba0a0, + 0xf6a45252, 0x4d763b3b, 0x61b7d6d6, 0xce7db3b3, + 0x7b522929, 0x3edde3e3, 0x715e2f2f, 0x97138484, + 0xf5a65353, 0x68b9d1d1, 0x0, 0x2cc1eded, + 0x60402020, 0x1fe3fcfc, 0xc879b1b1, 0xedb65b5b, + 0xbed46a6a, 0x468dcbcb, 0xd967bebe, 0x4b723939, + 0xde944a4a, 0xd4984c4c, 0xe8b05858, 0x4a85cfcf, + 0x6bbbd0d0, 0x2ac5efef, 0xe54faaaa, 0x16edfbfb, + 0xc5864343, 0xd79a4d4d, 0x55663333, 0x94118585, + 0xcf8a4545, 0x10e9f9f9, 0x6040202, 0x81fe7f7f, + 0xf0a05050, 0x44783c3c, 0xba259f9f, 0xe34ba8a8, + 0xf3a25151, 0xfe5da3a3, 0xc0804040, 0x8a058f8f, + 0xad3f9292, 0xbc219d9d, 0x48703838, 0x4f1f5f5, + 0xdf63bcbc, 0xc177b6b6, 0x75afdada, 0x63422121, + 0x30201010, 0x1ae5ffff, 0xefdf3f3, 0x6dbfd2d2, + 0x4c81cdcd, 0x14180c0c, 0x35261313, 0x2fc3ecec, + 0xe1be5f5f, 0xa2359797, 0xcc884444, 0x392e1717, + 0x5793c4c4, 0xf255a7a7, 0x82fc7e7e, 0x477a3d3d, + 0xacc86464, 0xe7ba5d5d, 0x2b321919, 0x95e67373, + 0xa0c06060, 0x98198181, 0xd19e4f4f, 0x7fa3dcdc, + 0x66442222, 0x7e542a2a, 0xab3b9090, 0x830b8888, + 0xca8c4646, 0x29c7eeee, 0xd36bb8b8, 0x3c281414, + 0x79a7dede, 0xe2bc5e5e, 0x1d160b0b, 0x76addbdb, + 0x3bdbe0e0, 0x56643232, 0x4e743a3a, 0x1e140a0a, + 0xdb924949, 0xa0c0606, 0x6c482424, 0xe4b85c5c, + 0x5d9fc2c2, 0x6ebdd3d3, 0xef43acac, 0xa6c46262, + 0xa8399191, 0xa4319595, 0x37d3e4e4, 0x8bf27979, + 0x32d5e7e7, 0x438bc8c8, 0x596e3737, 0xb7da6d6d, + 0x8c018d8d, 0x64b1d5d5, 0xd29c4e4e, 0xe049a9a9, + 0xb4d86c6c, 0xfaac5656, 0x7f3f4f4, 0x25cfeaea, + 0xafca6565, 0x8ef47a7a, 0xe947aeae, 0x18100808, + 0xd56fbaba, 0x88f07878, 0x6f4a2525, 0x725c2e2e, + 0x24381c1c, 0xf157a6a6, 0xc773b4b4, 0x5197c6c6, + 0x23cbe8e8, 0x7ca1dddd, 0x9ce87474, 0x213e1f1f, + 0xdd964b4b, 0xdc61bdbd, 0x860d8b8b, 0x850f8a8a, + 0x90e07070, 0x427c3e3e, 0xc471b5b5, 0xaacc6666, + 0xd8904848, 0x5060303, 0x1f7f6f6, 0x121c0e0e, + 0xa3c26161, 0x5f6a3535, 0xf9ae5757, 0xd069b9b9, + 0x91178686, 0x5899c1c1, 0x273a1d1d, 0xb9279e9e, + 0x38d9e1e1, 0x13ebf8f8, 0xb32b9898, 0x33221111, + 0xbbd26969, 0x70a9d9d9, 0x89078e8e, 0xa7339494, + 0xb62d9b9b, 0x223c1e1e, 0x92158787, 0x20c9e9e9, + 0x4987cece, 0xffaa5555, 0x78502828, 0x7aa5dfdf, + 0x8f038c8c, 0xf859a1a1, 0x80098989, 0x171a0d0d, + 0xda65bfbf, 0x31d7e6e6, 0xc6844242, 0xb8d06868, + 0xc3824141, 0xb0299999, 0x775a2d2d, 0x111e0f0f, + 0xcb7bb0b0, 0xfca85454, 0xd66dbbbb, 0x3a2c1616, +}; + +static uint32_t T2[256] = { + 0x63a5c663, 0x7c84f87c, 0x7799ee77, 0x7b8df67b, + 0xf20dfff2, 0x6bbdd66b, 0x6fb1de6f, 0xc55491c5, + 0x30506030, 0x1030201, 0x67a9ce67, 0x2b7d562b, + 0xfe19e7fe, 0xd762b5d7, 0xabe64dab, 0x769aec76, + 0xca458fca, 0x829d1f82, 0xc94089c9, 0x7d87fa7d, + 0xfa15effa, 0x59ebb259, 0x47c98e47, 0xf00bfbf0, + 0xadec41ad, 0xd467b3d4, 0xa2fd5fa2, 0xafea45af, + 0x9cbf239c, 0xa4f753a4, 0x7296e472, 0xc05b9bc0, + 0xb7c275b7, 0xfd1ce1fd, 0x93ae3d93, 0x266a4c26, + 0x365a6c36, 0x3f417e3f, 0xf702f5f7, 0xcc4f83cc, + 0x345c6834, 0xa5f451a5, 0xe534d1e5, 0xf108f9f1, + 0x7193e271, 0xd873abd8, 0x31536231, 0x153f2a15, + 0x40c0804, 0xc75295c7, 0x23654623, 0xc35e9dc3, + 0x18283018, 0x96a13796, 0x50f0a05, 0x9ab52f9a, + 0x7090e07, 0x12362412, 0x809b1b80, 0xe23ddfe2, + 0xeb26cdeb, 0x27694e27, 0xb2cd7fb2, 0x759fea75, + 0x91b1209, 0x839e1d83, 0x2c74582c, 0x1a2e341a, + 0x1b2d361b, 0x6eb2dc6e, 0x5aeeb45a, 0xa0fb5ba0, + 0x52f6a452, 0x3b4d763b, 0xd661b7d6, 0xb3ce7db3, + 0x297b5229, 0xe33edde3, 0x2f715e2f, 0x84971384, + 0x53f5a653, 0xd168b9d1, 0x0, 0xed2cc1ed, + 0x20604020, 0xfc1fe3fc, 0xb1c879b1, 0x5bedb65b, + 0x6abed46a, 0xcb468dcb, 0xbed967be, 0x394b7239, + 0x4ade944a, 0x4cd4984c, 0x58e8b058, 0xcf4a85cf, + 0xd06bbbd0, 0xef2ac5ef, 0xaae54faa, 0xfb16edfb, + 0x43c58643, 0x4dd79a4d, 0x33556633, 0x85941185, + 0x45cf8a45, 0xf910e9f9, 0x2060402, 0x7f81fe7f, + 0x50f0a050, 0x3c44783c, 0x9fba259f, 0xa8e34ba8, + 0x51f3a251, 0xa3fe5da3, 0x40c08040, 0x8f8a058f, + 0x92ad3f92, 0x9dbc219d, 0x38487038, 0xf504f1f5, + 0xbcdf63bc, 0xb6c177b6, 0xda75afda, 0x21634221, + 0x10302010, 0xff1ae5ff, 0xf30efdf3, 0xd26dbfd2, + 0xcd4c81cd, 0xc14180c, 0x13352613, 0xec2fc3ec, + 0x5fe1be5f, 0x97a23597, 0x44cc8844, 0x17392e17, + 0xc45793c4, 0xa7f255a7, 0x7e82fc7e, 0x3d477a3d, + 0x64acc864, 0x5de7ba5d, 0x192b3219, 0x7395e673, + 0x60a0c060, 0x81981981, 0x4fd19e4f, 0xdc7fa3dc, + 0x22664422, 0x2a7e542a, 0x90ab3b90, 0x88830b88, + 0x46ca8c46, 0xee29c7ee, 0xb8d36bb8, 0x143c2814, + 0xde79a7de, 0x5ee2bc5e, 0xb1d160b, 0xdb76addb, + 0xe03bdbe0, 0x32566432, 0x3a4e743a, 0xa1e140a, + 0x49db9249, 0x60a0c06, 0x246c4824, 0x5ce4b85c, + 0xc25d9fc2, 0xd36ebdd3, 0xacef43ac, 0x62a6c462, + 0x91a83991, 0x95a43195, 0xe437d3e4, 0x798bf279, + 0xe732d5e7, 0xc8438bc8, 0x37596e37, 0x6db7da6d, + 0x8d8c018d, 0xd564b1d5, 0x4ed29c4e, 0xa9e049a9, + 0x6cb4d86c, 0x56faac56, 0xf407f3f4, 0xea25cfea, + 0x65afca65, 0x7a8ef47a, 0xaee947ae, 0x8181008, + 0xbad56fba, 0x7888f078, 0x256f4a25, 0x2e725c2e, + 0x1c24381c, 0xa6f157a6, 0xb4c773b4, 0xc65197c6, + 0xe823cbe8, 0xdd7ca1dd, 0x749ce874, 0x1f213e1f, + 0x4bdd964b, 0xbddc61bd, 0x8b860d8b, 0x8a850f8a, + 0x7090e070, 0x3e427c3e, 0xb5c471b5, 0x66aacc66, + 0x48d89048, 0x3050603, 0xf601f7f6, 0xe121c0e, + 0x61a3c261, 0x355f6a35, 0x57f9ae57, 0xb9d069b9, + 0x86911786, 0xc15899c1, 0x1d273a1d, 0x9eb9279e, + 0xe138d9e1, 0xf813ebf8, 0x98b32b98, 0x11332211, + 0x69bbd269, 0xd970a9d9, 0x8e89078e, 0x94a73394, + 0x9bb62d9b, 0x1e223c1e, 0x87921587, 0xe920c9e9, + 0xce4987ce, 0x55ffaa55, 0x28785028, 0xdf7aa5df, + 0x8c8f038c, 0xa1f859a1, 0x89800989, 0xd171a0d, + 0xbfda65bf, 0xe631d7e6, 0x42c68442, 0x68b8d068, + 0x41c38241, 0x99b02999, 0x2d775a2d, 0xf111e0f, + 0xb0cb7bb0, 0x54fca854, 0xbbd66dbb, 0x163a2c16, +}; + +static uint32_t T3[256] = { + 0x6363a5c6, 0x7c7c84f8, 0x777799ee, 0x7b7b8df6, + 0xf2f20dff, 0x6b6bbdd6, 0x6f6fb1de, 0xc5c55491, + 0x30305060, 0x1010302, 0x6767a9ce, 0x2b2b7d56, + 0xfefe19e7, 0xd7d762b5, 0xababe64d, 0x76769aec, + 0xcaca458f, 0x82829d1f, 0xc9c94089, 0x7d7d87fa, + 0xfafa15ef, 0x5959ebb2, 0x4747c98e, 0xf0f00bfb, + 0xadadec41, 0xd4d467b3, 0xa2a2fd5f, 0xafafea45, + 0x9c9cbf23, 0xa4a4f753, 0x727296e4, 0xc0c05b9b, + 0xb7b7c275, 0xfdfd1ce1, 0x9393ae3d, 0x26266a4c, + 0x36365a6c, 0x3f3f417e, 0xf7f702f5, 0xcccc4f83, + 0x34345c68, 0xa5a5f451, 0xe5e534d1, 0xf1f108f9, + 0x717193e2, 0xd8d873ab, 0x31315362, 0x15153f2a, + 0x4040c08, 0xc7c75295, 0x23236546, 0xc3c35e9d, + 0x18182830, 0x9696a137, 0x5050f0a, 0x9a9ab52f, + 0x707090e, 0x12123624, 0x80809b1b, 0xe2e23ddf, + 0xebeb26cd, 0x2727694e, 0xb2b2cd7f, 0x75759fea, + 0x9091b12, 0x83839e1d, 0x2c2c7458, 0x1a1a2e34, + 0x1b1b2d36, 0x6e6eb2dc, 0x5a5aeeb4, 0xa0a0fb5b, + 0x5252f6a4, 0x3b3b4d76, 0xd6d661b7, 0xb3b3ce7d, + 0x29297b52, 0xe3e33edd, 0x2f2f715e, 0x84849713, + 0x5353f5a6, 0xd1d168b9, 0x0, 0xeded2cc1, + 0x20206040, 0xfcfc1fe3, 0xb1b1c879, 0x5b5bedb6, + 0x6a6abed4, 0xcbcb468d, 0xbebed967, 0x39394b72, + 0x4a4ade94, 0x4c4cd498, 0x5858e8b0, 0xcfcf4a85, + 0xd0d06bbb, 0xefef2ac5, 0xaaaae54f, 0xfbfb16ed, + 0x4343c586, 0x4d4dd79a, 0x33335566, 0x85859411, + 0x4545cf8a, 0xf9f910e9, 0x2020604, 0x7f7f81fe, + 0x5050f0a0, 0x3c3c4478, 0x9f9fba25, 0xa8a8e34b, + 0x5151f3a2, 0xa3a3fe5d, 0x4040c080, 0x8f8f8a05, + 0x9292ad3f, 0x9d9dbc21, 0x38384870, 0xf5f504f1, + 0xbcbcdf63, 0xb6b6c177, 0xdada75af, 0x21216342, + 0x10103020, 0xffff1ae5, 0xf3f30efd, 0xd2d26dbf, + 0xcdcd4c81, 0xc0c1418, 0x13133526, 0xecec2fc3, + 0x5f5fe1be, 0x9797a235, 0x4444cc88, 0x1717392e, + 0xc4c45793, 0xa7a7f255, 0x7e7e82fc, 0x3d3d477a, + 0x6464acc8, 0x5d5de7ba, 0x19192b32, 0x737395e6, + 0x6060a0c0, 0x81819819, 0x4f4fd19e, 0xdcdc7fa3, + 0x22226644, 0x2a2a7e54, 0x9090ab3b, 0x8888830b, + 0x4646ca8c, 0xeeee29c7, 0xb8b8d36b, 0x14143c28, + 0xdede79a7, 0x5e5ee2bc, 0xb0b1d16, 0xdbdb76ad, + 0xe0e03bdb, 0x32325664, 0x3a3a4e74, 0xa0a1e14, + 0x4949db92, 0x6060a0c, 0x24246c48, 0x5c5ce4b8, + 0xc2c25d9f, 0xd3d36ebd, 0xacacef43, 0x6262a6c4, + 0x9191a839, 0x9595a431, 0xe4e437d3, 0x79798bf2, + 0xe7e732d5, 0xc8c8438b, 0x3737596e, 0x6d6db7da, + 0x8d8d8c01, 0xd5d564b1, 0x4e4ed29c, 0xa9a9e049, + 0x6c6cb4d8, 0x5656faac, 0xf4f407f3, 0xeaea25cf, + 0x6565afca, 0x7a7a8ef4, 0xaeaee947, 0x8081810, + 0xbabad56f, 0x787888f0, 0x25256f4a, 0x2e2e725c, + 0x1c1c2438, 0xa6a6f157, 0xb4b4c773, 0xc6c65197, + 0xe8e823cb, 0xdddd7ca1, 0x74749ce8, 0x1f1f213e, + 0x4b4bdd96, 0xbdbddc61, 0x8b8b860d, 0x8a8a850f, + 0x707090e0, 0x3e3e427c, 0xb5b5c471, 0x6666aacc, + 0x4848d890, 0x3030506, 0xf6f601f7, 0xe0e121c, + 0x6161a3c2, 0x35355f6a, 0x5757f9ae, 0xb9b9d069, + 0x86869117, 0xc1c15899, 0x1d1d273a, 0x9e9eb927, + 0xe1e138d9, 0xf8f813eb, 0x9898b32b, 0x11113322, + 0x6969bbd2, 0xd9d970a9, 0x8e8e8907, 0x9494a733, + 0x9b9bb62d, 0x1e1e223c, 0x87879215, 0xe9e920c9, + 0xcece4987, 0x5555ffaa, 0x28287850, 0xdfdf7aa5, + 0x8c8c8f03, 0xa1a1f859, 0x89898009, 0xd0d171a, + 0xbfbfda65, 0xe6e631d7, 0x4242c684, 0x6868b8d0, + 0x4141c382, 0x9999b029, 0x2d2d775a, 0xf0f111e, + 0xb0b0cb7b, 0x5454fca8, 0xbbbbd66d, 0x16163a2c, +}; + +static uint32_t U0[256] = { + 0x51f4a750, 0x7e416553, 0x1a17a4c3, 0x3a275e96, + 0x3bab6bcb, 0x1f9d45f1, 0xacfa58ab, 0x4be30393, + 0x2030fa55, 0xad766df6, 0x88cc7691, 0xf5024c25, + 0x4fe5d7fc, 0xc52acbd7, 0x26354480, 0xb562a38f, + 0xdeb15a49, 0x25ba1b67, 0x45ea0e98, 0x5dfec0e1, + 0xc32f7502, 0x814cf012, 0x8d4697a3, 0x6bd3f9c6, + 0x38f5fe7, 0x15929c95, 0xbf6d7aeb, 0x955259da, + 0xd4be832d, 0x587421d3, 0x49e06929, 0x8ec9c844, + 0x75c2896a, 0xf48e7978, 0x99583e6b, 0x27b971dd, + 0xbee14fb6, 0xf088ad17, 0xc920ac66, 0x7dce3ab4, + 0x63df4a18, 0xe51a3182, 0x97513360, 0x62537f45, + 0xb16477e0, 0xbb6bae84, 0xfe81a01c, 0xf9082b94, + 0x70486858, 0x8f45fd19, 0x94de6c87, 0x527bf8b7, + 0xab73d323, 0x724b02e2, 0xe31f8f57, 0x6655ab2a, + 0xb2eb2807, 0x2fb5c203, 0x86c57b9a, 0xd33708a5, + 0x302887f2, 0x23bfa5b2, 0x2036aba, 0xed16825c, + 0x8acf1c2b, 0xa779b492, 0xf307f2f0, 0x4e69e2a1, + 0x65daf4cd, 0x605bed5, 0xd134621f, 0xc4a6fe8a, + 0x342e539d, 0xa2f355a0, 0x58ae132, 0xa4f6eb75, + 0xb83ec39, 0x4060efaa, 0x5e719f06, 0xbd6e1051, + 0x3e218af9, 0x96dd063d, 0xdd3e05ae, 0x4de6bd46, + 0x91548db5, 0x71c45d05, 0x406d46f, 0x605015ff, + 0x1998fb24, 0xd6bde997, 0x894043cc, 0x67d99e77, + 0xb0e842bd, 0x7898b88, 0xe7195b38, 0x79c8eedb, + 0xa17c0a47, 0x7c420fe9, 0xf8841ec9, 0x0, + 0x9808683, 0x322bed48, 0x1e1170ac, 0x6c5a724e, + 0xfd0efffb, 0xf853856, 0x3daed51e, 0x362d3927, + 0xa0fd964, 0x685ca621, 0x9b5b54d1, 0x24362e3a, + 0xc0a67b1, 0x9357e70f, 0xb4ee96d2, 0x1b9b919e, + 0x80c0c54f, 0x61dc20a2, 0x5a774b69, 0x1c121a16, + 0xe293ba0a, 0xc0a02ae5, 0x3c22e043, 0x121b171d, + 0xe090d0b, 0xf28bc7ad, 0x2db6a8b9, 0x141ea9c8, + 0x57f11985, 0xaf75074c, 0xee99ddbb, 0xa37f60fd, + 0xf701269f, 0x5c72f5bc, 0x44663bc5, 0x5bfb7e34, + 0x8b432976, 0xcb23c6dc, 0xb6edfc68, 0xb8e4f163, + 0xd731dcca, 0x42638510, 0x13972240, 0x84c61120, + 0x854a247d, 0xd2bb3df8, 0xaef93211, 0xc729a16d, + 0x1d9e2f4b, 0xdcb230f3, 0xd8652ec, 0x77c1e3d0, + 0x2bb3166c, 0xa970b999, 0x119448fa, 0x47e96422, + 0xa8fc8cc4, 0xa0f03f1a, 0x567d2cd8, 0x223390ef, + 0x87494ec7, 0xd938d1c1, 0x8ccaa2fe, 0x98d40b36, + 0xa6f581cf, 0xa57ade28, 0xdab78e26, 0x3fadbfa4, + 0x2c3a9de4, 0x5078920d, 0x6a5fcc9b, 0x547e4662, + 0xf68d13c2, 0x90d8b8e8, 0x2e39f75e, 0x82c3aff5, + 0x9f5d80be, 0x69d0937c, 0x6fd52da9, 0xcf2512b3, + 0xc8ac993b, 0x10187da7, 0xe89c636e, 0xdb3bbb7b, + 0xcd267809, 0x6e5918f4, 0xec9ab701, 0x834f9aa8, + 0xe6956e65, 0xaaffe67e, 0x21bccf08, 0xef15e8e6, + 0xbae79bd9, 0x4a6f36ce, 0xea9f09d4, 0x29b07cd6, + 0x31a4b2af, 0x2a3f2331, 0xc6a59430, 0x35a266c0, + 0x744ebc37, 0xfc82caa6, 0xe090d0b0, 0x33a7d815, + 0xf104984a, 0x41ecdaf7, 0x7fcd500e, 0x1791f62f, + 0x764dd68d, 0x43efb04d, 0xccaa4d54, 0xe49604df, + 0x9ed1b5e3, 0x4c6a881b, 0xc12c1fb8, 0x4665517f, + 0x9d5eea04, 0x18c355d, 0xfa877473, 0xfb0b412e, + 0xb3671d5a, 0x92dbd252, 0xe9105633, 0x6dd64713, + 0x9ad7618c, 0x37a10c7a, 0x59f8148e, 0xeb133c89, + 0xcea927ee, 0xb761c935, 0xe11ce5ed, 0x7a47b13c, + 0x9cd2df59, 0x55f2733f, 0x1814ce79, 0x73c737bf, + 0x53f7cdea, 0x5ffdaa5b, 0xdf3d6f14, 0x7844db86, + 0xcaaff381, 0xb968c43e, 0x3824342c, 0xc2a3405f, + 0x161dc372, 0xbce2250c, 0x283c498b, 0xff0d9541, + 0x39a80171, 0x80cb3de, 0xd8b4e49c, 0x6456c190, + 0x7bcb8461, 0xd532b670, 0x486c5c74, 0xd0b85742 +}; + +static uint32_t U1[256] = { + 0x5051f4a7, 0x537e4165, 0xc31a17a4, 0x963a275e, + 0xcb3bab6b, 0xf11f9d45, 0xabacfa58, 0x934be303, + 0x552030fa, 0xf6ad766d, 0x9188cc76, 0x25f5024c, + 0xfc4fe5d7, 0xd7c52acb, 0x80263544, 0x8fb562a3, + 0x49deb15a, 0x6725ba1b, 0x9845ea0e, 0xe15dfec0, + 0x2c32f75, 0x12814cf0, 0xa38d4697, 0xc66bd3f9, + 0xe7038f5f, 0x9515929c, 0xebbf6d7a, 0xda955259, + 0x2dd4be83, 0xd3587421, 0x2949e069, 0x448ec9c8, + 0x6a75c289, 0x78f48e79, 0x6b99583e, 0xdd27b971, + 0xb6bee14f, 0x17f088ad, 0x66c920ac, 0xb47dce3a, + 0x1863df4a, 0x82e51a31, 0x60975133, 0x4562537f, + 0xe0b16477, 0x84bb6bae, 0x1cfe81a0, 0x94f9082b, + 0x58704868, 0x198f45fd, 0x8794de6c, 0xb7527bf8, + 0x23ab73d3, 0xe2724b02, 0x57e31f8f, 0x2a6655ab, + 0x7b2eb28, 0x32fb5c2, 0x9a86c57b, 0xa5d33708, + 0xf2302887, 0xb223bfa5, 0xba02036a, 0x5ced1682, + 0x2b8acf1c, 0x92a779b4, 0xf0f307f2, 0xa14e69e2, + 0xcd65daf4, 0xd50605be, 0x1fd13462, 0x8ac4a6fe, + 0x9d342e53, 0xa0a2f355, 0x32058ae1, 0x75a4f6eb, + 0x390b83ec, 0xaa4060ef, 0x65e719f, 0x51bd6e10, + 0xf93e218a, 0x3d96dd06, 0xaedd3e05, 0x464de6bd, + 0xb591548d, 0x571c45d, 0x6f0406d4, 0xff605015, + 0x241998fb, 0x97d6bde9, 0xcc894043, 0x7767d99e, + 0xbdb0e842, 0x8807898b, 0x38e7195b, 0xdb79c8ee, + 0x47a17c0a, 0xe97c420f, 0xc9f8841e, 0x0, + 0x83098086, 0x48322bed, 0xac1e1170, 0x4e6c5a72, + 0xfbfd0eff, 0x560f8538, 0x1e3daed5, 0x27362d39, + 0x640a0fd9, 0x21685ca6, 0xd19b5b54, 0x3a24362e, + 0xb10c0a67, 0xf9357e7, 0xd2b4ee96, 0x9e1b9b91, + 0x4f80c0c5, 0xa261dc20, 0x695a774b, 0x161c121a, + 0xae293ba, 0xe5c0a02a, 0x433c22e0, 0x1d121b17, + 0xb0e090d, 0xadf28bc7, 0xb92db6a8, 0xc8141ea9, + 0x8557f119, 0x4caf7507, 0xbbee99dd, 0xfda37f60, + 0x9ff70126, 0xbc5c72f5, 0xc544663b, 0x345bfb7e, + 0x768b4329, 0xdccb23c6, 0x68b6edfc, 0x63b8e4f1, + 0xcad731dc, 0x10426385, 0x40139722, 0x2084c611, + 0x7d854a24, 0xf8d2bb3d, 0x11aef932, 0x6dc729a1, + 0x4b1d9e2f, 0xf3dcb230, 0xec0d8652, 0xd077c1e3, + 0x6c2bb316, 0x99a970b9, 0xfa119448, 0x2247e964, + 0xc4a8fc8c, 0x1aa0f03f, 0xd8567d2c, 0xef223390, + 0xc787494e, 0xc1d938d1, 0xfe8ccaa2, 0x3698d40b, + 0xcfa6f581, 0x28a57ade, 0x26dab78e, 0xa43fadbf, + 0xe42c3a9d, 0xd507892, 0x9b6a5fcc, 0x62547e46, + 0xc2f68d13, 0xe890d8b8, 0x5e2e39f7, 0xf582c3af, + 0xbe9f5d80, 0x7c69d093, 0xa96fd52d, 0xb3cf2512, + 0x3bc8ac99, 0xa710187d, 0x6ee89c63, 0x7bdb3bbb, + 0x9cd2678, 0xf46e5918, 0x1ec9ab7, 0xa8834f9a, + 0x65e6956e, 0x7eaaffe6, 0x821bccf, 0xe6ef15e8, + 0xd9bae79b, 0xce4a6f36, 0xd4ea9f09, 0xd629b07c, + 0xaf31a4b2, 0x312a3f23, 0x30c6a594, 0xc035a266, + 0x37744ebc, 0xa6fc82ca, 0xb0e090d0, 0x1533a7d8, + 0x4af10498, 0xf741ecda, 0xe7fcd50, 0x2f1791f6, + 0x8d764dd6, 0x4d43efb0, 0x54ccaa4d, 0xdfe49604, + 0xe39ed1b5, 0x1b4c6a88, 0xb8c12c1f, 0x7f466551, + 0x49d5eea, 0x5d018c35, 0x73fa8774, 0x2efb0b41, + 0x5ab3671d, 0x5292dbd2, 0x33e91056, 0x136dd647, + 0x8c9ad761, 0x7a37a10c, 0x8e59f814, 0x89eb133c, + 0xeecea927, 0x35b761c9, 0xede11ce5, 0x3c7a47b1, + 0x599cd2df, 0x3f55f273, 0x791814ce, 0xbf73c737, + 0xea53f7cd, 0x5b5ffdaa, 0x14df3d6f, 0x867844db, + 0x81caaff3, 0x3eb968c4, 0x2c382434, 0x5fc2a340, + 0x72161dc3, 0xcbce225, 0x8b283c49, 0x41ff0d95, + 0x7139a801, 0xde080cb3, 0x9cd8b4e4, 0x906456c1, + 0x617bcb84, 0x70d532b6, 0x74486c5c, 0x42d0b857 +}; + +static uint32_t U2[256] = { + 0xa75051f4, 0x65537e41, 0xa4c31a17, 0x5e963a27, + 0x6bcb3bab, 0x45f11f9d, 0x58abacfa, 0x3934be3, + 0xfa552030, 0x6df6ad76, 0x769188cc, 0x4c25f502, + 0xd7fc4fe5, 0xcbd7c52a, 0x44802635, 0xa38fb562, + 0x5a49deb1, 0x1b6725ba, 0xe9845ea, 0xc0e15dfe, + 0x7502c32f, 0xf012814c, 0x97a38d46, 0xf9c66bd3, + 0x5fe7038f, 0x9c951592, 0x7aebbf6d, 0x59da9552, + 0x832dd4be, 0x21d35874, 0x692949e0, 0xc8448ec9, + 0x896a75c2, 0x7978f48e, 0x3e6b9958, 0x71dd27b9, + 0x4fb6bee1, 0xad17f088, 0xac66c920, 0x3ab47dce, + 0x4a1863df, 0x3182e51a, 0x33609751, 0x7f456253, + 0x77e0b164, 0xae84bb6b, 0xa01cfe81, 0x2b94f908, + 0x68587048, 0xfd198f45, 0x6c8794de, 0xf8b7527b, + 0xd323ab73, 0x2e2724b, 0x8f57e31f, 0xab2a6655, + 0x2807b2eb, 0xc2032fb5, 0x7b9a86c5, 0x8a5d337, + 0x87f23028, 0xa5b223bf, 0x6aba0203, 0x825ced16, + 0x1c2b8acf, 0xb492a779, 0xf2f0f307, 0xe2a14e69, + 0xf4cd65da, 0xbed50605, 0x621fd134, 0xfe8ac4a6, + 0x539d342e, 0x55a0a2f3, 0xe132058a, 0xeb75a4f6, + 0xec390b83, 0xefaa4060, 0x9f065e71, 0x1051bd6e, + 0x8af93e21, 0x63d96dd, 0x5aedd3e, 0xbd464de6, + 0x8db59154, 0x5d0571c4, 0xd46f0406, 0x15ff6050, + 0xfb241998, 0xe997d6bd, 0x43cc8940, 0x9e7767d9, + 0x42bdb0e8, 0x8b880789, 0x5b38e719, 0xeedb79c8, + 0xa47a17c, 0xfe97c42, 0x1ec9f884, 0x0, + 0x86830980, 0xed48322b, 0x70ac1e11, 0x724e6c5a, + 0xfffbfd0e, 0x38560f85, 0xd51e3dae, 0x3927362d, + 0xd9640a0f, 0xa621685c, 0x54d19b5b, 0x2e3a2436, + 0x67b10c0a, 0xe70f9357, 0x96d2b4ee, 0x919e1b9b, + 0xc54f80c0, 0x20a261dc, 0x4b695a77, 0x1a161c12, + 0xba0ae293, 0x2ae5c0a0, 0xe0433c22, 0x171d121b, + 0xd0b0e09, 0xc7adf28b, 0xa8b92db6, 0xa9c8141e, + 0x198557f1, 0x74caf75, 0xddbbee99, 0x60fda37f, + 0x269ff701, 0xf5bc5c72, 0x3bc54466, 0x7e345bfb, + 0x29768b43, 0xc6dccb23, 0xfc68b6ed, 0xf163b8e4, + 0xdccad731, 0x85104263, 0x22401397, 0x112084c6, + 0x247d854a, 0x3df8d2bb, 0x3211aef9, 0xa16dc729, + 0x2f4b1d9e, 0x30f3dcb2, 0x52ec0d86, 0xe3d077c1, + 0x166c2bb3, 0xb999a970, 0x48fa1194, 0x642247e9, + 0x8cc4a8fc, 0x3f1aa0f0, 0x2cd8567d, 0x90ef2233, + 0x4ec78749, 0xd1c1d938, 0xa2fe8cca, 0xb3698d4, + 0x81cfa6f5, 0xde28a57a, 0x8e26dab7, 0xbfa43fad, + 0x9de42c3a, 0x920d5078, 0xcc9b6a5f, 0x4662547e, + 0x13c2f68d, 0xb8e890d8, 0xf75e2e39, 0xaff582c3, + 0x80be9f5d, 0x937c69d0, 0x2da96fd5, 0x12b3cf25, + 0x993bc8ac, 0x7da71018, 0x636ee89c, 0xbb7bdb3b, + 0x7809cd26, 0x18f46e59, 0xb701ec9a, 0x9aa8834f, + 0x6e65e695, 0xe67eaaff, 0xcf0821bc, 0xe8e6ef15, + 0x9bd9bae7, 0x36ce4a6f, 0x9d4ea9f, 0x7cd629b0, + 0xb2af31a4, 0x23312a3f, 0x9430c6a5, 0x66c035a2, + 0xbc37744e, 0xcaa6fc82, 0xd0b0e090, 0xd81533a7, + 0x984af104, 0xdaf741ec, 0x500e7fcd, 0xf62f1791, + 0xd68d764d, 0xb04d43ef, 0x4d54ccaa, 0x4dfe496, + 0xb5e39ed1, 0x881b4c6a, 0x1fb8c12c, 0x517f4665, + 0xea049d5e, 0x355d018c, 0x7473fa87, 0x412efb0b, + 0x1d5ab367, 0xd25292db, 0x5633e910, 0x47136dd6, + 0x618c9ad7, 0xc7a37a1, 0x148e59f8, 0x3c89eb13, + 0x27eecea9, 0xc935b761, 0xe5ede11c, 0xb13c7a47, + 0xdf599cd2, 0x733f55f2, 0xce791814, 0x37bf73c7, + 0xcdea53f7, 0xaa5b5ffd, 0x6f14df3d, 0xdb867844, + 0xf381caaf, 0xc43eb968, 0x342c3824, 0x405fc2a3, + 0xc372161d, 0x250cbce2, 0x498b283c, 0x9541ff0d, + 0x17139a8, 0xb3de080c, 0xe49cd8b4, 0xc1906456, + 0x84617bcb, 0xb670d532, 0x5c74486c, 0x5742d0b8 +}; + +static uint32_t U3[256] = { + 0xf4a75051, 0x4165537e, 0x17a4c31a, 0x275e963a, + 0xab6bcb3b, 0x9d45f11f, 0xfa58abac, 0xe303934b, + 0x30fa5520, 0x766df6ad, 0xcc769188, 0x24c25f5, + 0xe5d7fc4f, 0x2acbd7c5, 0x35448026, 0x62a38fb5, + 0xb15a49de, 0xba1b6725, 0xea0e9845, 0xfec0e15d, + 0x2f7502c3, 0x4cf01281, 0x4697a38d, 0xd3f9c66b, + 0x8f5fe703, 0x929c9515, 0x6d7aebbf, 0x5259da95, + 0xbe832dd4, 0x7421d358, 0xe0692949, 0xc9c8448e, + 0xc2896a75, 0x8e7978f4, 0x583e6b99, 0xb971dd27, + 0xe14fb6be, 0x88ad17f0, 0x20ac66c9, 0xce3ab47d, + 0xdf4a1863, 0x1a3182e5, 0x51336097, 0x537f4562, + 0x6477e0b1, 0x6bae84bb, 0x81a01cfe, 0x82b94f9, + 0x48685870, 0x45fd198f, 0xde6c8794, 0x7bf8b752, + 0x73d323ab, 0x4b02e272, 0x1f8f57e3, 0x55ab2a66, + 0xeb2807b2, 0xb5c2032f, 0xc57b9a86, 0x3708a5d3, + 0x2887f230, 0xbfa5b223, 0x36aba02, 0x16825ced, + 0xcf1c2b8a, 0x79b492a7, 0x7f2f0f3, 0x69e2a14e, + 0xdaf4cd65, 0x5bed506, 0x34621fd1, 0xa6fe8ac4, + 0x2e539d34, 0xf355a0a2, 0x8ae13205, 0xf6eb75a4, + 0x83ec390b, 0x60efaa40, 0x719f065e, 0x6e1051bd, + 0x218af93e, 0xdd063d96, 0x3e05aedd, 0xe6bd464d, + 0x548db591, 0xc45d0571, 0x6d46f04, 0x5015ff60, + 0x98fb2419, 0xbde997d6, 0x4043cc89, 0xd99e7767, + 0xe842bdb0, 0x898b8807, 0x195b38e7, 0xc8eedb79, + 0x7c0a47a1, 0x420fe97c, 0x841ec9f8, 0x0, + 0x80868309, 0x2bed4832, 0x1170ac1e, 0x5a724e6c, + 0xefffbfd, 0x8538560f, 0xaed51e3d, 0x2d392736, + 0xfd9640a, 0x5ca62168, 0x5b54d19b, 0x362e3a24, + 0xa67b10c, 0x57e70f93, 0xee96d2b4, 0x9b919e1b, + 0xc0c54f80, 0xdc20a261, 0x774b695a, 0x121a161c, + 0x93ba0ae2, 0xa02ae5c0, 0x22e0433c, 0x1b171d12, + 0x90d0b0e, 0x8bc7adf2, 0xb6a8b92d, 0x1ea9c814, + 0xf1198557, 0x75074caf, 0x99ddbbee, 0x7f60fda3, + 0x1269ff7, 0x72f5bc5c, 0x663bc544, 0xfb7e345b, + 0x4329768b, 0x23c6dccb, 0xedfc68b6, 0xe4f163b8, + 0x31dccad7, 0x63851042, 0x97224013, 0xc6112084, + 0x4a247d85, 0xbb3df8d2, 0xf93211ae, 0x29a16dc7, + 0x9e2f4b1d, 0xb230f3dc, 0x8652ec0d, 0xc1e3d077, + 0xb3166c2b, 0x70b999a9, 0x9448fa11, 0xe9642247, + 0xfc8cc4a8, 0xf03f1aa0, 0x7d2cd856, 0x3390ef22, + 0x494ec787, 0x38d1c1d9, 0xcaa2fe8c, 0xd40b3698, + 0xf581cfa6, 0x7ade28a5, 0xb78e26da, 0xadbfa43f, + 0x3a9de42c, 0x78920d50, 0x5fcc9b6a, 0x7e466254, + 0x8d13c2f6, 0xd8b8e890, 0x39f75e2e, 0xc3aff582, + 0x5d80be9f, 0xd0937c69, 0xd52da96f, 0x2512b3cf, + 0xac993bc8, 0x187da710, 0x9c636ee8, 0x3bbb7bdb, + 0x267809cd, 0x5918f46e, 0x9ab701ec, 0x4f9aa883, + 0x956e65e6, 0xffe67eaa, 0xbccf0821, 0x15e8e6ef, + 0xe79bd9ba, 0x6f36ce4a, 0x9f09d4ea, 0xb07cd629, + 0xa4b2af31, 0x3f23312a, 0xa59430c6, 0xa266c035, + 0x4ebc3774, 0x82caa6fc, 0x90d0b0e0, 0xa7d81533, + 0x4984af1, 0xecdaf741, 0xcd500e7f, 0x91f62f17, + 0x4dd68d76, 0xefb04d43, 0xaa4d54cc, 0x9604dfe4, + 0xd1b5e39e, 0x6a881b4c, 0x2c1fb8c1, 0x65517f46, + 0x5eea049d, 0x8c355d01, 0x877473fa, 0xb412efb, + 0x671d5ab3, 0xdbd25292, 0x105633e9, 0xd647136d, + 0xd7618c9a, 0xa10c7a37, 0xf8148e59, 0x133c89eb, + 0xa927eece, 0x61c935b7, 0x1ce5ede1, 0x47b13c7a, + 0xd2df599c, 0xf2733f55, 0x14ce7918, 0xc737bf73, + 0xf7cdea53, 0xfdaa5b5f, 0x3d6f14df, 0x44db8678, + 0xaff381ca, 0x68c43eb9, 0x24342c38, 0xa3405fc2, + 0x1dc37216, 0xe2250cbc, 0x3c498b28, 0xd9541ff, + 0xa8017139, 0xcb3de08, 0xb4e49cd8, 0x56c19064, + 0xcb84617b, 0x32b670d5, 0x6c5c7448, 0xb85742d0 +}; + +#endif + +/* + * the following tables (aes_sbox, aes_inv_sbox, T4, U4) are + * endian-neutral + */ + +static uint8_t +aes_sbox[256] = { + 0x63, 0x7c, 0x77, 0x7b, 0xf2, 0x6b, 0x6f, 0xc5, + 0x30, 0x01, 0x67, 0x2b, 0xfe, 0xd7, 0xab, 0x76, + 0xca, 0x82, 0xc9, 0x7d, 0xfa, 0x59, 0x47, 0xf0, + 0xad, 0xd4, 0xa2, 0xaf, 0x9c, 0xa4, 0x72, 0xc0, + 0xb7, 0xfd, 0x93, 0x26, 0x36, 0x3f, 0xf7, 0xcc, + 0x34, 0xa5, 0xe5, 0xf1, 0x71, 0xd8, 0x31, 0x15, + 0x04, 0xc7, 0x23, 0xc3, 0x18, 0x96, 0x05, 0x9a, + 0x07, 0x12, 0x80, 0xe2, 0xeb, 0x27, 0xb2, 0x75, + 0x09, 0x83, 0x2c, 0x1a, 0x1b, 0x6e, 0x5a, 0xa0, + 0x52, 0x3b, 0xd6, 0xb3, 0x29, 0xe3, 0x2f, 0x84, + 0x53, 0xd1, 0x00, 0xed, 0x20, 0xfc, 0xb1, 0x5b, + 0x6a, 0xcb, 0xbe, 0x39, 0x4a, 0x4c, 0x58, 0xcf, + 0xd0, 0xef, 0xaa, 0xfb, 0x43, 0x4d, 0x33, 0x85, + 0x45, 0xf9, 0x02, 0x7f, 0x50, 0x3c, 0x9f, 0xa8, + 0x51, 0xa3, 0x40, 0x8f, 0x92, 0x9d, 0x38, 0xf5, + 0xbc, 0xb6, 0xda, 0x21, 0x10, 0xff, 0xf3, 0xd2, + 0xcd, 0x0c, 0x13, 0xec, 0x5f, 0x97, 0x44, 0x17, + 0xc4, 0xa7, 0x7e, 0x3d, 0x64, 0x5d, 0x19, 0x73, + 0x60, 0x81, 0x4f, 0xdc, 0x22, 0x2a, 0x90, 0x88, + 0x46, 0xee, 0xb8, 0x14, 0xde, 0x5e, 0x0b, 0xdb, + 0xe0, 0x32, 0x3a, 0x0a, 0x49, 0x06, 0x24, 0x5c, + 0xc2, 0xd3, 0xac, 0x62, 0x91, 0x95, 0xe4, 0x79, + 0xe7, 0xc8, 0x37, 0x6d, 0x8d, 0xd5, 0x4e, 0xa9, + 0x6c, 0x56, 0xf4, 0xea, 0x65, 0x7a, 0xae, 0x08, + 0xba, 0x78, 0x25, 0x2e, 0x1c, 0xa6, 0xb4, 0xc6, + 0xe8, 0xdd, 0x74, 0x1f, 0x4b, 0xbd, 0x8b, 0x8a, + 0x70, 0x3e, 0xb5, 0x66, 0x48, 0x03, 0xf6, 0x0e, + 0x61, 0x35, 0x57, 0xb9, 0x86, 0xc1, 0x1d, 0x9e, + 0xe1, 0xf8, 0x98, 0x11, 0x69, 0xd9, 0x8e, 0x94, + 0x9b, 0x1e, 0x87, 0xe9, 0xce, 0x55, 0x28, 0xdf, + 0x8c, 0xa1, 0x89, 0x0d, 0xbf, 0xe6, 0x42, 0x68, + 0x41, 0x99, 0x2d, 0x0f, 0xb0, 0x54, 0xbb, 0x16 +}; + +#ifndef CPU_RISC +static uint8_t +aes_inv_sbox[256] = { + 0x52, 0x09, 0x6a, 0xd5, 0x30, 0x36, 0xa5, 0x38, + 0xbf, 0x40, 0xa3, 0x9e, 0x81, 0xf3, 0xd7, 0xfb, + 0x7c, 0xe3, 0x39, 0x82, 0x9b, 0x2f, 0xff, 0x87, + 0x34, 0x8e, 0x43, 0x44, 0xc4, 0xde, 0xe9, 0xcb, + 0x54, 0x7b, 0x94, 0x32, 0xa6, 0xc2, 0x23, 0x3d, + 0xee, 0x4c, 0x95, 0x0b, 0x42, 0xfa, 0xc3, 0x4e, + 0x08, 0x2e, 0xa1, 0x66, 0x28, 0xd9, 0x24, 0xb2, + 0x76, 0x5b, 0xa2, 0x49, 0x6d, 0x8b, 0xd1, 0x25, + 0x72, 0xf8, 0xf6, 0x64, 0x86, 0x68, 0x98, 0x16, + 0xd4, 0xa4, 0x5c, 0xcc, 0x5d, 0x65, 0xb6, 0x92, + 0x6c, 0x70, 0x48, 0x50, 0xfd, 0xed, 0xb9, 0xda, + 0x5e, 0x15, 0x46, 0x57, 0xa7, 0x8d, 0x9d, 0x84, + 0x90, 0xd8, 0xab, 0x00, 0x8c, 0xbc, 0xd3, 0x0a, + 0xf7, 0xe4, 0x58, 0x05, 0xb8, 0xb3, 0x45, 0x06, + 0xd0, 0x2c, 0x1e, 0x8f, 0xca, 0x3f, 0x0f, 0x02, + 0xc1, 0xaf, 0xbd, 0x03, 0x01, 0x13, 0x8a, 0x6b, + 0x3a, 0x91, 0x11, 0x41, 0x4f, 0x67, 0xdc, 0xea, + 0x97, 0xf2, 0xcf, 0xce, 0xf0, 0xb4, 0xe6, 0x73, + 0x96, 0xac, 0x74, 0x22, 0xe7, 0xad, 0x35, 0x85, + 0xe2, 0xf9, 0x37, 0xe8, 0x1c, 0x75, 0xdf, 0x6e, + 0x47, 0xf1, 0x1a, 0x71, 0x1d, 0x29, 0xc5, 0x89, + 0x6f, 0xb7, 0x62, 0x0e, 0xaa, 0x18, 0xbe, 0x1b, + 0xfc, 0x56, 0x3e, 0x4b, 0xc6, 0xd2, 0x79, 0x20, + 0x9a, 0xdb, 0xc0, 0xfe, 0x78, 0xcd, 0x5a, 0xf4, + 0x1f, 0xdd, 0xa8, 0x33, 0x88, 0x07, 0xc7, 0x31, + 0xb1, 0x12, 0x10, 0x59, 0x27, 0x80, 0xec, 0x5f, + 0x60, 0x51, 0x7f, 0xa9, 0x19, 0xb5, 0x4a, 0x0d, + 0x2d, 0xe5, 0x7a, 0x9f, 0x93, 0xc9, 0x9c, 0xef, + 0xa0, 0xe0, 0x3b, 0x4d, 0xae, 0x2a, 0xf5, 0xb0, + 0xc8, 0xeb, 0xbb, 0x3c, 0x83, 0x53, 0x99, 0x61, + 0x17, 0x2b, 0x04, 0x7e, 0xba, 0x77, 0xd6, 0x26, + 0xe1, 0x69, 0x14, 0x63, 0x55, 0x21, 0x0c, 0x7d +}; +#endif /* ! CPU_RISC */ + +#ifdef CPU_RISC +static uint32_t +T4[256] = { + 0x63636363, 0x7c7c7c7c, 0x77777777, 0x7b7b7b7b, + 0xf2f2f2f2, 0x6b6b6b6b, 0x6f6f6f6f, 0xc5c5c5c5, + 0x30303030, 0x01010101, 0x67676767, 0x2b2b2b2b, + 0xfefefefe, 0xd7d7d7d7, 0xabababab, 0x76767676, + 0xcacacaca, 0x82828282, 0xc9c9c9c9, 0x7d7d7d7d, + 0xfafafafa, 0x59595959, 0x47474747, 0xf0f0f0f0, + 0xadadadad, 0xd4d4d4d4, 0xa2a2a2a2, 0xafafafaf, + 0x9c9c9c9c, 0xa4a4a4a4, 0x72727272, 0xc0c0c0c0, + 0xb7b7b7b7, 0xfdfdfdfd, 0x93939393, 0x26262626, + 0x36363636, 0x3f3f3f3f, 0xf7f7f7f7, 0xcccccccc, + 0x34343434, 0xa5a5a5a5, 0xe5e5e5e5, 0xf1f1f1f1, + 0x71717171, 0xd8d8d8d8, 0x31313131, 0x15151515, + 0x04040404, 0xc7c7c7c7, 0x23232323, 0xc3c3c3c3, + 0x18181818, 0x96969696, 0x05050505, 0x9a9a9a9a, + 0x07070707, 0x12121212, 0x80808080, 0xe2e2e2e2, + 0xebebebeb, 0x27272727, 0xb2b2b2b2, 0x75757575, + 0x09090909, 0x83838383, 0x2c2c2c2c, 0x1a1a1a1a, + 0x1b1b1b1b, 0x6e6e6e6e, 0x5a5a5a5a, 0xa0a0a0a0, + 0x52525252, 0x3b3b3b3b, 0xd6d6d6d6, 0xb3b3b3b3, + 0x29292929, 0xe3e3e3e3, 0x2f2f2f2f, 0x84848484, + 0x53535353, 0xd1d1d1d1, 0x00000000, 0xedededed, + 0x20202020, 0xfcfcfcfc, 0xb1b1b1b1, 0x5b5b5b5b, + 0x6a6a6a6a, 0xcbcbcbcb, 0xbebebebe, 0x39393939, + 0x4a4a4a4a, 0x4c4c4c4c, 0x58585858, 0xcfcfcfcf, + 0xd0d0d0d0, 0xefefefef, 0xaaaaaaaa, 0xfbfbfbfb, + 0x43434343, 0x4d4d4d4d, 0x33333333, 0x85858585, + 0x45454545, 0xf9f9f9f9, 0x02020202, 0x7f7f7f7f, + 0x50505050, 0x3c3c3c3c, 0x9f9f9f9f, 0xa8a8a8a8, + 0x51515151, 0xa3a3a3a3, 0x40404040, 0x8f8f8f8f, + 0x92929292, 0x9d9d9d9d, 0x38383838, 0xf5f5f5f5, + 0xbcbcbcbc, 0xb6b6b6b6, 0xdadadada, 0x21212121, + 0x10101010, 0xffffffff, 0xf3f3f3f3, 0xd2d2d2d2, + 0xcdcdcdcd, 0x0c0c0c0c, 0x13131313, 0xecececec, + 0x5f5f5f5f, 0x97979797, 0x44444444, 0x17171717, + 0xc4c4c4c4, 0xa7a7a7a7, 0x7e7e7e7e, 0x3d3d3d3d, + 0x64646464, 0x5d5d5d5d, 0x19191919, 0x73737373, + 0x60606060, 0x81818181, 0x4f4f4f4f, 0xdcdcdcdc, + 0x22222222, 0x2a2a2a2a, 0x90909090, 0x88888888, + 0x46464646, 0xeeeeeeee, 0xb8b8b8b8, 0x14141414, + 0xdededede, 0x5e5e5e5e, 0x0b0b0b0b, 0xdbdbdbdb, + 0xe0e0e0e0, 0x32323232, 0x3a3a3a3a, 0x0a0a0a0a, + 0x49494949, 0x06060606, 0x24242424, 0x5c5c5c5c, + 0xc2c2c2c2, 0xd3d3d3d3, 0xacacacac, 0x62626262, + 0x91919191, 0x95959595, 0xe4e4e4e4, 0x79797979, + 0xe7e7e7e7, 0xc8c8c8c8, 0x37373737, 0x6d6d6d6d, + 0x8d8d8d8d, 0xd5d5d5d5, 0x4e4e4e4e, 0xa9a9a9a9, + 0x6c6c6c6c, 0x56565656, 0xf4f4f4f4, 0xeaeaeaea, + 0x65656565, 0x7a7a7a7a, 0xaeaeaeae, 0x08080808, + 0xbabababa, 0x78787878, 0x25252525, 0x2e2e2e2e, + 0x1c1c1c1c, 0xa6a6a6a6, 0xb4b4b4b4, 0xc6c6c6c6, + 0xe8e8e8e8, 0xdddddddd, 0x74747474, 0x1f1f1f1f, + 0x4b4b4b4b, 0xbdbdbdbd, 0x8b8b8b8b, 0x8a8a8a8a, + 0x70707070, 0x3e3e3e3e, 0xb5b5b5b5, 0x66666666, + 0x48484848, 0x03030303, 0xf6f6f6f6, 0x0e0e0e0e, + 0x61616161, 0x35353535, 0x57575757, 0xb9b9b9b9, + 0x86868686, 0xc1c1c1c1, 0x1d1d1d1d, 0x9e9e9e9e, + 0xe1e1e1e1, 0xf8f8f8f8, 0x98989898, 0x11111111, + 0x69696969, 0xd9d9d9d9, 0x8e8e8e8e, 0x94949494, + 0x9b9b9b9b, 0x1e1e1e1e, 0x87878787, 0xe9e9e9e9, + 0xcececece, 0x55555555, 0x28282828, 0xdfdfdfdf, + 0x8c8c8c8c, 0xa1a1a1a1, 0x89898989, 0x0d0d0d0d, + 0xbfbfbfbf, 0xe6e6e6e6, 0x42424242, 0x68686868, + 0x41414141, 0x99999999, 0x2d2d2d2d, 0x0f0f0f0f, + 0xb0b0b0b0, 0x54545454, 0xbbbbbbbb, 0x16161616 +}; + +static uint32_t U4[256] = { + 0x52525252, 0x9090909, 0x6a6a6a6a, 0xd5d5d5d5, + 0x30303030, 0x36363636, 0xa5a5a5a5, 0x38383838, + 0xbfbfbfbf, 0x40404040, 0xa3a3a3a3, 0x9e9e9e9e, + 0x81818181, 0xf3f3f3f3, 0xd7d7d7d7, 0xfbfbfbfb, + 0x7c7c7c7c, 0xe3e3e3e3, 0x39393939, 0x82828282, + 0x9b9b9b9b, 0x2f2f2f2f, 0xffffffff, 0x87878787, + 0x34343434, 0x8e8e8e8e, 0x43434343, 0x44444444, + 0xc4c4c4c4, 0xdededede, 0xe9e9e9e9, 0xcbcbcbcb, + 0x54545454, 0x7b7b7b7b, 0x94949494, 0x32323232, + 0xa6a6a6a6, 0xc2c2c2c2, 0x23232323, 0x3d3d3d3d, + 0xeeeeeeee, 0x4c4c4c4c, 0x95959595, 0xb0b0b0b, + 0x42424242, 0xfafafafa, 0xc3c3c3c3, 0x4e4e4e4e, + 0x8080808, 0x2e2e2e2e, 0xa1a1a1a1, 0x66666666, + 0x28282828, 0xd9d9d9d9, 0x24242424, 0xb2b2b2b2, + 0x76767676, 0x5b5b5b5b, 0xa2a2a2a2, 0x49494949, + 0x6d6d6d6d, 0x8b8b8b8b, 0xd1d1d1d1, 0x25252525, + 0x72727272, 0xf8f8f8f8, 0xf6f6f6f6, 0x64646464, + 0x86868686, 0x68686868, 0x98989898, 0x16161616, + 0xd4d4d4d4, 0xa4a4a4a4, 0x5c5c5c5c, 0xcccccccc, + 0x5d5d5d5d, 0x65656565, 0xb6b6b6b6, 0x92929292, + 0x6c6c6c6c, 0x70707070, 0x48484848, 0x50505050, + 0xfdfdfdfd, 0xedededed, 0xb9b9b9b9, 0xdadadada, + 0x5e5e5e5e, 0x15151515, 0x46464646, 0x57575757, + 0xa7a7a7a7, 0x8d8d8d8d, 0x9d9d9d9d, 0x84848484, + 0x90909090, 0xd8d8d8d8, 0xabababab, 0x0, + 0x8c8c8c8c, 0xbcbcbcbc, 0xd3d3d3d3, 0xa0a0a0a, + 0xf7f7f7f7, 0xe4e4e4e4, 0x58585858, 0x5050505, + 0xb8b8b8b8, 0xb3b3b3b3, 0x45454545, 0x6060606, + 0xd0d0d0d0, 0x2c2c2c2c, 0x1e1e1e1e, 0x8f8f8f8f, + 0xcacacaca, 0x3f3f3f3f, 0xf0f0f0f, 0x2020202, + 0xc1c1c1c1, 0xafafafaf, 0xbdbdbdbd, 0x3030303, + 0x1010101, 0x13131313, 0x8a8a8a8a, 0x6b6b6b6b, + 0x3a3a3a3a, 0x91919191, 0x11111111, 0x41414141, + 0x4f4f4f4f, 0x67676767, 0xdcdcdcdc, 0xeaeaeaea, + 0x97979797, 0xf2f2f2f2, 0xcfcfcfcf, 0xcececece, + 0xf0f0f0f0, 0xb4b4b4b4, 0xe6e6e6e6, 0x73737373, + 0x96969696, 0xacacacac, 0x74747474, 0x22222222, + 0xe7e7e7e7, 0xadadadad, 0x35353535, 0x85858585, + 0xe2e2e2e2, 0xf9f9f9f9, 0x37373737, 0xe8e8e8e8, + 0x1c1c1c1c, 0x75757575, 0xdfdfdfdf, 0x6e6e6e6e, + 0x47474747, 0xf1f1f1f1, 0x1a1a1a1a, 0x71717171, + 0x1d1d1d1d, 0x29292929, 0xc5c5c5c5, 0x89898989, + 0x6f6f6f6f, 0xb7b7b7b7, 0x62626262, 0xe0e0e0e, + 0xaaaaaaaa, 0x18181818, 0xbebebebe, 0x1b1b1b1b, + 0xfcfcfcfc, 0x56565656, 0x3e3e3e3e, 0x4b4b4b4b, + 0xc6c6c6c6, 0xd2d2d2d2, 0x79797979, 0x20202020, + 0x9a9a9a9a, 0xdbdbdbdb, 0xc0c0c0c0, 0xfefefefe, + 0x78787878, 0xcdcdcdcd, 0x5a5a5a5a, 0xf4f4f4f4, + 0x1f1f1f1f, 0xdddddddd, 0xa8a8a8a8, 0x33333333, + 0x88888888, 0x7070707, 0xc7c7c7c7, 0x31313131, + 0xb1b1b1b1, 0x12121212, 0x10101010, 0x59595959, + 0x27272727, 0x80808080, 0xecececec, 0x5f5f5f5f, + 0x60606060, 0x51515151, 0x7f7f7f7f, 0xa9a9a9a9, + 0x19191919, 0xb5b5b5b5, 0x4a4a4a4a, 0xd0d0d0d, + 0x2d2d2d2d, 0xe5e5e5e5, 0x7a7a7a7a, 0x9f9f9f9f, + 0x93939393, 0xc9c9c9c9, 0x9c9c9c9c, 0xefefefef, + 0xa0a0a0a0, 0xe0e0e0e0, 0x3b3b3b3b, 0x4d4d4d4d, + 0xaeaeaeae, 0x2a2a2a2a, 0xf5f5f5f5, 0xb0b0b0b0, + 0xc8c8c8c8, 0xebebebeb, 0xbbbbbbbb, 0x3c3c3c3c, + 0x83838383, 0x53535353, 0x99999999, 0x61616161, + 0x17171717, 0x2b2b2b2b, 0x4040404, 0x7e7e7e7e, + 0xbabababa, 0x77777777, 0xd6d6d6d6, 0x26262626, + 0xe1e1e1e1, 0x69696969, 0x14141414, 0x63636363, + 0x55555555, 0x21212121, 0xc0c0c0c, 0x7d7d7d7d +}; +#endif /* CPU_RISC */ + + +/* aes internals */ + +extern debug_module_t mod_aes_icm; + +inline void +aes_expand_encryption_key(const v128_t *key, + aes_expanded_key_t expanded_key) { + int i; + gf2_8 rc; + + /* initialize round constant */ + rc = 1; + + expanded_key[0].v32[0] = key->v32[0]; + expanded_key[0].v32[1] = key->v32[1]; + expanded_key[0].v32[2] = key->v32[2]; + expanded_key[0].v32[3] = key->v32[3]; + +#if 0 + debug_print(mod_aes_icm, + "expanded key[0]: %s", v128_hex_string(&expanded_key[0])); +#endif + + /* loop over round keys */ + for (i=1; i < 11; i++) { + + /* munge first word of round key */ + expanded_key[i].v8[0] = aes_sbox[expanded_key[i-1].v8[13]] ^ rc; + expanded_key[i].v8[1] = aes_sbox[expanded_key[i-1].v8[14]]; + expanded_key[i].v8[2] = aes_sbox[expanded_key[i-1].v8[15]]; + expanded_key[i].v8[3] = aes_sbox[expanded_key[i-1].v8[12]]; + + expanded_key[i].v32[0] ^= expanded_key[i-1].v32[0]; + + /* set remaining 32 bit words to the exor of the one previous with + * the one four words previous */ + + expanded_key[i].v32[1] = + expanded_key[i].v32[0] ^ expanded_key[i-1].v32[1]; + + expanded_key[i].v32[2] = + expanded_key[i].v32[1] ^ expanded_key[i-1].v32[2]; + + expanded_key[i].v32[3] = + expanded_key[i].v32[2] ^ expanded_key[i-1].v32[3]; + +#if 0 + debug_print2(mod_aes_icm, + "expanded key[%d]: %s", i,v128_hex_string(&expanded_key[i])); +#endif + + /* modify round constant */ + rc = gf2_8_shift(rc); + + } +} + +inline void +aes_expand_decryption_key(const v128_t *key, + aes_expanded_key_t expanded_key) { + int i; + + aes_expand_encryption_key(key, expanded_key); + + /* invert the order of the round keys */ + for (i=0; i < 5; i++) { + v128_t tmp; + v128_copy(&tmp, &expanded_key[10-i]); + v128_copy(&expanded_key[10-i], &expanded_key[i]); + v128_copy(&expanded_key[i], &tmp); + } + + /* + * apply the inverse mixColumn transform to the round keys (except + * for the first and the last) + * + * mixColumn is implemented by using the tables U0, U1, U2, U3, + * followed by the T4 table (which cancels out the use of the sbox + * in the U-tables) + */ + for (i=1; i < 10; i++) { +#ifdef CPU_RISC + uint32_t tmp; + + tmp = expanded_key[i].v32[0]; + expanded_key[i].v32[0] = + U0[T4[(tmp >> 24) ] & 0xff] ^ + U1[T4[(tmp >> 16) & 0xff] & 0xff] ^ + U2[T4[(tmp >> 8) & 0xff] & 0xff] ^ + U3[T4[(tmp) & 0xff] & 0xff]; + + tmp = expanded_key[i].v32[1]; + expanded_key[i].v32[1] = + U0[T4[(tmp >> 24) ] & 0xff] ^ + U1[T4[(tmp >> 16) & 0xff] & 0xff] ^ + U2[T4[(tmp >> 8) & 0xff] & 0xff] ^ + U3[T4[(tmp) & 0xff] & 0xff]; + + tmp = expanded_key[i].v32[2]; + expanded_key[i].v32[2] = + U0[T4[(tmp >> 24) ] & 0xff] ^ + U1[T4[(tmp >> 16) & 0xff] & 0xff] ^ + U2[T4[(tmp >> 8) & 0xff] & 0xff] ^ + U3[T4[(tmp) & 0xff] & 0xff]; + + tmp = expanded_key[i].v32[3]; + expanded_key[i].v32[3] = + U0[T4[(tmp >> 24) ] & 0xff] ^ + U1[T4[(tmp >> 16) & 0xff] & 0xff] ^ + U2[T4[(tmp >> 8) & 0xff] & 0xff] ^ + U3[T4[(tmp) & 0xff] & 0xff]; +#else /* assume CPU_CISC */ + + uint32_t c0, c1, c2, c3; + + c0 = U0[aes_sbox[expanded_key[i].v8[0]]] + ^ U1[aes_sbox[expanded_key[i].v8[1]]] + ^ U2[aes_sbox[expanded_key[i].v8[2]]] + ^ U3[aes_sbox[expanded_key[i].v8[3]]]; + + c1 = U0[aes_sbox[expanded_key[i].v8[4]]] + ^ U1[aes_sbox[expanded_key[i].v8[5]]] + ^ U2[aes_sbox[expanded_key[i].v8[6]]] + ^ U3[aes_sbox[expanded_key[i].v8[7]]]; + + c2 = U0[aes_sbox[expanded_key[i].v8[8]]] + ^ U1[aes_sbox[expanded_key[i].v8[9]]] + ^ U2[aes_sbox[expanded_key[i].v8[10]]] + ^ U3[aes_sbox[expanded_key[i].v8[11]]]; + + c3 = U0[aes_sbox[expanded_key[i].v8[12]]] + ^ U1[aes_sbox[expanded_key[i].v8[13]]] + ^ U2[aes_sbox[expanded_key[i].v8[14]]] + ^ U3[aes_sbox[expanded_key[i].v8[15]]]; + + expanded_key[i].v32[0] = c0; + expanded_key[i].v32[1] = c1; + expanded_key[i].v32[2] = c2; + expanded_key[i].v32[3] = c3; + +#endif + } +} + +#ifdef CPU_CISC + + +static inline void +aes_round(v128_t *state, const v128_t *round_key) { + uint32_t column0, column1, column2, column3; + + /* compute the columns of the output square in terms of the octets + of state, using the tables T0, T1, T2, T3 */ + + column0 = T0[state->v8[0]] ^ T1[state->v8[5]] + ^ T2[state->v8[10]] ^ T3[state->v8[15]]; + + column1 = T0[state->v8[4]] ^ T1[state->v8[9]] + ^ T2[state->v8[14]] ^ T3[state->v8[3]]; + + column2 = T0[state->v8[8]] ^ T1[state->v8[13]] + ^ T2[state->v8[2]] ^ T3[state->v8[7]]; + + column3 = T0[state->v8[12]] ^ T1[state->v8[1]] + ^ T2[state->v8[6]] ^ T3[state->v8[11]]; + + state->v32[0] = column0 ^ round_key->v32[0]; + state->v32[1] = column1 ^ round_key->v32[1]; + state->v32[2] = column2 ^ round_key->v32[2]; + state->v32[3] = column3 ^ round_key->v32[3]; + +} + + +static inline void +aes_inv_round(v128_t *state, const v128_t *round_key) { + uint32_t column0, column1, column2, column3; + + /* compute the columns of the output square in terms of the octets + of state, using the tables U0, U1, U2, U3 */ + + column0 = U0[state->v8[0]] ^ U1[state->v8[13]] + ^ U2[state->v8[10]] ^ U3[state->v8[7]]; + + column1 = U0[state->v8[4]] ^ U1[state->v8[1]] + ^ U2[state->v8[14]] ^ U3[state->v8[11]]; + + column2 = U0[state->v8[8]] ^ U1[state->v8[5]] + ^ U2[state->v8[2]] ^ U3[state->v8[15]]; + + column3 = U0[state->v8[12]] ^ U1[state->v8[9]] + ^ U2[state->v8[6]] ^ U3[state->v8[3]]; + + state->v32[0] = column0 ^ round_key->v32[0]; + state->v32[1] = column1 ^ round_key->v32[1]; + state->v32[2] = column2 ^ round_key->v32[2]; + state->v32[3] = column3 ^ round_key->v32[3]; + +} + +static inline void +aes_final_round(v128_t *state, const v128_t *round_key) { + uint8_t tmp; + + /* byte substitutions and row shifts */ + /* first row - no shift */ + state->v8[0] = aes_sbox[state->v8[0]]; + state->v8[4] = aes_sbox[state->v8[4]]; + state->v8[8] = aes_sbox[state->v8[8]]; + state->v8[12] = aes_sbox[state->v8[12]]; + + /* second row - shift one left */ + tmp = aes_sbox[state->v8[1]]; + state->v8[1] = aes_sbox[state->v8[5]]; + state->v8[5] = aes_sbox[state->v8[9]]; + state->v8[9] = aes_sbox[state->v8[13]]; + state->v8[13] = tmp; + + /* third row - shift two left */ + tmp = aes_sbox[state->v8[10]]; + state->v8[10] = aes_sbox[state->v8[2]]; + state->v8[2] = tmp; + tmp = aes_sbox[state->v8[14]]; + state->v8[14] = aes_sbox[state->v8[6]]; + state->v8[6] = tmp; + + /* fourth row - shift three left */ + tmp = aes_sbox[state->v8[15]]; + state->v8[15] = aes_sbox[state->v8[11]]; + state->v8[11] = aes_sbox[state->v8[7]]; + state->v8[7] = aes_sbox[state->v8[3]]; + state->v8[3] = tmp; + + v128_xor_eq(state, round_key); +} + +static inline void +aes_inv_final_round(v128_t *state, const v128_t *round_key) { + uint8_t tmp; + + /* byte substitutions and row shifts */ + /* first row - no shift */ + state->v8[0] = aes_inv_sbox[state->v8[0]]; + state->v8[4] = aes_inv_sbox[state->v8[4]]; + state->v8[8] = aes_inv_sbox[state->v8[8]]; + state->v8[12] = aes_inv_sbox[state->v8[12]]; + + /* second row - shift one right */ + tmp = aes_inv_sbox[state->v8[13]]; + state->v8[13] = aes_inv_sbox[state->v8[9]]; + state->v8[9] = aes_inv_sbox[state->v8[5]]; + state->v8[5] = aes_inv_sbox[state->v8[1]]; + state->v8[1] = tmp; + + /* third row - shift two right */ + tmp = aes_inv_sbox[state->v8[2]]; + state->v8[2] = aes_inv_sbox[state->v8[10]]; + state->v8[10] = tmp; + tmp = aes_inv_sbox[state->v8[6]]; + state->v8[6] = aes_inv_sbox[state->v8[14]]; + state->v8[14] = tmp; + + /* fourth row - shift three right */ + tmp = aes_inv_sbox[state->v8[3]]; + state->v8[3] = aes_inv_sbox[state->v8[7]]; + state->v8[7] = aes_inv_sbox[state->v8[11]]; + state->v8[11] = aes_inv_sbox[state->v8[15]]; + state->v8[15] = tmp; + + v128_xor_eq(state, round_key); +} + + +#elif CPU_RISC + +static inline void +aes_round(v128_t *state, const v128_t *round_key) { + uint32_t column0, column1, column2, column3; + + /* compute the columns of the output square in terms of the octets + of state, using the tables T0, T1, T2, T3 */ +#ifdef WORDS_BIGENDIAN + column0 = T0[state->v32[0] >> 24] ^ T1[(state->v32[1] >> 16) & 0xff] + ^ T2[(state->v32[2] >> 8) & 0xff] ^ T3[state->v32[3] & 0xff]; + + column1 = T0[state->v32[1] >> 24] ^ T1[(state->v32[2] >> 16) & 0xff] + ^ T2[(state->v32[3] >> 8) & 0xff] ^ T3[state->v32[0] & 0xff]; + + column2 = T0[state->v32[2] >> 24] ^ T1[(state->v32[3] >> 16) & 0xff] + ^ T2[(state->v32[0] >> 8) & 0xff] ^ T3[state->v32[1] & 0xff]; + + column3 = T0[state->v32[3] >> 24] ^ T1[(state->v32[0] >> 16) & 0xff] + ^ T2[(state->v32[1] >> 8) & 0xff] ^ T3[state->v32[2] & 0xff]; +#else + column0 = T0[state->v32[0] & 0xff] ^ T1[(state->v32[1] >> 8) & 0xff] + ^ T2[(state->v32[2] >> 16) & 0xff] ^ T3[state->v32[3] >> 24]; + + column1 = T0[state->v32[1] & 0xff] ^ T1[(state->v32[2] >> 8) & 0xff] + ^ T2[(state->v32[3] >> 16) & 0xff] ^ T3[state->v32[0] >> 24]; + + column2 = T0[state->v32[2] & 0xff] ^ T1[(state->v32[3] >> 8) & 0xff] + ^ T2[(state->v32[0] >> 16) & 0xff] ^ T3[state->v32[1] >> 24]; + + column3 = T0[state->v32[3] & 0xff] ^ T1[(state->v32[0] >> 8) & 0xff] + ^ T2[(state->v32[1] >> 16) & 0xff] ^ T3[state->v32[2] >> 24]; +#endif /* WORDS_BIGENDIAN */ + + state->v32[0] = column0 ^ round_key->v32[0]; + state->v32[1] = column1 ^ round_key->v32[1]; + state->v32[2] = column2 ^ round_key->v32[2]; + state->v32[3] = column3 ^ round_key->v32[3]; + +} + +static inline void +aes_inv_round(v128_t *state, const v128_t *round_key) { + uint32_t column0, column1, column2, column3; + + /* compute the columns of the output square in terms of the octets + of state, using the tables U0, U1, U2, U3 */ + +#ifdef WORDS_BIGENDIAN + /* FIX! WRong indexes */ + column0 = U0[state->v32[0] >> 24] ^ U1[(state->v32[3] >> 16) & 0xff] + ^ U2[(state->v32[2] >> 8) & 0xff] ^ U3[state->v32[1] & 0xff]; + + column1 = U0[state->v32[1] >> 24] ^ U1[(state->v32[0] >> 16) & 0xff] + ^ U2[(state->v32[3] >> 8) & 0xff] ^ U3[state->v32[2] & 0xff]; + + column2 = U0[state->v32[2] >> 24] ^ U1[(state->v32[1] >> 16) & 0xff] + ^ U2[(state->v32[0] >> 8) & 0xff] ^ U3[state->v32[3] & 0xff]; + + column3 = U0[state->v32[3] >> 24] ^ U1[(state->v32[2] >> 16) & 0xff] + ^ U2[(state->v32[1] >> 8) & 0xff] ^ U3[state->v32[0] & 0xff]; +#else + column0 = U0[state->v32[0] & 0xff] ^ U1[(state->v32[1] >> 8) & 0xff] + ^ U2[(state->v32[2] >> 16) & 0xff] ^ U3[state->v32[3] >> 24]; + + column1 = U0[state->v32[1] & 0xff] ^ U1[(state->v32[2] >> 8) & 0xff] + ^ U2[(state->v32[3] >> 16) & 0xff] ^ U3[state->v32[0] >> 24]; + + column2 = U0[state->v32[2] & 0xff] ^ U1[(state->v32[3] >> 8) & 0xff] + ^ U2[(state->v32[0] >> 16) & 0xff] ^ U3[state->v32[1] >> 24]; + + column3 = U0[state->v32[3] & 0xff] ^ U1[(state->v32[0] >> 8) & 0xff] + ^ U2[(state->v32[1] >> 16) & 0xff] ^ U3[state->v32[2] >> 24]; +#endif /* WORDS_BIGENDIAN */ + + state->v32[0] = column0 ^ round_key->v32[0]; + state->v32[1] = column1 ^ round_key->v32[1]; + state->v32[2] = column2 ^ round_key->v32[2]; + state->v32[3] = column3 ^ round_key->v32[3]; + +} + +static inline void +aes_final_round(v128_t *state, const v128_t *round_key) { + uint32_t tmp0, tmp1, tmp2, tmp3; + + tmp0 = (T4[(state->v32[0] >> 24)] & 0xff000000) + ^ (T4[(state->v32[1] >> 16) & 0xff] & 0x00ff0000) + ^ (T4[(state->v32[2] >> 8) & 0xff] & 0x0000ff00) + ^ (T4[(state->v32[3] ) & 0xff] & 0x000000ff) + ^ round_key->v32[0]; + + tmp1 = (T4[(state->v32[1] >> 24)] & 0xff000000) + ^ (T4[(state->v32[2] >> 16) & 0xff] & 0x00ff0000) + ^ (T4[(state->v32[3] >> 8) & 0xff] & 0x0000ff00) + ^ (T4[(state->v32[0] ) & 0xff] & 0x000000ff) + ^ round_key->v32[1]; + + tmp2 = (T4[(state->v32[2] >> 24)] & 0xff000000) + ^ (T4[(state->v32[3] >> 16) & 0xff] & 0x00ff0000) + ^ (T4[(state->v32[0] >> 8) & 0xff] & 0x0000ff00) + ^ (T4[(state->v32[1] ) & 0xff] & 0x000000ff) + ^ round_key->v32[2]; + + tmp3 = (T4[(state->v32[3] >> 24)] & 0xff000000) + ^ (T4[(state->v32[0] >> 16) & 0xff] & 0x00ff0000) + ^ (T4[(state->v32[1] >> 8) & 0xff] & 0x0000ff00) + ^ (T4[(state->v32[2] ) & 0xff] & 0x000000ff) + ^ round_key->v32[3]; + + state->v32[0] = tmp0; + state->v32[1] = tmp1; + state->v32[2] = tmp2; + state->v32[3] = tmp3; + +} + +static inline void +aes_inv_final_round(v128_t *state, const v128_t *round_key) { + uint32_t tmp0, tmp1, tmp2, tmp3; + + tmp0 = (U4[(state->v32[0] >> 24)] & 0xff000000) + ^ (U4[(state->v32[3] >> 16) & 0xff] & 0x00ff0000) + ^ (U4[(state->v32[2] >> 8) & 0xff] & 0x0000ff00) + ^ (U4[(state->v32[1] ) & 0xff] & 0x000000ff) + ^ round_key->v32[0]; + + tmp1 = (U4[(state->v32[1] >> 24)] & 0xff000000) + ^ (U4[(state->v32[0] >> 16) & 0xff] & 0x00ff0000) + ^ (U4[(state->v32[3] >> 8) & 0xff] & 0x0000ff00) + ^ (U4[(state->v32[2] ) & 0xff] & 0x000000ff) + ^ round_key->v32[1]; + + tmp2 = (U4[(state->v32[2] >> 24)] & 0xff000000) + ^ (U4[(state->v32[1] >> 16) & 0xff] & 0x00ff0000) + ^ (U4[(state->v32[0] >> 8) & 0xff] & 0x0000ff00) + ^ (U4[(state->v32[3] ) & 0xff] & 0x000000ff) + ^ round_key->v32[2]; + + tmp3 = (U4[(state->v32[3] >> 24)] & 0xff000000) + ^ (U4[(state->v32[2] >> 16) & 0xff] & 0x00ff0000) + ^ (U4[(state->v32[1] >> 8) & 0xff] & 0x0000ff00) + ^ (U4[(state->v32[0] ) & 0xff] & 0x000000ff) + ^ round_key->v32[3]; + + state->v32[0] = tmp0; + state->v32[1] = tmp1; + state->v32[2] = tmp2; + state->v32[3] = tmp3; + +} + +#elif CPU_16 /* assume 16-bit word size on processor */ + +static inline void +aes_round(v128_t *state, const v128_t *round_key) { + uint32_t column0, column1, column2, column3; + uint16_t c + /* compute the columns of the output square in terms of the octets + of state, using the tables T0, T1, T2, T3 */ + + column0 = T0[state->v8[0]] ^ T1[state->v8[5]] + ^ T2[state->v8[10]] ^ T3[state->v8[15]]; + + column1 = T0[state->v8[4]] ^ T1[state->v8[9]] + ^ T2[state->v8[14]] ^ T3[state->v8[3]]; + + column2 = T0[state->v8[8]] ^ T1[state->v8[13]] + ^ T2[state->v8[2]] ^ T3[state->v8[7]]; + + column3 = T0[state->v8[12]] ^ T1[state->v8[1]] + ^ T2[state->v8[6]] ^ T3[state->v8[11]]; + + state->v32[0] = column0 ^ round_key->v32[0]; + state->v32[1] = column1 ^ round_key->v32[1]; + state->v32[2] = column2 ^ round_key->v32[2]; + state->v32[3] = column3 ^ round_key->v32[3]; + +} + + +static inline void +aes_inv_round(v128_t *state, const v128_t *round_key) { + uint32_t column0, column1, column2, column3; + + /* compute the columns of the output square in terms of the octets + of state, using the tables U0, U1, U2, U3 */ + + column0 = U0[state->v8[0]] ^ U1[state->v8[5]] + ^ U2[state->v8[10]] ^ U3[state->v8[15]]; + + column1 = U0[state->v8[4]] ^ U1[state->v8[9]] + ^ U2[state->v8[14]] ^ U3[state->v8[3]]; + + column2 = U0[state->v8[8]] ^ U1[state->v8[13]] + ^ U2[state->v8[2]] ^ U3[state->v8[7]]; + + column3 = U0[state->v8[12]] ^ U1[state->v8[1]] + ^ U2[state->v8[6]] ^ U3[state->v8[11]]; + + state->v32[0] = column0 ^ round_key->v32[0]; + state->v32[1] = column1 ^ round_key->v32[1]; + state->v32[2] = column2 ^ round_key->v32[2]; + state->v32[3] = column3 ^ round_key->v32[3]; + +} + +static inline void +aes_final_round(v128_t *state, const v128_t *round_key) { + uint8_t tmp; + + /* byte substitutions and row shifts */ + /* first row - no shift */ + state->v8[0] = aes_sbox[state->v8[0]]; + state->v8[4] = aes_sbox[state->v8[4]]; + state->v8[8] = aes_sbox[state->v8[8]]; + state->v8[12] = aes_sbox[state->v8[12]]; + + /* second row - shift one left */ + tmp = aes_sbox[state->v8[1]]; + state->v8[1] = aes_sbox[state->v8[5]]; + state->v8[5] = aes_sbox[state->v8[9]]; + state->v8[9] = aes_sbox[state->v8[13]]; + state->v8[13] = tmp; + + /* third row - shift two left */ + tmp = aes_sbox[state->v8[10]]; + state->v8[10] = aes_sbox[state->v8[2]]; + state->v8[2] = tmp; + tmp = aes_sbox[state->v8[14]]; + state->v8[14] = aes_sbox[state->v8[6]]; + state->v8[6] = tmp; + + /* fourth row - shift three left */ + tmp = aes_sbox[state->v8[15]]; + state->v8[15] = aes_sbox[state->v8[11]]; + state->v8[11] = aes_sbox[state->v8[7]]; + state->v8[7] = aes_sbox[state->v8[3]]; + state->v8[3] = tmp; + + v128_xor_eq(state, round_key); +} + +static inline void +aes_inv_final_round(v128_t *state, const v128_t *round_key) { + uint8_t tmp; + + /* byte substitutions and row shifts */ + /* first row - no shift */ + state->v8[0] = aes_inv_sbox[state->v8[0]]; + state->v8[4] = aes_inv_sbox[state->v8[4]]; + state->v8[8] = aes_inv_sbox[state->v8[8]]; + state->v8[12] = aes_inv_sbox[state->v8[12]]; + + /* second row - shift one left */ + tmp = aes_inv_sbox[state->v8[1]]; + state->v8[1] = aes_inv_sbox[state->v8[5]]; + state->v8[5] = aes_inv_sbox[state->v8[9]]; + state->v8[9] = aes_inv_sbox[state->v8[13]]; + state->v8[13] = tmp; + + /* third row - shift two left */ + tmp = aes_inv_sbox[state->v8[10]]; + state->v8[10] = aes_inv_sbox[state->v8[2]]; + state->v8[2] = tmp; + tmp = aes_inv_sbox[state->v8[14]]; + state->v8[14] = aes_inv_sbox[state->v8[6]]; + state->v8[6] = tmp; + + /* fourth row - shift three left */ + tmp = aes_inv_sbox[state->v8[15]]; + state->v8[15] = aes_inv_sbox[state->v8[11]]; + state->v8[11] = aes_inv_sbox[state->v8[7]]; + state->v8[7] = aes_inv_sbox[state->v8[3]]; + state->v8[3] = tmp; + + v128_xor_eq(state, round_key); +} + +#endif /* CPU type */ + + +void +aes_encrypt(v128_t *plaintext, const aes_expanded_key_t exp_key) { + + /* add in the subkey */ + v128_xor_eq(plaintext, exp_key + 0); + + /* now do nine rounds */ + aes_round(plaintext, exp_key + 1); + aes_round(plaintext, exp_key + 2); + aes_round(plaintext, exp_key + 3); + aes_round(plaintext, exp_key + 4); + aes_round(plaintext, exp_key + 5); + aes_round(plaintext, exp_key + 6); + aes_round(plaintext, exp_key + 7); + aes_round(plaintext, exp_key + 8); + aes_round(plaintext, exp_key + 9); + /* the last round is different */ + + aes_final_round(plaintext, exp_key + 10); +} + +void +aes_decrypt(v128_t *plaintext, const aes_expanded_key_t exp_key) { + + /* add in the subkey */ + v128_xor_eq(plaintext, exp_key + 0); + + /* now do nine rounds */ + aes_inv_round(plaintext, exp_key + 1); + aes_inv_round(plaintext, exp_key + 2); + aes_inv_round(plaintext, exp_key + 3); + aes_inv_round(plaintext, exp_key + 4); + aes_inv_round(plaintext, exp_key + 5); + aes_inv_round(plaintext, exp_key + 6); + aes_inv_round(plaintext, exp_key + 7); + aes_inv_round(plaintext, exp_key + 8); + aes_inv_round(plaintext, exp_key + 9); + /* the last round is different */ + aes_inv_final_round(plaintext, exp_key + 10); +} diff --git a/libs/srtp/crypto/cipher/aes_cbc.c b/libs/srtp/crypto/cipher/aes_cbc.c new file mode 100644 index 0000000000..fbfce629e6 --- /dev/null +++ b/libs/srtp/crypto/cipher/aes_cbc.c @@ -0,0 +1,444 @@ +/* + * aes_cbc.c + * + * AES Cipher Block Chaining Mode + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#include "aes_cbc.h" +#include "alloc.h" + +debug_module_t mod_aes_cbc = { + 0, /* debugging is off by default */ + "aes cbc" /* printable module name */ +}; + + + +err_status_t +aes_cbc_alloc(cipher_t **c, int key_len) { + extern cipher_type_t aes_cbc; + uint8_t *pointer; + int tmp; + + debug_print(mod_aes_cbc, + "allocating cipher with key length %d", key_len); + + if (key_len != 16) + return err_status_bad_param; + + /* allocate memory a cipher of type aes_icm */ + tmp = (sizeof(aes_cbc_ctx_t) + sizeof(cipher_t)); + pointer = crypto_alloc(tmp); + if (pointer == NULL) + return err_status_alloc_fail; + + /* set pointers */ + *c = (cipher_t *)pointer; + (*c)->type = &aes_cbc; + (*c)->state = pointer + sizeof(cipher_t); + + /* increment ref_count */ + aes_cbc.ref_count++; + + /* set key size */ + (*c)->key_len = key_len; + + return err_status_ok; +} + +err_status_t +aes_cbc_dealloc(cipher_t *c) { + extern cipher_type_t aes_cbc; + + /* zeroize entire state*/ + octet_string_set_to_zero((uint8_t *)c, + sizeof(aes_cbc_ctx_t) + sizeof(cipher_t)); + + /* free memory */ + crypto_free(c); + + /* decrement ref_count */ + aes_cbc.ref_count--; + + return err_status_ok; +} + +err_status_t +aes_cbc_context_init(aes_cbc_ctx_t *c, const uint8_t *key, + cipher_direction_t dir) { + v128_t tmp_key; + + /* set tmp_key (for alignment) */ + v128_copy_octet_string(&tmp_key, key); + + debug_print(mod_aes_cbc, + "key: %s", v128_hex_string(&tmp_key)); + + /* expand key for the appropriate direction */ + switch (dir) { + case (direction_encrypt): + aes_expand_encryption_key(&tmp_key, c->expanded_key); + break; + case (direction_decrypt): + aes_expand_decryption_key(&tmp_key, c->expanded_key); + break; + default: + return err_status_bad_param; + } + + + return err_status_ok; +} + + +err_status_t +aes_cbc_set_iv(aes_cbc_ctx_t *c, void *iv) { + int i; +/* v128_t *input = iv; */ + uint8_t *input = iv; + + /* set state and 'previous' block to iv */ + for (i=0; i < 16; i++) + c->previous.v8[i] = c->state.v8[i] = input[i]; + + debug_print(mod_aes_cbc, "setting iv: %s", v128_hex_string(&c->state)); + + return err_status_ok; +} + +err_status_t +aes_cbc_encrypt(aes_cbc_ctx_t *c, + unsigned char *data, + unsigned int *bytes_in_data) { + int i; + unsigned char *input = data; /* pointer to data being read */ + unsigned char *output = data; /* pointer to data being written */ + int bytes_to_encr = *bytes_in_data; + + /* + * verify that we're 16-octet aligned + */ + if (*bytes_in_data & 0xf) + return err_status_bad_param; + + /* + * note that we assume that the initialization vector has already + * been set, e.g. by calling aes_cbc_set_iv() + */ + debug_print(mod_aes_cbc, "iv: %s", + v128_hex_string(&c->state)); + + /* + * loop over plaintext blocks, exoring state into plaintext then + * encrypting and writing to output + */ + while (bytes_to_encr > 0) { + + /* exor plaintext into state */ + for (i=0; i < 16; i++) + c->state.v8[i] ^= *input++; + + debug_print(mod_aes_cbc, "inblock: %s", + v128_hex_string(&c->state)); + + aes_encrypt(&c->state, c->expanded_key); + + debug_print(mod_aes_cbc, "outblock: %s", + v128_hex_string(&c->state)); + + /* copy ciphertext to output */ + for (i=0; i < 16; i++) + *output++ = c->state.v8[i]; + + bytes_to_encr -= 16; + } + + return err_status_ok; +} + +err_status_t +aes_cbc_decrypt(aes_cbc_ctx_t *c, + unsigned char *data, + unsigned int *bytes_in_data) { + int i; + v128_t state, previous; + unsigned char *input = data; /* pointer to data being read */ + unsigned char *output = data; /* pointer to data being written */ + int bytes_to_encr = *bytes_in_data; + uint8_t tmp; + + /* + * verify that we're 16-octet aligned + */ + if (*bytes_in_data & 0x0f) + return err_status_bad_param; + + /* set 'previous' block to iv*/ + for (i=0; i < 16; i++) { + previous.v8[i] = c->previous.v8[i]; + } + + debug_print(mod_aes_cbc, "iv: %s", + v128_hex_string(&previous)); + + /* + * loop over ciphertext blocks, decrypting then exoring with state + * then writing plaintext to output + */ + while (bytes_to_encr > 0) { + + /* set state to ciphertext input block */ + for (i=0; i < 16; i++) { + state.v8[i] = *input++; + } + + debug_print(mod_aes_cbc, "inblock: %s", + v128_hex_string(&state)); + + /* decrypt state */ + aes_decrypt(&state, c->expanded_key); + + debug_print(mod_aes_cbc, "outblock: %s", + v128_hex_string(&state)); + + /* + * exor previous ciphertext block out of plaintext, and write new + * plaintext block to output, while copying old ciphertext block + * to the 'previous' block + */ + for (i=0; i < 16; i++) { + tmp = *output; + *output++ = state.v8[i] ^ previous.v8[i]; + previous.v8[i] = tmp; + } + + bytes_to_encr -= 16; + } + + return err_status_ok; +} + + +err_status_t +aes_cbc_nist_encrypt(aes_cbc_ctx_t *c, + unsigned char *data, + unsigned int *bytes_in_data) { + int i; + unsigned char *pad_start; + int num_pad_bytes; + err_status_t status; + + /* + * determine the number of padding bytes that we need to add - + * this value is always between 1 and 16, inclusive. + */ + num_pad_bytes = 16 - (*bytes_in_data & 0xf); + pad_start = data; + pad_start += *bytes_in_data; + *pad_start++ = 0xa0; + for (i=0; i < num_pad_bytes; i++) + *pad_start++ = 0x00; + + /* + * increment the data size + */ + *bytes_in_data += num_pad_bytes; + + /* + * now cbc encrypt the padded data + */ + status = aes_cbc_encrypt(c, data, bytes_in_data); + if (status) + return status; + + return err_status_ok; +} + + +err_status_t +aes_cbc_nist_decrypt(aes_cbc_ctx_t *c, + unsigned char *data, + unsigned int *bytes_in_data) { + unsigned char *pad_end; + int num_pad_bytes; + err_status_t status; + + /* + * cbc decrypt the padded data + */ + status = aes_cbc_decrypt(c, data, bytes_in_data); + if (status) + return status; + + /* + * determine the number of padding bytes in the decrypted plaintext + * - this value is always between 1 and 16, inclusive. + */ + num_pad_bytes = 1; + pad_end = data + (*bytes_in_data - 1); + while (*pad_end != 0xa0) { /* note: should check padding correctness */ + pad_end--; + num_pad_bytes++; + } + + /* decrement data size */ + *bytes_in_data -= num_pad_bytes; + + return err_status_ok; +} + + +char +aes_cbc_description[] = "aes cipher block chaining (cbc) mode"; + +/* + * Test case 0 is derived from FIPS 197 Appendix A; it uses an + * all-zero IV, so that the first block encryption matches the test + * case in that appendix. This property provides a check of the base + * AES encryption and decryption algorithms; if CBC fails on some + * particular platform, then you should print out AES intermediate + * data and compare with the detailed info provided in that appendix. + * + */ + + +uint8_t aes_cbc_test_case_0_key[16] = { + 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, + 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f +}; + +uint8_t aes_cbc_test_case_0_plaintext[64] = { + 0x00, 0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77, + 0x88, 0x99, 0xaa, 0xbb, 0xcc, 0xdd, 0xee, 0xff +}; + +uint8_t aes_cbc_test_case_0_ciphertext[80] = { + 0x69, 0xc4, 0xe0, 0xd8, 0x6a, 0x7b, 0x04, 0x30, + 0xd8, 0xcd, 0xb7, 0x80, 0x70, 0xb4, 0xc5, 0x5a, + 0x03, 0x35, 0xed, 0x27, 0x67, 0xf2, 0x6d, 0xf1, + 0x64, 0x83, 0x2e, 0x23, 0x44, 0x38, 0x70, 0x8b + +}; + +uint8_t aes_cbc_test_case_0_iv[16] = { + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 +}; + + +cipher_test_case_t aes_cbc_test_case_0 = { + 16, /* octets in key */ + aes_cbc_test_case_0_key, /* key */ + aes_cbc_test_case_0_iv, /* initialization vector */ + 16, /* octets in plaintext */ + aes_cbc_test_case_0_plaintext, /* plaintext */ + 32, /* octets in ciphertext */ + aes_cbc_test_case_0_ciphertext, /* ciphertext */ + NULL /* pointer to next testcase */ +}; + + +/* + * this test case is taken directly from Appendix F.2 of NIST Special + * Publication SP 800-38A + */ + +uint8_t aes_cbc_test_case_1_key[16] = { + 0x2b, 0x7e, 0x15, 0x16, 0x28, 0xae, 0xd2, 0xa6, + 0xab, 0xf7, 0x15, 0x88, 0x09, 0xcf, 0x4f, 0x3c, +}; + +uint8_t aes_cbc_test_case_1_plaintext[64] = { + 0x6b, 0xc1, 0xbe, 0xe2, 0x2e, 0x40, 0x9f, 0x96, + 0xe9, 0x3d, 0x7e, 0x11, 0x73, 0x93, 0x17, 0x2a, + 0xae, 0x2d, 0x8a, 0x57, 0x1e, 0x03, 0xac, 0x9c, + 0x9e, 0xb7, 0x6f, 0xac, 0x45, 0xaf, 0x8e, 0x51, + 0x30, 0xc8, 0x1c, 0x46, 0xa3, 0x5c, 0xe4, 0x11, + 0xe5, 0xfb, 0xc1, 0x19, 0x1a, 0x0a, 0x52, 0xef, + 0xf6, 0x9f, 0x24, 0x45, 0xdf, 0x4f, 0x9b, 0x17, + 0xad, 0x2b, 0x41, 0x7b, 0xe6, 0x6c, 0x37, 0x10 +}; + +uint8_t aes_cbc_test_case_1_ciphertext[80] = { + 0x76, 0x49, 0xab, 0xac, 0x81, 0x19, 0xb2, 0x46, + 0xce, 0xe9, 0x8e, 0x9b, 0x12, 0xe9, 0x19, 0x7d, + 0x50, 0x86, 0xcb, 0x9b, 0x50, 0x72, 0x19, 0xee, + 0x95, 0xdb, 0x11, 0x3a, 0x91, 0x76, 0x78, 0xb2, + 0x73, 0xbe, 0xd6, 0xb8, 0xe3, 0xc1, 0x74, 0x3b, + 0x71, 0x16, 0xe6, 0x9e, 0x22, 0x22, 0x95, 0x16, + 0x3f, 0xf1, 0xca, 0xa1, 0x68, 0x1f, 0xac, 0x09, + 0x12, 0x0e, 0xca, 0x30, 0x75, 0x86, 0xe1, 0xa7, + 0x39, 0x34, 0x07, 0x03, 0x36, 0xd0, 0x77, 0x99, + 0xe0, 0xc4, 0x2f, 0xdd, 0xa8, 0xdf, 0x4c, 0xa3 +}; + +uint8_t aes_cbc_test_case_1_iv[16] = { + 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, + 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f +}; + +cipher_test_case_t aes_cbc_test_case_1 = { + 16, /* octets in key */ + aes_cbc_test_case_1_key, /* key */ + aes_cbc_test_case_1_iv, /* initialization vector */ + 64, /* octets in plaintext */ + aes_cbc_test_case_1_plaintext, /* plaintext */ + 80, /* octets in ciphertext */ + aes_cbc_test_case_1_ciphertext, /* ciphertext */ + &aes_cbc_test_case_0 /* pointer to next testcase */ +}; + +cipher_type_t aes_cbc = { + (cipher_alloc_func_t) aes_cbc_alloc, + (cipher_dealloc_func_t) aes_cbc_dealloc, + (cipher_init_func_t) aes_cbc_context_init, + (cipher_encrypt_func_t) aes_cbc_nist_encrypt, + (cipher_decrypt_func_t) aes_cbc_nist_decrypt, + (cipher_set_iv_func_t) aes_cbc_set_iv, + (char *) aes_cbc_description, + (int) 0, /* instance count */ + (cipher_test_case_t *) &aes_cbc_test_case_0, + (debug_module_t *) &mod_aes_cbc +}; + + diff --git a/libs/srtp/crypto/cipher/aes_icm.c b/libs/srtp/crypto/cipher/aes_icm.c new file mode 100644 index 0000000000..0c6175544c --- /dev/null +++ b/libs/srtp/crypto/cipher/aes_icm.c @@ -0,0 +1,509 @@ +/* + * aes_icm.c + * + * AES Integer Counter Mode + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#define ALIGN_32 0 + +#include "aes_icm.h" +#include "alloc.h" + + +debug_module_t mod_aes_icm = { + 0, /* debugging is off by default */ + "aes icm" /* printable module name */ +}; + +/* + * integer counter mode works as follows: + * + * 16 bits + * <-----> + * +------+------+------+------+------+------+------+------+ + * | nonce | pakcet index | ctr |---+ + * +------+------+------+------+------+------+------+------+ | + * | + * +------+------+------+------+------+------+------+------+ v + * | salt |000000|->(+) + * +------+------+------+------+------+------+------+------+ | + * | + * +---------+ + * | encrypt | + * +---------+ + * | + * +------+------+------+------+------+------+------+------+ | + * | keystream block |<--+ + * +------+------+------+------+------+------+------+------+ + * + * All fields are big-endian + * + * ctr is the block counter, which increments from zero for + * each packet (16 bits wide) + * + * packet index is distinct for each packet (48 bits wide) + * + * nonce can be distinct across many uses of the same key, or + * can be a fixed value per key, or can be per-packet randomness + * (64 bits) + * + */ + +err_status_t +aes_icm_alloc_ismacryp(cipher_t **c, int key_len, int forIsmacryp) { + extern cipher_type_t aes_icm; + uint8_t *pointer; + int tmp; + + debug_print(mod_aes_icm, + "allocating cipher with key length %d", key_len); + + // Ismacryp, for example, uses 16 byte key + 8 byte + // salt so this function is called with key_len = 24. + // The check for key_len = 30 does not apply. Our usage + // of aes functions with key_len = values other than 30 + // has not broken anything. Don't know what would be the + // effect of skipping this check for srtp in general. + if (!forIsmacryp && key_len != 30) + return err_status_bad_param; + + /* allocate memory a cipher of type aes_icm */ + tmp = (sizeof(aes_icm_ctx_t) + sizeof(cipher_t)); + pointer = crypto_alloc(tmp); + if (pointer == NULL) + return err_status_alloc_fail; + + /* set pointers */ + *c = (cipher_t *)pointer; + (*c)->type = &aes_icm; + (*c)->state = pointer + sizeof(cipher_t); + + /* increment ref_count */ + aes_icm.ref_count++; + + /* set key size */ + (*c)->key_len = key_len; + + return err_status_ok; +} + +err_status_t aes_icm_alloc(cipher_t **c, int key_len, int forIsmacryp) { + return aes_icm_alloc_ismacryp(c, key_len, 0); +} + +err_status_t +aes_icm_dealloc(cipher_t *c) { + extern cipher_type_t aes_icm; + + /* zeroize entire state*/ + octet_string_set_to_zero((uint8_t *)c, + sizeof(aes_icm_ctx_t) + sizeof(cipher_t)); + + /* free memory */ + crypto_free(c); + + /* decrement ref_count */ + aes_icm.ref_count--; + + return err_status_ok; +} + + +/* + * aes_icm_context_init(...) initializes the aes_icm_context + * using the value in key[]. + * + * the key is the secret key + * + * the salt is unpredictable (but not necessarily secret) data which + * randomizes the starting point in the keystream + */ + +err_status_t +aes_icm_context_init(aes_icm_ctx_t *c, const uint8_t *key) { + v128_t tmp_key; + + /* set counter and initial values to 'offset' value */ + /* FIX!!! this assumes the salt is at key + 16, and thus that the */ + /* FIX!!! cipher key length is 16! Also note this copies past the + end of the 'key' array by 2 bytes! */ + v128_copy_octet_string(&c->counter, key + 16); + v128_copy_octet_string(&c->offset, key + 16); + + /* force last two octets of the offset to zero (for srtp compatibility) */ + c->offset.v8[14] = c->offset.v8[15] = 0; + c->counter.v8[14] = c->counter.v8[15] = 0; + + /* set tmp_key (for alignment) */ + v128_copy_octet_string(&tmp_key, key); + + debug_print(mod_aes_icm, + "key: %s", v128_hex_string(&tmp_key)); + debug_print(mod_aes_icm, + "offset: %s", v128_hex_string(&c->offset)); + + /* expand key */ + aes_expand_encryption_key(&tmp_key, c->expanded_key); + + /* indicate that the keystream_buffer is empty */ + c->bytes_in_buffer = 0; + + return err_status_ok; +} + +/* + * aes_icm_set_octet(c, i) sets the counter of the context which it is + * passed so that the next octet of keystream that will be generated + * is the ith octet + */ + +err_status_t +aes_icm_set_octet(aes_icm_ctx_t *c, + uint64_t octet_num) { + +#ifdef NO_64BIT_MATH + int tail_num = low32(octet_num) & 0x0f; + /* 64-bit right-shift 4 */ + uint64_t block_num = make64(high32(octet_num) >> 4, + ((high32(octet_num) & 0x0f)<<(32-4)) | + (low32(octet_num) >> 4)); +#else + int tail_num = octet_num % 16; + uint64_t block_num = octet_num / 16; +#endif + + + /* set counter value */ + /* FIX - There's no way this is correct */ + c->counter.v64[0] = c->offset.v64[0]; +#ifdef NO_64BIT_MATH + c->counter.v64[0] = make64(high32(c->offset.v64[0]) ^ high32(block_num), + low32(c->offset.v64[0]) ^ low32(block_num)); +#else + c->counter.v64[0] = c->offset.v64[0] ^ block_num; +#endif + + debug_print(mod_aes_icm, + "set_octet: %s", v128_hex_string(&c->counter)); + + /* fill keystream buffer, if needed */ + if (tail_num) { + v128_copy(&c->keystream_buffer, &c->counter); + aes_encrypt(&c->keystream_buffer, c->expanded_key); + c->bytes_in_buffer = sizeof(v128_t); + + debug_print(mod_aes_icm, "counter: %s", + v128_hex_string(&c->counter)); + debug_print(mod_aes_icm, "ciphertext: %s", + v128_hex_string(&c->keystream_buffer)); + + /* indicate number of bytes in keystream_buffer */ + c->bytes_in_buffer = sizeof(v128_t) - tail_num; + + } else { + + /* indicate that keystream_buffer is empty */ + c->bytes_in_buffer = 0; + } + + return err_status_ok; +} + +/* + * aes_icm_set_iv(c, iv) sets the counter value to the exor of iv with + * the offset + */ + +err_status_t +aes_icm_set_iv(aes_icm_ctx_t *c, void *iv) { + v128_t *nonce = iv; + + debug_print(mod_aes_icm, + "setting iv: %s", v128_hex_string(nonce)); + + v128_xor(&c->counter, &c->offset, nonce); + + debug_print(mod_aes_icm, + "set_counter: %s", v128_hex_string(&c->counter)); + + /* indicate that the keystream_buffer is empty */ + c->bytes_in_buffer = 0; + + return err_status_ok; +} + + + +/* + * aes_icm_advance(...) refills the keystream_buffer and + * advances the block index of the sicm_context forward by one + * + * this is an internal, hopefully inlined function + */ + +inline void +aes_icm_advance_ismacryp(aes_icm_ctx_t *c, uint8_t forIsmacryp) { + /* fill buffer with new keystream */ + v128_copy(&c->keystream_buffer, &c->counter); + aes_encrypt(&c->keystream_buffer, c->expanded_key); + c->bytes_in_buffer = sizeof(v128_t); + + debug_print(mod_aes_icm, "counter: %s", + v128_hex_string(&c->counter)); + debug_print(mod_aes_icm, "ciphertext: %s", + v128_hex_string(&c->keystream_buffer)); + + /* clock counter forward */ + + if (forIsmacryp) { + uint32_t temp; + //alex's clock counter forward + temp = ntohl(c->counter.v32[3]); + c->counter.v32[3] = htonl(++temp); + } else { + if (!++(c->counter.v8[15])) + ++(c->counter.v8[14]); + } +} + +inline void aes_icm_advance(aes_icm_ctx_t *c) { + aes_icm_advance_ismacryp(c, 0); +} + + +/*e + * icm_encrypt deals with the following cases: + * + * bytes_to_encr < bytes_in_buffer + * - add keystream into data + * + * bytes_to_encr > bytes_in_buffer + * - add keystream into data until keystream_buffer is depleted + * - loop over blocks, filling keystream_buffer and then + * adding keystream into data + * - fill buffer then add in remaining (< 16) bytes of keystream + */ + +err_status_t +aes_icm_encrypt_ismacryp(aes_icm_ctx_t *c, + unsigned char *buf, unsigned int *enc_len, + int forIsmacryp) { + unsigned int bytes_to_encr = *enc_len; + int i; + uint32_t *b; + + /* check that there's enough segment left but not for ismacryp*/ + if (!forIsmacryp && (bytes_to_encr + htons(c->counter.v16[7])) > 0xffff) + return err_status_terminus; + + debug_print(mod_aes_icm, "block index: %d", + htons(c->counter.v16[7])); + if (bytes_to_encr <= c->bytes_in_buffer) { + + /* deal with odd case of small bytes_to_encr */ + for (i = (sizeof(v128_t) - c->bytes_in_buffer); + i < (sizeof(v128_t) - c->bytes_in_buffer + bytes_to_encr); i++) + { + *buf++ ^= c->keystream_buffer.v8[i]; + } + + c->bytes_in_buffer -= bytes_to_encr; + + /* return now to avoid the main loop */ + return err_status_ok; + + } else { + + /* encrypt bytes until the remaining data is 16-byte aligned */ + for (i=(sizeof(v128_t) - c->bytes_in_buffer); i < sizeof(v128_t); i++) + *buf++ ^= c->keystream_buffer.v8[i]; + + bytes_to_encr -= c->bytes_in_buffer; + c->bytes_in_buffer = 0; + + } + + /* now loop over entire 16-byte blocks of keystream */ + for (i=0; i < (bytes_to_encr/sizeof(v128_t)); i++) { + + /* fill buffer with new keystream */ + aes_icm_advance_ismacryp(c, forIsmacryp); + + /* + * add keystream into the data buffer (this would be a lot faster + * if we could assume 32-bit alignment!) + */ + +#if ALIGN_32 + b = (uint32_t *)buf; + *b++ ^= c->keystream_buffer.v32[0]; + *b++ ^= c->keystream_buffer.v32[1]; + *b++ ^= c->keystream_buffer.v32[2]; + *b++ ^= c->keystream_buffer.v32[3]; + buf = (uint8_t *)b; +#else + if ((((unsigned long) buf) & 0x03) != 0) { + *buf++ ^= c->keystream_buffer.v8[0]; + *buf++ ^= c->keystream_buffer.v8[1]; + *buf++ ^= c->keystream_buffer.v8[2]; + *buf++ ^= c->keystream_buffer.v8[3]; + *buf++ ^= c->keystream_buffer.v8[4]; + *buf++ ^= c->keystream_buffer.v8[5]; + *buf++ ^= c->keystream_buffer.v8[6]; + *buf++ ^= c->keystream_buffer.v8[7]; + *buf++ ^= c->keystream_buffer.v8[8]; + *buf++ ^= c->keystream_buffer.v8[9]; + *buf++ ^= c->keystream_buffer.v8[10]; + *buf++ ^= c->keystream_buffer.v8[11]; + *buf++ ^= c->keystream_buffer.v8[12]; + *buf++ ^= c->keystream_buffer.v8[13]; + *buf++ ^= c->keystream_buffer.v8[14]; + *buf++ ^= c->keystream_buffer.v8[15]; + } else { + b = (uint32_t *)buf; + *b++ ^= c->keystream_buffer.v32[0]; + *b++ ^= c->keystream_buffer.v32[1]; + *b++ ^= c->keystream_buffer.v32[2]; + *b++ ^= c->keystream_buffer.v32[3]; + buf = (uint8_t *)b; + } +#endif /* #if ALIGN_32 */ + + } + + /* if there is a tail end of the data, process it */ + if ((bytes_to_encr & 0xf) != 0) { + + /* fill buffer with new keystream */ + aes_icm_advance_ismacryp(c, forIsmacryp); + + for (i=0; i < (bytes_to_encr & 0xf); i++) + *buf++ ^= c->keystream_buffer.v8[i]; + + /* reset the keystream buffer size to right value */ + c->bytes_in_buffer = sizeof(v128_t) - i; + } else { + + /* no tail, so just reset the keystream buffer size to zero */ + c->bytes_in_buffer = 0; + + } + + return err_status_ok; +} + +err_status_t +aes_icm_encrypt(aes_icm_ctx_t *c, unsigned char *buf, unsigned int *enc_len) { + return aes_icm_encrypt_ismacryp(c, buf, enc_len, 0); +} + +err_status_t +aes_icm_output(aes_icm_ctx_t *c, uint8_t *buffer, int num_octets_to_output) { + unsigned int len = num_octets_to_output; + + /* zeroize the buffer */ + octet_string_set_to_zero(buffer, num_octets_to_output); + + /* exor keystream into buffer */ + return aes_icm_encrypt(c, buffer, &len); +} + + +char +aes_icm_description[] = "aes integer counter mode"; + +uint8_t aes_icm_test_case_0_key[30] = { + 0x2b, 0x7e, 0x15, 0x16, 0x28, 0xae, 0xd2, 0xa6, + 0xab, 0xf7, 0x15, 0x88, 0x09, 0xcf, 0x4f, 0x3c, + 0xf0, 0xf1, 0xf2, 0xf3, 0xf4, 0xf5, 0xf6, 0xf7, + 0xf8, 0xf9, 0xfa, 0xfb, 0xfc, 0xfd +}; + +uint8_t aes_icm_test_case_0_nonce[16] = { + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 +}; + +uint8_t aes_icm_test_case_0_plaintext[32] = { + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +}; + +uint8_t aes_icm_test_case_0_ciphertext[32] = { + 0xe0, 0x3e, 0xad, 0x09, 0x35, 0xc9, 0x5e, 0x80, + 0xe1, 0x66, 0xb1, 0x6d, 0xd9, 0x2b, 0x4e, 0xb4, + 0xd2, 0x35, 0x13, 0x16, 0x2b, 0x02, 0xd0, 0xf7, + 0x2a, 0x43, 0xa2, 0xfe, 0x4a, 0x5f, 0x97, 0xab +}; + +cipher_test_case_t aes_icm_test_case_0 = { + 30, /* octets in key */ + aes_icm_test_case_0_key, /* key */ + aes_icm_test_case_0_nonce, /* packet index */ + 32, /* octets in plaintext */ + aes_icm_test_case_0_plaintext, /* plaintext */ + 32, /* octets in ciphertext */ + aes_icm_test_case_0_ciphertext, /* ciphertext */ + NULL /* pointer to next testcase */ +}; + + +/* + * note: the encrypt function is identical to the decrypt function + */ + +cipher_type_t aes_icm = { + (cipher_alloc_func_t) aes_icm_alloc, + (cipher_dealloc_func_t) aes_icm_dealloc, + (cipher_init_func_t) aes_icm_context_init, + (cipher_encrypt_func_t) aes_icm_encrypt, + (cipher_decrypt_func_t) aes_icm_encrypt, + (cipher_set_iv_func_t) aes_icm_set_iv, + (char *) aes_icm_description, + (int) 0, /* instance count */ + (cipher_test_case_t *) &aes_icm_test_case_0, + (debug_module_t *) &mod_aes_icm +}; + diff --git a/libs/srtp/crypto/cipher/cipher.c b/libs/srtp/crypto/cipher/cipher.c new file mode 100644 index 0000000000..d81a8f0f1b --- /dev/null +++ b/libs/srtp/crypto/cipher/cipher.c @@ -0,0 +1,407 @@ +/* + * cipher.c + * + * cipher meta-functions + * + * David A. McGrew + * Cisco Systems, Inc. + * + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include "cipher.h" +#include "rand_source.h" /* used in invertibiltiy tests */ +#include "alloc.h" /* for crypto_alloc(), crypto_free() */ + +debug_module_t mod_cipher = { + 0, /* debugging is off by default */ + "cipher" /* printable module name */ +}; + +err_status_t +cipher_output(cipher_t *c, uint8_t *buffer, int num_octets_to_output) { + + /* zeroize the buffer */ + octet_string_set_to_zero(buffer, num_octets_to_output); + + /* exor keystream into buffer */ + return cipher_encrypt(c, buffer, (unsigned int *) &num_octets_to_output); +} + +/* some bookkeeping functions */ + +int +cipher_get_key_length(const cipher_t *c) { + return c->key_len; +} + +/* + * cipher_type_self_test(ct) tests a cipher of type ct against test cases + * provided in an array of values of key, salt, xtd_seq_num_t, + * plaintext, and ciphertext that is known to be good + */ + +#define SELF_TEST_BUF_OCTETS 128 +#define NUM_RAND_TESTS 128 +#define MAX_KEY_LEN 64 + +err_status_t +cipher_type_self_test(const cipher_type_t *ct) { + const cipher_test_case_t *test_case = ct->test_data; + cipher_t *c; + err_status_t status; + uint8_t buffer[SELF_TEST_BUF_OCTETS]; + uint8_t buffer2[SELF_TEST_BUF_OCTETS]; + unsigned int len; + int i, j, case_num = 0; + + debug_print(mod_cipher, "running self-test for cipher %s", + ct->description); + + /* + * check to make sure that we have at least one test case, and + * return an error if we don't - we need to be paranoid here + */ + if (test_case == NULL) + return err_status_cant_check; + + /* + * loop over all test cases, perform known-answer tests of both the + * encryption and decryption functions + */ + while (test_case != NULL) { + + /* allocate cipher */ + status = cipher_type_alloc(ct, &c, test_case->key_length_octets); + if (status) + return status; + + /* + * test the encrypt function + */ + debug_print(mod_cipher, "testing encryption", NULL); + + /* initialize cipher */ + status = cipher_init(c, test_case->key, direction_encrypt); + if (status) { + cipher_dealloc(c); + return status; + } + + /* copy plaintext into test buffer */ + if (test_case->ciphertext_length_octets > SELF_TEST_BUF_OCTETS) { + cipher_dealloc(c); + return err_status_bad_param; + } + for (i=0; i < test_case->plaintext_length_octets; i++) + buffer[i] = test_case->plaintext[i]; + + debug_print(mod_cipher, "plaintext: %s", + octet_string_hex_string(buffer, + test_case->plaintext_length_octets)); + + /* set the initialization vector */ + status = cipher_set_iv(c, test_case->idx); + if (status) { + cipher_dealloc(c); + return status; + } + + /* encrypt */ + len = test_case->plaintext_length_octets; + status = cipher_encrypt(c, buffer, &len); + if (status) { + cipher_dealloc(c); + return status; + } + + debug_print(mod_cipher, "ciphertext: %s", + octet_string_hex_string(buffer, + test_case->ciphertext_length_octets)); + + /* compare the resulting ciphertext with that in the test case */ + if (len != test_case->ciphertext_length_octets) + return err_status_algo_fail; + status = err_status_ok; + for (i=0; i < test_case->ciphertext_length_octets; i++) + if (buffer[i] != test_case->ciphertext[i]) { + status = err_status_algo_fail; + debug_print(mod_cipher, "test case %d failed", case_num); + debug_print(mod_cipher, "(failure at byte %d)", i); + break; + } + if (status) { + + debug_print(mod_cipher, "c computed: %s", + octet_string_hex_string(buffer, + 2*test_case->plaintext_length_octets)); + debug_print(mod_cipher, "c expected: %s", + octet_string_hex_string(test_case->ciphertext, + 2*test_case->plaintext_length_octets)); + + cipher_dealloc(c); + return err_status_algo_fail; + } + + /* + * test the decrypt function + */ + debug_print(mod_cipher, "testing decryption", NULL); + + /* re-initialize cipher for decryption */ + status = cipher_init(c, test_case->key, direction_decrypt); + if (status) { + cipher_dealloc(c); + return status; + } + + /* copy ciphertext into test buffer */ + if (test_case->ciphertext_length_octets > SELF_TEST_BUF_OCTETS) { + cipher_dealloc(c); + return err_status_bad_param; + } + for (i=0; i < test_case->ciphertext_length_octets; i++) + buffer[i] = test_case->ciphertext[i]; + + debug_print(mod_cipher, "ciphertext: %s", + octet_string_hex_string(buffer, + test_case->plaintext_length_octets)); + + /* set the initialization vector */ + status = cipher_set_iv(c, test_case->idx); + if (status) { + cipher_dealloc(c); + return status; + } + + /* decrypt */ + len = test_case->ciphertext_length_octets; + status = cipher_decrypt(c, buffer, &len); + if (status) { + cipher_dealloc(c); + return status; + } + + debug_print(mod_cipher, "plaintext: %s", + octet_string_hex_string(buffer, + test_case->plaintext_length_octets)); + + /* compare the resulting plaintext with that in the test case */ + if (len != test_case->plaintext_length_octets) + return err_status_algo_fail; + status = err_status_ok; + for (i=0; i < test_case->plaintext_length_octets; i++) + if (buffer[i] != test_case->plaintext[i]) { + status = err_status_algo_fail; + debug_print(mod_cipher, "test case %d failed", case_num); + debug_print(mod_cipher, "(failure at byte %d)", i); + } + if (status) { + + debug_print(mod_cipher, "p computed: %s", + octet_string_hex_string(buffer, + 2*test_case->plaintext_length_octets)); + debug_print(mod_cipher, "p expected: %s", + octet_string_hex_string(test_case->plaintext, + 2*test_case->plaintext_length_octets)); + + cipher_dealloc(c); + return err_status_algo_fail; + } + + /* deallocate the cipher */ + status = cipher_dealloc(c); + if (status) + return status; + + /* + * the cipher passed the test case, so move on to the next test + * case in the list; if NULL, we'l proceed to the next test + */ + test_case = test_case->next_test_case; + ++case_num; + } + + /* now run some random invertibility tests */ + + /* allocate cipher, using paramaters from the first test case */ + test_case = ct->test_data; + status = cipher_type_alloc(ct, &c, test_case->key_length_octets); + if (status) + return status; + + rand_source_init(); + + for (j=0; j < NUM_RAND_TESTS; j++) { + unsigned length; + int plaintext_len; + uint8_t key[MAX_KEY_LEN]; + uint8_t iv[MAX_KEY_LEN]; + + /* choose a length at random (leaving room for IV and padding) */ + length = rand() % (SELF_TEST_BUF_OCTETS - 64); + debug_print(mod_cipher, "random plaintext length %d\n", length); + status = rand_source_get_octet_string(buffer, length); + if (status) return status; + + debug_print(mod_cipher, "plaintext: %s", + octet_string_hex_string(buffer, length)); + + /* copy plaintext into second buffer */ + for (i=0; i < length; i++) + buffer2[i] = buffer[i]; + + /* choose a key at random */ + if (test_case->key_length_octets > MAX_KEY_LEN) + return err_status_cant_check; + status = rand_source_get_octet_string(key, test_case->key_length_octets); + if (status) return status; + + /* chose a random initialization vector */ + status = rand_source_get_octet_string(iv, MAX_KEY_LEN); + if (status) return status; + + /* initialize cipher */ + status = cipher_init(c, key, direction_encrypt); + if (status) { + cipher_dealloc(c); + return status; + } + + /* set initialization vector */ + status = cipher_set_iv(c, test_case->idx); + if (status) { + cipher_dealloc(c); + return status; + } + + /* encrypt buffer with cipher */ + plaintext_len = length; + status = cipher_encrypt(c, buffer, &length); + if (status) { + cipher_dealloc(c); + return status; + } + debug_print(mod_cipher, "ciphertext: %s", + octet_string_hex_string(buffer, length)); + + /* + * re-initialize cipher for decryption, re-set the iv, then + * decrypt the ciphertext + */ + status = cipher_init(c, key, direction_decrypt); + if (status) { + cipher_dealloc(c); + return status; + } + status = cipher_set_iv(c, test_case->idx); + if (status) { + cipher_dealloc(c); + return status; + } + status = cipher_decrypt(c, buffer, &length); + if (status) { + cipher_dealloc(c); + return status; + } + + debug_print(mod_cipher, "plaintext[2]: %s", + octet_string_hex_string(buffer, length)); + + /* compare the resulting plaintext with the original one */ + if (length != plaintext_len) + return err_status_algo_fail; + status = err_status_ok; + for (i=0; i < plaintext_len; i++) + if (buffer[i] != buffer2[i]) { + status = err_status_algo_fail; + debug_print(mod_cipher, "random test case %d failed", case_num); + debug_print(mod_cipher, "(failure at byte %d)", i); + } + if (status) { + cipher_dealloc(c); + return err_status_algo_fail; + } + + } + + return err_status_ok; +} + + +/* + * cipher_bits_per_second(c, l, t) computes (an estimate of) the + * number of bits that a cipher implementation can encrypt in a second + * + * c is a cipher (which MUST be allocated and initialized already), l + * is the length in octets of the test data to be encrypted, and t is + * the number of trials + * + * if an error is encountered, the value 0 is returned + */ + +uint64_t +cipher_bits_per_second(cipher_t *c, int octets_in_buffer, int num_trials) { + int i; + v128_t nonce; + clock_t timer; + unsigned char *enc_buf; + unsigned int len = octets_in_buffer; + + enc_buf = crypto_alloc(octets_in_buffer); + if (enc_buf == NULL) + return 0; /* indicate bad parameters by returning null */ + + /* time repeated trials */ + v128_set_to_zero(&nonce); + timer = clock(); + for(i=0; i < num_trials; i++, nonce.v32[3] = i) { + cipher_set_iv(c, &nonce); + cipher_encrypt(c, enc_buf, &len); + } + timer = clock() - timer; + + crypto_free(enc_buf); + + if (timer == 0) { + /* Too fast! */ + return 0; + } + + return (uint64_t)CLOCKS_PER_SEC * num_trials * 8 * octets_in_buffer / timer; +} diff --git a/libs/srtp/crypto/cipher/null_cipher.c b/libs/srtp/crypto/cipher/null_cipher.c new file mode 100644 index 0000000000..812d679f4f --- /dev/null +++ b/libs/srtp/crypto/cipher/null_cipher.c @@ -0,0 +1,152 @@ +/* + * null_cipher.c + * + * A null cipher implementation. This cipher leaves the plaintext + * unchanged. + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include "datatypes.h" +#include "null_cipher.h" +#include "alloc.h" + +/* the null_cipher uses the cipher debug module */ + +extern debug_module_t mod_cipher; + +err_status_t +null_cipher_alloc(cipher_t **c, int key_len) { + extern cipher_type_t null_cipher; + uint8_t *pointer; + + debug_print(mod_cipher, + "allocating cipher with key length %d", key_len); + + /* allocate memory a cipher of type null_cipher */ + pointer = crypto_alloc(sizeof(null_cipher_ctx_t) + sizeof(cipher_t)); + if (pointer == NULL) + return err_status_alloc_fail; + + /* set pointers */ + *c = (cipher_t *)pointer; + (*c)->type = &null_cipher; + (*c)->state = pointer + sizeof(cipher_t); + + /* set key size */ + (*c)->key_len = key_len; + + /* increment ref_count */ + null_cipher.ref_count++; + + return err_status_ok; + +} + +err_status_t +null_cipher_dealloc(cipher_t *c) { + extern cipher_type_t null_cipher; + + /* zeroize entire state*/ + octet_string_set_to_zero((uint8_t *)c, + sizeof(null_cipher_ctx_t) + sizeof(cipher_t)); + + /* free memory of type null_cipher */ + crypto_free(c); + + /* decrement reference count */ + null_cipher.ref_count--; + + return err_status_ok; + +} + +err_status_t +null_cipher_init(null_cipher_ctx_t *ctx, const uint8_t *key) { + + debug_print(mod_cipher, "initializing null cipher", NULL); + + return err_status_ok; +} + +err_status_t +null_cipher_set_iv(null_cipher_ctx_t *c, void *iv) { + return err_status_ok; +} + +err_status_t +null_cipher_encrypt(null_cipher_ctx_t *c, + unsigned char *buf, unsigned int *bytes_to_encr) { + return err_status_ok; +} + +char +null_cipher_description[] = "null cipher"; + +cipher_test_case_t +null_cipher_test_0 = { + 0, /* octets in key */ + NULL, /* key */ + 0, /* packet index */ + 0, /* octets in plaintext */ + NULL, /* plaintext */ + 0, /* octets in plaintext */ + NULL, /* ciphertext */ + NULL /* pointer to next testcase */ +}; + + +/* + * note: the decrypt function is idential to the encrypt function + */ + +cipher_type_t null_cipher = { + (cipher_alloc_func_t) null_cipher_alloc, + (cipher_dealloc_func_t) null_cipher_dealloc, + (cipher_init_func_t) null_cipher_init, + (cipher_encrypt_func_t) null_cipher_encrypt, + (cipher_decrypt_func_t) null_cipher_encrypt, + (cipher_set_iv_func_t) null_cipher_set_iv, + (char *) null_cipher_description, + (int) 0, + (cipher_test_case_t *) &null_cipher_test_0, + (debug_module_t *) NULL +}; + diff --git a/libs/srtp/crypto/hash/CVS/Entries b/libs/srtp/crypto/hash/CVS/Entries new file mode 100644 index 0000000000..c536dd328f --- /dev/null +++ b/libs/srtp/crypto/hash/CVS/Entries @@ -0,0 +1,5 @@ +/auth.c/1.2/Thu Sep 29 12:36:43 2005// +/hmac.c/1.5/Mon Oct 3 15:53:26 2005// +/null_auth.c/1.2/Thu Sep 29 12:36:43 2005// +/sha1.c/1.5/Fri Mar 17 17:41:33 2006// +D diff --git a/libs/srtp/crypto/hash/CVS/Repository b/libs/srtp/crypto/hash/CVS/Repository new file mode 100644 index 0000000000..ccc8664015 --- /dev/null +++ b/libs/srtp/crypto/hash/CVS/Repository @@ -0,0 +1 @@ +srtp/crypto/hash diff --git a/libs/srtp/crypto/hash/CVS/Root b/libs/srtp/crypto/hash/CVS/Root new file mode 100644 index 0000000000..b4b74237d9 --- /dev/null +++ b/libs/srtp/crypto/hash/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@cvs.sourceforge.net:/cvsroot/srtp diff --git a/libs/srtp/crypto/hash/auth.c b/libs/srtp/crypto/hash/auth.c new file mode 100644 index 0000000000..2773c16cc3 --- /dev/null +++ b/libs/srtp/crypto/hash/auth.c @@ -0,0 +1,173 @@ +/* + * auth.c + * + * some bookkeeping functions for authentication functions + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include "auth.h" + +/* the debug module for authentiation */ + +debug_module_t mod_auth = { + 0, /* debugging is off by default */ + "auth func" /* printable name for module */ +}; + + +int +auth_get_key_length(const auth_t *a) { + return a->key_len; +} + +int +auth_get_tag_length(const auth_t *a) { + return a->out_len; +} + +int +auth_get_prefix_length(const auth_t *a) { + return a->prefix_len; +} + +int +auth_type_get_ref_count(const auth_type_t *at) { + return at->ref_count; +} + +/* + * auth_type_self_test() tests an auth function of type ct against + * test cases provided in an array of values of key, data, and tag + * that is known to be good + */ + +/* should be big enough for most occasions */ +#define SELF_TEST_TAG_BUF_OCTETS 32 + +err_status_t +auth_type_self_test(const auth_type_t *at) { + auth_test_case_t *test_case = at->test_data; + auth_t *a; + err_status_t status; + uint8_t tag[SELF_TEST_TAG_BUF_OCTETS]; + int i, case_num = 0; + + debug_print(mod_auth, "running self-test for auth function %s", + at->description); + + /* + * check to make sure that we have at least one test case, and + * return an error if we don't - we need to be paranoid here + */ + if (test_case == NULL) + return err_status_cant_check; + + /* loop over all test cases */ + while (test_case != NULL) { + + /* check test case parameters */ + if (test_case->tag_length_octets > SELF_TEST_TAG_BUF_OCTETS) + return err_status_bad_param; + + /* allocate auth */ + status = auth_type_alloc(at, &a, test_case->key_length_octets, + test_case->tag_length_octets); + if (status) + return status; + + /* initialize auth */ + status = auth_init(a, test_case->key); + if (status) { + auth_dealloc(a); + return status; + } + + /* zeroize tag then compute */ + octet_string_set_to_zero(tag, test_case->tag_length_octets); + status = auth_compute(a, test_case->data, + test_case->data_length_octets, tag); + if (status) { + auth_dealloc(a); + return status; + } + + debug_print(mod_auth, "key: %s", + octet_string_hex_string(test_case->key, + test_case->key_length_octets)); + debug_print(mod_auth, "data: %s", + octet_string_hex_string(test_case->data, + test_case->data_length_octets)); + debug_print(mod_auth, "tag computed: %s", + octet_string_hex_string(tag, test_case->tag_length_octets)); + debug_print(mod_auth, "tag expected: %s", + octet_string_hex_string(test_case->tag, + test_case->tag_length_octets)); + + /* check the result */ + status = err_status_ok; + for (i=0; i < test_case->tag_length_octets; i++) + if (tag[i] != test_case->tag[i]) { + status = err_status_algo_fail; + debug_print(mod_auth, "test case %d failed", case_num); + debug_print(mod_auth, " (mismatch at octet %d)", i); + } + if (status) { + auth_dealloc(a); + return err_status_algo_fail; + } + + /* deallocate the auth function */ + status = auth_dealloc(a); + if (status) + return status; + + /* + * the auth function passed the test case, so move on to the next test + * case in the list; if NULL, we'll quit and return an OK + */ + test_case = test_case->next_test_case; + ++case_num; + } + + return err_status_ok; +} + + diff --git a/libs/srtp/crypto/hash/hmac.c b/libs/srtp/crypto/hash/hmac.c new file mode 100644 index 0000000000..a786523c50 --- /dev/null +++ b/libs/srtp/crypto/hash/hmac.c @@ -0,0 +1,269 @@ +/* + * hmac.c + * + * implementation of hmac auth_type_t + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright(c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include "hmac.h" +#include "alloc.h" + +/* the debug module for authentiation */ + +debug_module_t mod_hmac = { + 0, /* debugging is off by default */ + "hmac sha-1" /* printable name for module */ +}; + + +err_status_t +hmac_alloc(auth_t **a, int key_len, int out_len) { + extern auth_type_t hmac; + uint8_t *pointer; + hmac_ctx_t *new_hmac_ctx; + + debug_print(mod_hmac, "allocating auth func with key length %d", key_len); + debug_print(mod_hmac, " tag length %d", out_len); + + /* + * check key length - note that we don't support keys larger + * than 20 bytes yet + */ + if (key_len > 20) + return err_status_bad_param; + + /* check output length - should be less than 20 bytes */ + if (out_len > 20) + return err_status_bad_param; + + /* allocate memory for auth and hmac_ctx_t structures */ + pointer = crypto_alloc(sizeof(hmac_ctx_t) + sizeof(auth_t)); + if (pointer == NULL) + return err_status_alloc_fail; + + /* set pointers */ + *a = (auth_t *)pointer; + (*a)->type = &hmac; + (*a)->state = pointer + sizeof(auth_t); + (*a)->out_len = out_len; + (*a)->key_len = key_len; + (*a)->prefix_len = 0; + new_hmac_ctx = (hmac_ctx_t *)((*a)->state); + + /* increment global count of all hmac uses */ + hmac.ref_count++; + + return err_status_ok; +} + +err_status_t +hmac_dealloc(auth_t *a) { + extern auth_type_t hmac; + + /* zeroize entire state*/ + octet_string_set_to_zero((uint8_t *)a, + sizeof(hmac_ctx_t) + sizeof(auth_t)); + + /* free memory */ + crypto_free(a); + + /* decrement global count of all hmac uses */ + hmac.ref_count--; + + return err_status_ok; +} + +err_status_t +hmac_init(hmac_ctx_t *state, const uint8_t *key, int key_len) { + int i; + uint8_t ipad[64]; + + /* + * check key length - note that we don't support keys larger + * than 20 bytes yet + */ + if (key_len > 20) + return err_status_bad_param; + + /* + * set values of ipad and opad by exoring the key into the + * appropriate constant values + */ + for (i=0; i < key_len; i++) { + ipad[i] = key[i] ^ 0x36; + state->opad[i] = key[i] ^ 0x5c; + } + /* set the rest of ipad, opad to constant values */ + for ( ; i < 64; i++) { + ipad[i] = 0x36; + ((uint8_t *)state->opad)[i] = 0x5c; + } + + debug_print(mod_hmac, "ipad: %s", octet_string_hex_string(ipad, 64)); + + /* initialize sha1 context */ + sha1_init(&state->init_ctx); + + /* hash ipad ^ key */ + sha1_update(&state->init_ctx, ipad, 64); + memcpy(&state->ctx, &state->init_ctx, sizeof(sha1_ctx_t)); + + return err_status_ok; +} + +err_status_t +hmac_start(hmac_ctx_t *state) { + + memcpy(&state->ctx, &state->init_ctx, sizeof(sha1_ctx_t)); + + return err_status_ok; +} + +err_status_t +hmac_update(hmac_ctx_t *state, const uint8_t *message, int msg_octets) { + + debug_print(mod_hmac, "input: %s", + octet_string_hex_string(message, msg_octets)); + + /* hash message into sha1 context */ + sha1_update(&state->ctx, message, msg_octets); + + return err_status_ok; +} + +err_status_t +hmac_compute(hmac_ctx_t *state, const void *message, + int msg_octets, int tag_len, uint8_t *result) { + uint32_t hash_value[5]; + uint32_t H[5]; + int i; + + /* check tag length, return error if we can't provide the value expected */ + if (tag_len > 20) + return err_status_bad_param; + + /* hash message, copy output into H */ + hmac_update(state, message, msg_octets); + sha1_final(&state->ctx, H); + + /* + * note that we don't need to debug_print() the input, since the + * function hmac_update() already did that for us + */ + debug_print(mod_hmac, "intermediate state: %s", + octet_string_hex_string((uint8_t *)H, 20)); + + /* re-initialize hash context */ + sha1_init(&state->ctx); + + /* hash opad ^ key */ + sha1_update(&state->ctx, (uint8_t *)state->opad, 64); + + /* hash the result of the inner hash */ + sha1_update(&state->ctx, (uint8_t *)H, 20); + + /* the result is returned in the array hash_value[] */ + sha1_final(&state->ctx, hash_value); + + /* copy hash_value to *result */ + for (i=0; i < tag_len; i++) + result[i] = ((uint8_t *)hash_value)[i]; + + debug_print(mod_hmac, "output: %s", + octet_string_hex_string((uint8_t *)hash_value, tag_len)); + + return err_status_ok; +} + + +/* begin test case 0 */ + +uint8_t +hmac_test_case_0_key[20] = { + 0x0b, 0x0b, 0x0b, 0x0b, 0x0b, 0x0b, 0x0b, 0x0b, + 0x0b, 0x0b, 0x0b, 0x0b, 0x0b, 0x0b, 0x0b, 0x0b, + 0x0b, 0x0b, 0x0b, 0x0b +}; + +uint8_t +hmac_test_case_0_data[8] = { + 0x48, 0x69, 0x20, 0x54, 0x68, 0x65, 0x72, 0x65 /* "Hi There" */ +}; + +uint8_t +hmac_test_case_0_tag[20] = { + 0xb6, 0x17, 0x31, 0x86, 0x55, 0x05, 0x72, 0x64, + 0xe2, 0x8b, 0xc0, 0xb6, 0xfb, 0x37, 0x8c, 0x8e, + 0xf1, 0x46, 0xbe, 0x00 +}; + +auth_test_case_t +hmac_test_case_0 = { + 20, /* octets in key */ + hmac_test_case_0_key, /* key */ + 8, /* octets in data */ + hmac_test_case_0_data, /* data */ + 20, /* octets in tag */ + hmac_test_case_0_tag, /* tag */ + NULL /* pointer to next testcase */ +}; + +/* end test case 0 */ + +char hmac_description[] = "hmac sha-1 authentication function"; + +/* + * auth_type_t hmac is the hmac metaobject + */ + +auth_type_t +hmac = { + (auth_alloc_func) hmac_alloc, + (auth_dealloc_func) hmac_dealloc, + (auth_init_func) hmac_init, + (auth_compute_func) hmac_compute, + (auth_update_func) hmac_update, + (auth_start_func) hmac_start, + (char *) hmac_description, + (int) 0, /* instance count */ + (auth_test_case_t *) &hmac_test_case_0, + (debug_module_t *) &mod_hmac +}; + diff --git a/libs/srtp/crypto/hash/null_auth.c b/libs/srtp/crypto/hash/null_auth.c new file mode 100644 index 0000000000..77e37da70d --- /dev/null +++ b/libs/srtp/crypto/hash/null_auth.c @@ -0,0 +1,160 @@ +/* + * null_auth.c + * + * implements the do-nothing auth algorithm + * + * David A. McGrew + * Cisco Systems, Inc. + * + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#include "null_auth.h" +#include "alloc.h" + +/* null_auth uses the auth debug module */ + +extern debug_module_t mod_auth; + +err_status_t +null_auth_alloc(auth_t **a, int key_len, int out_len) { + extern auth_type_t null_auth; + uint8_t *pointer; + + debug_print(mod_auth, "allocating auth func with key length %d", key_len); + debug_print(mod_auth, " tag length %d", out_len); + + /* allocate memory for auth and null_auth_ctx_t structures */ + pointer = crypto_alloc(sizeof(null_auth_ctx_t) + sizeof(auth_t)); + if (pointer == NULL) + return err_status_alloc_fail; + + /* set pointers */ + *a = (auth_t *)pointer; + (*a)->type = &null_auth; + (*a)->state = pointer + sizeof (auth_t); + (*a)->out_len = out_len; + (*a)->prefix_len = out_len; + (*a)->key_len = key_len; + + /* increment global count of all null_auth uses */ + null_auth.ref_count++; + + return err_status_ok; +} + +err_status_t +null_auth_dealloc(auth_t *a) { + extern auth_type_t null_auth; + + /* zeroize entire state*/ + octet_string_set_to_zero((uint8_t *)a, + sizeof(null_auth_ctx_t) + sizeof(auth_t)); + + /* free memory */ + crypto_free(a); + + /* decrement global count of all null_auth uses */ + null_auth.ref_count--; + + return err_status_ok; +} + +err_status_t +null_auth_init(null_auth_ctx_t *state, const uint8_t *key, int key_len) { + + /* accept any length of key, and do nothing */ + + return err_status_ok; +} + +err_status_t +null_auth_compute(null_auth_ctx_t *state, uint8_t *message, + int msg_octets, int tag_len, uint8_t *result) { + + return err_status_ok; +} + +err_status_t +null_auth_update(null_auth_ctx_t *state, uint8_t *message, + int msg_octets) { + + return err_status_ok; +} + +err_status_t +null_auth_start(null_auth_ctx_t *state) { + return err_status_ok; +} + +/* + * auth_type_t - defines description, test case, and null_auth + * metaobject + */ + +/* begin test case 0 */ + +auth_test_case_t +null_auth_test_case_0 = { + 0, /* octets in key */ + NULL, /* key */ + 0, /* octets in data */ + NULL, /* data */ + 0, /* octets in tag */ + NULL, /* tag */ + NULL /* pointer to next testcase */ +}; + +/* end test case 0 */ + +char null_auth_description[] = "null authentication function"; + +auth_type_t +null_auth = { + (auth_alloc_func) null_auth_alloc, + (auth_dealloc_func) null_auth_dealloc, + (auth_init_func) null_auth_init, + (auth_compute_func) null_auth_compute, + (auth_update_func) null_auth_update, + (auth_start_func) null_auth_start, + (char *) null_auth_description, + (int) 0, /* instance count */ + (auth_test_case_t *) &null_auth_test_case_0 +}; + diff --git a/libs/srtp/crypto/hash/sha1.c b/libs/srtp/crypto/hash/sha1.c new file mode 100644 index 0000000000..d4eba3cc26 --- /dev/null +++ b/libs/srtp/crypto/hash/sha1.c @@ -0,0 +1,404 @@ +/* + * sha1.c + * + * an implementation of the Secure Hash Algorithm v.1 (SHA-1), + * specified in FIPS 180-1 + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#include "sha1.h" + +debug_module_t mod_sha1 = { + 0, /* debugging is off by default */ + "sha-1" /* printable module name */ +}; + +/* SN == Rotate left N bits */ +#define S1(X) ((X << 1) | (X >> 31)) +#define S5(X) ((X << 5) | (X >> 27)) +#define S30(X) ((X << 30) | (X >> 2)) + +#define f0(B,C,D) ((B & C) | (~B & D)) +#define f1(B,C,D) (B ^ C ^ D) +#define f2(B,C,D) ((B & C) | (B & D) | (C & D)) +#define f3(B,C,D) (B ^ C ^ D) + +/* + * nota bene: the variable K0 appears in the curses library, so we + * give longer names to these variables to avoid spurious warnings + * on systems that uses curses + */ + +uint32_t SHA_K0 = 0x5A827999; /* Kt for 0 <= t <= 19 */ +uint32_t SHA_K1 = 0x6ED9EBA1; /* Kt for 20 <= t <= 39 */ +uint32_t SHA_K2 = 0x8F1BBCDC; /* Kt for 40 <= t <= 59 */ +uint32_t SHA_K3 = 0xCA62C1D6; /* Kt for 60 <= t <= 79 */ + +void +sha1(const uint8_t *msg, int octets_in_msg, uint32_t hash_value[5]) { + sha1_ctx_t ctx; + + sha1_init(&ctx); + sha1_update(&ctx, msg, octets_in_msg); + sha1_final(&ctx, hash_value); + +} + +/* + * sha1_core(M, H) computes the core compression function, where M is + * the next part of the message (in network byte order) and H is the + * intermediate state { H0, H1, ...} (in host byte order) + * + * this function does not do any of the padding required in the + * complete SHA1 function + * + * this function is used in the SEAL 3.0 key setup routines + * (crypto/cipher/seal.c) + */ + +void +sha1_core(const uint32_t M[16], uint32_t hash_value[5]) { + uint32_t H0; + uint32_t H1; + uint32_t H2; + uint32_t H3; + uint32_t H4; + uint32_t W[80]; + uint32_t A, B, C, D, E, TEMP; + int t; + + /* copy hash_value into H0, H1, H2, H3, H4 */ + H0 = hash_value[0]; + H1 = hash_value[1]; + H2 = hash_value[2]; + H3 = hash_value[3]; + H4 = hash_value[4]; + + /* copy/xor message into array */ + + W[0] = be32_to_cpu(M[0]); + W[1] = be32_to_cpu(M[1]); + W[2] = be32_to_cpu(M[2]); + W[3] = be32_to_cpu(M[3]); + W[4] = be32_to_cpu(M[4]); + W[5] = be32_to_cpu(M[5]); + W[6] = be32_to_cpu(M[6]); + W[7] = be32_to_cpu(M[7]); + W[8] = be32_to_cpu(M[8]); + W[9] = be32_to_cpu(M[9]); + W[10] = be32_to_cpu(M[10]); + W[11] = be32_to_cpu(M[11]); + W[12] = be32_to_cpu(M[12]); + W[13] = be32_to_cpu(M[13]); + W[14] = be32_to_cpu(M[14]); + W[15] = be32_to_cpu(M[15]); + TEMP = W[13] ^ W[8] ^ W[2] ^ W[0]; W[16] = S1(TEMP); + TEMP = W[14] ^ W[9] ^ W[3] ^ W[1]; W[17] = S1(TEMP); + TEMP = W[15] ^ W[10] ^ W[4] ^ W[2]; W[18] = S1(TEMP); + TEMP = W[16] ^ W[11] ^ W[5] ^ W[3]; W[19] = S1(TEMP); + TEMP = W[17] ^ W[12] ^ W[6] ^ W[4]; W[20] = S1(TEMP); + TEMP = W[18] ^ W[13] ^ W[7] ^ W[5]; W[21] = S1(TEMP); + TEMP = W[19] ^ W[14] ^ W[8] ^ W[6]; W[22] = S1(TEMP); + TEMP = W[20] ^ W[15] ^ W[9] ^ W[7]; W[23] = S1(TEMP); + TEMP = W[21] ^ W[16] ^ W[10] ^ W[8]; W[24] = S1(TEMP); + TEMP = W[22] ^ W[17] ^ W[11] ^ W[9]; W[25] = S1(TEMP); + TEMP = W[23] ^ W[18] ^ W[12] ^ W[10]; W[26] = S1(TEMP); + TEMP = W[24] ^ W[19] ^ W[13] ^ W[11]; W[27] = S1(TEMP); + TEMP = W[25] ^ W[20] ^ W[14] ^ W[12]; W[28] = S1(TEMP); + TEMP = W[26] ^ W[21] ^ W[15] ^ W[13]; W[29] = S1(TEMP); + TEMP = W[27] ^ W[22] ^ W[16] ^ W[14]; W[30] = S1(TEMP); + TEMP = W[28] ^ W[23] ^ W[17] ^ W[15]; W[31] = S1(TEMP); + + /* process the remainder of the array */ + for (t=32; t < 80; t++) { + TEMP = W[t-3] ^ W[t-8] ^ W[t-14] ^ W[t-16]; + W[t] = S1(TEMP); + } + + A = H0; B = H1; C = H2; D = H3; E = H4; + + for (t=0; t < 20; t++) { + TEMP = S5(A) + f0(B,C,D) + E + W[t] + SHA_K0; + E = D; D = C; C = S30(B); B = A; A = TEMP; + } + for ( ; t < 40; t++) { + TEMP = S5(A) + f1(B,C,D) + E + W[t] + SHA_K1; + E = D; D = C; C = S30(B); B = A; A = TEMP; + } + for ( ; t < 60; t++) { + TEMP = S5(A) + f2(B,C,D) + E + W[t] + SHA_K2; + E = D; D = C; C = S30(B); B = A; A = TEMP; + } + for ( ; t < 80; t++) { + TEMP = S5(A) + f3(B,C,D) + E + W[t] + SHA_K3; + E = D; D = C; C = S30(B); B = A; A = TEMP; + } + + hash_value[0] = H0 + A; + hash_value[1] = H1 + B; + hash_value[2] = H2 + C; + hash_value[3] = H3 + D; + hash_value[4] = H4 + E; + + return; +} + +void +sha1_init(sha1_ctx_t *ctx) { + + /* initialize state vector */ + ctx->H[0] = 0x67452301; + ctx->H[1] = 0xefcdab89; + ctx->H[2] = 0x98badcfe; + ctx->H[3] = 0x10325476; + ctx->H[4] = 0xc3d2e1f0; + + /* indicate that message buffer is empty */ + ctx->octets_in_buffer = 0; + + /* reset message bit-count to zero */ + ctx->num_bits_in_msg = 0; + +} + +void +sha1_update(sha1_ctx_t *ctx, const uint8_t *msg, int octets_in_msg) { + int i; + uint8_t *buf = (uint8_t *)ctx->M; + + /* update message bit-count */ + ctx->num_bits_in_msg += octets_in_msg * 8; + + /* loop over 16-word blocks of M */ + while (octets_in_msg > 0) { + + if (octets_in_msg + ctx->octets_in_buffer >= 64) { + + /* + * copy words of M into msg buffer until that buffer is full, + * converting them into host byte order as needed + */ + octets_in_msg -= (64 - ctx->octets_in_buffer); + for (i=ctx->octets_in_buffer; i < 64; i++) + buf[i] = *msg++; + ctx->octets_in_buffer = 0; + + /* process a whole block */ + + debug_print(mod_sha1, "(update) running sha1_core()", NULL); + + sha1_core(ctx->M, ctx->H); + + } else { + + debug_print(mod_sha1, "(update) not running sha1_core()", NULL); + + for (i=ctx->octets_in_buffer; + i < (ctx->octets_in_buffer + octets_in_msg); i++) + buf[i] = *msg++; + ctx->octets_in_buffer += octets_in_msg; + octets_in_msg = 0; + } + + } + +} + +/* + * sha1_final(ctx, output) computes the result for ctx and copies it + * into the twenty octets located at *output + */ + +void +sha1_final(sha1_ctx_t *ctx, uint32_t *output) { + uint32_t A, B, C, D, E, TEMP; + uint32_t W[80]; + int i, t; + + /* + * process the remaining octets_in_buffer, padding and terminating as + * necessary + */ + { + int tail = ctx->octets_in_buffer % 4; + + /* copy/xor message into array */ + for (i=0; i < (ctx->octets_in_buffer+3)/4; i++) + W[i] = be32_to_cpu(ctx->M[i]); + + /* set the high bit of the octet immediately following the message */ + switch (tail) { + case (3): + W[i-1] = (be32_to_cpu(ctx->M[i-1]) & 0xffffff00) | 0x80; + W[i] = 0x0; + break; + case (2): + W[i-1] = (be32_to_cpu(ctx->M[i-1]) & 0xffff0000) | 0x8000; + W[i] = 0x0; + break; + case (1): + W[i-1] = (be32_to_cpu(ctx->M[i-1]) & 0xff000000) | 0x800000; + W[i] = 0x0; + break; + case (0): + W[i] = 0x80000000; + break; + } + + /* zeroize remaining words */ + for (i++ ; i < 15; i++) + W[i] = 0x0; + + /* + * if there is room at the end of the word array, then set the + * last word to the bit-length of the message; otherwise, set that + * word to zero and then we need to do one more run of the + * compression algo. + */ + if (ctx->octets_in_buffer < 56) + W[15] = ctx->num_bits_in_msg; + else if (ctx->octets_in_buffer < 60) + W[15] = 0x0; + + /* process the word array */ for (t=16; t < 80; t++) { + TEMP = W[t-3] ^ W[t-8] ^ W[t-14] ^ W[t-16]; + W[t] = S1(TEMP); + } + + A = ctx->H[0]; + B = ctx->H[1]; + C = ctx->H[2]; + D = ctx->H[3]; + E = ctx->H[4]; + + for (t=0; t < 20; t++) { + TEMP = S5(A) + f0(B,C,D) + E + W[t] + SHA_K0; + E = D; D = C; C = S30(B); B = A; A = TEMP; + } + for ( ; t < 40; t++) { + TEMP = S5(A) + f1(B,C,D) + E + W[t] + SHA_K1; + E = D; D = C; C = S30(B); B = A; A = TEMP; + } + for ( ; t < 60; t++) { + TEMP = S5(A) + f2(B,C,D) + E + W[t] + SHA_K2; + E = D; D = C; C = S30(B); B = A; A = TEMP; + } + for ( ; t < 80; t++) { + TEMP = S5(A) + f3(B,C,D) + E + W[t] + SHA_K3; + E = D; D = C; C = S30(B); B = A; A = TEMP; + } + + ctx->H[0] += A; + ctx->H[1] += B; + ctx->H[2] += C; + ctx->H[3] += D; + ctx->H[4] += E; + + } + + debug_print(mod_sha1, "(final) running sha1_core()", NULL); + + if (ctx->octets_in_buffer >= 56) { + + debug_print(mod_sha1, "(final) running sha1_core() again", NULL); + + /* we need to do one final run of the compression algo */ + + /* + * set initial part of word array to zeros, and set the + * final part to the number of bits in the message + */ + for (i=0; i < 15; i++) + W[i] = 0x0; + W[15] = ctx->num_bits_in_msg; + + /* process the word array */ + for (t=16; t < 80; t++) { + TEMP = W[t-3] ^ W[t-8] ^ W[t-14] ^ W[t-16]; + W[t] = S1(TEMP); + } + + A = ctx->H[0]; + B = ctx->H[1]; + C = ctx->H[2]; + D = ctx->H[3]; + E = ctx->H[4]; + + for (t=0; t < 20; t++) { + TEMP = S5(A) + f0(B,C,D) + E + W[t] + SHA_K0; + E = D; D = C; C = S30(B); B = A; A = TEMP; + } + for ( ; t < 40; t++) { + TEMP = S5(A) + f1(B,C,D) + E + W[t] + SHA_K1; + E = D; D = C; C = S30(B); B = A; A = TEMP; + } + for ( ; t < 60; t++) { + TEMP = S5(A) + f2(B,C,D) + E + W[t] + SHA_K2; + E = D; D = C; C = S30(B); B = A; A = TEMP; + } + for ( ; t < 80; t++) { + TEMP = S5(A) + f3(B,C,D) + E + W[t] + SHA_K3; + E = D; D = C; C = S30(B); B = A; A = TEMP; + } + + ctx->H[0] += A; + ctx->H[1] += B; + ctx->H[2] += C; + ctx->H[3] += D; + ctx->H[4] += E; + } + + /* copy result into output buffer */ + output[0] = be32_to_cpu(ctx->H[0]); + output[1] = be32_to_cpu(ctx->H[1]); + output[2] = be32_to_cpu(ctx->H[2]); + output[3] = be32_to_cpu(ctx->H[3]); + output[4] = be32_to_cpu(ctx->H[4]); + + /* indicate that message buffer in context is empty */ + ctx->octets_in_buffer = 0; + + return; +} + + + diff --git a/libs/srtp/crypto/include/.cvsignore b/libs/srtp/crypto/include/.cvsignore new file mode 100644 index 0000000000..0e56cf2f8c --- /dev/null +++ b/libs/srtp/crypto/include/.cvsignore @@ -0,0 +1 @@ +config.h diff --git a/libs/srtp/crypto/include/CVS/Entries b/libs/srtp/crypto/include/CVS/Entries new file mode 100644 index 0000000000..6d680b1666 --- /dev/null +++ b/libs/srtp/crypto/include/CVS/Entries @@ -0,0 +1,29 @@ +/.cvsignore/1.1/Thu Sep 29 11:59:01 2005// +/aes.h/1.3/Sat Oct 8 16:06:05 2005// +/aes_cbc.h/1.2/Thu Sep 29 12:36:43 2005// +/aes_icm.h/1.3/Fri Mar 17 21:00:46 2006// +/alloc.h/1.3/Mon Oct 3 15:52:50 2005// +/auth.h/1.2/Thu Sep 29 12:36:43 2005// +/cipher.h/1.4/Wed Nov 30 18:47:18 2005// +/crypto.h/1.1.1.1/Wed Sep 21 22:51:39 2005// +/crypto_kernel.h/1.2/Thu Sep 29 12:48:41 2005// +/crypto_math.h/1.3/Mon Oct 3 16:17:23 2005// +/crypto_types.h/1.1.1.1/Wed Sep 21 22:51:39 2005// +/cryptoalg.h/1.2/Wed Sep 28 14:23:06 2005// +/datatypes.h/1.11/Sat Oct 8 16:35:31 2005// +/err.h/1.9/Tue Oct 18 15:25:46 2005// +/gf2_8.h/1.3/Thu Sep 29 12:36:43 2005// +/hmac.h/1.4/Mon Oct 3 14:33:59 2005// +/integers.h/1.6/Mon Oct 3 15:36:44 2005// +/kernel_compat.h/1.3/Tue Oct 18 15:27:35 2005// +/key.h/1.1.1.1/Wed Sep 21 22:51:39 2005// +/null_auth.h/1.2/Thu Sep 29 12:36:43 2005// +/null_cipher.h/1.2/Thu Sep 29 12:36:43 2005// +/prng.h/1.5/Mon Oct 3 15:52:19 2005// +/rand_source.h/1.3/Thu Sep 29 12:48:42 2005// +/rdb.h/1.1/Mon Sep 26 20:41:14 2005// +/rdbx.h/1.1.1.1/Wed Sep 21 22:51:40 2005// +/sha1.h/1.3/Mon Oct 3 14:33:59 2005// +/stat.h/1.3/Fri Mar 17 20:51:24 2006// +/xfm.h/1.2/Wed Sep 28 14:23:06 2005// +D diff --git a/libs/srtp/crypto/include/CVS/Repository b/libs/srtp/crypto/include/CVS/Repository new file mode 100644 index 0000000000..c29d21a005 --- /dev/null +++ b/libs/srtp/crypto/include/CVS/Repository @@ -0,0 +1 @@ +srtp/crypto/include diff --git a/libs/srtp/crypto/include/CVS/Root b/libs/srtp/crypto/include/CVS/Root new file mode 100644 index 0000000000..b4b74237d9 --- /dev/null +++ b/libs/srtp/crypto/include/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@cvs.sourceforge.net:/cvsroot/srtp diff --git a/libs/srtp/crypto/include/aes.h b/libs/srtp/crypto/include/aes.h new file mode 100644 index 0000000000..6a8b7e2552 --- /dev/null +++ b/libs/srtp/crypto/include/aes.h @@ -0,0 +1,84 @@ +/* + * aes.h + * + * header file for the AES block cipher + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#ifndef _AES_H +#define _AES_H + +#include "config.h" + +#include "datatypes.h" +#include "gf2_8.h" + +/* aes internals */ + +typedef v128_t aes_expanded_key_t[11]; + +inline void +aes_expand_encryption_key(const v128_t *key, + aes_expanded_key_t expanded_key); + +inline void +aes_expand_decryption_key(const v128_t *key, + aes_expanded_key_t expanded_key); + +void +aes_encrypt(v128_t *plaintext, const aes_expanded_key_t exp_key); + +void +aes_decrypt(v128_t *plaintext, const aes_expanded_key_t exp_key); + +#if 0 +/* + * internal functions + */ + +void +aes_init_sbox(void); + +void +aes_compute_tables(void); +#endif + +#endif /* _AES_H */ diff --git a/libs/srtp/crypto/include/aes_cbc.h b/libs/srtp/crypto/include/aes_cbc.h new file mode 100644 index 0000000000..9fb6682bd0 --- /dev/null +++ b/libs/srtp/crypto/include/aes_cbc.h @@ -0,0 +1,50 @@ +/* + * aes_cbc.h + * + * Header for AES Cipher Blobk Chaining Mode. + * + * David A. McGrew + * Cisco Systems, Inc. + * + */ + +#ifndef AES_CBC_H +#define AES_CBC_H + +#include "aes.h" +#include "cipher.h" + +typedef struct { + v128_t state; /* cipher chaining state */ + v128_t previous; /* previous ciphertext block */ + aes_expanded_key_t expanded_key; /* the cipher key */ +} aes_cbc_ctx_t; + +err_status_t +aes_cbc_set_key(aes_cbc_ctx_t *c, + const unsigned char *key); + +err_status_t +aes_cbc_encrypt(aes_cbc_ctx_t *c, + unsigned char *buf, + unsigned int *bytes_in_data); + +err_status_t +aes_cbc_context_init(aes_cbc_ctx_t *c, const uint8_t *key, + cipher_direction_t dir); + +err_status_t +aes_cbc_set_iv(aes_cbc_ctx_t *c, void *iv); + +err_status_t +aes_cbc_nist_encrypt(aes_cbc_ctx_t *c, + unsigned char *data, + unsigned int *bytes_in_data); + +err_status_t +aes_cbc_nist_decrypt(aes_cbc_ctx_t *c, + unsigned char *data, + unsigned int *bytes_in_data); + +#endif /* AES_CBC_H */ + diff --git a/libs/srtp/crypto/include/aes_icm.h b/libs/srtp/crypto/include/aes_icm.h new file mode 100644 index 0000000000..17a1ddbadd --- /dev/null +++ b/libs/srtp/crypto/include/aes_icm.h @@ -0,0 +1,56 @@ +/* + * aes_icm.h + * + * Header for AES Integer Counter Mode. + * + * David A. McGrew + * Cisco Systems, Inc. + * + */ + +#ifndef AES_ICM_H +#define AES_ICM_H + +#include "aes.h" +#include "cipher.h" + +typedef struct { + v128_t counter; /* holds the counter value */ + v128_t offset; /* initial offset value */ + v128_t keystream_buffer; /* buffers bytes of keystream */ + aes_expanded_key_t expanded_key; /* the cipher key */ + int bytes_in_buffer; /* number of unused bytes in buffer */ +} aes_icm_ctx_t; + + +err_status_t +aes_icm_context_init(aes_icm_ctx_t *c, + const unsigned char *key); + +err_status_t +aes_icm_set_iv(aes_icm_ctx_t *c, void *iv); + +err_status_t +aes_icm_encrypt(aes_icm_ctx_t *c, + unsigned char *buf, unsigned int *bytes_to_encr); + +err_status_t +aes_icm_output(aes_icm_ctx_t *c, + unsigned char *buf, int bytes_to_output); + +err_status_t +aes_icm_dealloc(cipher_t *c); + +err_status_t +aes_icm_encrypt_ismacryp(aes_icm_ctx_t *c, + unsigned char *buf, + unsigned int *enc_len, + int forIsmacryp); + +err_status_t +aes_icm_alloc_ismacryp(cipher_t **c, + int key_len, + int forIsmacryp); + +#endif /* AES_ICM_H */ + diff --git a/libs/srtp/crypto/include/alloc.h b/libs/srtp/crypto/include/alloc.h new file mode 100644 index 0000000000..5eb508c719 --- /dev/null +++ b/libs/srtp/crypto/include/alloc.h @@ -0,0 +1,57 @@ +/* + * alloc.h + * + * interface to memory allocation and deallocation, with optional debugging + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright (c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#ifndef CRYPTO_ALLOC_H +#define CRYPTO_ALLOC_H + +#include "datatypes.h" + +void * +crypto_alloc(size_t size); + +void +crypto_free(void *ptr); + +#endif /* CRYPTO_ALLOC_H */ diff --git a/libs/srtp/crypto/include/auth.h b/libs/srtp/crypto/include/auth.h new file mode 100644 index 0000000000..6e32480daa --- /dev/null +++ b/libs/srtp/crypto/include/auth.h @@ -0,0 +1,159 @@ +/* + * auth.h + * + * common interface to authentication functions + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#ifndef AUTH_H +#define AUTH_H + +#include "datatypes.h" +#include "err.h" /* error codes */ + +typedef struct auth_type_t *auth_type_pointer; +typedef struct auth_t *auth_pointer_t; + +typedef err_status_t (*auth_alloc_func) + (auth_pointer_t *ap, int key_len, int out_len); + +typedef err_status_t (*auth_init_func) + (void *state, const uint8_t *key, int key_len); + +typedef err_status_t (*auth_dealloc_func)(auth_pointer_t ap); + +typedef err_status_t (*auth_compute_func) + (void *state, uint8_t *buffer, int octets_to_auth, + int tag_len, uint8_t *tag); + +typedef err_status_t (*auth_update_func) + (void *state, uint8_t *buffer, int octets_to_auth); + +typedef err_status_t (*auth_start_func)(void *state); + +/* some syntactic sugar on these function types */ + +#define auth_type_alloc(at, a, klen, outlen) \ + ((at)->alloc((a), (klen), (outlen))) + +#define auth_init(a, key) \ + (((a)->type)->init((a)->state, (key), ((a)->key_len))) + +#define auth_compute(a, buf, len, res) \ + (((a)->type)->compute((a)->state, (buf), (len), (a)->out_len, (res))) + +#define auth_update(a, buf, len) \ + (((a)->type)->update((a)->state, (buf), (len))) + +#define auth_start(a)(((a)->type)->start((a)->state)) + +#define auth_dealloc(c) (((c)->type)->dealloc(c)) + +/* functions to get information about a particular auth_t */ + +int +auth_get_key_length(const struct auth_t *a); + +int +auth_get_tag_length(const struct auth_t *a); + +int +auth_get_prefix_length(const struct auth_t *a); + +/* + * auth_test_case_t is a (list of) key/message/tag values that are + * known to be correct for a particular cipher. this data can be used + * to test an implementation in an on-the-fly self test of the + * correcness of the implementation. (see the auth_type_self_test() + * function below) + */ + +typedef struct auth_test_case_t { + int key_length_octets; /* octets in key */ + uint8_t *key; /* key */ + int data_length_octets; /* octets in data */ + uint8_t *data; /* data */ + int tag_length_octets; /* octets in tag */ + uint8_t *tag; /* tag */ + struct auth_test_case_t *next_test_case; /* pointer to next testcase */ +} auth_test_case_t; + +/* auth_type_t */ + +typedef struct auth_type_t { + auth_alloc_func alloc; + auth_dealloc_func dealloc; + auth_init_func init; + auth_compute_func compute; + auth_update_func update; + auth_start_func start; + char *description; + int ref_count; + auth_test_case_t *test_data; + debug_module_t *debug; +} auth_type_t; + +typedef struct auth_t { + auth_type_t *type; + void *state; + int out_len; /* length of output tag in octets */ + int key_len; /* length of key in octets */ + int prefix_len; /* length of keystream prefix */ +} auth_t; + +/* + * auth_type_self_test() tests an auth_type against test cases + * provided in an array of values of key/message/tag that is known to + * be good + */ + +err_status_t +auth_type_self_test(const auth_type_t *at); + +/* + * auth_type_get_ref_count(at) returns the reference count (the number + * of instantiations) of the auth_type_t at + */ + +int +auth_type_get_ref_count(const auth_type_t *at); + +#endif /* AUTH_H */ diff --git a/libs/srtp/crypto/include/cipher.h b/libs/srtp/crypto/include/cipher.h new file mode 100644 index 0000000000..5c37c38413 --- /dev/null +++ b/libs/srtp/crypto/include/cipher.h @@ -0,0 +1,218 @@ +/* + * cipher.h + * + * common interface to ciphers + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#ifndef CIPHER_H +#define CIPHER_H + +#include "datatypes.h" +#include "rdbx.h" /* for xtd_seq_num_t */ +#include "err.h" /* for error codes */ + + +/** + * @brief cipher_direction_t defines a particular cipher operation. + * + * A cipher_direction_t is an enum that describes a particular cipher + * operation, i.e. encryption or decryption. For some ciphers, this + * distinction does not matter, but for others, it is essential. + */ + +typedef enum { + direction_encrypt, /**< encryption (convert plaintext to ciphertext) */ + direction_decrypt, /**< decryption (convert ciphertext to plaintext) */ + direction_any /**< encryption or decryption */ +} cipher_direction_t; + +/* + * the cipher_pointer and cipher_type_pointer definitions are needed + * as cipher_t and cipher_type_t are not yet defined + */ + +typedef struct cipher_type_t *cipher_type_pointer_t; +typedef struct cipher_t *cipher_pointer_t; + +/* + * a cipher_alloc_func_t allocates (but does not initialize) a cipher_t + */ + +typedef err_status_t (*cipher_alloc_func_t) + (cipher_pointer_t *cp, int key_len); + +/* + * a cipher_init_func_t [re-]initializes a cipher_t with a given key + * and direction (i.e., encrypt or decrypt) + */ + +typedef err_status_t (*cipher_init_func_t) + (void *state, const uint8_t *key, cipher_direction_t dir); + +/* a cipher_dealloc_func_t de-allocates a cipher_t */ + +typedef err_status_t (*cipher_dealloc_func_t)(cipher_pointer_t cp); + +/* a cipher_set_segment_func_t sets the segment index of a cipher_t */ + +typedef err_status_t (*cipher_set_segment_func_t) + (void *state, xtd_seq_num_t idx); + +/* a cipher_encrypt_func_t encrypts data in-place */ + +typedef err_status_t (*cipher_encrypt_func_t) + (void *state, uint8_t *buffer, unsigned int *octets_to_encrypt); + +/* a cipher_decrypt_func_t decrypts data in-place */ + +typedef err_status_t (*cipher_decrypt_func_t) + (void *state, uint8_t *buffer, unsigned int *octets_to_decrypt); + +/* + * a cipher_set_nonce_seq_func_t function sets both the nonce + * and the extended sequence number + */ + +typedef err_status_t (*cipher_set_iv_func_t) + (cipher_pointer_t cp, void *iv); + +/* + * cipher_test_case_t is a (list of) key, salt, xtd_seq_num_t, + * plaintext, and ciphertext values that are known to be correct for a + * particular cipher. this data can be used to test an implementation + * in an on-the-fly self test of the correcness of the implementation. + * (see the cipher_type_self_test() function below) + */ + +typedef struct cipher_test_case_t { + int key_length_octets; /* octets in key */ + uint8_t *key; /* key */ + uint8_t *idx; /* packet index */ + int plaintext_length_octets; /* octets in plaintext */ + uint8_t *plaintext; /* plaintext */ + int ciphertext_length_octets; /* octets in plaintext */ + uint8_t *ciphertext; /* ciphertext */ + struct cipher_test_case_t *next_test_case; /* pointer to next testcase */ +} cipher_test_case_t; + +/* cipher_type_t defines the 'metadata' for a particular cipher type */ + +typedef struct cipher_type_t { + cipher_alloc_func_t alloc; + cipher_dealloc_func_t dealloc; + cipher_init_func_t init; + cipher_encrypt_func_t encrypt; + cipher_encrypt_func_t decrypt; + cipher_set_iv_func_t set_iv; + char *description; + int ref_count; + cipher_test_case_t *test_data; + debug_module_t *debug; +} cipher_type_t; + +/* + * cipher_t defines an instantiation of a particular cipher, with fixed + * key length, key and salt values + */ + +typedef struct cipher_t { + cipher_type_t *type; + void *state; + int key_len; +#ifdef FORCE_64BIT_ALIGN + int pad; +#endif +} cipher_t; + +/* some syntactic sugar on these function types */ + +#define cipher_type_alloc(ct, c, klen) ((ct)->alloc((c), (klen))) + +#define cipher_dealloc(c) (((c)->type)->dealloc(c)) + +#define cipher_init(c, k, dir) (((c)->type)->init(((c)->state), (k), (dir))) + +#define cipher_encrypt(c, buf, len) \ + (((c)->type)->encrypt(((c)->state), (buf), (len))) + +#define cipher_decrypt(c, buf, len) \ + (((c)->type)->decrypt(((c)->state), (buf), (len))) + +#define cipher_set_iv(c, n) \ + ((c) ? (((c)->type)->set_iv(((c)->state), (n))) : \ + err_status_no_such_op) + +err_status_t +cipher_output(cipher_t *c, uint8_t *buffer, int num_octets_to_output); + + +/* some bookkeeping functions */ + +int +cipher_get_key_length(const cipher_t *c); + + +/* + * cipher_type_self_test() tests a cipher against test cases provided in + * an array of values of key/xtd_seq_num_t/plaintext/ciphertext + * that is known to be good + */ + +err_status_t +cipher_type_self_test(const cipher_type_t *ct); + + +/* + * cipher_bits_per_second(c, l, t) computes (and estimate of) the + * number of bits that a cipher implementation can encrypt in a second + * + * c is a cipher (which MUST be allocated and initialized already), l + * is the length in octets of the test data to be encrypted, and t is + * the number of trials + * + * if an error is encountered, then the value 0 is returned + */ + +uint64_t +cipher_bits_per_second(cipher_t *c, int octets_in_buffer, int num_trials); + +#endif /* CIPHER_H */ diff --git a/libs/srtp/crypto/include/config.h b/libs/srtp/crypto/include/config.h new file mode 100644 index 0000000000..5163cf9b9c --- /dev/null +++ b/libs/srtp/crypto/include/config.h @@ -0,0 +1,171 @@ +/* crypto/include/config.h. Generated by configure. */ +/* config_in.h. Generated from configure.in by autoheader. */ + +/* Define if building for a CISC machine (e.g. Intel). */ +#define CPU_CISC 1 + +/* Define if building for a RISC machine (assume slow byte access). */ +/* #undef CPU_RISC */ + +/* Path to random device */ +#define DEV_URANDOM "/dev/urandom" + +/* Define to compile in dynamic debugging system. */ +#define ENABLE_DEBUGGING 1 + +/* Report errors to this file. */ +/* #undef ERR_REPORTING_FILE */ + +/* Define to use logging to stdout. */ +#define ERR_REPORTING_STDOUT 1 + +/* Define this to use ISMAcryp code. */ +/* #undef GENERIC_AESICM */ + +/* Define to 1 if you have the header file. */ +#define HAVE_ARPA_INET_H 1 + +/* Define to 1 if you have the header file. */ +#define HAVE_BYTESWAP_H 1 + +/* Define to 1 if you have the `inet_aton' function. */ +#define HAVE_INET_ATON 1 + +/* Define to 1 if the system has the type `int16_t'. */ +#define HAVE_INT16_T 1 + +/* Define to 1 if the system has the type `int32_t'. */ +#define HAVE_INT32_T 1 + +/* Define to 1 if the system has the type `int8_t'. */ +#define HAVE_INT8_T 1 + +/* Define to 1 if you have the header file. */ +#define HAVE_INTTYPES_H 1 + +/* Define to 1 if you have the `socket' library (-lsocket). */ +/* #undef HAVE_LIBSOCKET */ + +/* Define to 1 if you have the header file. */ +/* #undef HAVE_MACHINE_TYPES_H */ + +/* Define to 1 if you have the header file. */ +#define HAVE_MEMORY_H 1 + +/* Define to 1 if you have the header file. */ +#define HAVE_NETINET_IN_H 1 + +/* Define to 1 if you have the `socket' function. */ +#define HAVE_SOCKET 1 + +/* Define to 1 if you have the header file. */ +#define HAVE_STDINT_H 1 + +/* Define to 1 if you have the header file. */ +#define HAVE_STDLIB_H 1 + +/* Define to 1 if you have the header file. */ +#define HAVE_STRINGS_H 1 + +/* Define to 1 if you have the header file. */ +#define HAVE_STRING_H 1 + +/* Define to 1 if you have the header file. */ +#define HAVE_SYSLOG_H 1 + +/* Define to 1 if you have the header file. */ +/* #undef HAVE_SYS_INT_TYPES_H */ + +/* Define to 1 if you have the header file. */ +#define HAVE_SYS_SOCKET_H 1 + +/* Define to 1 if you have the header file. */ +#define HAVE_SYS_STAT_H 1 + +/* Define to 1 if you have the header file. */ +#define HAVE_SYS_TYPES_H 1 + +/* Define to 1 if you have the header file. */ +#define HAVE_SYS_UIO_H 1 + +/* Define to 1 if the system has the type `uint16_t'. */ +#define HAVE_UINT16_T 1 + +/* Define to 1 if the system has the type `uint32_t'. */ +#define HAVE_UINT32_T 1 + +/* Define to 1 if the system has the type `uint64_t'. */ +#define HAVE_UINT64_T 1 + +/* Define to 1 if the system has the type `uint8_t'. */ +#define HAVE_UINT8_T 1 + +/* Define to 1 if you have the header file. */ +#define HAVE_UNISTD_H 1 + +/* Define to 1 if you have the `usleep' function. */ +#define HAVE_USLEEP 1 + +/* Define to 1 if you have the header file. */ +/* #undef HAVE_WINDOWS_H */ + +/* Define to 1 if you have the header file. */ +/* #undef HAVE_WINSOCK2_H */ + +/* Define to use X86 inlined assembly code */ +/* #undef HAVE_X86 */ + +/* Define to the address where bug reports for this package should be sent. */ +#define PACKAGE_BUGREPORT "" + +/* Define to the full name of this package. */ +#define PACKAGE_NAME "" + +/* Define to the full name and version of this package. */ +#define PACKAGE_STRING "" + +/* Define to the one symbol short name of this package. */ +#define PACKAGE_TARNAME "" + +/* Define to the version of this package. */ +#define PACKAGE_VERSION "" + +/* The size of a `unsigned long', as computed by sizeof. */ +#define SIZEOF_UNSIGNED_LONG 8 + +/* The size of a `unsigned long long', as computed by sizeof. */ +#define SIZEOF_UNSIGNED_LONG_LONG 8 + +/* Define to use GDOI. */ +/* #undef SRTP_GDOI */ + +/* Define to compile for kernel contexts. */ +/* #undef SRTP_KERNEL */ + +/* Define to compile for Linux kernel context. */ +/* #undef SRTP_KERNEL_LINUX */ + +/* Define to 1 if you have the ANSI C header files. */ +#define STDC_HEADERS 1 + +/* Write errors to this file */ +/* #undef USE_ERR_REPORTING_FILE */ + +/* Define to use syslog logging. */ +/* #undef USE_SYSLOG */ + +/* Define to 1 if your processor stores words with the most significant byte + first (like Motorola and SPARC, unlike Intel and VAX). */ +/* #undef WORDS_BIGENDIAN */ + +/* Define to empty if `const' does not conform to ANSI C. */ +/* #undef const */ + +/* Define to `__inline__' or `__inline' if that's what the C compiler + calls it, or to nothing if 'inline' is not supported under any name. */ +#ifndef __cplusplus +/* #undef inline */ +#endif + +/* Define to `unsigned' if does not define. */ +/* #undef size_t */ diff --git a/libs/srtp/crypto/include/crypto.h b/libs/srtp/crypto/include/crypto.h new file mode 100644 index 0000000000..dc52821e43 --- /dev/null +++ b/libs/srtp/crypto/include/crypto.h @@ -0,0 +1,17 @@ +/* + * crypto.h + * + * API for libcrypto + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +#ifndef CRYPTO_H +#define CRYPTO_H + +#include "crypto_kernel.h" + +#endif /* CRYPTO_H */ + + diff --git a/libs/srtp/crypto/include/crypto_kernel.h b/libs/srtp/crypto/include/crypto_kernel.h new file mode 100644 index 0000000000..029f1c885a --- /dev/null +++ b/libs/srtp/crypto/include/crypto_kernel.h @@ -0,0 +1,288 @@ +/* + * crypto_kernel.h + * + * header for the cryptographic kernel + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright(c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#ifndef CRYPTO_KERNEL +#define CRYPTO_KERNEL + +#include "rand_source.h" +#include "prng.h" +#include "cipher.h" +#include "auth.h" +#include "cryptoalg.h" +#include "stat.h" +#include "err.h" +#include "crypto_types.h" +#include "key.h" + + +/* + * crypto_kernel_state_t defines the possible states: + * + * insecure - not yet initialized + * secure - initialized and passed self-tests + */ + +typedef enum { + crypto_kernel_state_insecure, + crypto_kernel_state_secure +} crypto_kernel_state_t; + + +/** + * @brief A cipher_type_id_t is an identifier for a particular cipher + * type. + * + * A cipher_type_id_t is an integer that represents a particular + * cipher type, e.g. the Advanced Encryption Standard (AES). A + * NULL_CIPHER is avaliable; this cipher leaves the data unchanged, + * and can be selected to indicate that no encryption is to take + * place. + * + * @ingroup Ciphers + */ +typedef uint32_t cipher_type_id_t; + +/** + * @brief An auth_type_id_t is an identifier for a particular authentication + * function. + * + * An auth_type_id_t is an integer that represents a particular + * authentication function type, e.g. HMAC-SHA1. A NULL_AUTH is + * avaliable; this authentication function performs no computation, + * and can be selected to indicate that no authentication is to take + * place. + * + * @ingroup Authentication + */ +typedef uint32_t auth_type_id_t; + + +/* + * linked list of cipher types + */ + +typedef struct kernel_cipher_type { + cipher_type_id_t id; + cipher_type_t *cipher_type; + struct kernel_cipher_type *next; +} kernel_cipher_type_t; + +/* + * linked list of auth types + */ + +typedef struct kernel_auth_type { + auth_type_id_t id; + auth_type_t *auth_type; + struct kernel_auth_type *next; +} kernel_auth_type_t; + +/* + * linked list of debug modules + */ + +typedef struct kernel_debug_module { + debug_module_t *mod; + struct kernel_debug_module *next; +} kernel_debug_module_t; + + +/* + * crypto_kernel_t is the data structure for the crypto kernel + * + * note that there is *exactly one* instance of this data type, + * a global variable defined in crypto_kernel.c + */ + +typedef struct { + crypto_kernel_state_t state; /* current state of kernel */ + kernel_cipher_type_t *cipher_type_list; /* list of all cipher types */ + kernel_auth_type_t *auth_type_list; /* list of all auth func types */ + kernel_debug_module_t *debug_module_list; /* list of all debug modules */ +} crypto_kernel_t; + + +/* + * crypto_kernel_t external api + */ + + +/* + * The function crypto_kernel_init() initialized the crypto kernel and + * runs the self-test operations on the random number generators and + * crypto algorithms. Possible return values are: + * + * err_status_ok initialization successful + * init failure + * + * If any value other than err_status_ok is returned, the + * crypto_kernel MUST NOT be used. + */ + +err_status_t +crypto_kernel_init(void); + + +/* + * The function crypto_kernel_shutdown() de-initializes the + * crypto_kernel, zeroizes keys and other cryptographic material, and + * deallocates any dynamically allocated memory. Possible return + * values are: + * + * err_status_ok shutdown successful + * shutdown failure + * + */ + +err_status_t +crypto_kernel_shutdown(void); + +/* + * The function crypto_kernel_stats() checks the the crypto_kernel, + * running tests on the ciphers, auth funcs, and rng, and prints out a + * status report. Possible return values are: + * + * err_status_ok all tests were passed + * a test failed + * + */ + +err_status_t +crypto_kernel_status(void); + + +/* + * crypto_kernel_list_debug_modules() outputs a list of debugging modules + * + */ + +err_status_t +crypto_kernel_list_debug_modules(void); + +/* + * crypto_kernel_load_cipher_type() + * + */ + +err_status_t +crypto_kernel_load_cipher_type(cipher_type_t *ct, cipher_type_id_t id); + +err_status_t +crypto_kernel_load_auth_type(auth_type_t *ct, auth_type_id_t id); + +err_status_t +crypto_kernel_load_debug_module(debug_module_t *new_dm); + +/* + * crypto_kernel_alloc_cipher(id, cp, key_len); + * + * allocates a cipher of type id at location *cp, with key length + * key_len octets. Return values are: + * + * err_status_ok no problems + * err_status_alloc_fail an allocation failure occured + * err_status_fail couldn't find cipher with identifier 'id' + */ + +err_status_t +crypto_kernel_alloc_cipher(cipher_type_id_t id, + cipher_pointer_t *cp, + int key_len); + +/* + * crypto_kernel_alloc_auth(id, ap, key_len, tag_len); + * + * allocates an auth function of type id at location *ap, with key + * length key_len octets and output tag length of tag_len. Return + * values are: + * + * err_status_ok no problems + * err_status_alloc_fail an allocation failure occured + * err_status_fail couldn't find auth with identifier 'id' + */ + +err_status_t +crypto_kernel_alloc_auth(auth_type_id_t id, + auth_pointer_t *ap, + int key_len, + int tag_len); + + +/* + * crypto_kernel_set_debug_module(mod_name, v) + * + * sets dynamic debugging to the value v (0 for off, 1 for on) for the + * debug module with the name mod_name + * + * returns err_status_ok on success, err_status_fail otherwise + */ + +err_status_t +crypto_kernel_set_debug_module(char *mod_name, int v); + +/** + * @brief writes a random octet string. + * + * The function call crypto_get_random(dest, len) writes len octets of + * random data to the location to which dest points, and returns an + * error code. This error code @b must be checked, and if a failure is + * reported, the data in the buffer @b must @b not be used. + * + * @warning If the return code is not checked, then non-random + * data may be in the buffer. This function will fail + * unless it is called after crypto_kernel_init(). + * + * @return + * - err_status_ok if no problems occured. + * - [other] a problem occured, and no assumptions should + * be made about the contents of the destination + * buffer. + * + * @ingroup SRTP + */ +err_status_t +crypto_get_random(unsigned char *buffer, unsigned int length); + +#endif /* CRYPTO_KERNEL */ diff --git a/libs/srtp/crypto/include/crypto_math.h b/libs/srtp/crypto/include/crypto_math.h new file mode 100644 index 0000000000..be6558f638 --- /dev/null +++ b/libs/srtp/crypto/include/crypto_math.h @@ -0,0 +1,273 @@ +/* + * math.h + * + * crypto math operations and data types + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright (c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#ifndef MATH_H +#define MATH_H + +#include "datatypes.h" + +unsigned char +v32_weight(v32_t a); + +unsigned char +v32_distance(v32_t x, v32_t y); + +unsigned int +v32_dot_product(v32_t a, v32_t b); + +char * +v16_bit_string(v16_t x); + +char * +v32_bit_string(v32_t x); + +char * +v64_bit_string(const v64_t *x); + +char * +octet_hex_string(uint8_t x); + +char * +v16_hex_string(v16_t x); + +char * +v32_hex_string(v32_t x); + +char * +v64_hex_string(const v64_t *x); + +int +hex_char_to_nibble(uint8_t c); + +int +is_hex_string(char *s); + +v16_t +hex_string_to_v16(char *s); + +v32_t +hex_string_to_v32(char *s); + +v64_t +hex_string_to_v64(char *s); + +/* the matrix A[] is stored in column format, i.e., A[i] is + the ith column of the matrix */ + +uint8_t +A_times_x_plus_b(uint8_t A[8], uint8_t x, uint8_t b); + +void +v16_copy_octet_string(v16_t *x, const uint8_t s[2]); + +void +v32_copy_octet_string(v32_t *x, const uint8_t s[4]); + +void +v64_copy_octet_string(v64_t *x, const uint8_t s[8]); + +void +v128_add(v128_t *z, v128_t *x, v128_t *y); + +int +octet_string_is_eq(uint8_t *a, uint8_t *b, int len); + +void +octet_string_set_to_zero(uint8_t *s, int len); + + + +/* + * the matrix A[] is stored in column format, i.e., A[i] is the ith + * column of the matrix +*/ +uint8_t +A_times_x_plus_b(uint8_t A[8], uint8_t x, uint8_t b); + + +#if 0 +#if WORDS_BIGENDIAN + +#define _v128_add(z, x, y) { \ + uint64_t tmp; \ + \ + tmp = x->v32[3] + y->v32[3]; \ + z->v32[3] = (uint32_t) tmp; \ + \ + tmp = x->v32[2] + y->v32[2] + (tmp >> 32); \ + z->v32[2] = (uint32_t) tmp; \ + \ + tmp = x->v32[1] + y->v32[1] + (tmp >> 32); \ + z->v32[1] = (uint32_t) tmp; \ + \ + tmp = x->v32[0] + y->v32[0] + (tmp >> 32); \ + z->v32[0] = (uint32_t) tmp; \ +} + +#else /* assume little endian architecture */ + +#define _v128_add(z, x, y) { \ + uint64_t tmp; \ + \ + tmp = htonl(x->v32[3]) + htonl(y->v32[3]); \ + z->v32[3] = ntohl((uint32_t) tmp); \ + \ + tmp = htonl(x->v32[2]) + htonl(y->v32[2]) \ + + htonl(tmp >> 32); \ + z->v32[2] = ntohl((uint32_t) tmp); \ + \ + tmp = htonl(x->v32[1]) + htonl(y->v32[1]) \ + + htonl(tmp >> 32); \ + z->v32[1] = ntohl((uint32_t) tmp); \ + \ + tmp = htonl(x->v32[0]) + htonl(y->v32[0]) \ + + htonl(tmp >> 32); \ + z->v32[0] = ntohl((uint32_t) tmp); \ +} + +#endif /* WORDS_BIGENDIAN */ +#endif + +#ifdef DATATYPES_USE_MACROS /* little functions are really macros */ + +#define v128_set_to_zero(z) _v128_set_to_zero(z) +#define v128_copy(z, x) _v128_copy(z, x) +#define v128_xor(z, x, y) _v128_xor(z, x, y) +#define v128_and(z, x, y) _v128_and(z, x, y) +#define v128_or(z, x, y) _v128_or(z, x, y) +#define v128_complement(x) _v128_complement(x) +#define v128_is_eq(x, y) _v128_is_eq(x, y) +#define v128_xor_eq(x, y) _v128_xor_eq(x, y) +#define v128_get_bit(x, i) _v128_get_bit(x, i) +#define v128_set_bit(x, i) _v128_set_bit(x, i) +#define v128_clear_bit(x, i) _v128_clear_bit(x, i) +#define v128_set_bit_to(x, i, y) _v128_set_bit_to(x, i, y) + +#else + +void +v128_set_to_zero(v128_t *x); + +int +v128_is_eq(const v128_t *x, const v128_t *y); + +void +v128_copy(v128_t *x, const v128_t *y); + +void +v128_xor(v128_t *z, v128_t *x, v128_t *y); + +void +v128_and(v128_t *z, v128_t *x, v128_t *y); + +void +v128_or(v128_t *z, v128_t *x, v128_t *y); + +void +v128_complement(v128_t *x); + +int +v128_get_bit(const v128_t *x, int i); + +void +v128_set_bit(v128_t *x, int i) ; + +void +v128_clear_bit(v128_t *x, int i); + +void +v128_set_bit_to(v128_t *x, int i, int y); + +#endif /* DATATYPES_USE_MACROS */ + +/* + * octet_string_is_eq(a,b, len) returns 1 if the length len strings a + * and b are not equal, returns 0 otherwise + */ + +int +octet_string_is_eq(uint8_t *a, uint8_t *b, int len); + +void +octet_string_set_to_zero(uint8_t *s, int len); + + +/* + * functions manipulating bit_vector_t + * + * A bitvector_t consists of an array of words and an integer + * representing the number of significant bits stored in the array. + * The bits are packed as follows: the least significant bit is that + * of word[0], while the most significant bit is the nth most + * significant bit of word[m], where length = bits_per_word * m + n. + * + */ + +#define bits_per_word 32 +#define bytes_per_word 4 + +typedef struct { + uint32_t length; + uint32_t *word; +} bitvector_t; + +int +bitvector_alloc(bitvector_t *v, unsigned long length); + +void +bitvector_set_bit(bitvector_t *v, int bit_index); + +int +bitvector_get_bit(const bitvector_t *v, int bit_index); + +int +bitvector_print_hex(const bitvector_t *v, FILE *stream); + +int +bitvector_set_from_hex(bitvector_t *v, char *string); + +#endif /* MATH_H */ + + + diff --git a/libs/srtp/crypto/include/crypto_types.h b/libs/srtp/crypto/include/crypto_types.h new file mode 100644 index 0000000000..e20c7715b0 --- /dev/null +++ b/libs/srtp/crypto/include/crypto_types.h @@ -0,0 +1,206 @@ +/* + * crypto_types.h + * + * constants for cipher types and auth func types + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright(c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#ifndef CRYPTO_TYPES_H +#define CRYPTO_TYPES_H + +/** + * @defgroup Algos Cryptographic Algorithms + * + * + * This library provides several different cryptographic algorithms, + * each of which can be selected by using the cipher_type_id_t and + * auth_type_id_t. These algorithms are documented below. + * + * Authentication functions that use the Universal Security Transform + * (UST) must be used in conjunction with a cipher other than the null + * cipher. These functions require a per-message pseudorandom input + * that is generated by the cipher. + * + * The identifiers STRONGHOLD_AUTH and STRONGHOLD_CIPHER identify the + * strongest available authentication function and cipher, + * respectively. They are resolved at compile time to the strongest + * available algorithm. The stronghold algorithms can serve as did + * the keep of a medieval fortification; they provide the strongest + * defense (or the last refuge). + * + * @{ + */ + +/** + * @defgroup Ciphers Cipher Types + * + * @brief Each cipher type is identified by an unsigned integer. The + * cipher types available in this edition of libSRTP are given + * by the #defines below. + * + * A cipher_type_id_t is an identifier for a cipher_type; only values + * given by the #defines above (or those present in the file + * crypto_types.h) should be used. + * + * The identifier STRONGHOLD_CIPHER indicates the strongest available + * cipher, allowing an application to choose the strongest available + * algorithm without any advance knowledge about the avaliable + * algorithms. + * + * @{ + */ + +/** + * @brief The null cipher performs no encryption. + * + * The NULL_CIPHER leaves its inputs unaltered, during both the + * encryption and decryption operations. This cipher can be chosen + * to indicate that no encryption is to be performed. + */ +#define NULL_CIPHER 0 + +/** + * @brief AES-128 Integer Counter Mode (AES ICM) + * + * AES-128 ICM is the variant of counter mode that is used by Secure RTP. + * This cipher uses a 16-octet key and a 30-octet offset (or salt) value. + */ +#define AES_128_ICM 1 + +/** + * @brief SEAL 3.0 + * + * SEAL is the Software-Optimized Encryption Algorithm of Coppersmith + * and Rogaway. Nota bene: this cipher is IBM proprietary. + */ +#define SEAL 2 + +/** + * @brief AES-128 Integer Counter Mode (AES ICM) + * + * AES-128 ICM is the variant of counter mode that is used by Secure RTP. + * This cipher uses a 16-octet key and a 30-octet offset (or salt) value. + */ +#define AES_128_CBC 3 + +/** + * @brief Strongest available cipher. + * + * This identifier resolves to the strongest cipher type available. + */ +#define STRONGHOLD_CIPHER AES_128_ICM + +/** + * @} + */ + + + +/** + * @defgroup Authentication Authentication Function Types + * + * @brief Each authentication function type is identified by an + * unsigned integer. The authentication function types available in + * this edition of libSRTP are given by the #defines below. + * + * An auth_type_id_t is an identifier for an authentication function type; + * only values given by the #defines above (or those present in the + * file crypto_types.h) should be used. + * + * The identifier STRONGHOLD_AUTH indicates the strongest available + * authentication function, allowing an application to choose the + * strongest available algorithm without any advance knowledge about + * the avaliable algorithms. The stronghold algorithms can serve as + * did the keep of a medieval fortification; they provide the + * strongest defense (or the last refuge). + * + * @{ + */ + +/** + * @brief The null authentication function performs no authentication. + * + * The NULL_AUTH function does nothing, and can be selected to indicate + * that authentication should not be performed. + */ +#define NULL_AUTH 0 + +/** + * @brief UST with TMMH Version 2 + * + * UST_TMMHv2 implements the Truncated Multi-Modular Hash using + * UST. This function must be used in conjunction with a cipher other + * than the null cipher. + * with a cipher. + */ +#define UST_TMMHv2 1 + +/** + * @brief (UST) AES-128 XORMAC + * + * UST_AES_128_XMAC implements AES-128 XORMAC, using UST. Nota bene: + * the XORMAC algorithm is IBM proprietary. + */ +#define UST_AES_128_XMAC 2 + +/** + * @brief HMAC-SHA1 + * + * HMAC_SHA1 implements the Hash-based MAC using the NIST Secure + * Hash Algorithm version 1 (SHA1). + */ +#define HMAC_SHA1 3 + +/** + * @brief Strongest available authentication function. + * + * This identifier resolves to the strongest available authentication + * function. + */ +#define STRONGHOLD_AUTH HMAC_SHA1 + +/** + * @} + */ +/** + * @} + */ + +#endif /* CRYPTO_TYPES_H */ diff --git a/libs/srtp/crypto/include/cryptoalg.h b/libs/srtp/crypto/include/cryptoalg.h new file mode 100644 index 0000000000..9024a71117 --- /dev/null +++ b/libs/srtp/crypto/include/cryptoalg.h @@ -0,0 +1,133 @@ +/* + * cryptoalg.h + * + * API for authenticated encryption crypto algorithms + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright (c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#ifndef CRYPTOALG_H +#define CRYPTOALG_H + +#include "err.h" + +/** + * @defgroup Crypto Cryptography + * + * Zed uses a simple interface to a cryptographic transform. + * + * @{ + */ + +/** + * @brief applies a crypto algorithm + * + * The function pointer cryptoalg_func_t points to a function that + * implements a crypto transform, and provides a uniform API for + * accessing crypto mechanisms. + * + * @param key location of secret key + * + * @param clear data to be authenticated but not encrypted + * + * @param clear_len length of data to be authenticated but not encrypted + * + * @param iv location to write the Initialization Vector (IV) + * + * @param protect location of the data to be encrypted and + * authenticated (before the function call), and the ciphertext + * and authentication tag (after the call) + * + * @param protected_len location of the length of the data to be + * encrypted and authenticated (before the function call), and the + * length of the ciphertext (after the call) + * + */ + +typedef err_status_t (*cryptoalg_func_t) + (void *key, + const void *clear, + unsigned clear_len, + void *iv, + void *protect, + unsigned *protected_len); + +typedef +err_status_t (*cryptoalg_inv_t) + (void *key, /* location of secret key */ + const void *clear, /* data to be authenticated only */ + unsigned clear_len, /* length of data to be authenticated only */ + void *iv, /* location of iv */ + void *opaque, /* data to be decrypted and authenticated */ + unsigned *opaque_len /* location of the length of data to be + * decrypted and authd (before and after) + */ + ); + +typedef struct cryptoalg_ctx_t { + cryptoalg_func_t enc; + cryptoalg_inv_t dec; + unsigned key_len; + unsigned iv_len; + unsigned auth_tag_len; + unsigned max_expansion; +} cryptoalg_ctx_t; + +typedef cryptoalg_ctx_t *cryptoalg_t; + +#define cryptoalg_get_key_len(cryptoalg) ((cryptoalg)->key_len) + +#define cryptoalg_get_iv_len(cryptoalg) ((cryptoalg)->iv_len) + +#define cryptoalg_get_auth_tag_len(cryptoalg) ((cryptoalg)->auth_tag_len) + +int +cryptoalg_get_id(cryptoalg_t c); + +cryptoalg_t +cryptoalg_find_by_id(int id); + + +/** + * @} + */ + +#endif /* CRYPTOALG_H */ + + diff --git a/libs/srtp/crypto/include/datatypes.h b/libs/srtp/crypto/include/datatypes.h new file mode 100644 index 0000000000..75f88a72d4 --- /dev/null +++ b/libs/srtp/crypto/include/datatypes.h @@ -0,0 +1,426 @@ +/* + * datatypes.h + * + * data types for bit vectors and finite fields + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#ifndef _DATATYPES_H +#define _DATATYPES_H + +#include "integers.h" /* definitions of uint32_t, et cetera */ +#include "alloc.h" + +#include + +#ifndef SRTP_KERNEL +# include +# include +# include +# ifdef HAVE_NETINET_IN_H +# include +# elif defined HAVE_WINSOCK2_H +# include +# endif +#endif + +/* if DATATYPES_USE_MACROS is defined, then little functions are macros */ +#define DATATYPES_USE_MACROS + +typedef union { + uint8_t v8[2]; + uint16_t value; +} v16_t; + +typedef union { + uint8_t v8[4]; + uint16_t v16[2]; + uint32_t value; +} v32_t; + +typedef union { + uint8_t v8[8]; + uint16_t v16[4]; + uint32_t v32[2]; + uint64_t value; +} v64_t; + +typedef union { + uint8_t v8[16]; + uint16_t v16[8]; + uint32_t v32[4]; + uint64_t v64[2]; +} v128_t; + + + +/* some useful and simple math functions */ + +#define pow_2(X) ( (unsigned int)1 << (X) ) /* 2^X */ + +#define pow_minus_one(X) ( (X) ? -1 : 1 ) /* (-1)^X */ + + +/* + * octet_get_weight(x) returns the hamming weight (number of bits equal to + * one) in the octet x + */ + +int +octet_get_weight(uint8_t octet); + +char * +octet_bit_string(uint8_t x); + +#define MAX_PRINT_STRING_LEN 1024 + +char * +octet_string_hex_string(const void *str, int length); + +char * +v128_bit_string(v128_t *x); + +char * +v128_hex_string(v128_t *x); + +uint8_t +nibble_to_hex_char(uint8_t nibble); + +char * +char_to_hex_string(char *x, int num_char); + +uint8_t +hex_string_to_octet(char *s); + +/* + * hex_string_to_octet_string(raw, hex, len) converts the hexadecimal + * string at *hex (of length len octets) to the equivalent raw data + * and writes it to *raw. + * + * if a character in the hex string that is not a hexadeciaml digit + * (0123456789abcdefABCDEF) is encountered, the function stops writing + * data to *raw + * + * the number of hex digits copied (which is two times the number of + * octets in *raw) is returned + */ + +int +hex_string_to_octet_string(char *raw, char *hex, int len); + +v128_t +hex_string_to_v128(char *s); + +void +v128_copy_octet_string(v128_t *x, const uint8_t s[16]); + +void +v128_left_shift(v128_t *x, int index); + +void +v128_right_shift(v128_t *x, int index); + +/* + * the following macros define the data manipulation functions + * + * If DATATYPES_USE_MACROS is defined, then these macros are used + * directly (and function call overhead is avoided). Otherwise, + * the macros are used through the functions defined in datatypes.c + * (and the compiler provides better warnings). + */ + +#define _v128_set_to_zero(x) \ +( \ + (x)->v32[0] = 0, \ + (x)->v32[1] = 0, \ + (x)->v32[2] = 0, \ + (x)->v32[3] = 0 \ +) + +#define _v128_copy(x, y) \ +( \ + (x)->v32[0] = (y)->v32[0], \ + (x)->v32[1] = (y)->v32[1], \ + (x)->v32[2] = (y)->v32[2], \ + (x)->v32[3] = (y)->v32[3] \ +) + +#define _v128_xor(z, x, y) \ +( \ + (z)->v32[0] = (x)->v32[0] ^ (y)->v32[0], \ + (z)->v32[1] = (x)->v32[1] ^ (y)->v32[1], \ + (z)->v32[2] = (x)->v32[2] ^ (y)->v32[2], \ + (z)->v32[3] = (x)->v32[3] ^ (y)->v32[3] \ +) + +#define _v128_and(z, x, y) \ +( \ + (z)->v32[0] = (x)->v32[0] & (y)->v32[0], \ + (z)->v32[1] = (x)->v32[1] & (y)->v32[1], \ + (z)->v32[2] = (x)->v32[2] & (y)->v32[2], \ + (z)->v32[3] = (x)->v32[3] & (y)->v32[3] \ +) + +#define _v128_or(z, x, y) \ +( \ + (z)->v32[0] = (x)->v32[0] | (y)->v32[0], \ + (z)->v32[1] = (x)->v32[1] | (y)->v32[1], \ + (z)->v32[2] = (x)->v32[2] | (y)->v32[2], \ + (z)->v32[3] = (x)->v32[3] | (y)->v32[3] \ +) + +#define _v128_complement(x) \ +( \ + (x)->v32[0] = ~(x)->v32[0], \ + (x)->v32[1] = ~(x)->v32[1], \ + (x)->v32[2] = ~(x)->v32[2], \ + (x)->v32[3] = ~(x)->v32[3] \ +) + +/* ok for NO_64BIT_MATH if it can compare uint64_t's (even as structures) */ +#define _v128_is_eq(x, y) \ + (((x)->v64[0] == (y)->v64[0]) && ((x)->v64[1] == (y)->v64[1])) + + +#ifdef NO_64BIT_MATH +#define _v128_xor_eq(z, x) \ +( \ + (z)->v32[0] ^= (x)->v32[0], \ + (z)->v32[1] ^= (x)->v32[1], \ + (z)->v32[2] ^= (x)->v32[2], \ + (z)->v32[3] ^= (x)->v32[3] \ +) +#else +#define _v128_xor_eq(z, x) \ +( \ + (z)->v64[0] ^= (x)->v64[0], \ + (z)->v64[1] ^= (x)->v64[1] \ +) +#endif + +/* NOTE! This assumes an odd ordering! */ +/* This will not be compatible directly with math on some processors */ +/* bit 0 is first 32-bit word, low order bit. in little-endian, that's + the first byte of the first 32-bit word. In big-endian, that's + the 3rd byte of the first 32-bit word */ +/* The get/set bit code is used by the replay code ONLY, and it doesn't + really care which bit is which. AES does care which bit is which, but + doesn't use the 128-bit get/set or 128-bit shifts */ + +#define _v128_get_bit(x, bit) \ +( \ + ((((x)->v32[(bit) >> 5]) >> ((bit) & 31)) & 1) \ +) + +#define _v128_set_bit(x, bit) \ +( \ + (((x)->v32[(bit) >> 5]) |= ((uint32_t)1 << ((bit) & 31))) \ +) + +#define _v128_clear_bit(x, bit) \ +( \ + (((x)->v32[(bit) >> 5]) &= ~((uint32_t)1 << ((bit) & 31))) \ +) + +#define _v128_set_bit_to(x, bit, value) \ +( \ + (value) ? _v128_set_bit(x, bit) : \ + _v128_clear_bit(x, bit) \ +) + + +#if 0 +/* nothing uses this */ +#ifdef WORDS_BIGENDIAN + +#define _v128_add(z, x, y) { \ + uint64_t tmp; \ + \ + tmp = x->v32[3] + y->v32[3]; \ + z->v32[3] = (uint32_t) tmp; \ + \ + tmp = x->v32[2] + y->v32[2] + (tmp >> 32); \ + z->v32[2] = (uint32_t) tmp; \ + \ + tmp = x->v32[1] + y->v32[1] + (tmp >> 32); \ + z->v32[1] = (uint32_t) tmp; \ + \ + tmp = x->v32[0] + y->v32[0] + (tmp >> 32); \ + z->v32[0] = (uint32_t) tmp; \ +} + +#else /* assume little endian architecture */ + +#define _v128_add(z, x, y) { \ + uint64_t tmp; \ + \ + tmp = htonl(x->v32[3]) + htonl(y->v32[3]); \ + z->v32[3] = ntohl((uint32_t) tmp); \ + \ + tmp = htonl(x->v32[2]) + htonl(y->v32[2]) \ + + htonl(tmp >> 32); \ + z->v32[2] = ntohl((uint32_t) tmp); \ + \ + tmp = htonl(x->v32[1]) + htonl(y->v32[1]) \ + + htonl(tmp >> 32); \ + z->v32[1] = ntohl((uint32_t) tmp); \ + \ + tmp = htonl(x->v32[0]) + htonl(y->v32[0]) \ + + htonl(tmp >> 32); \ + z->v32[0] = ntohl((uint32_t) tmp); \ +} +#endif /* WORDS_BIGENDIAN */ +#endif /* 0 */ + + +#ifdef DATATYPES_USE_MACROS /* little functions are really macros */ + +#define v128_set_to_zero(z) _v128_set_to_zero(z) +#define v128_copy(z, x) _v128_copy(z, x) +#define v128_xor(z, x, y) _v128_xor(z, x, y) +#define v128_and(z, x, y) _v128_and(z, x, y) +#define v128_or(z, x, y) _v128_or(z, x, y) +#define v128_complement(x) _v128_complement(x) +#define v128_is_eq(x, y) _v128_is_eq(x, y) +#define v128_xor_eq(x, y) _v128_xor_eq(x, y) +#define v128_get_bit(x, i) _v128_get_bit(x, i) +#define v128_set_bit(x, i) _v128_set_bit(x, i) +#define v128_clear_bit(x, i) _v128_clear_bit(x, i) +#define v128_set_bit_to(x, i, y) _v128_set_bit_to(x, i, y) + +#else + +void +v128_set_to_zero(v128_t *x); + +int +v128_is_eq(const v128_t *x, const v128_t *y); + +void +v128_copy(v128_t *x, const v128_t *y); + +void +v128_xor(v128_t *z, v128_t *x, v128_t *y); + +void +v128_and(v128_t *z, v128_t *x, v128_t *y); + +void +v128_or(v128_t *z, v128_t *x, v128_t *y); + +void +v128_complement(v128_t *x); + +int +v128_get_bit(const v128_t *x, int i); + +void +v128_set_bit(v128_t *x, int i) ; + +void +v128_clear_bit(v128_t *x, int i); + +void +v128_set_bit_to(v128_t *x, int i, int y); + +#endif /* DATATYPES_USE_MACROS */ + +/* + * octet_string_is_eq(a,b, len) returns 1 if the length len strings a + * and b are not equal, returns 0 otherwise + */ + +int +octet_string_is_eq(uint8_t *a, uint8_t *b, int len); + +void +octet_string_set_to_zero(uint8_t *s, int len); + + +#ifndef SRTP_KERNEL_LINUX + +/* + * Convert big endian integers to CPU byte order. + */ +#ifdef WORDS_BIGENDIAN +/* Nothing to do. */ +# define be32_to_cpu(x) (x) +# define be64_to_cpu(x) (x) +#elif defined(HAVE_BYTESWAP_H) +/* We have (hopefully) optimized versions in byteswap.h */ +# include +# define be32_to_cpu(x) bswap_32((x)) +# define be64_to_cpu(x) bswap_64((x)) +#else + +# ifdef HAVE_X86 +/* Fall back. */ +static inline uint32_t be32_to_cpu(uint32_t v) { + /* optimized for x86. */ + asm("bswap %0" : "=r" (v) : "0" (v)); + return v; +} +# else /* HAVE_X86 */ +# ifdef HAVE_NETINET_IN_H +# include +# elif defined HAVE_WINSOCK2_H +# include +# endif +# define be32_to_cpu(x) ntohl((x)) +# endif /* HAVE_X86 */ + +static inline uint64_t be64_to_cpu(uint64_t v) { +# ifdef NO_64BIT_MATH + /* use the make64 functions to do 64-bit math */ + v = make64(htonl(low32(v)),htonl(high32(v))); +# else + /* use the native 64-bit math */ + v= (be32_to_cpu(v >> 32)) | (((uint64_t)be32_to_cpu((uint32_t)v)) << 32); +# endif + return v; +} + +#endif /* ! SRTP_KERNEL_LINUX */ + +#endif /* WORDS_BIGENDIAN */ + +#endif /* _DATATYPES_H */ diff --git a/libs/srtp/crypto/include/err.h b/libs/srtp/crypto/include/err.h new file mode 100644 index 0000000000..37d3cb93e8 --- /dev/null +++ b/libs/srtp/crypto/include/err.h @@ -0,0 +1,174 @@ +/* + * err.h + * + * error status codes + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#ifndef ERR_H +#define ERR_H + +#include "datatypes.h" + +/** + * @defgroup Error Error Codes + * + * Error status codes are represented by the enumeration err_status_t. + * + * @{ + */ + + +/* + * @brief err_status_t defines error codes. + * + * The enumeration err_status_t defines error codes. Note that the + * value of err_status_ok is equal to zero, which can simplify error + * checking somewhat. + * + */ +typedef enum { + err_status_ok = 0, /**< nothing to report */ + err_status_fail = 1, /**< unspecified failure */ + err_status_bad_param = 2, /**< unsupported parameter */ + err_status_alloc_fail = 3, /**< couldn't allocate memory */ + err_status_dealloc_fail = 4, /**< couldn't deallocate properly */ + err_status_init_fail = 5, /**< couldn't initialize */ + err_status_terminus = 6, /**< can't process as much data as requested */ + err_status_auth_fail = 7, /**< authentication failure */ + err_status_cipher_fail = 8, /**< cipher failure */ + err_status_replay_fail = 9, /**< replay check failed (bad index) */ + err_status_replay_old = 10, /**< replay check failed (index too old) */ + err_status_algo_fail = 11, /**< algorithm failed test routine */ + err_status_no_such_op = 12, /**< unsupported operation */ + err_status_no_ctx = 13, /**< no appropriate context found */ + err_status_cant_check = 14, /**< unable to perform desired validation */ + err_status_key_expired = 15, /**< can't use key any more */ + err_status_socket_err = 16, /**< error in use of socket */ + err_status_signal_err = 17, /**< error in use POSIX signals */ + err_status_nonce_bad = 18, /**< nonce check failed */ + err_status_read_fail = 19, /**< couldn't read data */ + err_status_write_fail = 20, /**< couldn't write data */ + err_status_parse_err = 21, /**< error pasring data */ + err_status_encode_err = 22, /**< error encoding data */ + err_status_semaphore_err = 23,/**< error while using semaphores */ + err_status_pfkey_err = 24 ,/**< error while using pfkey */ +} err_status_t; + +/** + * @} + */ + +typedef enum { + err_level_emergency = 0, + err_level_alert, + err_level_critical, + err_level_error, + err_level_warning, + err_level_notice, + err_level_info, + err_level_debug, + err_level_none +} err_reporting_level_t; + +/* + * err_reporting_init prepares the error system. If + * ERR_REPORTING_SYSLOG is defined, it will open syslog. + * + * The ident argument is a string that will be prepended to + * all syslog messages. It is conventionally argv[0]. + */ + +err_status_t +err_reporting_init(char *ident); + +#ifdef SRTP_KERNEL_LINUX +extern err_reporting_level_t err_level; +#else + +/* + * keydaemon_report_error reports a 'printf' formatted error + * string, followed by a an arg list. The priority argument + * is equivalent to that defined for syslog. + * + * Errors will be reported to ERR_REPORTING_FILE, if defined, and to + * syslog, if ERR_REPORTING_SYSLOG is defined. + * + */ + +void +err_report(int priority, char *format, ...); +#endif /* ! SRTP_KERNEL_LINUX */ + + +/* + * debug_module_t defines a debug module + */ + +typedef struct { + int on; /* 1 if debugging is on, 0 if it is off */ + char *name; /* printable name for debug module */ +} debug_module_t; + +#ifdef ENABLE_DEBUGGING + +#define debug_on(mod) (mod).on = 1 + +#define debug_off(mod) (mod).on = 0 + +/* use err_report() to report debug message */ +#define debug_print(mod, format, arg) \ + if (mod.on) err_report(err_level_debug, ("%s: " format "\n"), mod.name, arg) +#define debug_print2(mod, format, arg1,arg2) \ + if (mod.on) err_report(err_level_debug, ("%s: " format "\n"), mod.name, arg1,arg2) + +#else + +/* define macros to do nothing */ +#define debug_print(mod, format, arg) + +#define debug_on(mod) + +#define debug_off(mod) + +#endif + +#endif /* ERR_H */ diff --git a/libs/srtp/crypto/include/gf2_8.h b/libs/srtp/crypto/include/gf2_8.h new file mode 100644 index 0000000000..ea40a4412b --- /dev/null +++ b/libs/srtp/crypto/include/gf2_8.h @@ -0,0 +1,79 @@ +/* + * gf2_8.h + * + * GF(256) implementation + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#ifndef GF2_8_H +#define GF2_8_H + +#include "datatypes.h" /* for uint8_t definition */ + +typedef uint8_t gf2_8; + +#define gf2_8_field_polynomial 0x1B + +/* + * gf2_8_shift(x) returns + */ + +/* + * gf2_8_shift(z) returns the result of the GF(2^8) 'multiply by x' + * operation, using the field representation from AES; that is, the + * next gf2_8 value in the cyclic representation of that field. The + * value z should be an uint8_t. + */ + +#define gf2_8_shift(z) (((z) & 128) ? \ + (((z) << 1) ^ gf2_8_field_polynomial) : ((z) << 1)) + +gf2_8 +gf2_8_compute_inverse(gf2_8 x); + +void +test_gf2_8(void); + +gf2_8 +gf2_8_multiply(gf2_8 x, gf2_8 y); + +#endif /* GF2_8_H */ diff --git a/libs/srtp/crypto/include/hmac.h b/libs/srtp/crypto/include/hmac.h new file mode 100644 index 0000000000..6abfdf8d51 --- /dev/null +++ b/libs/srtp/crypto/include/hmac.h @@ -0,0 +1,78 @@ +/* + * hmac.h + * + * interface to hmac auth_type_t + * + * David A. McGrew + * Cisco Systems, Inc. + * + */ +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#ifndef HMAC_H +#define HMAC_H + +#include "auth.h" +#include "sha1.h" + +typedef struct { + uint8_t opad[64]; + sha1_ctx_t ctx; + sha1_ctx_t init_ctx; +} hmac_ctx_t; + +err_status_t +hmac_alloc(auth_t **a, int key_len, int out_len); + +err_status_t +hmac_dealloc(auth_t *a); + +err_status_t +hmac_init(hmac_ctx_t *state, const uint8_t *key, int key_len); + +err_status_t +hmac_start(hmac_ctx_t *state); + +err_status_t +hmac_update(hmac_ctx_t *state, const uint8_t *message, int msg_octets); + +err_status_t +hmac_compute(hmac_ctx_t *state, const void *message, + int msg_octets, int tag_len, uint8_t *result); + + +#endif /* HMAC_H */ diff --git a/libs/srtp/crypto/include/integers.h b/libs/srtp/crypto/include/integers.h new file mode 100644 index 0000000000..8b55720a91 --- /dev/null +++ b/libs/srtp/crypto/include/integers.h @@ -0,0 +1,147 @@ +/* + * integers.h + * + * defines integer types (or refers to their definitions) + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#ifndef INTEGERS_H +#define INTEGERS_H + +#include "config.h" /* configuration file, using autoconf */ + +#ifdef SRTP_KERNEL + +#include "kernel_compat.h" + +#else /* SRTP_KERNEL */ + +/* use standard integer definitions, if they're available */ +#ifdef HAVE_STDLIB_H +# include +#endif +#ifdef HAVE_STDINT_H +# include +#endif +#ifdef HAVE_INTTYPES_H +# include +#endif +#ifdef HAVE_SYS_TYPES_H +# include +#endif +#ifdef HAVE_SYS_INT_TYPES_H +# include /* this exists on Sun OS */ +#endif +#ifdef HAVE_MACHINE_TYPES_H +# include +#endif + +/* Can we do 64 bit integers? */ +#ifndef HAVE_UINT64_T +# if SIZEOF_UNSIGNED_LONG == 8 +typedef unsigned long uint64_t; +# elif SIZEOF_UNSIGNED_LONG_LONG == 8 +typedef unsigned long long uint64_t; +# else +# define NO_64BIT_MATH 1 +# endif +#endif + +/* Reasonable defaults for 32 bit machines - you may need to + * edit these definitions for your own machine. */ +#ifndef HAVE_UINT8_T +typedef unsigned char uint8_t; +#endif +#ifndef HAVE_UINT16_T +typedef unsigned short int uint16_t; +#endif +#ifndef HAVE_UINT32_T +typedef unsigned int uint32_t; +#endif + + +#ifdef NO_64BIT_MATH +typedef double uint64_t; +/* assert that sizeof(double) == 8 */ +extern uint64_t make64(uint32_t high, uint32_t low); +extern uint32_t high32(uint64_t value); +extern uint32_t low32(uint64_t value); +#endif + +#endif /* SRTP_KERNEL */ + +/* These macros are to load and store 32-bit values from un-aligned + addresses. This is required for processors that do not allow unaligned + loads. */ +#ifdef ALIGNMENT_32BIT_REQUIRED +// Note that if it's in a variable, you can memcpy it +#ifdef WORDS_BIGENDIAN +#define PUT_32(addr,value) \ + { \ + ((unsigned char *) (addr))[0] = (value >> 24); \ + ((unsigned char *) (addr))[1] = (value >> 16) & 0xff; \ + ((unsigned char *) (addr))[2] = (value >> 8) & 0xff; \ + ((unsigned char *) (addr))[3] = (value) & 0xff; \ + } +#define GET_32(addr) ((((unsigned char *) (addr))[0] << 24) | \ + (((unsigned char *) (addr))[1] << 16) | \ + (((unsigned char *) (addr))[2] << 8) | \ + (((unsigned char *) (addr))[3])) +#else +#define PUT_32(addr,value) \ + { \ + ((unsigned char *) (addr))[3] = (value >> 24); \ + ((unsigned char *) (addr))[2] = (value >> 16) & 0xff; \ + ((unsigned char *) (addr))[1] = (value >> 8) & 0xff; \ + ((unsigned char *) (addr))[0] = (value) & 0xff; \ + } +#define GET_32(addr) ((((unsigned char *) (addr))[3] << 24) | \ + (((unsigned char *) (addr))[2] << 16) | \ + (((unsigned char *) (addr))[1] << 8) | \ + (((unsigned char *) (addr))[0])) +#endif // WORDS_BIGENDIAN +#else +#define PUT_32(addr,value) *(((uint32_t *) (addr)) = (value) +#define GET_32(addr) (*(((uint32_t *) (addr))) +#endif + +#endif /* INTEGERS_H */ diff --git a/libs/srtp/crypto/include/kernel_compat.h b/libs/srtp/crypto/include/kernel_compat.h new file mode 100644 index 0000000000..59d1898e1b --- /dev/null +++ b/libs/srtp/crypto/include/kernel_compat.h @@ -0,0 +1,84 @@ +/* + * kernel_compat.h + * + * Compatibility stuff for building in kernel context where standard + * C headers and library are not available. + * + * Marcus Sundberg + * Ingate Systems AB + */ +/* + * + * Copyright(c) 2005 Ingate Systems AB + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the author(s) nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#ifndef KERNEL_COMPAT_H +#define KERNEL_COMPAT_H + +#ifdef SRTP_KERNEL_LINUX + +#include +#include +#include +#include +#include + + +#define err_report(priority, ...) \ + do {\ + if (priority <= err_level) {\ + printk(__VA_ARGS__);\ + }\ + }while(0) + +#define clock() (jiffies) +#define time(x) (jiffies) + +/* rand() implementation. */ +#define RAND_MAX 32767 + +static inline int rand(void) +{ + uint32_t temp; + get_random_bytes(&temp, sizeof(temp)); + return temp % (RAND_MAX+1); +} + +/* stdio/stdlib implementation. */ +#define printf(...) printk(__VA_ARGS__) +#define exit(n) panic("%s:%d: exit(%d)\n", __FILE__, __LINE__, (n)) + +#endif /* SRTP_KERNEL_LINUX */ + +#endif /* KERNEL_COMPAT_H */ diff --git a/libs/srtp/crypto/include/key.h b/libs/srtp/crypto/include/key.h new file mode 100644 index 0000000000..05a0669c0a --- /dev/null +++ b/libs/srtp/crypto/include/key.h @@ -0,0 +1,82 @@ +/* + * key.h + * + * key usage limits enforcement + * + * David A. Mcgrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright (c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#ifndef KEY_H +#define KEY_H + +#include "rdbx.h" /* for xtd_seq_num_t */ +#include "err.h" + +typedef struct key_limit_ctx_t *key_limit_t; + +typedef enum { + key_event_normal, + key_event_soft_limit, + key_event_hard_limit +} key_event_t; + +err_status_t +key_limit_set(key_limit_t key, xtd_seq_num_t s); + +err_status_t +key_limit_clone(key_limit_t original, key_limit_t *new_key); + +err_status_t +key_limit_check(const key_limit_t key); + +key_event_t +key_limit_update(key_limit_t key); + +typedef enum { + key_state_normal, + key_state_past_soft_limit, + key_state_expired +} key_state_t; + +typedef struct key_limit_ctx_t { + xtd_seq_num_t num_left; + key_state_t state; +} key_limit_ctx_t; + +#endif /* KEY_H */ diff --git a/libs/srtp/crypto/include/null_auth.h b/libs/srtp/crypto/include/null_auth.h new file mode 100644 index 0000000000..cebb86b1a5 --- /dev/null +++ b/libs/srtp/crypto/include/null_auth.h @@ -0,0 +1,68 @@ +/* + * null-auth.h + * + * David A. McGrew + * Cisco Systems, Inc. + * + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#ifndef NULL_AUTH_H +#define NULL_AUTH_H + +#include "auth.h" + +typedef struct { + char foo; +} null_auth_ctx_t; + +err_status_t +null_auth_alloc(auth_t **a, int key_len, int out_len); + +err_status_t +null_auth_dealloc(auth_t *a); + +err_status_t +null_auth_init(null_auth_ctx_t *state, const uint8_t *key, int key_len); + +err_status_t +null_auth_compute (null_auth_ctx_t *state, uint8_t *message, + int msg_octets, int tag_len, uint8_t *result); + + +#endif /* NULL_AUTH_H */ diff --git a/libs/srtp/crypto/include/null_cipher.h b/libs/srtp/crypto/include/null_cipher.h new file mode 100644 index 0000000000..7f348be5ff --- /dev/null +++ b/libs/srtp/crypto/include/null_cipher.h @@ -0,0 +1,80 @@ +/* + * null-cipher.h + * + * header file for the null cipher + * + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#ifndef NULL_CIPHER_H +#define NULL_CIPHER_H + +#include "datatypes.h" +#include "cipher.h" + +typedef struct { + char foo ;/* empty, for now */ +} null_cipher_ctx_t; + + +/* + * none of these functions do anything (though future versions may keep + * track of bytes encrypted, number of instances, and/or other info). + */ + +err_status_t +null_cipher_init(null_cipher_ctx_t *c, const uint8_t *key); + +err_status_t +null_cipher_set_segment(null_cipher_ctx_t *c, + unsigned long index); + +err_status_t +null_cipher_encrypt(null_cipher_ctx_t *c, + unsigned char *buf, unsigned int *bytes_to_encr); + + +err_status_t +null_cipher_encrypt_aligned(null_cipher_ctx_t *c, + unsigned char *buf, int bytes_to_encr); + +#endif /* NULL_CIPHER_H */ diff --git a/libs/srtp/crypto/include/prng.h b/libs/srtp/crypto/include/prng.h new file mode 100644 index 0000000000..fb96b5ebaa --- /dev/null +++ b/libs/srtp/crypto/include/prng.h @@ -0,0 +1,54 @@ +/* + * prng.h + * + * pseudorandom source + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +#ifndef PRNG_H +#define PRNG_H + +#include "rand_source.h" /* for rand_source_func_t definition */ +#include "aes.h" /* for aes */ +#include "aes_icm.h" /* for aes ctr */ + +#define MAX_PRNG_OUT_LEN 0xffffffffU + +/* + * x917_prng is an ANSI X9.17-like AES-based PRNG + */ + +typedef struct { + v128_t state; /* state data */ + aes_expanded_key_t key; /* secret key */ + uint32_t octet_count; /* number of octets output since last init */ + rand_source_func_t rand; /* random source for re-initialization */ +} x917_prng_t; + +err_status_t +x917_prng_init(rand_source_func_t random_source); + +err_status_t +x917_prng_get_octet_string(uint8_t *dest, uint32_t len); + + +/* + * ctr_prng is an AES-CTR based PRNG + */ + +typedef struct { + uint32_t octet_count; /* number of octets output since last init */ + aes_icm_ctx_t state; /* state data */ + rand_source_func_t rand; /* random source for re-initialization */ +} ctr_prng_t; + +err_status_t +ctr_prng_init(rand_source_func_t random_source); + +err_status_t +ctr_prng_get_octet_string(void *dest, uint32_t len); + + +#endif diff --git a/libs/srtp/crypto/include/rand_source.h b/libs/srtp/crypto/include/rand_source.h new file mode 100644 index 0000000000..19a3f94b1b --- /dev/null +++ b/libs/srtp/crypto/include/rand_source.h @@ -0,0 +1,91 @@ +/* + * rand_source.h + * + * implements a random source based on /dev/random + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright(c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#ifndef RAND_SOURCE +#define RAND_SOURCE + +#include "err.h" +#include "datatypes.h" + +err_status_t +rand_source_init(void); + +/* + * rand_source_get_octet_string() writes a random octet string. + * + * The function call rand_source_get_octet_string(dest, len) writes + * len octets of random data to the location to which dest points, + * and returns an error code. This error code should be checked, + * and if a failure is reported, the data in the buffer MUST NOT + * be used. + * + * warning: If the return code is not checked, then non-random + * data may inadvertently be used. + * + * returns: + * - err_status_ok if no problems occured. + * - [other] a problem occured, and no assumptions should + * be made about the contents of the destination + * buffer. + */ + +err_status_t +rand_source_get_octet_string(void *dest, uint32_t length); + +err_status_t +rand_source_deinit(void); + +/* + * function prototype for a random source function + * + * A rand_source_func_t writes num_octets at the location indicated by + * dest and returns err_status_ok. Any other return value indicates + * failure. + */ + +typedef err_status_t (*rand_source_func_t) + (void *dest, uint32_t num_octets); + +#endif /* RAND_SOURCE */ diff --git a/libs/srtp/crypto/include/rdb.h b/libs/srtp/crypto/include/rdb.h new file mode 100644 index 0000000000..5a26c5e303 --- /dev/null +++ b/libs/srtp/crypto/include/rdb.h @@ -0,0 +1,94 @@ +/* + * replay-database.h + * + * interface for a replay database for packet security + * + * David A. McGrew + * Cisco Systems, Inc. + */ + + +#ifndef REPLAY_DB_H +#define REPLAY_DB_H + +#include "integers.h" /* for uint32_t */ +#include "datatypes.h" /* for v128_t */ +#include "err.h" /* for err_status_t */ + +/* + * if the ith least significant bit is one, then the packet index + * window_end-i is in the database + */ + +typedef struct { + uint32_t window_start; /* packet index of the first bit in bitmask */ + v128_t bitmask; +} rdb_t; + +#define rdb_bits_in_bitmask (8*sizeof(v128_t)) + +/* + * rdb init + * + * initalizes rdb + * + * returns err_status_ok on success, err_status_t_fail otherwise + */ + +err_status_t +rdb_init(rdb_t *rdb); + + +/* + * rdb_check + * + * checks to see if index appears in rdb + * + * returns err_status_fail if the index already appears in rdb, + * returns err_status_ok otherwise + */ + +err_status_t +rdb_check(const rdb_t *rdb, uint32_t index); + +/* + * rdb_add_index + * + * adds index to rdb_t (and does *not* check if index appears in db) + * + * returns err_status_ok on success, err_status_fail otherwise + * + */ + +err_status_t +rdb_add_index(rdb_t *rdb, uint32_t index); + +/* + * the functions rdb_increment() and rdb_get_value() are for use by + * senders, not receivers - DO NOT use these functions on the same + * rdb_t upon which rdb_add_index is used! + */ + + +/* + * rdb_increment(db) increments the sequence number in db, if it is + * not too high + * + * return values: + * + * err_status_ok no problem + * err_status_key_expired sequence number too high + * + */ +err_status_t +rdb_increment(rdb_t *rdb); + +/* + * rdb_get_value(db) returns the current sequence number of db + */ + +uint32_t +rdb_get_value(const rdb_t *rdb); + + +#endif /* REPLAY_DB_H */ diff --git a/libs/srtp/crypto/include/rdbx.h b/libs/srtp/crypto/include/rdbx.h new file mode 100644 index 0000000000..ce9ecf6f47 --- /dev/null +++ b/libs/srtp/crypto/include/rdbx.h @@ -0,0 +1,146 @@ +/* + * rdbx.h + * + * replay database with extended packet indices, using a rollover counter + * + * David A. McGrew + * Cisco Systems, Inc. + * + */ + +#ifndef RDBX_H +#define RDBX_H + +#include "datatypes.h" +#include "err.h" + +/* #define ROC_TEST */ + +#ifndef ROC_TEST + +typedef uint16_t sequence_number_t; /* 16 bit sequence number */ +typedef uint32_t rollover_counter_t; /* 32 bit rollover counter */ + +#else /* use small seq_num and roc datatypes for testing purposes */ + +typedef unsigned char sequence_number_t; /* 8 bit sequence number */ +typedef uint16_t rollover_counter_t; /* 16 bit rollover counter */ + +#endif + +#define seq_num_median (1 << (8*sizeof(sequence_number_t) - 1)) +#define seq_num_max (1 << (8*sizeof(sequence_number_t))) + +/* + * An xtd_seq_num_t is a 64-bit unsigned integer used as an 'extended' + * sequence number. + */ + +typedef uint64_t xtd_seq_num_t; + + +/* + * An rdbx_t is a replay database with extended range; it uses an + * xtd_seq_num_t and a bitmask of recently received indices. + */ + +typedef struct { + xtd_seq_num_t index; + v128_t bitmask; +} rdbx_t; + + +/* + * rdbx_init(rdbx_ptr) + * + * initializes the rdbx pointed to by its argument, setting the + * rollover counter and sequence number to zero + */ + +err_status_t +rdbx_init(rdbx_t *rdbx); + + +/* + * rdbx_estimate_index(rdbx, guess, s) + * + * given an rdbx and a sequence number s (from a newly arrived packet), + * sets the contents of *guess to contain the best guess of the packet + * index to which s corresponds, and returns the difference between + * *guess and the locally stored synch info + */ + +int +rdbx_estimate_index(const rdbx_t *rdbx, + xtd_seq_num_t *guess, + sequence_number_t s); + +/* + * rdbx_check(rdbx, delta); + * + * rdbx_check(&r, delta) checks to see if the xtd_seq_num_t + * which is at rdbx->window_start + delta is in the rdb + * + */ + +err_status_t +rdbx_check(const rdbx_t *rdbx, int difference); + +/* + * replay_add_index(rdbx, delta) + * + * adds the xtd_seq_num_t at rdbx->window_start + delta to replay_db + * (and does *not* check if that xtd_seq_num_t appears in db) + * + * this function should be called *only* after replay_check has + * indicated that the index does not appear in the rdbx, and a mutex + * should protect the rdbx between these calls if necessary. + */ + +err_status_t +rdbx_add_index(rdbx_t *rdbx, int delta); + +/* + * xtd_seq_num_t functions - these are *internal* functions of rdbx, and + * shouldn't be used to manipulate rdbx internal values. use the rdbx + * api instead! + */ + + +/* index_init(&pi) initializes a packet index pi (sets it to zero) */ + +void +index_init(xtd_seq_num_t *pi); + +/* index_advance(&pi, s) advances a xtd_seq_num_t forward by s */ + +void +index_advance(xtd_seq_num_t *pi, sequence_number_t s); + + +/* + * index_guess(local, guess, s) + * + * given a xtd_seq_num_t local (which represents the highest + * known-to-be-good index) and a sequence number s (from a newly + * arrived packet), sets the contents of *guess to contain the best + * guess of the packet index to which s corresponds, and returns the + * difference between *guess and *local + */ + +int +index_guess(const xtd_seq_num_t *local, + xtd_seq_num_t *guess, + sequence_number_t s); + + +#endif /* RDBX_H */ + + + + + + + + + diff --git a/libs/srtp/crypto/include/sha1.h b/libs/srtp/crypto/include/sha1.h new file mode 100644 index 0000000000..1d61c5832c --- /dev/null +++ b/libs/srtp/crypto/include/sha1.h @@ -0,0 +1,108 @@ +/* + * sha1.h + * + * interface to the Secure Hash Algorithm v.1 (SHA-1), specified in + * FIPS 180-1 + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#ifndef SHA1_H +#define SHA1_H + +#include "err.h" +#include "datatypes.h" + +typedef struct { + uint32_t H[5]; /* state vector */ + uint32_t M[16]; /* message buffer */ + int octets_in_buffer; /* octets of message in buffer */ + uint32_t num_bits_in_msg; /* total number of bits in message */ +} sha1_ctx_t; + +/* + * sha1(&ctx, msg, len, output) hashes the len octets starting at msg + * into the SHA1 context, then writes the result to the 20 octets at + * output + * + */ + +void +sha1(const uint8_t *message, int octets_in_msg, uint32_t output[5]); + +/* + * sha1_init(&ctx) initializes the SHA1 context ctx + * + * sha1_update(&ctx, msg, len) hashes the len octets starting at msg + * into the SHA1 context + * + * sha1_final(&ctx, output) performs the final processing of the SHA1 + * context and writes the result to the 20 octets at output + * + */ + +void +sha1_init(sha1_ctx_t *ctx); + +void +sha1_update(sha1_ctx_t *ctx, const uint8_t *M, int octets_in_msg); + +void +sha1_final(sha1_ctx_t *ctx, uint32_t output[5]); + +/* + * The sha1_core function is INTERNAL to SHA-1, but it is declared + * here because it is also used by the cipher SEAL 3.0 in its key + * setup algorithm. + */ + +/* + * sha1_core(M, H) computes the core sha1 compression function, where M is + * the next part of the message and H is the intermediate state {H0, + * H1, ...} + * + * this function does not do any of the padding required in the + * complete sha1 function + */ + +void +sha1_core(const uint32_t M[16], uint32_t hash_value[5]); + +#endif /* SHA1_H */ diff --git a/libs/srtp/crypto/include/stat.h b/libs/srtp/crypto/include/stat.h new file mode 100644 index 0000000000..1fe4f2f481 --- /dev/null +++ b/libs/srtp/crypto/include/stat.h @@ -0,0 +1,69 @@ +/* + * stats.h + * + * interface to statistical test functions + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright(c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#ifndef STAT_H +#define STAT_H + +#include "datatypes.h" /* for uint8_t */ +#include "err.h" /* for err_status_t */ +#include "rand_source.h" /* for rand_source_func_t definition */ + +err_status_t +stat_test_monobit(uint8_t *data); + +err_status_t +stat_test_poker(uint8_t *data); + +err_status_t +stat_test_runs(uint8_t *data); + +err_status_t +stat_test_rand_source(rand_source_func_t rs); + +err_status_t +stat_test_rand_source_with_repetition(rand_source_func_t source, unsigned num_trials); + +#endif /* STAT_H */ diff --git a/libs/srtp/crypto/include/xfm.h b/libs/srtp/crypto/include/xfm.h new file mode 100644 index 0000000000..5837149b61 --- /dev/null +++ b/libs/srtp/crypto/include/xfm.h @@ -0,0 +1,139 @@ +/* + * xfm.h + * + * interface for abstract crypto transform + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +#ifndef XFM_H +#define XFM_H + +#include "crypto_kernel.h" +#include "err.h" + +/** + * @defgroup Crypto Cryptography + * + * A simple interface to an abstract cryptographic transform that + * provides both confidentiality and message authentication. + * + * @{ + */ + +/** + * @brief applies a crypto transform + * + * The function pointer xfm_func_t points to a function that + * implements a crypto transform, and provides a uniform API for + * accessing crypto mechanisms. + * + * @param key location of secret key + * + * @param clear data to be authenticated only + * + * @param clear_len length of data to be authenticated only + * + * @param iv location to write the Initialization Vector (IV) + * + * @param protect location of the data to be encrypted and + * authenticated (before the function call), and the ciphertext + * and authentication tag (after the call) + * + * @param protected_len location of the length of the data to be + * encrypted and authenticated (before the function call), and the + * length of the ciphertext (after the call) + * + * @param auth_tag location to write auth tag + */ + +typedef err_status_t (*xfm_func_t) + (void *key, + void *clear, + unsigned clear_len, + void *iv, + void *protect, + unsigned *protected_len, + void *auth_tag + ); + +typedef +err_status_t (*xfm_inv_t) + (void *key, /* location of secret key */ + void *clear, /* data to be authenticated only */ + unsigned clear_len, /* length of data to be authenticated only */ + void *iv, /* location of iv */ + void *opaque, /* data to be decrypted and authenticated */ + unsigned *opaque_len, /* location of the length of data to be + * decrypted and authd (before and after) + */ + void *auth_tag /* location of auth tag */ + ); + +typedef struct xfm_ctx_t { + xfm_func_t func; + xfm_inv_t inv; + unsigned key_len; + unsigned iv_len; + unsigned auth_tag_len; +} xfm_ctx_t; + +typedef xfm_ctx_t *xfm_t; + +#define xfm_get_key_len(xfm) ((xfm)->key_len) + +#define xfm_get_iv_len(xfm) ((xfm)->iv_len) + +#define xfm_get_auth_tag_len(xfm) ((xfm)->auth_tag_len) + + +/* cryptoalgo - 5/28 */ + +typedef err_status_t (*cryptoalg_func_t) + (void *key, + void *clear, + unsigned clear_len, + void *iv, + void *opaque, + unsigned *opaque_len + ); + +typedef +err_status_t (*cryptoalg_inv_t) + (void *key, /* location of secret key */ + void *clear, /* data to be authenticated only */ + unsigned clear_len, /* length of data to be authenticated only */ + void *iv, /* location of iv */ + void *opaque, /* data to be decrypted and authenticated */ + unsigned *opaque_len /* location of the length of data to be + * decrypted and authd (before and after) + */ + ); + +typedef struct cryptoalg_ctx_t { + cryptoalg_func_t enc; + cryptoalg_inv_t dec; + unsigned key_len; + unsigned iv_len; + unsigned auth_tag_len; + unsigned max_expansion; +} cryptoalg_ctx_t; + +typedef cryptoalg_ctx_t *cryptoalg_t; + +#define cryptoalg_get_key_len(cryptoalg) ((cryptoalg)->key_len) + +#define cryptoalg_get_iv_len(cryptoalg) ((cryptoalg)->iv_len) + +#define cryptoalg_get_auth_tag_len(cryptoalg) ((cryptoalg)->auth_tag_len) + + + +/** + * @} + */ + +#endif /* XFM_H */ + + diff --git a/libs/srtp/crypto/kernel/CVS/Entries b/libs/srtp/crypto/kernel/CVS/Entries new file mode 100644 index 0000000000..64880fadec --- /dev/null +++ b/libs/srtp/crypto/kernel/CVS/Entries @@ -0,0 +1,5 @@ +/alloc.c/1.3/Wed Oct 5 11:50:56 2005// +/crypto_kernel.c/1.5/Fri Mar 17 20:51:24 2006// +/err.c/1.5/Tue Oct 18 15:26:31 2005// +/key.c/1.6/Sun Oct 2 20:33:10 2005// +D diff --git a/libs/srtp/crypto/kernel/CVS/Repository b/libs/srtp/crypto/kernel/CVS/Repository new file mode 100644 index 0000000000..87ab491c42 --- /dev/null +++ b/libs/srtp/crypto/kernel/CVS/Repository @@ -0,0 +1 @@ +srtp/crypto/kernel diff --git a/libs/srtp/crypto/kernel/CVS/Root b/libs/srtp/crypto/kernel/CVS/Root new file mode 100644 index 0000000000..b4b74237d9 --- /dev/null +++ b/libs/srtp/crypto/kernel/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@cvs.sourceforge.net:/cvsroot/srtp diff --git a/libs/srtp/crypto/kernel/alloc.c b/libs/srtp/crypto/kernel/alloc.c new file mode 100644 index 0000000000..9ff5898f06 --- /dev/null +++ b/libs/srtp/crypto/kernel/alloc.c @@ -0,0 +1,119 @@ +/* + * alloc.c + * + * memory allocation and deallocation + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright (c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include "alloc.h" +#include "crypto_kernel.h" + +/* the debug module for memory allocation */ + +debug_module_t mod_alloc = { + 0, /* debugging is off by default */ + "alloc" /* printable name for module */ +}; + +/* + * Nota bene: the debugging statements for crypto_alloc() and + * crypto_free() have identical prefixes, which include the addresses + * of the memory locations on which they are operating. This fact can + * be used to locate memory leaks, by turning on memory debugging, + * grepping for 'alloc', then matching alloc and free calls by + * address. + */ + +#ifdef SRTP_KERNEL_LINUX + +#include + +void * +crypto_alloc(size_t size) { + void *ptr; + + ptr = kmalloc(size, in_interrupt() ? GFP_ATOMIC : GFP_KERNEL); + + if (ptr) { + debug_print(mod_alloc, "(location: %p) allocated", ptr); + } else + debug_print(mod_alloc, "allocation failed (asked for %d bytes)\n", size); + + return ptr; +} + +void +crypto_free(void *ptr) { + + debug_print(mod_alloc, "(location: %p) freed", ptr); + + kfree(ptr); +} + + +#elif defined(HAVE_STDLIB_H) + +void * +crypto_alloc(size_t size) { + void *ptr; + + ptr = malloc(size); + + if (ptr) { + debug_print(mod_alloc, "(location: %p) allocated", ptr); + } else + debug_print(mod_alloc, "allocation failed (asked for %d bytes)\n", size); + + return ptr; +} + +void +crypto_free(void *ptr) { + + debug_print(mod_alloc, "(location: %p) freed", ptr); + + free(ptr); +} + +#else /* we need to define our own memory allocation routines */ + +#error no memory allocation defined yet + +#endif diff --git a/libs/srtp/crypto/kernel/crypto_kernel.c b/libs/srtp/crypto/kernel/crypto_kernel.c new file mode 100644 index 0000000000..23131c9c54 --- /dev/null +++ b/libs/srtp/crypto/kernel/crypto_kernel.c @@ -0,0 +1,523 @@ +/* + * crypto_kernel.c + * + * header for the cryptographic kernel + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright(c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#include "alloc.h" + +#include "crypto_kernel.h" + +/* the debug module for the crypto_kernel */ + +debug_module_t mod_crypto_kernel = { + 0, /* debugging is off by default */ + "crypto kernel" /* printable name for module */ +}; + +/* + * other debug modules that can be included in the kernel + */ + +extern debug_module_t mod_auth; +extern debug_module_t mod_cipher; +extern debug_module_t mod_stat; +extern debug_module_t mod_alloc; + +/* + * cipher types that can be included in the kernel + */ + +extern cipher_type_t null_cipher; +extern cipher_type_t aes_icm; +extern cipher_type_t aes_cbc; + + +/* + * auth func types that can be included in the kernel + */ + +extern auth_type_t null_auth; +extern auth_type_t hmac; + +/* crypto_kernel is a global variable, the only one of its datatype */ + +crypto_kernel_t +crypto_kernel = { + crypto_kernel_state_insecure, /* start off in insecure state */ + NULL, /* no cipher types yet */ + NULL, /* no auth types yet */ + NULL /* no debug modules yet */ +}; + +#define MAX_RNG_TRIALS 25 + +err_status_t +crypto_kernel_init() { + err_status_t status; + + /* check the security state */ + if (crypto_kernel.state == crypto_kernel_state_secure) { + + /* + * we're already in the secure state, but we've been asked to + * re-initialize, so we just re-run the self-tests and then return + */ + return crypto_kernel_status(); + } + + /* initialize error reporting system */ + status = err_reporting_init("crypto"); + if (status) + return status; + + /* load debug modules */ + status = crypto_kernel_load_debug_module(&mod_crypto_kernel); + if (status) + return status; + status = crypto_kernel_load_debug_module(&mod_auth); + if (status) + return status; + status = crypto_kernel_load_debug_module(&mod_cipher); + if (status) + return status; + status = crypto_kernel_load_debug_module(&mod_stat); + if (status) + return status; + status = crypto_kernel_load_debug_module(&mod_alloc); + if (status) + return status; + + /* initialize random number generator */ + status = rand_source_init(); + if (status) + return status; + + /* run FIPS-140 statistical tests on rand_source */ + status = stat_test_rand_source_with_repetition(rand_source_get_octet_string, MAX_RNG_TRIALS); + if (status) + return status; + + /* initialize pseudorandom number generator */ + status = ctr_prng_init(rand_source_get_octet_string); + if (status) + return status; + + /* run FIPS-140 statistical tests on ctr_prng */ + status = stat_test_rand_source_with_repetition(ctr_prng_get_octet_string, MAX_RNG_TRIALS); + if (status) + return status; + + /* load cipher types */ + status = crypto_kernel_load_cipher_type(&null_cipher, NULL_CIPHER); + if (status) + return status; + status = crypto_kernel_load_cipher_type(&aes_icm, AES_128_ICM); + if (status) + return status; + status = crypto_kernel_load_cipher_type(&aes_cbc, AES_128_CBC); + if (status) + return status; + + /* load auth func types */ + status = crypto_kernel_load_auth_type(&null_auth, NULL_AUTH); + if (status) + return status; + status = crypto_kernel_load_auth_type(&hmac, HMAC_SHA1); + if (status) + return status; + + /* change state to secure */ + crypto_kernel.state = crypto_kernel_state_secure; + + return err_status_ok; +} + +err_status_t +crypto_kernel_status() { + err_status_t status; + kernel_cipher_type_t *ctype = crypto_kernel.cipher_type_list; + kernel_auth_type_t *atype = crypto_kernel.auth_type_list; + kernel_debug_module_t *dm = crypto_kernel.debug_module_list; + + /* run FIPS-140 statistical tests on rand_source */ + printf("testing rand_source..."); + status = stat_test_rand_source_with_repetition(rand_source_get_octet_string, MAX_RNG_TRIALS); + if (status) { + printf("failed\n"); + crypto_kernel.state = crypto_kernel_state_insecure; + return status; + } + printf("passed\n"); + + /* for each cipher type, describe and test */ + while(ctype != NULL) { + printf("cipher: %s\n", ctype->cipher_type->description); + printf(" instance count: %d\n", ctype->cipher_type->ref_count); + printf(" self-test: "); + status = cipher_type_self_test(ctype->cipher_type); + if (status) { + printf("failed with error code %d\n", status); + exit(status); + } + printf("passed\n"); + ctype = ctype->next; + } + + /* for each auth type, describe and test */ + while(atype != NULL) { + printf("auth func: %s\n", atype->auth_type->description); + printf(" instance count: %d\n", atype->auth_type->ref_count); + printf(" self-test: "); + status = auth_type_self_test(atype->auth_type); + if (status) { + printf("failed with error code %d\n", status); + exit(status); + } + printf("passed\n"); + atype = atype->next; + } + + /* describe each debug module */ + printf("debug modules loaded:\n"); + while (dm != NULL) { + printf(" %s ", dm->mod->name); + if (dm->mod->on) + printf("(on)\n"); + else + printf("(off)\n"); + dm = dm->next; + } + + return err_status_ok; +} + +err_status_t +crypto_kernel_list_debug_modules() { + kernel_debug_module_t *dm = crypto_kernel.debug_module_list; + + /* describe each debug module */ + printf("debug modules loaded:\n"); + while (dm != NULL) { + printf(" %s ", dm->mod->name); + if (dm->mod->on) + printf("(on)\n"); + else + printf("(off)\n"); + dm = dm->next; + } + + return err_status_ok; +} + +err_status_t +crypto_kernel_shutdown() { + err_status_t status; + + /* + * free dynamic memory used in crypto_kernel at present + */ + + /* walk down cipher type list, freeing memory */ + while (crypto_kernel.cipher_type_list != NULL) { + kernel_cipher_type_t *ctype = crypto_kernel.cipher_type_list; + crypto_kernel.cipher_type_list = ctype->next; + debug_print(mod_crypto_kernel, + "freeing memory for cipher %s", + ctype->cipher_type->description); + crypto_free(ctype); + } + + /* walk down authetication module list, freeing memory */ + while (crypto_kernel.auth_type_list != NULL) { + kernel_auth_type_t *atype = crypto_kernel.auth_type_list; + crypto_kernel.auth_type_list = atype->next; + debug_print(mod_crypto_kernel, + "freeing memory for authentication %s", + atype->auth_type->description); + crypto_free(atype); + } + + /* walk down debug module list, freeing memory */ + while (crypto_kernel.debug_module_list != NULL) { + kernel_debug_module_t *kdm = crypto_kernel.debug_module_list; + crypto_kernel.debug_module_list = kdm->next; + debug_print(mod_crypto_kernel, + "freeing memory for debug module %s", + kdm->mod->name); + crypto_free(kdm); + } + + /* de-initialize random number generator */ status = rand_source_deinit(); + if (status) + return status; + + /* return to insecure state */ + crypto_kernel.state = crypto_kernel_state_insecure; + + return err_status_ok; +} + +err_status_t +crypto_kernel_load_cipher_type(cipher_type_t *new_ct, cipher_type_id_t id) { + kernel_cipher_type_t *ctype, *new; + err_status_t status; + + /* defensive coding */ + if (new_ct == NULL) + return err_status_bad_param; + + /* check cipher type by running self-test */ + status = cipher_type_self_test(new_ct); + if (status) { + return status; + } + + /* walk down list, checking if this type is in the list already */ + ctype = crypto_kernel.cipher_type_list; + while (ctype != NULL) { + if ((new_ct == ctype->cipher_type) || (id == ctype->id)) + return err_status_bad_param; + ctype = ctype->next; + } + + /* put new_ct at the head of the list */ + /* allocate memory */ + new = (kernel_cipher_type_t *) crypto_alloc(sizeof(kernel_cipher_type_t)); + if (new == NULL) + return err_status_alloc_fail; + + /* set fields */ + new->cipher_type = new_ct; + new->id = id; + new->next = crypto_kernel.cipher_type_list; + + /* set head of list to new cipher type */ + crypto_kernel.cipher_type_list = new; + + /* load debug module, if there is one present */ + if (new_ct->debug != NULL) + crypto_kernel_load_debug_module(new_ct->debug); + /* we could check for errors here */ + + return err_status_ok; +} + +err_status_t +crypto_kernel_load_auth_type(auth_type_t *new_at, auth_type_id_t id) { + kernel_auth_type_t *atype, *new; + err_status_t status; + + /* defensive coding */ + if (new_at == NULL) + return err_status_bad_param; + + /* check auth type by running self-test */ + status = auth_type_self_test(new_at); + if (status) { + return status; + } + + /* walk down list, checking if this type is in the list already */ + atype = crypto_kernel.auth_type_list; + while (atype != NULL) { + if ((new_at == atype->auth_type) || (id == atype->id)) + return err_status_bad_param; + atype = atype->next; + } + + /* put new_at at the head of the list */ + /* allocate memory */ + new = (kernel_auth_type_t *)crypto_alloc(sizeof(kernel_auth_type_t)); + if (new == NULL) + return err_status_alloc_fail; + + /* set fields */ + new->auth_type = new_at; + new->id = id; + new->next = crypto_kernel.auth_type_list; + + /* set head of list to new auth type */ + crypto_kernel.auth_type_list = new; + + /* load debug module, if there is one present */ + if (new_at->debug != NULL) + crypto_kernel_load_debug_module(new_at->debug); + /* we could check for errors here */ + + return err_status_ok; + +} + + +cipher_type_t * +crypto_kernel_get_cipher_type(cipher_type_id_t id) { + kernel_cipher_type_t *ctype; + + /* walk down list, looking for id */ + ctype = crypto_kernel.cipher_type_list; + while (ctype != NULL) { + if (id == ctype->id) + return ctype->cipher_type; + ctype = ctype->next; + } + + /* haven't found the right one, indicate failure by returning NULL */ + return NULL; +} + + +err_status_t +crypto_kernel_alloc_cipher(cipher_type_id_t id, + cipher_pointer_t *cp, + int key_len) { + cipher_type_t *ct; + + /* + * if the crypto_kernel is not yet initialized, we refuse to allocate + * any ciphers - this is a bit extra-paranoid + */ + if (crypto_kernel.state != crypto_kernel_state_secure) + return err_status_init_fail; + + ct = crypto_kernel_get_cipher_type(id); + if (!ct) + return err_status_fail; + + return ((ct)->alloc(cp, key_len)); +} + + + +auth_type_t * +crypto_kernel_get_auth_type(auth_type_id_t id) { + kernel_auth_type_t *atype; + + /* walk down list, looking for id */ + atype = crypto_kernel.auth_type_list; + while (atype != NULL) { + if (id == atype->id) + return atype->auth_type; + atype = atype->next; + } + + /* haven't found the right one, indicate failure by returning NULL */ + return NULL; +} + +err_status_t +crypto_kernel_alloc_auth(auth_type_id_t id, + auth_pointer_t *ap, + int key_len, + int tag_len) { + auth_type_t *at; + + /* + * if the crypto_kernel is not yet initialized, we refuse to allocate + * any auth functions - this is a bit extra-paranoid + */ + if (crypto_kernel.state != crypto_kernel_state_secure) + return err_status_init_fail; + + at = crypto_kernel_get_auth_type(id); + if (!at) + return err_status_fail; + + return ((at)->alloc(ap, key_len, tag_len)); +} + +err_status_t +crypto_kernel_load_debug_module(debug_module_t *new_dm) { + kernel_debug_module_t *kdm, *new; + + /* defensive coding */ + if (new_dm == NULL) + return err_status_bad_param; + + /* walk down list, checking if this type is in the list already */ + kdm = crypto_kernel.debug_module_list; + while (kdm != NULL) { + if (strncmp(new_dm->name, kdm->mod->name, 64) == 0) + return err_status_bad_param; + kdm = kdm->next; + } + + /* put new_dm at the head of the list */ + /* allocate memory */ + new = (kernel_debug_module_t *)crypto_alloc(sizeof(kernel_debug_module_t)); + if (new == NULL) + return err_status_alloc_fail; + + /* set fields */ + new->mod = new_dm; + new->next = crypto_kernel.debug_module_list; + + /* set head of list to new cipher type */ + crypto_kernel.debug_module_list = new; + + return err_status_ok; +} + +err_status_t +crypto_kernel_set_debug_module(char *name, int on) { + kernel_debug_module_t *kdm; + + /* walk down list, checking if this type is in the list already */ + kdm = crypto_kernel.debug_module_list; + while (kdm != NULL) { + if (strncmp(name, kdm->mod->name, 64) == 0) { + kdm->mod->on = on; + return err_status_ok; + } + kdm = kdm->next; + } + + return err_status_fail; +} + +err_status_t +crypto_get_random(unsigned char *buffer, unsigned int length) { + if (crypto_kernel.state == crypto_kernel_state_secure) + return ctr_prng_get_octet_string(buffer, length); + else + return err_status_fail; +} diff --git a/libs/srtp/crypto/kernel/err.c b/libs/srtp/crypto/kernel/err.c new file mode 100644 index 0000000000..e64719c346 --- /dev/null +++ b/libs/srtp/crypto/kernel/err.c @@ -0,0 +1,148 @@ +/* + * err.c + * + * error status reporting functions + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright(c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include "err.h" + +#ifdef ERR_REPORTING_SYSLOG +# ifdef HAVE_SYSLOG_H +# include +# endif +#endif + + +/* err_level reflects the level of errors that are reported */ + +err_reporting_level_t err_level = err_level_none; + +#ifdef SRTP_KERNEL_LINUX +err_status_t +err_reporting_init(char *ident) { + + return err_status_ok; +} + +#else /* SRTP_KERNEL_LINUX */ + +/* err_file is the FILE to which errors are reported */ + +static FILE *err_file = NULL; + +err_status_t +err_reporting_init(char *ident) { +#ifdef ERR_REPORTING_SYSLOG + openlog(ident, LOG_PID, LOG_AUTHPRIV); +#endif + + /* + * Believe it or not, openlog doesn't return an error on failure. + * But then, neither does the syslog() call... + */ + +#ifdef ERR_REPORTING_STDOUT + err_file = stdout; +#elif defined(USE_ERR_REPORTING_FILE) + /* open file for error reporting */ + err_file = fopen(ERR_REPORTING_FILE, "w"); + if (err_file == NULL) + return err_status_init_fail; +#endif + + return err_status_ok; +} + +void +err_report(int priority, char *format, ...) { + va_list args; + + if (priority <= err_level) { + + va_start(args, format); + if (err_file != NULL) { + vfprintf(err_file, format, args); + /* fprintf(err_file, "\n"); */ + } +#ifdef ERR_REPORTING_SYSLOG + if (1) { /* FIXME: Make this a runtime option. */ + int syslogpri; + + switch (priority) { + case err_level_emergency: + syslogpri = LOG_EMERG; + break; + case err_level_alert: + syslogpri = LOG_ALERT; + break; + case err_level_critical: + syslogpri = LOG_CRIT; + break; + case err_level_error: + syslogpri = LOG_ERR; + break; + case err_level_warning: + syslogpri = LOG_WARNING; + break; + case err_level_notice: + syslogpri = LOG_NOTICE; + break; + case err_level_info: + syslogpri = LOG_INFO; + break; + case err_level_debug: + case err_level_none: + default: + syslogpri = LOG_DEBUG; + break; + } + + vsyslog(syslogpri, format, args); +#endif + va_end(args); + } +} +#endif /* SRTP_KERNEL_LINUX */ + +void +err_reporting_set_level(err_reporting_level_t lvl) { + err_level = lvl; +} diff --git a/libs/srtp/crypto/kernel/key.c b/libs/srtp/crypto/kernel/key.c new file mode 100644 index 0000000000..8c86b89f26 --- /dev/null +++ b/libs/srtp/crypto/kernel/key.c @@ -0,0 +1,115 @@ +/* + * key.c + * + * key usage limits enforcement + * + * David A. Mcgrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright (c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include "key.h" + +#define soft_limit 0x10000 + +err_status_t +key_limit_set(key_limit_t key, const xtd_seq_num_t s) { +#ifdef NO_64BIT_MATH + if (high32(s) == 0 && low32(s) < soft_limit) + return err_status_bad_param; +#else + if (s < soft_limit) + return err_status_bad_param; +#endif + key->num_left = s; + key->state = key_state_normal; + return err_status_ok; +} + +err_status_t +key_limit_clone(key_limit_t original, key_limit_t *new_key) { + if (original == NULL) + return err_status_bad_param; + *new_key = original; + return err_status_ok; +} + +err_status_t +key_limit_check(const key_limit_t key) { + if (key->state == key_state_expired) + return err_status_key_expired; + return err_status_ok; +} + +key_event_t +key_limit_update(key_limit_t key) { +#ifdef NO_64BIT_MATH + if (low32(key->num_left) == 0) + { + // carry + key->num_left = make64(high32(key->num_left)-1,low32(key->num_left) - 1); + } + else + { + // no carry + key->num_left = make64(high32(key->num_left),low32(key->num_left) - 1); + } + if (high32(key->num_left) != 0 || low32(key->num_left) >= soft_limit) { + return key_event_normal; /* we're above the soft limit */ + } +#else + key->num_left--; + if (key->num_left >= soft_limit) { + return key_event_normal; /* we're above the soft limit */ + } +#endif + if (key->state == key_state_normal) { + /* we just passed the soft limit, so change the state */ + key->state = key_state_past_soft_limit; + } +#ifdef NO_64BIT_MATH + if (low32(key->num_left) == 0 && high32(key->num_left == 0)) +#else + if (key->num_left < 1) +#endif + { /* we just hit the hard limit */ + key->state = key_state_expired; + return key_event_hard_limit; + } + return key_event_soft_limit; +} + diff --git a/libs/srtp/crypto/math/CVS/Entries b/libs/srtp/crypto/math/CVS/Entries new file mode 100644 index 0000000000..7d1780a636 --- /dev/null +++ b/libs/srtp/crypto/math/CVS/Entries @@ -0,0 +1,5 @@ +/datatypes.c/1.6/Sat Oct 8 16:38:06 2005// +/gf2_8.c/1.2/Fri Sep 23 19:34:12 2005// +/math.c/1.5/Sat Oct 8 16:38:06 2005// +/stat.c/1.4/Fri Mar 17 20:51:25 2006// +D diff --git a/libs/srtp/crypto/math/CVS/Repository b/libs/srtp/crypto/math/CVS/Repository new file mode 100644 index 0000000000..739b84bf18 --- /dev/null +++ b/libs/srtp/crypto/math/CVS/Repository @@ -0,0 +1 @@ +srtp/crypto/math diff --git a/libs/srtp/crypto/math/CVS/Root b/libs/srtp/crypto/math/CVS/Root new file mode 100644 index 0000000000..b4b74237d9 --- /dev/null +++ b/libs/srtp/crypto/math/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@cvs.sourceforge.net:/cvsroot/srtp diff --git a/libs/srtp/crypto/math/datatypes.c b/libs/srtp/crypto/math/datatypes.c new file mode 100644 index 0000000000..1b5989b7fa --- /dev/null +++ b/libs/srtp/crypto/math/datatypes.c @@ -0,0 +1,600 @@ +/* + * datatypes.c + * + * data types for finite fields and functions for input, output, and + * manipulation + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright (c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include "datatypes.h" + +int +octet_weight[256] = { + 0, 1, 1, 2, 1, 2, 2, 3, + 1, 2, 2, 3, 2, 3, 3, 4, + 1, 2, 2, 3, 2, 3, 3, 4, + 2, 3, 3, 4, 3, 4, 4, 5, + 1, 2, 2, 3, 2, 3, 3, 4, + 2, 3, 3, 4, 3, 4, 4, 5, + 2, 3, 3, 4, 3, 4, 4, 5, + 3, 4, 4, 5, 4, 5, 5, 6, + 1, 2, 2, 3, 2, 3, 3, 4, + 2, 3, 3, 4, 3, 4, 4, 5, + 2, 3, 3, 4, 3, 4, 4, 5, + 3, 4, 4, 5, 4, 5, 5, 6, + 2, 3, 3, 4, 3, 4, 4, 5, + 3, 4, 4, 5, 4, 5, 5, 6, + 3, 4, 4, 5, 4, 5, 5, 6, + 4, 5, 5, 6, 5, 6, 6, 7, + 1, 2, 2, 3, 2, 3, 3, 4, + 2, 3, 3, 4, 3, 4, 4, 5, + 2, 3, 3, 4, 3, 4, 4, 5, + 3, 4, 4, 5, 4, 5, 5, 6, + 2, 3, 3, 4, 3, 4, 4, 5, + 3, 4, 4, 5, 4, 5, 5, 6, + 3, 4, 4, 5, 4, 5, 5, 6, + 4, 5, 5, 6, 5, 6, 6, 7, + 2, 3, 3, 4, 3, 4, 4, 5, + 3, 4, 4, 5, 4, 5, 5, 6, + 3, 4, 4, 5, 4, 5, 5, 6, + 4, 5, 5, 6, 5, 6, 6, 7, + 3, 4, 4, 5, 4, 5, 5, 6, + 4, 5, 5, 6, 5, 6, 6, 7, + 4, 5, 5, 6, 5, 6, 6, 7, + 5, 6, 6, 7, 6, 7, 7, 8 +}; + +int +octet_get_weight(uint8_t octet) { + extern int octet_weight[256]; + + return octet_weight[octet]; +} + +/* + * bit_string is a buffer that is used to hold output strings, e.g. + * for printing. + */ + +/* the value MAX_PRINT_STRING_LEN is defined in datatypes.h */ + +static char bit_string[MAX_PRINT_STRING_LEN]; + +uint8_t +nibble_to_hex_char(uint8_t nibble) { + char buf[16] = {'0', '1', '2', '3', '4', '5', '6', '7', + '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' }; + return buf[nibble & 0xF]; +} + +char * +octet_string_hex_string(const void *s, int length) { + const uint8_t *str = s; + int i; + + /* double length, since one octet takes two hex characters */ + length *= 2; + + /* truncate string if it would be too long */ + if (length > MAX_PRINT_STRING_LEN) + length = MAX_PRINT_STRING_LEN-1; + + for (i=0; i < length; i+=2) { + bit_string[i] = nibble_to_hex_char(*str >> 4); + bit_string[i+1] = nibble_to_hex_char(*str++ & 0xF); + } + bit_string[i] = 0; /* null terminate string */ + return bit_string; +} + +inline int +hex_char_to_nibble(uint8_t c) { + switch(c) { + case ('0'): return 0x0; + case ('1'): return 0x1; + case ('2'): return 0x2; + case ('3'): return 0x3; + case ('4'): return 0x4; + case ('5'): return 0x5; + case ('6'): return 0x6; + case ('7'): return 0x7; + case ('8'): return 0x8; + case ('9'): return 0x9; + case ('a'): return 0xa; + case ('A'): return 0xa; + case ('b'): return 0xb; + case ('B'): return 0xb; + case ('c'): return 0xc; + case ('C'): return 0xc; + case ('d'): return 0xd; + case ('D'): return 0xd; + case ('e'): return 0xe; + case ('E'): return 0xe; + case ('f'): return 0xf; + case ('F'): return 0xf; + default: return -1; /* this flags an error */ + } + /* NOTREACHED */ + return -1; /* this keeps compilers from complaining */ +} + +int +is_hex_string(char *s) { + while(*s != 0) + if (hex_char_to_nibble(*s++) == -1) + return 0; + return 1; +} + +/* + * hex_string_to_octet_string converts a hexadecimal string + * of length 2 * len to a raw octet string of length len + */ + +int +hex_string_to_octet_string(char *raw, char *hex, int len) { + uint8_t x; + int tmp; + int hex_len; + + hex_len = 0; + while (hex_len < len) { + tmp = hex_char_to_nibble(hex[0]); + if (tmp == -1) + return hex_len; + x = (tmp << 4); + hex_len++; + tmp = hex_char_to_nibble(hex[1]); + if (tmp == -1) + return hex_len; + x |= (tmp & 0xff); + hex_len++; + *raw++ = x; + hex += 2; + } + return hex_len; +} + +char * +v128_hex_string(v128_t *x) { + int i, j; + + for (i=j=0; i < 16; i++) { + bit_string[j++] = nibble_to_hex_char(x->v8[i] >> 4); + bit_string[j++] = nibble_to_hex_char(x->v8[i] & 0xF); + } + + bit_string[j] = 0; /* null terminate string */ + return bit_string; +} + +char * +v128_bit_string(v128_t *x) { + int j, index; + uint32_t mask; + + for (j=index=0; j < 4; j++) { + for (mask=0x80000000; mask > 0; mask >>= 1) { + if (x->v32[j] & mask) + bit_string[index] = '1'; + else + bit_string[index] = '0'; + ++index; + } + } + bit_string[128] = 0; /* null terminate string */ + + return bit_string; +} + +void +v128_copy_octet_string(v128_t *x, const uint8_t s[16]) { +#ifdef ALIGNMENT_32BIT_REQUIRED + if ((((uint32_t) &s[0]) & 0x3) != 0) +#endif + { + x->v8[0] = s[0]; + x->v8[1] = s[1]; + x->v8[2] = s[2]; + x->v8[3] = s[3]; + x->v8[4] = s[4]; + x->v8[5] = s[5]; + x->v8[6] = s[6]; + x->v8[7] = s[7]; + x->v8[8] = s[8]; + x->v8[9] = s[9]; + x->v8[10] = s[10]; + x->v8[11] = s[11]; + x->v8[12] = s[12]; + x->v8[13] = s[13]; + x->v8[14] = s[14]; + x->v8[15] = s[15]; + } +#ifdef ALIGNMENT_32BIT_REQUIRED + else + { + v128_t *v = (v128_t *) &s[0]; + + v128_copy(x,v); + } +#endif +} + +#ifndef DATATYPES_USE_MACROS /* little functions are not macros */ + +void +v128_set_to_zero(v128_t *x) { + _v128_set_to_zero(x); +} + +void +v128_copy(v128_t *x, const v128_t *y) { + _v128_copy(x, y); +} + +void +v128_xor(v128_t *z, v128_t *x, v128_t *y) { + _v128_xor(z, x, y); +} + +void +v128_and(v128_t *z, v128_t *x, v128_t *y) { + _v128_and(z, x, y); +} + +void +v128_or(v128_t *z, v128_t *x, v128_t *y) { + _v128_or(z, x, y); +} + +void +v128_complement(v128_t *x) { + _v128_complement(x); +} + +int +v128_is_eq(const v128_t *x, const v128_t *y) { + return _v128_is_eq(x, y); +} + +int +v128_xor_eq(v128_t *x, const v128_t *y) { + return _v128_xor_eq(x, y); +} + +int +v128_get_bit(const v128_t *x, int i) { + return _v128_get_bit(x, i); +} + +void +v128_set_bit(v128_t *x, int i) { + _v128_set_bit(x, i); +} + +void +v128_clear_bit(v128_t *x, int i){ + _v128_clear_bit(x, i); +} + +void +v128_set_bit_to(v128_t *x, int i, int y){ + _v128_set_bit_to(x, i, y); +} + + +#endif /* DATATYPES_USE_MACROS */ + +void +v128_right_shift(v128_t *x, int index) { + const int base_index = index >> 5; + const int bit_index = index & 31; + int i, from; + uint32_t b; + + if (index > 127) { + v128_set_to_zero(x); + return; + } + + if (bit_index == 0) { + + /* copy each word from left size to right side */ + x->v32[4-1] = x->v32[4-1-base_index]; + for (i=4-1; i > base_index; i--) + x->v32[i-1] = x->v32[i-1-base_index]; + + } else { + + /* set each word to the "or" of the two bit-shifted words */ + for (i = 4; i > base_index; i--) { + from = i-1 - base_index; + b = x->v32[from] << bit_index; + if (from > 0) + b |= x->v32[from-1] >> (32-bit_index); + x->v32[i-1] = b; + } + + } + + /* now wrap up the final portion */ + for (i=0; i < base_index; i++) + x->v32[i] = 0; + +} + +void +v128_left_shift(v128_t *x, int index) { + int i; + const int base_index = index >> 5; + const int bit_index = index & 31; + + if (index > 127) { + v128_set_to_zero(x); + return; + } + + if (bit_index == 0) { + for (i=0; i < 4 - base_index; i++) + x->v32[i] = x->v32[i+base_index]; + } else { + for (i=0; i < 4 - base_index - 1; i++) + x->v32[i] = (x->v32[i+base_index] >> bit_index) ^ + (x->v32[i+base_index+1] << (32 - bit_index)); + x->v32[4 - base_index-1] = x->v32[4-1] >> bit_index; + } + + /* now wrap up the final portion */ + for (i = 4 - base_index; i < 4; i++) + x->v32[i] = 0; + +} + + +int +octet_string_is_eq(uint8_t *a, uint8_t *b, int len) { + uint8_t *end = b + len; + while (b < end) + if (*a++ != *b++) + return 1; + return 0; +} + +void +octet_string_set_to_zero(uint8_t *s, int len) { + uint8_t *end = s + len; + + do { + *s = 0; + } while (++s < end); + +} + + +/* + * From RFC 1521: The Base64 Alphabet + * + * Value Encoding Value Encoding Value Encoding Value Encoding + * 0 A 17 R 34 i 51 z + * 1 B 18 S 35 j 52 0 + * 2 C 19 T 36 k 53 1 + * 3 D 20 U 37 l 54 2 + * 4 E 21 V 38 m 55 3 + * 5 F 22 W 39 n 56 4 + * 6 G 23 X 40 o 57 5 + * 7 H 24 Y 41 p 58 6 + * 8 I 25 Z 42 q 59 7 + * 9 J 26 a 43 r 60 8 + * 10 K 27 b 44 s 61 9 + * 11 L 28 c 45 t 62 + + * 12 M 29 d 46 u 63 / + * 13 N 30 e 47 v + * 14 O 31 f 48 w (pad) = + * 15 P 32 g 49 x + * 16 Q 33 h 50 y + */ + +int +base64_char_to_sextet(uint8_t c) { + switch(c) { + case 'A': + return 0; + case 'B': + return 1; + case 'C': + return 2; + case 'D': + return 3; + case 'E': + return 4; + case 'F': + return 5; + case 'G': + return 6; + case 'H': + return 7; + case 'I': + return 8; + case 'J': + return 9; + case 'K': + return 10; + case 'L': + return 11; + case 'M': + return 12; + case 'N': + return 13; + case 'O': + return 14; + case 'P': + return 15; + case 'Q': + return 16; + case 'R': + return 17; + case 'S': + return 18; + case 'T': + return 19; + case 'U': + return 20; + case 'V': + return 21; + case 'W': + return 22; + case 'X': + return 23; + case 'Y': + return 24; + case 'Z': + return 25; + case 'a': + return 26; + case 'b': + return 27; + case 'c': + return 28; + case 'd': + return 29; + case 'e': + return 30; + case 'f': + return 31; + case 'g': + return 32; + case 'h': + return 33; + case 'i': + return 34; + case 'j': + return 35; + case 'k': + return 36; + case 'l': + return 37; + case 'm': + return 38; + case 'n': + return 39; + case 'o': + return 40; + case 'p': + return 41; + case 'q': + return 42; + case 'r': + return 43; + case 's': + return 44; + case 't': + return 45; + case 'u': + return 46; + case 'v': + return 47; + case 'w': + return 48; + case 'x': + return 49; + case 'y': + return 50; + case 'z': + return 51; + case '0': + return 52; + case '1': + return 53; + case '2': + return 54; + case '3': + return 55; + case '4': + return 56; + case '5': + return 57; + case '6': + return 58; + case '7': + return 59; + case '8': + return 60; + case '9': + return 61; + case '+': + return 62; + case '/': + return 63; + case '=': + return 64; + default: + return -1; + } + return -1; +} + +/* + * base64_string_to_octet_string converts a hexadecimal string + * of length 2 * len to a raw octet string of length len + */ + +int +base64_string_to_octet_string(char *raw, char *base64, int len) { + uint8_t x; + int tmp; + int base64_len; + + base64_len = 0; + while (base64_len < len) { + tmp = base64_char_to_sextet(base64[0]); + if (tmp == -1) + return base64_len; + x = (tmp << 6); + base64_len++; + tmp = base64_char_to_sextet(base64[1]); + if (tmp == -1) + return base64_len; + x |= (tmp & 0xffff); + base64_len++; + *raw++ = x; + base64 += 2; + } + return base64_len; +} diff --git a/libs/srtp/crypto/math/gf2_8.c b/libs/srtp/crypto/math/gf2_8.c new file mode 100644 index 0000000000..7bd5a0dea2 --- /dev/null +++ b/libs/srtp/crypto/math/gf2_8.c @@ -0,0 +1,83 @@ +/* + * gf2_8.c + * + * GF(256) finite field implementation, with the representation used + * in the AES cipher. + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#include "datatypes.h" +#include "gf2_8.h" + +/* gf2_8_shift() moved to gf2_8.h as an inline function */ + +inline gf2_8 +gf2_8_multiply(gf2_8 x, gf2_8 y) { + gf2_8 z = 0; + + if (y & 1) z ^= x; x = gf2_8_shift(x); + if (y & 2) z ^= x; x = gf2_8_shift(x); + if (y & 4) z ^= x; x = gf2_8_shift(x); + if (y & 8) z ^= x; x = gf2_8_shift(x); + if (y & 16) z ^= x; x = gf2_8_shift(x); + if (y & 32) z ^= x; x = gf2_8_shift(x); + if (y & 64) z ^= x; x = gf2_8_shift(x); + if (y & 128) z ^= x; + + return z; +} + + +/* this should use the euclidean algorithm */ + +gf2_8 +gf2_8_compute_inverse(gf2_8 x) { + unsigned int i; + + if (x == 0) return 0; /* zero is a special case */ + for (i=0; i < 256; i++) + if (gf2_8_multiply((gf2_8) i, x) == 1) + return (gf2_8) i; + + return 0; +} + diff --git a/libs/srtp/crypto/math/math.c b/libs/srtp/crypto/math/math.c new file mode 100644 index 0000000000..f3aaf65014 --- /dev/null +++ b/libs/srtp/crypto/math/math.c @@ -0,0 +1,962 @@ +/* + * math.c + * + * crypto math operations and data types + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright (c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include "crypto_math.h" +#include /* malloc() used in bitvector_alloc */ + +int +octet_weight[256] = { + 0, 1, 1, 2, 1, 2, 2, 3, + 1, 2, 2, 3, 2, 3, 3, 4, + 1, 2, 2, 3, 2, 3, 3, 4, + 2, 3, 3, 4, 3, 4, 4, 5, + 1, 2, 2, 3, 2, 3, 3, 4, + 2, 3, 3, 4, 3, 4, 4, 5, + 2, 3, 3, 4, 3, 4, 4, 5, + 3, 4, 4, 5, 4, 5, 5, 6, + 1, 2, 2, 3, 2, 3, 3, 4, + 2, 3, 3, 4, 3, 4, 4, 5, + 2, 3, 3, 4, 3, 4, 4, 5, + 3, 4, 4, 5, 4, 5, 5, 6, + 2, 3, 3, 4, 3, 4, 4, 5, + 3, 4, 4, 5, 4, 5, 5, 6, + 3, 4, 4, 5, 4, 5, 5, 6, + 4, 5, 5, 6, 5, 6, 6, 7, + 1, 2, 2, 3, 2, 3, 3, 4, + 2, 3, 3, 4, 3, 4, 4, 5, + 2, 3, 3, 4, 3, 4, 4, 5, + 3, 4, 4, 5, 4, 5, 5, 6, + 2, 3, 3, 4, 3, 4, 4, 5, + 3, 4, 4, 5, 4, 5, 5, 6, + 3, 4, 4, 5, 4, 5, 5, 6, + 4, 5, 5, 6, 5, 6, 6, 7, + 2, 3, 3, 4, 3, 4, 4, 5, + 3, 4, 4, 5, 4, 5, 5, 6, + 3, 4, 4, 5, 4, 5, 5, 6, + 4, 5, 5, 6, 5, 6, 6, 7, + 3, 4, 4, 5, 4, 5, 5, 6, + 4, 5, 5, 6, 5, 6, 6, 7, + 4, 5, 5, 6, 5, 6, 6, 7, + 5, 6, 6, 7, 6, 7, 7, 8 +}; + +int +low_bit[256] = { + -1, 0, 1, 0, 2, 0, 1, 0, + 3, 0, 1, 0, 2, 0, 1, 0, + 4, 0, 1, 0, 2, 0, 1, 0, + 3, 0, 1, 0, 2, 0, 1, 0, + 5, 0, 1, 0, 2, 0, 1, 0, + 3, 0, 1, 0, 2, 0, 1, 0, + 4, 0, 1, 0, 2, 0, 1, 0, + 3, 0, 1, 0, 2, 0, 1, 0, + 6, 0, 1, 0, 2, 0, 1, 0, + 3, 0, 1, 0, 2, 0, 1, 0, + 4, 0, 1, 0, 2, 0, 1, 0, + 3, 0, 1, 0, 2, 0, 1, 0, + 5, 0, 1, 0, 2, 0, 1, 0, + 3, 0, 1, 0, 2, 0, 1, 0, + 4, 0, 1, 0, 2, 0, 1, 0, + 3, 0, 1, 0, 2, 0, 1, 0, + 7, 0, 1, 0, 2, 0, 1, 0, + 3, 0, 1, 0, 2, 0, 1, 0, + 4, 0, 1, 0, 2, 0, 1, 0, + 3, 0, 1, 0, 2, 0, 1, 0, + 5, 0, 1, 0, 2, 0, 1, 0, + 3, 0, 1, 0, 2, 0, 1, 0, + 4, 0, 1, 0, 2, 0, 1, 0, + 3, 0, 1, 0, 2, 0, 1, 0, + 6, 0, 1, 0, 2, 0, 1, 0, + 3, 0, 1, 0, 2, 0, 1, 0, + 4, 0, 1, 0, 2, 0, 1, 0, + 3, 0, 1, 0, 2, 0, 1, 0, + 5, 0, 1, 0, 2, 0, 1, 0, + 3, 0, 1, 0, 2, 0, 1, 0, + 4, 0, 1, 0, 2, 0, 1, 0, + 3, 0, 1, 0, 2, 0, 1, 0 +}; + + +int +high_bit[256] = { + -1, 0, 1, 1, 2, 2, 2, 2, + 3, 3, 3, 3, 3, 3, 3, 3, + 4, 4, 4, 4, 4, 4, 4, 4, + 4, 4, 4, 4, 4, 4, 4, 4, + 5, 5, 5, 5, 5, 5, 5, 5, + 5, 5, 5, 5, 5, 5, 5, 5, + 5, 5, 5, 5, 5, 5, 5, 5, + 5, 5, 5, 5, 5, 5, 5, 5, + 6, 6, 6, 6, 6, 6, 6, 6, + 6, 6, 6, 6, 6, 6, 6, 6, + 6, 6, 6, 6, 6, 6, 6, 6, + 6, 6, 6, 6, 6, 6, 6, 6, + 6, 6, 6, 6, 6, 6, 6, 6, + 6, 6, 6, 6, 6, 6, 6, 6, + 6, 6, 6, 6, 6, 6, 6, 6, + 6, 6, 6, 6, 6, 6, 6, 6, + 7, 7, 7, 7, 7, 7, 7, 7, + 7, 7, 7, 7, 7, 7, 7, 7, + 7, 7, 7, 7, 7, 7, 7, 7, + 7, 7, 7, 7, 7, 7, 7, 7, + 7, 7, 7, 7, 7, 7, 7, 7, + 7, 7, 7, 7, 7, 7, 7, 7, + 7, 7, 7, 7, 7, 7, 7, 7, + 7, 7, 7, 7, 7, 7, 7, 7, + 7, 7, 7, 7, 7, 7, 7, 7, + 7, 7, 7, 7, 7, 7, 7, 7, + 7, 7, 7, 7, 7, 7, 7, 7, + 7, 7, 7, 7, 7, 7, 7, 7, + 7, 7, 7, 7, 7, 7, 7, 7, + 7, 7, 7, 7, 7, 7, 7, 7, + 7, 7, 7, 7, 7, 7, 7, 7, + 7, 7, 7, 7, 7, 7, 7, 7 +}; + +int +octet_get_weight(uint8_t octet) { + extern int octet_weight[256]; + + return octet_weight[octet]; +} + +unsigned char +v32_weight(v32_t a) { + unsigned int wt = 0; + + wt += octet_weight[a.v8[0]]; /* note: endian-ness makes no difference */ + wt += octet_weight[a.v8[1]]; + wt += octet_weight[a.v8[2]]; + wt += octet_weight[a.v8[3]]; + + return wt; +} + +inline unsigned char +v32_distance(v32_t x, v32_t y) { + x.value ^= y.value; + return v32_weight(x); +} + +unsigned int +v32_dot_product(v32_t a, v32_t b) { + a.value &= b.value; + return v32_weight(a) & 1; +} + +/* + * _bit_string returns a NULL-terminated character string suitable for + * printing + */ + +#define MAX_STRING_LENGTH 1024 + +static char bit_string[MAX_STRING_LENGTH]; + +char * +octet_bit_string(uint8_t x) { + int mask, index; + + for (mask = 1, index = 0; mask < 256; mask <<= 1) + if ((x & mask) == 0) + bit_string[index++] = '0'; + else + bit_string[index++] = '1'; + + bit_string[index++] = 0; /* NULL terminate string */ + + return bit_string; +} + +char * +v16_bit_string(v16_t x) { + int i, mask, index; + + for (i = index = 0; i < 2; i++) { + for (mask = 1; mask < 256; mask <<= 1) + if ((x.v8[i] & mask) == 0) + bit_string[index++] = '0'; + else + bit_string[index++] = '1'; + } + bit_string[index++] = 0; /* NULL terminate string */ + return bit_string; +} + +char * +v32_bit_string(v32_t x) { + int i, mask, index; + + for (i = index = 0; i < 4; i++) { + for (mask = 128; mask > 0; mask >>= 1) + if ((x.v8[i] & mask) == 0) + bit_string[index++] = '0'; + else + bit_string[index++] = '1'; + } + bit_string[index++] = 0; /* NULL terminate string */ + return bit_string; +} + +char * +v64_bit_string(const v64_t *x) { + int i, mask, index; + + for (i = index = 0; i < 8; i++) { + for (mask = 1; mask < 256; mask <<= 1) + if ((x->v8[i] & mask) == 0) + bit_string[index++] = '0'; + else + bit_string[index++] = '1'; + } + bit_string[index++] = 0; /* NULL terminate string */ + return bit_string; +} + +char * +v128_bit_string(v128_t *x) { + int j, index; + uint32_t mask; + + for (j=index=0; j < 4; j++) { + for (mask=0x80000000; mask > 0; mask >>= 1) { + if (x->v32[j] & mask) + bit_string[index] = '1'; + else + bit_string[index] = '0'; + ++index; + } + } + bit_string[128] = 0; /* null terminate string */ + + return bit_string; +} + +uint8_t +nibble_to_hex_char(uint8_t nibble) { + char buf[16] = {'0', '1', '2', '3', '4', '5', '6', '7', + '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' }; + return buf[nibble & 0xF]; +} + +char * +octet_hex_string(uint8_t x) { + + bit_string[0] = nibble_to_hex_char(x >> 4); + bit_string[1] = nibble_to_hex_char(x & 0xF); + + bit_string[2] = 0; /* null terminate string */ + return bit_string; +} + +char * +octet_string_hex_string(const void *str, int length) { + const uint8_t *s = str; + int i; + + /* double length, since one octet takes two hex characters */ + length *= 2; + + /* truncate string if it would be too long */ + if (length > MAX_STRING_LENGTH) + length = MAX_STRING_LENGTH-1; + + for (i=0; i < length; i+=2) { + bit_string[i] = nibble_to_hex_char(*s >> 4); + bit_string[i+1] = nibble_to_hex_char(*s++ & 0xF); + } + bit_string[i] = 0; /* null terminate string */ + return bit_string; +} + +char * +v16_hex_string(v16_t x) { + int i, j; + + for (i=j=0; i < 2; i++) { + bit_string[j++] = nibble_to_hex_char(x.v8[i] >> 4); + bit_string[j++] = nibble_to_hex_char(x.v8[i] & 0xF); + } + + bit_string[j] = 0; /* null terminate string */ + return bit_string; +} + +char * +v32_hex_string(v32_t x) { + int i, j; + + for (i=j=0; i < 4; i++) { + bit_string[j++] = nibble_to_hex_char(x.v8[i] >> 4); + bit_string[j++] = nibble_to_hex_char(x.v8[i] & 0xF); + } + + bit_string[j] = 0; /* null terminate string */ + return bit_string; +} + +char * +v64_hex_string(const v64_t *x) { + int i, j; + + for (i=j=0; i < 8; i++) { + bit_string[j++] = nibble_to_hex_char(x->v8[i] >> 4); + bit_string[j++] = nibble_to_hex_char(x->v8[i] & 0xF); + } + + bit_string[j] = 0; /* null terminate string */ + return bit_string; +} + +char * +v128_hex_string(v128_t *x) { + int i, j; + + for (i=j=0; i < 16; i++) { + bit_string[j++] = nibble_to_hex_char(x->v8[i] >> 4); + bit_string[j++] = nibble_to_hex_char(x->v8[i] & 0xF); + } + + bit_string[j] = 0; /* null terminate string */ + return bit_string; +} + +char * +char_to_hex_string(char *x, int num_char) { + int i, j; + + if (num_char >= 16) + num_char = 16; + for (i=j=0; i < num_char; i++) { + bit_string[j++] = nibble_to_hex_char(x[i] >> 4); + bit_string[j++] = nibble_to_hex_char(x[i] & 0xF); + } + + bit_string[j] = 0; /* null terminate string */ + return bit_string; +} + +int +hex_char_to_nibble(uint8_t c) { + switch(c) { + case ('0'): return 0x0; + case ('1'): return 0x1; + case ('2'): return 0x2; + case ('3'): return 0x3; + case ('4'): return 0x4; + case ('5'): return 0x5; + case ('6'): return 0x6; + case ('7'): return 0x7; + case ('8'): return 0x8; + case ('9'): return 0x9; + case ('a'): return 0xa; + case ('A'): return 0xa; + case ('b'): return 0xb; + case ('B'): return 0xb; + case ('c'): return 0xc; + case ('C'): return 0xc; + case ('d'): return 0xd; + case ('D'): return 0xd; + case ('e'): return 0xe; + case ('E'): return 0xe; + case ('f'): return 0xf; + case ('F'): return 0xf; + default: return -1; /* this flags an error */ + } + /* NOTREACHED */ + return -1; /* this keeps compilers from complaining */ +} + +int +is_hex_string(char *s) { + while(*s != 0) + if (hex_char_to_nibble(*s++) == -1) + return 0; + return 1; +} + +uint8_t +hex_string_to_octet(char *s) { + uint8_t x; + + x = (hex_char_to_nibble(s[0]) << 4) + | hex_char_to_nibble(s[1] & 0xFF); + + return x; +} + +/* + * hex_string_to_octet_string converts a hexadecimal string + * of length 2 * len to a raw octet string of length len + */ + +int +hex_string_to_octet_string(char *raw, char *hex, int len) { + uint8_t x; + int tmp; + int hex_len; + + hex_len = 0; + while (hex_len < len) { + tmp = hex_char_to_nibble(hex[0]); + if (tmp == -1) + return hex_len; + x = (tmp << 4); + hex_len++; + tmp = hex_char_to_nibble(hex[1]); + if (tmp == -1) + return hex_len; + x |= (tmp & 0xff); + hex_len++; + *raw++ = x; + hex += 2; + } + return hex_len; +} + +v16_t +hex_string_to_v16(char *s) { + v16_t x; + int i, j; + + for (i=j=0; i < 4; i += 2, j++) { + x.v8[j] = (hex_char_to_nibble(s[i]) << 4) + | hex_char_to_nibble(s[i+1] & 0xFF); + } + return x; +} + +v32_t +hex_string_to_v32(char *s) { + v32_t x; + int i, j; + + for (i=j=0; i < 8; i += 2, j++) { + x.v8[j] = (hex_char_to_nibble(s[i]) << 4) + | hex_char_to_nibble(s[i+1] & 0xFF); + } + return x; +} + +v64_t +hex_string_to_v64(char *s) { + v64_t x; + int i, j; + + for (i=j=0; i < 16; i += 2, j++) { + x.v8[j] = (hex_char_to_nibble(s[i]) << 4) + | hex_char_to_nibble(s[i+1] & 0xFF); + } + return x; +} + +v128_t +hex_string_to_v128(char *s) { + v128_t x; + int i, j; + + for (i=j=0; i < 32; i += 2, j++) { + x.v8[j] = (hex_char_to_nibble(s[i]) << 4) + | hex_char_to_nibble(s[i+1] & 0xFF); + } + return x; +} + + + +/* + * the matrix A[] is stored in column format, i.e., A[i] is the ith + * column of the matrix + */ + +uint8_t +A_times_x_plus_b(uint8_t A[8], uint8_t x, uint8_t b) { + int index = 0; + unsigned mask; + + for (mask=1; mask < 256; mask *= 2) { + if (x & mask) + b^= A[index]; + ++index; + } + + return b; +} + +inline void +v16_copy_octet_string(v16_t *x, const uint8_t s[2]) { + x->v8[0] = s[0]; + x->v8[1] = s[1]; +} + +inline void +v32_copy_octet_string(v32_t *x, const uint8_t s[4]) { + x->v8[0] = s[0]; + x->v8[1] = s[1]; + x->v8[2] = s[2]; + x->v8[3] = s[3]; +} + +inline void +v64_copy_octet_string(v64_t *x, const uint8_t s[8]) { + x->v8[0] = s[0]; + x->v8[1] = s[1]; + x->v8[2] = s[2]; + x->v8[3] = s[3]; + x->v8[4] = s[4]; + x->v8[5] = s[5]; + x->v8[6] = s[6]; + x->v8[7] = s[7]; +} + +void +v128_copy_octet_string(v128_t *x, const uint8_t s[16]) { + x->v8[0] = s[0]; + x->v8[1] = s[1]; + x->v8[2] = s[2]; + x->v8[3] = s[3]; + x->v8[4] = s[4]; + x->v8[5] = s[5]; + x->v8[6] = s[6]; + x->v8[7] = s[7]; + x->v8[8] = s[8]; + x->v8[9] = s[9]; + x->v8[10] = s[10]; + x->v8[11] = s[11]; + x->v8[12] = s[12]; + x->v8[13] = s[13]; + x->v8[14] = s[14]; + x->v8[15] = s[15]; + +} + +#ifndef DATATYPES_USE_MACROS /* little functions are not macros */ + +void +v128_set_to_zero(v128_t *x) { + _v128_set_to_zero(x); +} + +void +v128_copy(v128_t *x, const v128_t *y) { + _v128_copy(x, y); +} + +void +v128_xor(v128_t *z, v128_t *x, v128_t *y) { + _v128_xor(z, x, y); +} + +void +v128_and(v128_t *z, v128_t *x, v128_t *y) { + _v128_and(z, x, y); +} + +void +v128_or(v128_t *z, v128_t *x, v128_t *y) { + _v128_or(z, x, y); +} + +void +v128_complement(v128_t *x) { + _v128_complement(x); +} + +int +v128_is_eq(const v128_t *x, const v128_t *y) { + return _v128_is_eq(x, y); +} + +int +v128_get_bit(const v128_t *x, int i) { + return _v128_get_bit(x, i); +} + +void +v128_set_bit(v128_t *x, int i) { + _v128_set_bit(x, i); +} + +void +v128_clear_bit(v128_t *x, int i){ + _v128_clear_bit(x, i); +} + +void +v128_set_bit_to(v128_t *x, int i, int y){ + _v128_set_bit_to(x, i, y); +} + + +#endif /* DATATYPES_USE_MACROS */ + + +inline void +v128_left_shift2(v128_t *x, int num_bits) { + int i; + int word_shift = num_bits >> 5; + int bit_shift = num_bits & 31; + + for (i=0; i < (4-word_shift); i++) { + x->v32[i] = x->v32[i+word_shift] << bit_shift; + } + + for ( ; i < word_shift; i++) { + x->v32[i] = 0; + } + +} + +void +v128_right_shift(v128_t *x, int index) { + const int base_index = index >> 5; + const int bit_index = index & 31; + int i, from; + uint32_t b; + + if (index > 127) { + v128_set_to_zero(x); + return; + } + + if (bit_index == 0) { + + /* copy each word from left size to right side */ + x->v32[4-1] = x->v32[4-1-base_index]; + for (i=4-1; i > base_index; i--) + x->v32[i-1] = x->v32[i-1-base_index]; + + } else { + + /* set each word to the "or" of the two bit-shifted words */ + for (i = 4; i > base_index; i--) { + from = i-1 - base_index; + b = x->v32[from] << bit_index; + if (from > 0) + b |= x->v32[from-1] >> (32-bit_index); + x->v32[i-1] = b; + } + + } + + /* now wrap up the final portion */ + for (i=0; i < base_index; i++) + x->v32[i] = 0; + +} + +void +v128_left_shift(v128_t *x, int index) { + int i; + const int base_index = index >> 5; + const int bit_index = index & 31; + + if (index > 127) { + v128_set_to_zero(x); + return; + } + + if (bit_index == 0) { + for (i=0; i < 4 - base_index; i++) + x->v32[i] = x->v32[i+base_index]; + } else { + for (i=0; i < 4 - base_index - 1; i++) + x->v32[i] = (x->v32[i+base_index] << bit_index) ^ + (x->v32[i+base_index+1] >> (32 - bit_index)); + x->v32[4 - base_index-1] = x->v32[4-1] << bit_index; + } + + /* now wrap up the final portion */ + for (i = 4 - base_index; i < 4; i++) + x->v32[i] = 0; + +} + + +#if 0 +void +v128_add(v128_t *z, v128_t *x, v128_t *y) { + /* integer addition modulo 2^128 */ + +#ifdef WORDS_BIGENDIAN + uint64_t tmp; + + tmp = x->v32[3] + y->v32[3]; + z->v32[3] = (uint32_t) tmp; + + tmp = x->v32[2] + y->v32[2] + (tmp >> 32); + z->v32[2] = (uint32_t) tmp; + + tmp = x->v32[1] + y->v32[1] + (tmp >> 32); + z->v32[1] = (uint32_t) tmp; + + tmp = x->v32[0] + y->v32[0] + (tmp >> 32); + z->v32[0] = (uint32_t) tmp; + +#else /* assume little endian architecture */ + uint64_t tmp; + + tmp = htonl(x->v32[3]) + htonl(y->v32[3]); + z->v32[3] = ntohl((uint32_t) tmp); + + tmp = htonl(x->v32[2]) + htonl(y->v32[2]) + htonl(tmp >> 32); + z->v32[2] = ntohl((uint32_t) tmp); + + tmp = htonl(x->v32[1]) + htonl(y->v32[1]) + htonl(tmp >> 32); + z->v32[1] = ntohl((uint32_t) tmp); + + tmp = htonl(x->v32[0]) + htonl(y->v32[0]) + htonl(tmp >> 32); + z->v32[0] = ntohl((uint32_t) tmp); + +#endif /* WORDS_BIGENDIAN */ + +} +#endif + +int +octet_string_is_eq(uint8_t *a, uint8_t *b, int len) { + uint8_t *end = b + len; + while (b < end) + if (*a++ != *b++) + return 1; + return 0; +} + +void +octet_string_set_to_zero(uint8_t *s, int len) { + uint8_t *end = s + len; + + do { + *s = 0; + } while (++s < end); + +} + +/* functions manipulating bit_vector_t */ + +#define BITVECTOR_MAX_WORDS 5 + +int +bitvector_alloc(bitvector_t *v, unsigned long length) { + unsigned long l = (length + bytes_per_word - 1) / bytes_per_word; + int i; + + /* allocate memory, then set parameters */ + if (l > BITVECTOR_MAX_WORDS) + return -1; + else + l = BITVECTOR_MAX_WORDS; + v->word = malloc(l); + if (v->word == NULL) + return -1; + v->length = length; + + /* initialize bitvector to zero */ + for (i=0; i < (length >> 5); i++) { + v->word = 0; + } + + return 0; +} + +void +bitvector_set_bit(bitvector_t *v, int bit_index) { + + v->word[(bit_index >> 5)] |= (1 << (bit_index & 31)); + +} + +int +bitvector_get_bit(const bitvector_t *v, int bit_index) { + + return ((v->word[(bit_index >> 5)]) >> (bit_index & 31)) & 1; + +} + +#include + +int +bitvector_print_hex(const bitvector_t *v, FILE *stream) { + int i; + int m = v->length >> 5; + int n = v->length & 31; + char string[9]; + uint32_t tmp; + + /* if length isn't a multiple of four, we can't hex_print */ + if (n & 3) + return -1; + + /* if the length is zero, do nothing */ + if (v->length == 0) + return 0; + + /* + * loop over words from most significant to least significant - + */ + + for (i=m; i > 0; i++) { + char *str = string + 7; + tmp = v->word[i]; + + /* null terminate string */ + string[8] = 0; + + /* loop over nibbles */ + *str-- = nibble_to_hex_char(tmp & 0xf); tmp >>= 4; + *str-- = nibble_to_hex_char(tmp & 0xf); tmp >>= 4; + *str-- = nibble_to_hex_char(tmp & 0xf); tmp >>= 4; + *str-- = nibble_to_hex_char(tmp & 0xf); tmp >>= 4; + *str-- = nibble_to_hex_char(tmp & 0xf); tmp >>= 4; + *str-- = nibble_to_hex_char(tmp & 0xf); tmp >>= 4; + *str-- = nibble_to_hex_char(tmp & 0xf); tmp >>= 4; + *str-- = nibble_to_hex_char(tmp & 0xf); + + /* now print stream */ + fprintf(stream, string); + } + + return 0; + +} + + +int +hex_string_length(char *s) { + int count = 0; + + /* ignore leading zeros */ + while ((*s != 0) && *s == '0') + s++; + + /* count remaining characters */ + while (*s != 0) { + if (hex_char_to_nibble(*s++) == -1) + return -1; + count++; + } + + return count; +} + +int +bitvector_set_from_hex(bitvector_t *v, char *string) { + int num_hex_chars, m, n, i, j; + uint32_t tmp; + + num_hex_chars = hex_string_length(string); + if (num_hex_chars == -1) + return -1; + + /* set length */ + v->length = num_hex_chars * 4; + /* + * at this point, we should subtract away a bit if the high + * bit of the first character is zero, but we ignore that + * for now and assume that we're four-bit aligned - DAM + */ + + + m = num_hex_chars / 8; /* number of words */ + n = num_hex_chars % 8; /* number of nibbles in last word */ + + /* if the length is greater than the bitvector, return an error */ + if (m > BITVECTOR_MAX_WORDS) + return -1; + + /* + * loop over words from most significant - first word is a special + * case + */ + + if (n) { + tmp = 0; + for (i=0; i < n; i++) { + tmp = hex_char_to_nibble(*string++); + tmp <<= 4; + } + v->word[m] = tmp; + } + + /* now loop over the rest of the words */ + for (i=m-1; i >= 0; i--) { + tmp = 0; + for (j=0; j < 8; j++) { + tmp = hex_char_to_nibble(*string++); + tmp <<= 4; + } + v->word[i] = tmp; + } + + return 0; +} + + +/* functions below not yet tested! */ + +int +v32_low_bit(v32_t *w) { + int value; + + value = low_bit[w->v8[0]]; + if (value != -1) + return value; + value = low_bit[w->v8[1]]; + if (value != -1) + return value + 8; + value = low_bit[w->v8[2]]; + if (value != -1) + return value + 16; + value = low_bit[w->v8[3]]; + if (value == -1) + return -1; + return value + 24; +} + +/* high_bit not done yet */ + + + + + diff --git a/libs/srtp/crypto/math/stat.c b/libs/srtp/crypto/math/stat.c new file mode 100644 index 0000000000..3c6bbda596 --- /dev/null +++ b/libs/srtp/crypto/math/stat.c @@ -0,0 +1,367 @@ +/* + * stats.c + * + * statistical tests for randomness (FIPS 140-2, Section 4.9) + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +#include "stat.h" + +debug_module_t mod_stat = { + 0, /* debugging is off by default */ + (char *)"stat test" /* printable module name */ +}; + +/* + * each test assumes that 20,000 bits (2500 octets) of data is + * provided as input + */ + +#define STAT_TEST_DATA_LEN 2500 + +err_status_t +stat_test_monobit(uint8_t *data) { + uint8_t *data_end = data + STAT_TEST_DATA_LEN; + uint16_t ones_count; + + ones_count = 0; + while (data < data_end) { + ones_count += octet_get_weight(*data); + data++; + } + + debug_print(mod_stat, "bit count: %d", ones_count); + + if ((ones_count < 9725) || (ones_count > 10275)) + return err_status_algo_fail; + + return err_status_ok; +} + +err_status_t +stat_test_poker(uint8_t *data) { + int i; + uint8_t *data_end = data + STAT_TEST_DATA_LEN; + double poker; + uint16_t f[16] = { + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0 + }; + + while (data < data_end) { + f[*data & 0x0f]++; /* increment freq. count for low nibble */ + f[(*data) >> 4]++; /* increment freq. count for high nibble */ + data++; + } + + poker = 0.0; + for (i=0; i < 16; i++) + poker += (double) f[i] * f[i]; + + poker *= (16.0 / 5000.0); + poker -= 5000.0; + + debug_print(mod_stat, "poker test: %f\n", poker); + + if ((poker < 2.16) || (poker > 46.17)) + return err_status_algo_fail; + + return err_status_ok; +} + + +/* + * runs[i] holds the number of runs of size (i-1) + */ + +err_status_t +stat_test_runs(uint8_t *data) { + uint8_t *data_end = data + STAT_TEST_DATA_LEN; + uint16_t runs[6] = { 0, 0, 0, 0, 0, 0 }; + uint16_t gaps[6] = { 0, 0, 0, 0, 0, 0 }; + uint16_t lo_value[6] = { 2315, 1114, 527, 240, 103, 103 }; + uint16_t hi_value[6] = { 2685, 1386, 723, 384, 209, 209 }; + int16_t state = 0; + uint16_t mask; + int i; + + /* + * the state variable holds the number of bits in the + * current run (or gap, if negative) + */ + + while (data < data_end) { + + /* loop over the bits of this byte */ + for (mask = 1; mask < 256; mask <<= 1) { + if (*data & mask) { + + /* next bit is a one */ + if (state > 0) { + + /* prefix is a run, so increment the run-count */ + state++; + + /* check for long runs */ + if (state > 25) { + debug_print(mod_stat, ">25 runs: %d", state); + return err_status_algo_fail; + } + + } else if (state < 0) { + + /* prefix is a gap */ + if (state < -25) { + debug_print(mod_stat, ">25 gaps: %d", state); + return err_status_algo_fail; /* long-runs test failed */ + } + if (state < -6) { + state = -6; /* group together gaps > 5 */ + } + gaps[-1-state]++; /* increment gap count */ + state = 1; /* set state at one set bit */ + } else { + + /* state is zero; this happens only at initialization */ + state = 1; + } + } else { + + /* next bit is a zero */ + if (state > 0) { + + /* prefix is a run */ + if (state > 25) { + debug_print(mod_stat, ">25 runs (2): %d", state); + return err_status_algo_fail; /* long-runs test failed */ + } + if (state > 6) { + state = 6; /* group together runs > 5 */ + } + runs[state-1]++; /* increment run count */ + state = -1; /* set state at one zero bit */ + } else if (state < 0) { + + /* prefix is a gap, so increment gap-count (decrement state) */ + state--; + + /* check for long gaps */ + if (state < -25) { + debug_print(mod_stat, ">25 gaps (2): %d", state); + return err_status_algo_fail; + } + + } else { + + /* state is zero; this happens only at initialization */ + state = -1; + } + } + } + + /* move along to next octet */ + data++; + } + + if (mod_stat.on) { + debug_print(mod_stat, "runs test", NULL); + for (i=0; i < 6; i++) + debug_print(mod_stat, " runs[]: %d", runs[i]); + for (i=0; i < 6; i++) + debug_print(mod_stat, " gaps[]: %d", gaps[i]); + } + + /* check run and gap counts against the fixed limits */ + for (i=0; i < 6; i++) + if ( (runs[i] < lo_value[i] ) || (runs[i] > hi_value[i]) + || (gaps[i] < lo_value[i] ) || (gaps[i] > hi_value[i])) + return err_status_algo_fail; + + + return err_status_ok; +} + + +/* + * the function stat_test_rand_source applys the FIPS-140-2 statistical + * tests to the random source defined by rs + * + */ + +#define RAND_SRC_BUF_OCTETS 50 /* this value MUST divide 2500! */ + +err_status_t +stat_test_rand_source(rand_source_func_t get_rand_bytes) { + int i; + double poker; + uint8_t *data, *data_end; + uint16_t f[16] = { + 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0 + }; + uint8_t buffer[RAND_SRC_BUF_OCTETS]; + err_status_t status; + int ones_count = 0; + uint16_t runs[6] = { 0, 0, 0, 0, 0, 0 }; + uint16_t gaps[6] = { 0, 0, 0, 0, 0, 0 }; + uint16_t lo_value[6] = { 2315, 1114, 527, 240, 103, 103 }; + uint16_t hi_value[6] = { 2685, 1386, 723, 384, 209, 209 }; + int16_t state = 0; + uint16_t mask; + + /* counters for monobit, poker, and runs tests are initialized above */ + + /* main loop: fill buffer, update counters for stat tests */ + for (i=0; i < 2500; i+=RAND_SRC_BUF_OCTETS) { + + /* fill data buffer */ + status = get_rand_bytes(buffer, RAND_SRC_BUF_OCTETS); + if (status) { + debug_print(mod_stat, "couldn't get rand bytes: %d",status); + return status; + } + +#if 0 + debug_print(mod_stat, "%s", + octet_string_hex_string(buffer, RAND_SRC_BUF_OCTETS)); +#endif + + data = buffer; + data_end = data + RAND_SRC_BUF_OCTETS; + while (data < data_end) { + + /* update monobit test counter */ + ones_count += octet_get_weight(*data); + + /* update poker test counters */ + f[*data & 0x0f]++; /* increment freq. count for low nibble */ + f[(*data) >> 4]++; /* increment freq. count for high nibble */ + + /* update runs test counters */ + /* loop over the bits of this byte */ + for (mask = 1; mask < 256; mask <<= 1) { + if (*data & mask) { + + /* next bit is a one */ + if (state > 0) { + + /* prefix is a run, so increment the run-count */ + state++; + + /* check for long runs */ + if (state > 25) { + debug_print(mod_stat, ">25 runs (3): %d", state); + return err_status_algo_fail; + } + + } else if (state < 0) { + + /* prefix is a gap */ + if (state < -25) { + debug_print(mod_stat, ">25 gaps (3): %d", state); + return err_status_algo_fail; /* long-runs test failed */ + } + if (state < -6) { + state = -6; /* group together gaps > 5 */ + } + gaps[-1-state]++; /* increment gap count */ + state = 1; /* set state at one set bit */ + } else { + + /* state is zero; this happens only at initialization */ + state = 1; + } + } else { + + /* next bit is a zero */ + if (state > 0) { + + /* prefix is a run */ + if (state > 25) { + debug_print(mod_stat, ">25 runs (4): %d", state); + return err_status_algo_fail; /* long-runs test failed */ + } + if (state > 6) { + state = 6; /* group together runs > 5 */ + } + runs[state-1]++; /* increment run count */ + state = -1; /* set state at one zero bit */ + } else if (state < 0) { + + /* prefix is a gap, so increment gap-count (decrement state) */ + state--; + + /* check for long gaps */ + if (state < -25) { + debug_print(mod_stat, ">25 gaps (4): %d", state); + return err_status_algo_fail; + } + + } else { + + /* state is zero; this happens only at initialization */ + state = -1; + } + } + } + + /* advance data pointer */ + data++; + } + } + + /* check to see if test data is within bounds */ + + /* check monobit test data */ + + debug_print(mod_stat, "stat: bit count: %d", ones_count); + + if ((ones_count < 9725) || (ones_count > 10275)) { + debug_print(mod_stat, "stat: failed monobit test %d", ones_count); + return err_status_algo_fail; + } + + /* check poker test data */ + poker = 0.0; + for (i=0; i < 16; i++) + poker += (double) f[i] * f[i]; + + poker *= (16.0 / 5000.0); + poker -= 5000.0; + + debug_print(mod_stat, "stat: poker test: %f", poker); + + if ((poker < 2.16) || (poker > 46.17)) { + debug_print(mod_stat, "stat: failed poker test", NULL); + return err_status_algo_fail; + } + + /* check run and gap counts against the fixed limits */ + for (i=0; i < 6; i++) + if ((runs[i] < lo_value[i] ) || (runs[i] > hi_value[i]) + || (gaps[i] < lo_value[i] ) || (gaps[i] > hi_value[i])) { + debug_print(mod_stat, "stat: failed run/gap test", NULL); + return err_status_algo_fail; + } + + debug_print(mod_stat, "passed random stat test", NULL); + return err_status_ok; +} + +err_status_t +stat_test_rand_source_with_repetition(rand_source_func_t source, unsigned num_trials) { + int i; + err_status_t err = err_status_algo_fail; + + for (i=0; i < num_trials; i++) { + err = stat_test_rand_source(source); + if (err == err_status_ok) { + return err_status_ok; + } + debug_print(mod_stat, "failed stat test (try number %d)\n", i); + } + + return err; +} diff --git a/libs/srtp/crypto/replay/CVS/Entries b/libs/srtp/crypto/replay/CVS/Entries new file mode 100644 index 0000000000..09b4fa4d58 --- /dev/null +++ b/libs/srtp/crypto/replay/CVS/Entries @@ -0,0 +1,4 @@ +/rdb.c/1.2/Thu Mar 16 19:13:33 2006// +/rdbx.c/1.3/Sun Oct 2 20:35:26 2005// +/ut_sim.c/1.2/Sun Oct 2 20:36:02 2005// +D diff --git a/libs/srtp/crypto/replay/CVS/Repository b/libs/srtp/crypto/replay/CVS/Repository new file mode 100644 index 0000000000..f4ad54d9a0 --- /dev/null +++ b/libs/srtp/crypto/replay/CVS/Repository @@ -0,0 +1 @@ +srtp/crypto/replay diff --git a/libs/srtp/crypto/replay/CVS/Root b/libs/srtp/crypto/replay/CVS/Root new file mode 100644 index 0000000000..b4b74237d9 --- /dev/null +++ b/libs/srtp/crypto/replay/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@cvs.sourceforge.net:/cvsroot/srtp diff --git a/libs/srtp/crypto/replay/rdb.c b/libs/srtp/crypto/replay/rdb.c new file mode 100644 index 0000000000..e2e645be1b --- /dev/null +++ b/libs/srtp/crypto/replay/rdb.c @@ -0,0 +1,137 @@ +/* + * rdb.c + * + * Implements a replay database for packet security + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#include "rdb.h" + + +/* + * this implementation of a replay database works as follows: + * + * window_start is the index of the first packet in the window + * bitmask a bit-buffer, containing the most recently entered + * index as the leftmost bit + * + */ + +/* rdb_init initalizes rdb */ + +err_status_t +rdb_init(rdb_t *rdb) { + v128_set_to_zero(&rdb->bitmask); + rdb->window_start = 0; + return err_status_ok; +} + +/* + * rdb_check checks to see if index appears in rdb + */ + +err_status_t +rdb_check(const rdb_t *rdb, uint32_t index) { + + /* if the index appears after (or at very end of) the window, its good */ + if (index >= rdb->window_start + rdb_bits_in_bitmask) + return err_status_ok; + + /* if the index appears before the window, its bad */ + if (index < rdb->window_start) + return err_status_fail; + + /* otherwise, the index appears within the window, so check the bitmask */ + if (v128_get_bit(&rdb->bitmask, (index - rdb->window_start)) == 1) + return err_status_fail; + + /* otherwise, the index is okay */ + return err_status_ok; +} + +/* + * rdb_add_index adds index to rdb_t (and does *not* check if + * index appears in db) + * + * this function should be called only after rdb_check has + * indicated that the index does not appear in the rdb, e.g., a mutex + * should protect the rdb between these calls + */ + +err_status_t +rdb_add_index(rdb_t *rdb, uint32_t index) { + int delta; + + /* here we *assume* that index > rdb->window_start */ + + delta = (index - rdb->window_start); + if (delta < rdb_bits_in_bitmask) { + + /* if the index is within the window, set the appropriate bit */ + v128_set_bit(&rdb->bitmask, delta); + + } else { + + delta -= rdb_bits_in_bitmask - 1; + + /* shift the window forward by delta bits*/ + v128_left_shift(&rdb->bitmask, delta); + v128_set_bit(&rdb->bitmask, rdb_bits_in_bitmask-delta); + rdb->window_start += delta; + + } + + return err_status_ok; +} + +err_status_t +rdb_increment(rdb_t *rdb) { + + if (rdb->window_start++ > 0x7fffffff) + return err_status_key_expired; + return err_status_ok; +} + +uint32_t +rdb_get_value(const rdb_t *rdb) { + return rdb->window_start; +} diff --git a/libs/srtp/crypto/replay/rdbx.c b/libs/srtp/crypto/replay/rdbx.c new file mode 100644 index 0000000000..75ca70f457 --- /dev/null +++ b/libs/srtp/crypto/replay/rdbx.c @@ -0,0 +1,289 @@ +/* + * rdbx.c + * + * a replay database with extended range, using a rollover counter + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include "rdbx.h" + +#define rdbx_high_bit_in_bitmask 127 + +/* + * from draft-ietf-avt-srtp-00.txt: + * + * A receiver reconstructs the index i of a packet with sequence + * number s using the estimate + * + * i = 65,536 * t + s, + * + * where t is chosen from the set { r-1, r, r+1 } such that i is + * closest to the value 65,536 * r + s_l. If the value r+1 is used, + * then the rollover counter r in the cryptographic context is + * incremented by one (if the packet containing s is authentic). + */ + + + +/* + * rdbx implementation notes + * + * A xtd_seq_num_t is essentially a sequence number for which some of + * the data on the wire are implicit. It logically consists of a + * rollover counter and a sequence number; the sequence number is the + * explicit part, and the rollover counter is the implicit part. + * + * Upon receiving a sequence_number (e.g. in a newly received SRTP + * packet), the complete xtd_seq_num_t can be estimated by using a + * local xtd_seq_num_t as a basis. This is done using the function + * index_guess(&local, &guess, seq_from_packet). This function + * returns the difference of the guess and the local value. The local + * xtd_seq_num_t can be moved forward to the guess using the function + * index_advance(&guess, delta), where delta is the difference. + * + * + * A rdbx_t consists of a xtd_seq_num_t and a bitmask. The index is highest + * sequence number that has been received, and the bitmask indicates + * which of the recent indicies have been received as well. The + * highest bit in the bitmask corresponds to the index in the bitmask. + */ + + +void +index_init(xtd_seq_num_t *pi) { +#ifdef NO_64BIT_MATH + *pi = make64(0,0); +#else + *pi = 0; +#endif +} + +void +index_advance(xtd_seq_num_t *pi, sequence_number_t s) { +#ifdef NO_64BIT_MATH + /* a > ~b means a+b will generate a carry */ + /* s is uint16 here */ + *pi = make64(high32(*pi) + (s > ~low32(*pi) ? 1 : 0),low32(*pi) + s); +#else + *pi += s; +#endif +} + + +/* + * index_guess(local, guess, s) + * + * given a xtd_seq_num_t local (which represents the last + * known-to-be-good received xtd_seq_num_t) and a sequence number s + * (from a newly arrived packet), sets the contents of *guess to + * contain the best guess of the packet index to which s corresponds, + * and returns the difference between *guess and *local + * + * nota bene - the output is a signed integer, DON'T cast it to a + * unsigned integer! + */ + +int +index_guess(const xtd_seq_num_t *local, + xtd_seq_num_t *guess, + sequence_number_t s) { +#ifdef NO_64BIT_MATH + uint32_t local_roc = ((high32(*local) << 16) | + (low32(*local) >> 16)); + uint16_t local_seq = (uint16_t) (low32(*local)); +#else + uint32_t local_roc = (uint32_t)(*local >> 16); + uint16_t local_seq = (uint16_t) *local; +#endif +#ifdef NO_64BIT_MATH + uint32_t guess_roc = ((high32(*guess) << 16) | + (low32(*guess) >> 16)); + uint16_t guess_seq = (uint16_t) (low32(*guess)); +#else + uint32_t guess_roc = (uint32_t)(*guess >> 16); + uint16_t guess_seq = (uint16_t) *guess; +#endif + int difference; + + if (local_seq < seq_num_median) { + if (s - local_seq > seq_num_median) { + guess_roc = local_roc - 1; + difference = seq_num_max - s + local_seq; + } else { + guess_roc = local_roc; + difference = s - local_seq; + } + } else { + if (local_seq - seq_num_median > s) { + guess_roc = local_roc+1; + difference = seq_num_max - local_seq + s; + } else { + difference = s - local_seq; + guess_roc = local_roc; + } + } + guess_seq = s; + + /* Note: guess_roc is 32 bits, so this generates a 48-bit result! */ +#ifdef NO_64BIT_MATH + *guess = make64(guess_roc >> 16, + (guess_roc << 16) | guess_seq); +#else + *guess = (((uint64_t) guess_roc) << 16) | guess_seq; +#endif + + return difference; +} + +/* + * rdbx + * + */ + + +/* + * rdbx_init(&r) initalizes the rdbx_t pointed to by r + */ + +err_status_t +rdbx_init(rdbx_t *rdbx) { + v128_set_to_zero(&rdbx->bitmask); + index_init(&rdbx->index); + + return err_status_ok; +} + + +/* + * rdbx_check(&r, delta) checks to see if the xtd_seq_num_t + * which is at rdbx->index + delta is in the rdb + */ + +err_status_t +rdbx_check(const rdbx_t *rdbx, int delta) { + + if (delta > 0) { /* if delta is positive, it's good */ + return err_status_ok; + } else if (rdbx_high_bit_in_bitmask + delta < 0) { + /* if delta is lower than the bitmask, it's bad */ + return err_status_replay_old; + } else if (v128_get_bit(&rdbx->bitmask, + rdbx_high_bit_in_bitmask + delta) == 1) { + /* delta is within the window, so check the bitmask */ + return err_status_replay_fail; + } + /* otherwise, the index is okay */ + + return err_status_ok; +} + +/* + * rdbx_add_index adds the xtd_seq_num_t at rdbx->window_start + d to + * replay_db (and does *not* check if that xtd_seq_num_t appears in db) + * + * this function should be called only after replay_check has + * indicated that the index does not appear in the rdbx, e.g., a mutex + * should protect the rdbx between these calls if need be + */ + +err_status_t +rdbx_add_index(rdbx_t *rdbx, int delta) { + + if (delta > 0) { + /* shift forward by delta */ + index_advance(&rdbx->index, delta); + v128_left_shift(&rdbx->bitmask, delta); + v128_set_bit(&rdbx->bitmask, 127); + } else { + /* delta is in window, so flip bit in bitmask */ + v128_set_bit(&rdbx->bitmask, -delta); + } + + /* note that we need not consider the case that delta == 0 */ + + return err_status_ok; +} + + + +/* + * rdbx_estimate_index(rdbx, guess, s) + * + * given an rdbx and a sequence number s (from a newly arrived packet), + * sets the contents of *guess to contain the best guess of the packet + * index to which s corresponds, and returns the difference between + * *guess and the locally stored synch info + */ + +int +rdbx_estimate_index(const rdbx_t *rdbx, + xtd_seq_num_t *guess, + sequence_number_t s) { + + /* + * if the sequence number and rollover counter in the rdbx are + * non-zero, then use the index_guess(...) function, otherwise, just + * set the rollover counter to zero (since the index_guess(...) + * function might incorrectly guess that the rollover counter is + * 0xffffffff) + */ + +#ifdef NO_64BIT_MATH + /* seq_num_median = 0x8000 */ + if (high32(rdbx->index) > 0 || + low32(rdbx->index) > seq_num_median) +#else + if (rdbx->index > seq_num_median) +#endif + return index_guess(&rdbx->index, guess, s); + +#ifdef NO_64BIT_MATH + *guess = make64(0,(uint32_t) s); +#else + *guess = s; +#endif + +#ifdef NO_64BIT_MATH + return s - (uint16_t) low32(rdbx->index); +#else + return s - (uint16_t) rdbx->index; +#endif +} diff --git a/libs/srtp/crypto/replay/ut_sim.c b/libs/srtp/crypto/replay/ut_sim.c new file mode 100644 index 0000000000..18ec4fe440 --- /dev/null +++ b/libs/srtp/crypto/replay/ut_sim.c @@ -0,0 +1,105 @@ +/* + * ut_sim.c + * + * an unreliable transport simulator + * (for testing replay databases and suchlike) + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#include "ut_sim.h" + + +int +ut_compar(const void *a, const void *b) { + return rand() > (RAND_MAX/2) ? -1 : 1; +} + +void +ut_init(ut_connection *utc) { + int i; + utc->index = 0; + + for (i=0; i < UT_BUF; i++) + utc->buffer[i] = i; + + qsort(utc->buffer, UT_BUF, sizeof(uint32_t), ut_compar); + + utc->index = UT_BUF - 1; +} + +uint32_t +ut_next_index(ut_connection *utc) { + uint32_t tmp; + + tmp = utc->buffer[0]; + utc->index++; + utc->buffer[0] = utc->index; + + qsort(utc->buffer, UT_BUF, sizeof(uint32_t), ut_compar); + + return tmp; +} + + + +#ifdef UT_TEST + +#include + +int +main() { + uint32_t i, irecvd, idiff; + ut_connection utc; + + ut_init(&utc); + + for (i=0; i < 1000; i++) { + irecvd = ut_next_index(&utc); + idiff = i - irecvd; + printf("%lu\t%lu\t%d\n", i, irecvd, idiff); + } + + return 0; +} + + +#endif diff --git a/libs/srtp/crypto/rng/CVS/Entries b/libs/srtp/crypto/rng/CVS/Entries new file mode 100644 index 0000000000..1546f8f0ac --- /dev/null +++ b/libs/srtp/crypto/rng/CVS/Entries @@ -0,0 +1,5 @@ +/ctr_prng.c/1.4/Thu Sep 29 12:48:42 2005// +/prng.c/1.6/Sat Oct 8 16:38:06 2005// +/rand_linux_kernel.c/1.1/Mon Oct 3 15:29:10 2005// +/rand_source.c/1.6/Sun Oct 2 20:22:36 2005// +D diff --git a/libs/srtp/crypto/rng/CVS/Repository b/libs/srtp/crypto/rng/CVS/Repository new file mode 100644 index 0000000000..aeb47c266b --- /dev/null +++ b/libs/srtp/crypto/rng/CVS/Repository @@ -0,0 +1 @@ +srtp/crypto/rng diff --git a/libs/srtp/crypto/rng/CVS/Root b/libs/srtp/crypto/rng/CVS/Root new file mode 100644 index 0000000000..b4b74237d9 --- /dev/null +++ b/libs/srtp/crypto/rng/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@cvs.sourceforge.net:/cvsroot/srtp diff --git a/libs/srtp/crypto/rng/ctr_prng.c b/libs/srtp/crypto/rng/ctr_prng.c new file mode 100644 index 0000000000..91effb96ab --- /dev/null +++ b/libs/srtp/crypto/rng/ctr_prng.c @@ -0,0 +1,108 @@ +/* + * ctr_prng.c + * + * counter mode based pseudorandom source + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright(c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#include "prng.h" + +/* single, global prng structure */ + +static ctr_prng_t ctr_prng; + +err_status_t +ctr_prng_init(rand_source_func_t random_source) { + uint8_t tmp_key[32]; + err_status_t status; + + /* initialize output count to zero */ + ctr_prng.octet_count = 0; + + /* set random source */ + ctr_prng.rand = random_source; + + /* initialize secret key from random source */ + status = random_source(tmp_key, 32); + if (status) + return status; + + /* initialize aes ctr context with random key */ + status = aes_icm_context_init(&ctr_prng.state, tmp_key); + if (status) + return status; + + return err_status_ok; +} + +err_status_t +ctr_prng_get_octet_string(void *dest, uint32_t len) { + err_status_t status; + + /* + * if we need to re-initialize the prng, do so now + * + * avoid 32-bit overflows by subtracting instead of adding + */ + if (ctr_prng.octet_count > MAX_PRNG_OUT_LEN - len) { + status = ctr_prng_init(ctr_prng.rand); + if (status) + return status; + } + ctr_prng.octet_count += len; + + /* + * write prng output + */ + status = aes_icm_output(&ctr_prng.state, dest, len); + if (status) + return status; + + return err_status_ok; +} + +err_status_t +ctr_prng_deinit(void) { + + /* nothing */ + + return err_status_ok; +} diff --git a/libs/srtp/crypto/rng/prng.c b/libs/srtp/crypto/rng/prng.c new file mode 100644 index 0000000000..559b7dba81 --- /dev/null +++ b/libs/srtp/crypto/rng/prng.c @@ -0,0 +1,180 @@ +/* + * prng.c + * + * pseudorandom source + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright(c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#include "prng.h" + +/* single, global prng structure */ + +x917_prng_t x917_prng; + +err_status_t +x917_prng_init(rand_source_func_t random_source) { + v128_t tmp_key; + err_status_t status; + + /* initialize output count to zero */ + x917_prng.octet_count = 0; + + /* set random source */ + x917_prng.rand = random_source; + + /* initialize secret key from random source */ + status = random_source((uint8_t *)&tmp_key, 16); + if (status) + return status; + + /* expand aes key */ + aes_expand_encryption_key(&tmp_key, x917_prng.key); + + /* initialize prng state from random source */ + status = x917_prng.rand((uint8_t *)&x917_prng.state, 16); + if (status) + return status; + + return err_status_ok; +} + +err_status_t +x917_prng_get_octet_string(uint8_t *dest, uint32_t len) { + uint32_t t; + v128_t buffer; + int i, tail_len; + err_status_t status; + + /* + * if we need to re-initialize the prng, do so now + * + * avoid overflows by subtracting instead of adding + */ + if (x917_prng.octet_count > MAX_PRNG_OUT_LEN - len) { + status = x917_prng_init(x917_prng.rand); + if (status) + return status; + } + x917_prng.octet_count += len; + + /* find out the time */ + t = time(NULL); + + /* loop until we have output enough data */ + for (i=0; i < len/16; i++) { + + /* exor time into state */ + x917_prng.state.v32[0] ^= t; + + /* copy state into buffer */ + v128_copy(&buffer, &x917_prng.state); + + /* apply aes to buffer */ + aes_encrypt(&buffer, x917_prng.key); + + /* write data to output */ + *dest++ = buffer.v8[0]; + *dest++ = buffer.v8[1]; + *dest++ = buffer.v8[2]; + *dest++ = buffer.v8[3]; + *dest++ = buffer.v8[4]; + *dest++ = buffer.v8[5]; + *dest++ = buffer.v8[6]; + *dest++ = buffer.v8[7]; + *dest++ = buffer.v8[8]; + *dest++ = buffer.v8[9]; + *dest++ = buffer.v8[10]; + *dest++ = buffer.v8[11]; + *dest++ = buffer.v8[12]; + *dest++ = buffer.v8[13]; + *dest++ = buffer.v8[14]; + *dest++ = buffer.v8[15]; + + /* exor time into buffer */ + buffer.v32[0] ^= t; + + /* encrypt buffer */ + aes_encrypt(&buffer, x917_prng.key); + + /* copy buffer into state */ + v128_copy(&x917_prng.state, &buffer); + + } + + /* if we need to output any more octets, we'll do so now */ + tail_len = len % 16; + if (tail_len) { + + /* exor time into state */ + x917_prng.state.v32[0] ^= t; + + /* copy value into buffer */ + v128_copy(&buffer, &x917_prng.state); + + /* apply aes to buffer */ + aes_encrypt(&buffer, x917_prng.key); + + /* write data to output */ + for (i=0; i < tail_len; i++) { + *dest++ = buffer.v8[i]; + } + + /* now update the state one more time */ + + /* exor time into buffer */ + buffer.v32[0] ^= t; + + /* encrypt buffer */ + aes_encrypt(&buffer, x917_prng.key); + + /* copy buffer into state */ + v128_copy(&x917_prng.state, &buffer); + + } + + return err_status_ok; +} + +err_status_t +x917_prng_deinit(void) { + + return err_status_ok; +} diff --git a/libs/srtp/crypto/rng/rand_linux_kernel.c b/libs/srtp/crypto/rng/rand_linux_kernel.c new file mode 100644 index 0000000000..c51978e5e5 --- /dev/null +++ b/libs/srtp/crypto/rng/rand_linux_kernel.c @@ -0,0 +1,65 @@ +/* + * rand_linux_kernel.c + * + * implements a random source using Linux kernel functions + * + * Marcus Sundberg + * Ingate Systems AB + */ +/* + * + * Copyright(c) 2005 Ingate Systems AB + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the author(s) nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include "config.h" +#include "rand_source.h" + + +err_status_t +rand_source_init(void) { + return err_status_ok; +} + +err_status_t +rand_source_get_octet_string(void *dest, uint32_t len) { + + get_random_bytes(dest, len); + + return err_status_ok; +} + +err_status_t +rand_source_deinit(void) { + return err_status_ok; +} diff --git a/libs/srtp/crypto/rng/rand_source.c b/libs/srtp/crypto/rng/rand_source.c new file mode 100644 index 0000000000..48e2a6e079 --- /dev/null +++ b/libs/srtp/crypto/rng/rand_source.c @@ -0,0 +1,118 @@ +/* + * rand_source.c + * + * implements a random source based on /dev/random + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright(c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include "config.h" +#include "rand_source.h" + +#ifdef DEV_URANDOM +# include /* for open() */ +# include /* for close() */ +#else +# include +#endif + +/* global dev_rand_fdes is file descriptor for /dev/random */ + +static int dev_random_fdes = -1; + + +err_status_t +rand_source_init(void) { + if (dev_random_fdes >= 0) { + /* already open */ + return err_status_ok; + } +#ifdef DEV_URANDOM + /* open random source for reading */ + dev_random_fdes = open(DEV_URANDOM, O_RDONLY); + if (dev_random_fdes < 0) + return err_status_init_fail; +#else + /* no random source available; let the user know */ + fprintf(stderr, "WARNING: no real random source present!\n"); + dev_random_fdes = 17; +#endif + return err_status_ok; +} + +err_status_t +rand_source_get_octet_string(void *dest, uint32_t len) { + + /* + * read len octets from /dev/random to dest, and + * check return value to make sure enough octets were + * written + */ +#ifdef DEV_URANDOM + if (read(dev_random_fdes, dest, len) != len) + return err_status_fail; +#else + /* Generic C-library (rand()) version */ + /* This is a random source of last resort */ + uint8_t *dst = dest; + while (len) + { + int val = rand(); + /* rand() returns 0-32767 (ugh) */ + /* Is this a good enough way to get random bytes? + It is if it passes FIPS-140... */ + *dst++ = val & 0xff; + len--; + } +#endif + return err_status_ok; +} + +err_status_t +rand_source_deinit(void) { + if (dev_random_fdes < 0) + return err_status_dealloc_fail; /* well, we haven't really failed, * + * but there is something wrong */ +#ifdef DEV_URANDOM + close(dev_random_fdes); +#endif + dev_random_fdes = -1; + + return err_status_ok; +} diff --git a/libs/srtp/crypto/test/.cvsignore b/libs/srtp/crypto/test/.cvsignore new file mode 100644 index 0000000000..888bf597e1 --- /dev/null +++ b/libs/srtp/crypto/test/.cvsignore @@ -0,0 +1,8 @@ +aes_calc +cipher_driver +datatypes_driver +env +kernel_driver +rand_gen +sha1_driver +stat_driver diff --git a/libs/srtp/crypto/test/CVS/Entries b/libs/srtp/crypto/test/CVS/Entries new file mode 100644 index 0000000000..21ad14f19c --- /dev/null +++ b/libs/srtp/crypto/test/CVS/Entries @@ -0,0 +1,11 @@ +/.cvsignore/1.1/Thu Sep 29 12:06:41 2005// +/aes_calc.c/1.4/Sat Oct 8 16:23:59 2005// +/auth_driver.c/1.2/Thu Sep 29 12:36:43 2005// +/cipher_driver.c/1.5/Mon Oct 3 16:29:05 2005// +/datatypes_driver.c/1.5/Sat Oct 8 16:38:06 2005// +/env.c/1.4/Sun Oct 2 20:40:01 2005// +/kernel_driver.c/1.2/Sun Oct 2 20:40:29 2005// +/rand_gen.c/1.2/Thu Sep 29 12:36:43 2005// +/sha1_driver.c/1.4/Fri Mar 17 17:41:33 2006// +/stat_driver.c/1.3/Sun Oct 2 20:43:26 2005// +D diff --git a/libs/srtp/crypto/test/CVS/Repository b/libs/srtp/crypto/test/CVS/Repository new file mode 100644 index 0000000000..487778ee08 --- /dev/null +++ b/libs/srtp/crypto/test/CVS/Repository @@ -0,0 +1 @@ +srtp/crypto/test diff --git a/libs/srtp/crypto/test/CVS/Root b/libs/srtp/crypto/test/CVS/Root new file mode 100644 index 0000000000..b4b74237d9 --- /dev/null +++ b/libs/srtp/crypto/test/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@cvs.sourceforge.net:/cvsroot/srtp diff --git a/libs/srtp/crypto/test/aes_calc.c b/libs/srtp/crypto/test/aes_calc.c new file mode 100644 index 0000000000..2fac07ae35 --- /dev/null +++ b/libs/srtp/crypto/test/aes_calc.c @@ -0,0 +1,111 @@ +/* + * aes_calc.c + * + * A simple AES calculator for generating AES encryption values + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + + Example usage (with first NIST FIPS 197 test case): + +[sh]$ test/aes_calc 000102030405060708090a0b0c0d0e0f 00112233445566778899aabbccddeeff -v + plaintext: 00112233445566778899aabbccddeeff + key: 000102030405060708090a0b0c0d0e0f + ciphertext: 69c4e0d86a7b0430d8cdb78070b4c55a + + */ + +#include "aes.h" +#include +#include + +void +usage(char *prog_name) { + printf("usage: %s [-v]\n", prog_name); + exit(255); +} + +#define AES_KEY_LEN 16 + +int +main (int argc, char *argv[]) { + v128_t data, key; + aes_expanded_key_t exp_key; + int len; + int verbose; + + if (argc == 3) { + /* we're not in verbose mode */ + verbose = 0; + } else if (argc == 4) { + if (strncmp(argv[3], "-v", 2) == 0) { + /* we're in verbose mode */ + verbose = 1; + } else { + /* unrecognized flag, complain and exit */ + usage(argv[0]); + } + } else { + /* we've been fed the wrong number of arguments - compain and exit */ + usage(argv[0]); + } + + /* read in key, checking length */ + if (strlen(argv[1]) > AES_KEY_LEN*2) { + fprintf(stderr, + "error: too many digits in key " + "(should be %d hexadecimal digits, found %u)\n", + AES_KEY_LEN*2, (unsigned)strlen(argv[1])); + exit(1); + } + len = hex_string_to_octet_string((char *)&key, argv[1], AES_KEY_LEN*2); + /* check that hex string is the right length */ + if (len < AES_KEY_LEN*2) { + fprintf(stderr, + "error: too few digits in key " + "(should be %d hexadecimal digits, found %d)\n", + AES_KEY_LEN*2, len); + exit(1); + } + + /* read in plaintext, checking length */ + if (strlen(argv[2]) > 16*2) { + fprintf(stderr, + "error: too many digits in plaintext " + "(should be %d hexadecimal digits, found %u)\n", + 16*2, (unsigned)strlen(argv[2])); + exit(1); + } + len = hex_string_to_octet_string((char *)(&data), argv[2], 16*2); + /* check that hex string is the right length */ + if (len < 16*2) { + fprintf(stderr, + "error: too few digits in plaintext " + "(should be %d hexadecimal digits, found %d)\n", + 16*2, len); + exit(1); + } + + if (verbose) { + /* print out plaintext */ + printf("plaintext:\t%s\n", octet_string_hex_string((uint8_t *)&data, 16)); + } + + /* encrypt plaintext */ + aes_expand_encryption_key(&key, exp_key); + + aes_encrypt(&data, exp_key); + + /* write ciphertext to output */ + if (verbose) { + printf("key:\t\t%s\n", v128_hex_string(&key)); + printf("ciphertext:\t"); + } + printf("%s\n", v128_hex_string(&data)); + + return 0; +} + diff --git a/libs/srtp/crypto/test/auth_driver.c b/libs/srtp/crypto/test/auth_driver.c new file mode 100644 index 0000000000..3073d49500 --- /dev/null +++ b/libs/srtp/crypto/test/auth_driver.c @@ -0,0 +1,200 @@ +/* + * auth_driver.c + * + * a driver for auth functions + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#include <stdio.h> /* for printf() */ +#include <stdlib.h> /* for xalloc() */ +#include <unistd.h> /* for getopt() */ + +#include "auth.h" +#include "null_auth.h" + +#define PRINT_DEBUG_DATA 0 + +extern auth_type_t tmmhv2; + +const uint16_t msg0[9] = { + 0x6015, 0xf141, 0x5ba1, 0x29a0, 0xf604, 0xd1c, 0x2d9, 0xaa8a, 0x7931 +}; + +/* key1 is for TAG_WORDS = 2 */ + +const uint16_t key1[47] = { + 0xe627, 0x6a01, 0x5ea7, 0xf27a, 0xc536, 0x2192, 0x11be, 0xea35, + 0xdb9d, 0x63d6, 0xfa8a, 0xfc45, 0xe08b, 0xd216, 0xced2, 0x7853, + 0x1a82, 0x22f5, 0x90fb, 0x1c29, 0x708e, 0xd06f, 0x82c3, 0xbee6, + 0x4f21, 0x6f33, 0x65c0, 0xd211, 0xc25e, 0x9138, 0x4fa3, 0x7c1f, + 0x61ac, 0x3489, 0x2976, 0x8c19, 0x8252, 0xddbf, 0xcad3, 0xc28f, + 0x68d6, 0x58dd, 0x504f, 0x2bbf, 0x0278, 0x70b7, 0xcfca +}; + +double +auth_bits_per_second(auth_t *h, int msg_len); + + +void +usage(char *prog_name) { + printf("usage: %s [ -t | -v ]\n", prog_name); + exit(255); +} + +#define MAX_MSG_LEN 2048 + +int +main (int argc, char *argv[]) { + auth_t *a = NULL; + err_status_t status; + int i; + int c; + unsigned do_timing_test = 0; + unsigned do_validation = 0; + + /* process input arguments */ + while (1) { + c = getopt(argc, argv, "tv"); + if (c == -1) + break; + switch (c) { + case 't': + do_timing_test = 1; + break; + case 'v': + do_validation = 1; + break; + default: + usage(argv[0]); + } + } + + printf("auth driver\nDavid A. McGrew\nCisco Systems, Inc.\n"); + + if (!do_validation && !do_timing_test) + usage(argv[0]); + + if (do_validation) { + printf("running self-test for %s...", tmmhv2.description); + status = tmmhv2_add_big_test(); + if (status) { + printf("tmmhv2_add_big_test failed with error code %d\n", status); + exit(status); + } + status = auth_type_self_test(&tmmhv2); + if (status) { + printf("failed with error code %d\n", status); + exit(status); + } + printf("passed\n"); + } + + if (do_timing_test) { + + /* tmmhv2 timing test */ + status = auth_type_alloc(&tmmhv2, &a, 94, 4); + if (status) { + fprintf(stderr, "can't allocate tmmhv2\n"); + exit(status); + } + status = auth_init(a, (uint8_t *)key1); + if (status) { + printf("error initializaing auth function\n"); + exit(status); + } + + printf("timing %s (tag length %d)\n", + tmmhv2.description, auth_get_tag_length(a)); + for (i=8; i <= MAX_MSG_LEN; i *= 2) + printf("msg len: %d\tgigabits per second: %f\n", + i, auth_bits_per_second(a, i) / 1E9); + + status = auth_dealloc(a); + if (status) { + printf("error deallocating auth function\n"); + exit(status); + } + + } + + return 0; +} + +#define NUM_TRIALS 100000 + +#include <time.h> + +double +auth_bits_per_second(auth_t *a, int msg_len_octets) { + int i; + clock_t timer; + uint8_t *result; + int msg_len = (msg_len_octets + 1)/2; + uint16_t *msg_string; + + /* create random message */ + msg_string = (uint16_t *) crypto_alloc(msg_len_octets); + if (msg_string == NULL) + return 0.0; /* indicate failure */ + for (i=0; i < msg_len; i++) + msg_string[i] = (uint16_t) random(); + + /* allocate temporary storage for authentication tag */ + result = crypto_alloc(auth_get_tag_length(a)); + if (result == NULL) { + free(msg_string); + return 0.0; /* indicate failure */ + } + + timer = clock(); + for (i=0; i < NUM_TRIALS; i++) { + auth_compute(a, (uint8_t *)msg_string, msg_len_octets, (uint8_t *)result); + } + timer = clock() - timer; + + free(msg_string); + free(result); + + return (double) NUM_TRIALS * 8 * msg_len_octets * CLOCKS_PER_SEC / timer; +} + + diff --git a/libs/srtp/crypto/test/cipher_driver.c b/libs/srtp/crypto/test/cipher_driver.c new file mode 100644 index 0000000000..276e775b83 --- /dev/null +++ b/libs/srtp/crypto/test/cipher_driver.c @@ -0,0 +1,491 @@ +/* + * cipher_driver.c + * + * A driver for the generic cipher type + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include <stdio.h> /* for printf() */ +#include <stdlib.h> /* for rand() */ +#include <string.h> /* for memset() */ +#include <unistd.h> /* for getopt() */ +#include "cipher.h" +#include "aes_icm.h" +#include "null_cipher.h" + +#define PRINT_DEBUG 0 + +void +cipher_driver_test_throughput(cipher_t *c); + +err_status_t +cipher_driver_self_test(cipher_type_t *ct); + + +/* + * cipher_driver_test_buffering(ct) tests the cipher's output + * buffering for correctness by checking the consistency of succesive + * calls + */ + +err_status_t +cipher_driver_test_buffering(cipher_t *c); + + +/* + * functions for testing cipher cache thrash + */ +err_status_t +cipher_driver_test_array_throughput(cipher_type_t *ct, + int klen, int num_cipher); + +void +cipher_array_test_throughput(cipher_t *ca[], int num_cipher); + +uint64_t +cipher_array_bits_per_second(cipher_t *cipher_array[], int num_cipher, + unsigned octets_in_buffer, int num_trials); + +err_status_t +cipher_array_delete(cipher_t *cipher_array[], int num_cipher); + +err_status_t +cipher_array_alloc_init(cipher_t ***cipher_array, int num_ciphers, + cipher_type_t *ctype, int klen); + +void +usage(char *prog_name) { + printf("usage: %s [ -t | -v | -a ]\n", prog_name); + exit(255); +} + +void +check_status(err_status_t s) { + if (s) { + printf("error (code %d)\n", s); + exit(s); + } + return; +} + +/* + * null_cipher, aes_icm, and aes_cbc are the cipher meta-objects + * defined in the files in crypto/cipher subdirectory. these are + * declared external so that we can use these cipher types here + */ + +extern cipher_type_t null_cipher; +extern cipher_type_t aes_icm; +extern cipher_type_t aes_cbc; + +int +main(int argc, char *argv[]) { + cipher_t *c = NULL; + err_status_t status; + unsigned char test_key[20] = { + 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, + 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f, + 0x10, 0x11, 0x12, 0x13 + }; + int q; + unsigned do_timing_test = 0; + unsigned do_validation = 0; + unsigned do_array_timing_test = 0; + + /* process input arguments */ + while (1) { + q = getopt(argc, argv, "tva"); + if (q == -1) + break; + switch (q) { + case 't': + do_timing_test = 1; + break; + case 'v': + do_validation = 1; + break; + case 'a': + do_array_timing_test = 1; + break; + default: + usage(argv[0]); + } + } + + printf("cipher test driver\n" + "David A. McGrew\n" + "Cisco Systems, Inc.\n"); + + if (!do_validation && !do_timing_test && !do_array_timing_test) + usage(argv[0]); + + /* arry timing (cache thrash) test */ + if (do_array_timing_test) { + int max_num_cipher = 1 << 16; /* number of ciphers in cipher_array */ + int num_cipher; + + for (num_cipher=1; num_cipher < max_num_cipher; num_cipher *=8) + cipher_driver_test_array_throughput(&null_cipher, 0, num_cipher); + + for (num_cipher=1; num_cipher < max_num_cipher; num_cipher *=8) + cipher_driver_test_array_throughput(&aes_icm, 30, num_cipher); + + for (num_cipher=1; num_cipher < max_num_cipher; num_cipher *=8) + cipher_driver_test_array_throughput(&aes_cbc, 16, num_cipher); + + } + + if (do_validation) { + cipher_driver_self_test(&null_cipher); + cipher_driver_self_test(&aes_icm); + cipher_driver_self_test(&aes_cbc); + } + + /* do timing and/or buffer_test on null_cipher */ + status = cipher_type_alloc(&null_cipher, &c, 0); + check_status(status); + + status = cipher_init(c, NULL, direction_encrypt); + check_status(status); + + if (do_timing_test) + cipher_driver_test_throughput(c); + if (do_validation) { + status = cipher_driver_test_buffering(c); + check_status(status); + } + status = cipher_dealloc(c); + check_status(status); + + + /* run the throughput test on the aes_icm cipher */ + status = cipher_type_alloc(&aes_icm, &c, 30); + if (status) { + fprintf(stderr, "error: can't allocate cipher\n"); + exit(status); + } + + status = cipher_init(c, test_key, direction_encrypt); + check_status(status); + + if (do_timing_test) + cipher_driver_test_throughput(c); + + if (do_validation) { + status = cipher_driver_test_buffering(c); + check_status(status); + } + + status = cipher_dealloc(c); + check_status(status); + + return 0; +} + +void +cipher_driver_test_throughput(cipher_t *c) { + int i; + int min_enc_len = 32; + int max_enc_len = 2048; /* should be a power of two */ + int num_trials = 100000; + + printf("timing %s throughput:\n", c->type->description); + fflush(stdout); + for (i=min_enc_len; i <= max_enc_len; i = i * 2) + printf("msg len: %d\tgigabits per second: %f\n", + i, cipher_bits_per_second(c, i, num_trials) / 1e9); + +} + +err_status_t +cipher_driver_self_test(cipher_type_t *ct) { + err_status_t status; + + printf("running cipher self-test for %s...", ct->description); + status = cipher_type_self_test(ct); + if (status) { + printf("failed with error code %d\n", status); + exit(status); + } + printf("passed\n"); + + return err_status_ok; +} + +/* + * cipher_driver_test_buffering(ct) tests the cipher's output + * buffering for correctness by checking the consistency of succesive + * calls + */ + +err_status_t +cipher_driver_test_buffering(cipher_t *c) { + int i, j, num_trials = 1000; + unsigned len, buflen = 1024; + uint8_t buffer0[buflen], buffer1[buflen], *current, *end; + uint8_t idx[16] = { + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x12, 0x34 + }; + err_status_t status; + + printf("testing output buffering for cipher %s...", + c->type->description); + + for (i=0; i < num_trials; i++) { + + /* set buffers to zero */ + for (j=0; j < buflen; j++) + buffer0[j] = buffer1[j] = 0; + + /* initialize cipher */ + status = cipher_set_iv(c, idx); + if (status) + return status; + + /* generate 'reference' value by encrypting all at once */ + status = cipher_encrypt(c, buffer0, &buflen); + if (status) + return status; + + /* re-initialize cipher */ + status = cipher_set_iv(c, idx); + if (status) + return status; + + /* now loop over short lengths until buffer1 is encrypted */ + current = buffer1; + end = buffer1 + buflen; + while (current < end) { + + /* choose a short length */ + len = rand() & 0x01f; + + /* make sure that len doesn't cause us to overreach the buffer */ + if (current + len > end) + len = end - current; + + status = cipher_encrypt(c, current, &len); + if (status) + return status; + + /* advance pointer into buffer1 to reflect encryption */ + current += len; + + /* if buffer1 is all encrypted, break out of loop */ + if (current == end) + break; + } + + /* compare buffers */ + for (j=0; j < buflen; j++) + if (buffer0[j] != buffer1[j]) { +#if PRINT_DEBUG + printf("test case %d failed at byte %d\n", i, j); + printf("computed: %s\n", octet_string_hex_string(buffer1, buflen)); + printf("expected: %s\n", octet_string_hex_string(buffer0, buflen)); +#endif + return err_status_algo_fail; + } + } + + printf("passed\n"); + + return err_status_ok; +} + + +/* + * The function cipher_test_throughput_array() tests the effect of CPU + * cache thrash on cipher throughput. + * + * cipher_array_alloc_init(ctype, array, num_ciphers) creates an array + * of cipher_t of type ctype + */ + +err_status_t +cipher_array_alloc_init(cipher_t ***ca, int num_ciphers, + cipher_type_t *ctype, int klen) { + int i, j; + err_status_t status; + uint8_t *key; + cipher_t **cipher_array; + + /* allocate array of pointers to ciphers */ + cipher_array = (cipher_t **) malloc(sizeof(cipher_t *) * num_ciphers); + if (cipher_array == NULL) + return err_status_alloc_fail; + + /* set ca to location of cipher_array */ + *ca = cipher_array; + + /* allocate key */ + key = crypto_alloc(klen); + if (key == NULL) { + free(cipher_array); + return err_status_alloc_fail; + } + + /* allocate and initialize an array of ciphers */ + for (i=0; i < num_ciphers; i++) { + + /* allocate cipher */ + status = cipher_type_alloc(ctype, cipher_array, klen); + if (status) + return status; + + /* generate random key and initialize cipher */ + for (j=0; j < klen; j++) + key[j] = (uint8_t) rand(); + status = cipher_init(*cipher_array, key, direction_encrypt); + if (status) + return status; + +/* printf("%dth cipher is at %p\n", i, *cipher_array); */ +/* printf("%dth cipher description: %s\n", i, */ +/* (*cipher_array)->type->description); */ + + /* advance cipher array pointer */ + cipher_array++; + } + + return err_status_ok; +} + +err_status_t +cipher_array_delete(cipher_t *cipher_array[], int num_cipher) { + int i; + + for (i=0; i < num_cipher; i++) { + cipher_dealloc(cipher_array[i]); + } + + free(cipher_array); + + return err_status_ok; +} + + +/* + * cipher_array_bits_per_second(c, l, t) computes (an estimate of) the + * number of bits that a cipher implementation can encrypt in a second + * when distinct keys are used to encrypt distinct messages + * + * c is a cipher (which MUST be allocated an initialized already), l + * is the length in octets of the test data to be encrypted, and t is + * the number of trials + * + * if an error is encountered, the value 0 is returned + */ + +uint64_t +cipher_array_bits_per_second(cipher_t *cipher_array[], int num_cipher, + unsigned octets_in_buffer, int num_trials) { + int i; + v128_t nonce; + clock_t timer; + unsigned char *enc_buf; + int cipher_index = 0; + + + enc_buf = crypto_alloc(octets_in_buffer); + if (enc_buf == NULL) + return 0; /* indicate bad parameters by returning null */ + + /* time repeated trials */ + v128_set_to_zero(&nonce); + timer = clock(); + for(i=0; i < num_trials; i++, nonce.v32[3] = i) { + + /* choose a cipher at random from the array*/ + cipher_index = (*((uint32_t *)enc_buf)) % num_cipher; + + /* encrypt buffer with cipher */ + cipher_set_iv(cipher_array[cipher_index], &nonce); + cipher_encrypt(cipher_array[cipher_index], enc_buf, &octets_in_buffer); + } + timer = clock() - timer; + + free(enc_buf); + + if (timer == 0) { + /* Too fast! */ + return 0; + } + + return CLOCKS_PER_SEC * num_trials * 8 * octets_in_buffer / timer; +} + +void +cipher_array_test_throughput(cipher_t *ca[], int num_cipher) { + int i; + int min_enc_len = 16; + int max_enc_len = 2048; /* should be a power of two */ + int num_trials = 10000; + + printf("timing %s throughput with array size %d:\n", + (ca[0])->type->description, num_cipher); + fflush(stdout); + for (i=min_enc_len; i <= max_enc_len; i = i * 4) + printf("msg len: %d\tgigabits per second: %f\n", i, + cipher_array_bits_per_second(ca, num_cipher, i, num_trials) / 1e9); + +} + +err_status_t +cipher_driver_test_array_throughput(cipher_type_t *ct, + int klen, int num_cipher) { + cipher_t **ca = NULL; + err_status_t status; + + status = cipher_array_alloc_init(&ca, num_cipher, ct, klen); + if (status) { + printf("error: cipher_array_alloc_init() failed with error code %d\n", + status); + return status; + } + + cipher_array_test_throughput(ca, num_cipher); + + cipher_array_delete(ca, num_cipher); + + return err_status_ok; +} diff --git a/libs/srtp/crypto/test/datatypes_driver.c b/libs/srtp/crypto/test/datatypes_driver.c new file mode 100644 index 0000000000..f8e34ef942 --- /dev/null +++ b/libs/srtp/crypto/test/datatypes_driver.c @@ -0,0 +1,237 @@ +/* + * datatypes_driver.c + * + * a test driver for crypto/math datatypes + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#include <stdio.h> /* for printf() */ +#include <string.h> /* for strlen() */ +#include "datatypes.h" + +void +byte_order(void); + +void +test_hex_string_funcs(void); + +void +print_string(char *s); + +void +test_bswap(void); + +int +main (void) { + + /* + * this program includes various and sundry tests for fundamental + * datatypes. it's a grab-bag of throwaway code, retained only in + * case of future problems + */ + + int i, j; + v128_t x; + char *r = + "The Moving Finger writes; and, having writ,\n" + "Moves on: nor all thy Piety nor Wit\n" + "Shall lure it back to cancel half a Line,\n" + "Nor all thy Tears wash out a Word of it."; + char *s = "incomplet"; + + print_string(r); + print_string(s); + + byte_order(); + test_hex_string_funcs(); + + for (j=0; j < 128; j++) { + v128_set_to_zero(&x); + /* x.v32[0] = (1 << j); */ + v128_set_bit(&x, j); + printf("%s\n", v128_bit_string(&x)); + v128_clear_bit(&x, j); + printf("%s\n", v128_bit_string(&x)); + + } + + printf("----------------------------------------------\n"); + v128_set_to_zero(&x); + for (i=0; i < 128; i++) { + v128_set_bit(&x, i); + } + printf("%s\n", v128_bit_string(&x)); + + printf("----------------------------------------------\n"); + v128_set_to_zero(&x); + v128_set_bit(&x, 0); + for (i=0; i < 128; i++) { + printf("%s\n", v128_bit_string(&x)); + v128_right_shift(&x, 1); + } + printf("----------------------------------------------\n"); + v128_set_to_zero(&x); + v128_set_bit(&x, 127); + for (i=0; i < 128; i++) { + printf("%s\n", v128_bit_string(&x)); + v128_left_shift(&x, 1); + } + printf("----------------------------------------------\n"); + for (i=0; i < 128; i++) { + v128_set_to_zero(&x); + v128_set_bit(&x, 127); + v128_left_shift(&x, i); + printf("%s\n", v128_bit_string(&x)); + } + printf("----------------------------------------------\n"); + v128_set_to_zero(&x); + for (i=0; i < 128; i+=2) { + v128_set_bit(&x, i); + } + printf("bit_string: { %s }\n", v128_bit_string(&x)); + printf("get_bit: { "); + for (i=0; i < 128; i++) { + if (v128_get_bit(&x, i) == 1) + printf("1"); + else + printf("0"); + } + printf(" } \n"); + + test_bswap(); + + return 0; +} + + +/* byte_order() prints out byte ordering of datatypes */ + +void +byte_order(void) { + int i; + v128_t e; +#if 0 + v16_t b; + v32_t c; + v64_t d; + + for (i=0; i < sizeof(b); i++) + b.octet[i] = i; + for (i=0; i < sizeof(c); i++) + c.octet[i] = i; + for (i=0; i < sizeof(d); i++) + d.octet[i] = i; + + printf("v128_t:\t%s\n", v128_hex_string(&e)); + printf("v64_t:\t%s\n", v64_hex_string(&d)); + printf("v32_t:\t%s\n", v32_hex_string(c)); + printf("v16_t:\t%s\n", v16_hex_string(b)); + + c.value = 0x01020304; + printf("v32_t:\t%s\n", v32_hex_string(c)); + b.value = 0x0102; + printf("v16_t:\t%s\n", v16_hex_string(b)); + + printf("uint16_t ordering:\n"); + + c.value = 0x00010002; + printf("v32_t:\t%x%x\n", c.v16[0], c.v16[1]); +#endif + + printf("byte ordering of crypto/math datatypes:\n"); + for (i=0; i < sizeof(e); i++) + e.v8[i] = i; + printf("v128_t: %s\n", v128_hex_string(&e)); + +} + +void +test_hex_string_funcs(void) { + char hex1[] = "abadcafe"; + char hex2[] = "0123456789abcdefqqqqq"; + char raw[10]; + int len; + + len = hex_string_to_octet_string(raw, hex1, strlen(hex1)); + printf("computed length: %d\tstring: %s\n", len, + octet_string_hex_string(raw, len/2)); + printf("expected length: %u\tstring: %s\n", (unsigned)strlen(hex1), hex1); + + len = hex_string_to_octet_string(raw, hex2, strlen(hex2)); + printf("computed length: %d\tstring: %s\n", len, + octet_string_hex_string(raw, len/2)); + printf("expected length: %d\tstring: %s\n", 16, "0123456789abcdef"); + +} + +void +print_string(char *s) { + int i; + printf("%s\n", s); + printf("strlen(s) = %u\n", (unsigned)strlen(s)); + printf("{ "); + for (i=0; i < strlen(s); i++) { + printf("0x%x, ", s[i]); + if (((i+1) % 8) == 0) + printf("\n "); + } + printf("}\n"); +} + +void +test_bswap(void) { + uint32_t x = 0x11223344; + uint64_t y = 0x1122334455667788LL; + + printf("before: %0x\nafter: %0x\n", x, be32_to_cpu(x)); + printf("before: %0llx\nafter: %0llx\n", (unsigned long long)y, + (unsigned long long)be64_to_cpu(y)); + + y = 1234; + + printf("1234: %0llx\n", (unsigned long long)y); + printf("as octet string: %s\n", + octet_string_hex_string((uint8_t *) &y, 8)); + y = be64_to_cpu(y); + printf("bswapped octet string: %s\n", + octet_string_hex_string((uint8_t *) &y, 8)); +} diff --git a/libs/srtp/crypto/test/env.c b/libs/srtp/crypto/test/env.c new file mode 100644 index 0000000000..07b6f51527 --- /dev/null +++ b/libs/srtp/crypto/test/env.c @@ -0,0 +1,99 @@ +/* + * env.c + * + * prints out a brief report on the build environment + * + * David McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright (c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include <stdio.h> +#include <string.h> /* for srtcmp() */ +#include "config.h" + +int +main(void) { + int err_count = 0; + char *str; + +#ifdef WORDS_BIGENDIAN + printf("CPU set to big-endian\t\t\t(WORDS_BIGENDIAN == 1)\n"); +#else + printf("CPU set to little-endian\t\t(WORDS_BIGENDIAN == 0)\n"); +#endif + +#ifdef CPU_RISC + printf("CPU set to RISC\t\t\t\t(CPU_RISC == 1)\n"); +#elif defined(CPU_CISC) + printf("CPU set to CISC\t\t\t\t(CPU_CISC == 1)\n"); +#else + printf("CPU set to an unknown type, probably due to a configuration error\n"); + err_count++; +#endif + +#ifdef CPU_ALTIVEC + printf("CPU set to ALTIVEC\t\t\t\t(CPU_ALTIVEC == 0)\n"); +#endif + +#ifndef NO_64BIT_MATH + printf("using native 64-bit type\t\t(NO_64_BIT_MATH == 0)\n"); +#else + printf("using built-in 64-bit math\t\t(NO_64_BIT_MATH == 1)\n"); +#endif + +#ifdef ERR_REPORTING_STDOUT + printf("using stdout for error reporting\t(ERR_REPORTING_STDOUT == 1)\n"); +#endif + +#ifdef DEV_URANDOM + str = DEV_URANDOM; +#else + str = ""; +#endif + printf("using %s as a random source\t(DEV_URANDOM == %s)\n", + str, str); + if (strcmp("", str) == 0) { + err_count++; + } + + if (err_count) + printf("warning: configuration is probably in error " + "(found %d problems)\n", err_count); + + return err_count; +} diff --git a/libs/srtp/crypto/test/kernel_driver.c b/libs/srtp/crypto/test/kernel_driver.c new file mode 100644 index 0000000000..4271f8ee68 --- /dev/null +++ b/libs/srtp/crypto/test/kernel_driver.c @@ -0,0 +1,126 @@ +/* + * kernel_driver.c + * + * a test driver for the crypto_kernel + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright(c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#include <stdio.h> /* for printf() */ +#include <unistd.h> /* for getopt() */ +#include "crypto_kernel.h" + +void +usage(char *prog_name) { + printf("usage: %s [ -v ][ -d debug_module ]*\n", prog_name); + exit(255); +} + +int +main (int argc, char *argv[]) { + extern char *optarg; + int q; + int do_validation = 0; + err_status_t status; + + if (argc == 1) + usage(argv[0]); + + /* initialize kernel - we need to do this before anything else */ + status = crypto_kernel_init(); + if (status) { + printf("error: crypto_kernel init failed\n"); + exit(1); + } + printf("crypto_kernel successfully initalized\n"); + + /* process input arguments */ + while (1) { + q = getopt(argc, argv, "vd:"); + if (q == -1) + break; + switch (q) { + case 'v': + do_validation = 1; + break; + case 'd': + status = crypto_kernel_set_debug_module(optarg, 1); + if (status) { + printf("error: set debug module (%s) failed\n", optarg); + exit(1); + } + break; + default: + usage(argv[0]); + } + } + + if (do_validation) { + printf("checking crypto_kernel status...\n"); + status = crypto_kernel_status(); + if (status) { + printf("failed\n"); + exit(1); + } + printf("crypto_kernel passed self-tests\n"); + } + + status = crypto_kernel_shutdown(); + if (status) { + printf("error: crypto_kernel shutdown failed\n"); + exit(1); + } + printf("crypto_kernel successfully shut down\n"); + + return 0; +} + +/* + * crypto_kernel_cipher_test() is a test of the cipher interface + * of the crypto_kernel + */ + +err_status_t +crypto_kernel_cipher_test(void) { + + /* not implemented yet! */ + + return err_status_ok; +} diff --git a/libs/srtp/crypto/test/rand_gen.c b/libs/srtp/crypto/test/rand_gen.c new file mode 100644 index 0000000000..8fcf89cd66 --- /dev/null +++ b/libs/srtp/crypto/test/rand_gen.c @@ -0,0 +1,140 @@ +/* + * rand_gen.c + * + * a random source (random number generator) + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright(c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#include <stdio.h> /* for printf() */ +#include <unistd.h> /* for getopt() */ +#include "crypto_kernel.h" + +/* + * MAX_PRINT_STRING_LEN is defined in datatypes.h, and is the length + * of the largest hexadecimal string that can be generated by the + * function octet_string_hex_string(). + */ + +#define BUF_LEN (MAX_PRINT_STRING_LEN/2) + +void +usage(char *prog_name) { + printf("usage: %s -n <num_bytes> [-l][ -d debug_module ]*\n" + " -n <num> output <num> random bytes, where <num>" + " is between zero and %d\n" + " -l list the avaliable debug modules\n" + " -d <mod> turn on debugging module <mod>\n", + prog_name, BUF_LEN); + exit(255); +} + +int +main (int argc, char *argv[]) { + extern char *optarg; + int q; + int num_octets = 0; + unsigned do_list_mods = 0; + err_status_t status; + + if (argc == 1) + usage(argv[0]); + + /* initialize kernel - we need to do this before anything else */ + status = crypto_kernel_init(); + if (status) { + printf("error: crypto_kernel init failed\n"); + exit(1); + } + + /* process input arguments */ + while (1) { + q = getopt(argc, argv, "ld:n:"); + if (q == -1) + break; + switch (q) { + case 'd': + status = crypto_kernel_set_debug_module(optarg, 1); + if (status) { + printf("error: set debug module (%s) failed\n", optarg); + exit(1); + } + break; + case 'l': + do_list_mods = 1; + break; + case 'n': + num_octets = atoi(optarg); + if (num_octets < 0 || num_octets > BUF_LEN) + usage(argv[0]); + break; + default: + usage(argv[0]); + } + } + + if (do_list_mods) { + status = crypto_kernel_list_debug_modules(); + if (status) { + printf("error: list of debug modules failed\n"); + exit(1); + } + } + + if (num_octets > 0) { + uint8_t buffer[BUF_LEN]; + + status = crypto_get_random(buffer, num_octets); + if (status) { + printf("error: failure in random source\n"); + } else { + printf("%s\n", octet_string_hex_string(buffer, num_octets)); + } + } + + status = crypto_kernel_shutdown(); + if (status) { + printf("error: crypto_kernel shutdown failed\n"); + exit(1); + } + + return 0; +} + diff --git a/libs/srtp/crypto/test/sha1_driver.c b/libs/srtp/crypto/test/sha1_driver.c new file mode 100644 index 0000000000..f7cb6ca2a6 --- /dev/null +++ b/libs/srtp/crypto/test/sha1_driver.c @@ -0,0 +1,533 @@ +/* + * sha1_driver.c + * + * a test driver for SHA-1 + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2006, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include <stdio.h> +#include "sha1.h" + +#define SHA_PASS 0 +#define SHA_FAIL 1 + +#define MAX_HASH_DATA_LEN 1024 +#define MAX_HASH_OUT_LEN 20 + +typedef struct hash_test_case_t { + unsigned data_len; /* number of octets in data */ + unsigned hash_len; /* number of octets output by hash */ + uint8_t data[MAX_HASH_DATA_LEN]; /* message data */ + uint8_t hash[MAX_HASH_OUT_LEN]; /* expected hash output */ + struct hash_test_case_t *next_test_case; +} hash_test_case_t; + +hash_test_case_t *sha1_test_case_list; + +err_status_t +hash_test_case_add(hash_test_case_t **list_ptr, + char *hex_data, + unsigned data_len, + char *hex_hash, + unsigned hash_len) { + hash_test_case_t *list_head = *list_ptr; + hash_test_case_t *test_case; + unsigned tmp_len; + + test_case = malloc(sizeof(hash_test_case_t)); + if (test_case == NULL) + return err_status_alloc_fail; + + tmp_len = hex_string_to_octet_string((char *)test_case->data, hex_data, data_len*2); + if (tmp_len != data_len*2) + return err_status_parse_err; + + tmp_len = hex_string_to_octet_string((char *)test_case->hash, hex_hash, hash_len*2); + if (tmp_len != hash_len*2) + return err_status_parse_err; + + test_case->data_len = data_len; + test_case->hash_len = hash_len; + + /* add the new test case to the head of the list */ + test_case->next_test_case = list_head; + *list_ptr = test_case; + + return err_status_ok; +} + +err_status_t +sha1_test_case_validate(const hash_test_case_t *test_case) { + sha1_ctx_t ctx; + uint32_t hash_value[5]; + + if (test_case == NULL) + return err_status_bad_param; + + if (test_case->hash_len != 20) + return err_status_bad_param; + if (test_case->data_len > MAX_HASH_DATA_LEN) + return err_status_bad_param; + + sha1_init(&ctx); + sha1_update(&ctx, test_case->data, test_case->data_len); + sha1_final(&ctx, hash_value); + if (0 == memcmp(test_case->hash, hash_value, 20)) { +#if VERBOSE + printf("PASSED: reference value: %s\n", + octet_string_hex_string((uint8_t *)test_case->hash, 20)); + printf("PASSED: computed value: %s\n", + octet_string_hex_string((uint8_t *)hash_value, 20)); +#endif + return err_status_ok; + } + + printf("reference value: %s\n", + octet_string_hex_string((uint8_t *)test_case->hash, 20)); + printf("computed value: %s\n", + octet_string_hex_string((uint8_t *)hash_value, 20)); + + return err_status_algo_fail; + +} + +struct hex_sha1_test_case_t { + unsigned bit_len; + char hex_data[MAX_HASH_DATA_LEN*2]; + char hex_hash[40]; +}; + +err_status_t +sha1_add_test_cases() { + int i; + err_status_t err; + + /* + * these test cases are taken from the "SHA-1 Sample Vectors" + * provided by NIST at http://csrc.nist.gov/cryptval/shs.html + */ + + struct hex_sha1_test_case_t tc[] = { + { + 0, + "", + "da39a3ee5e6b4b0d3255bfef95601890afd80709" + }, + { + 8, + "a8", + "99f2aa95e36f95c2acb0eaf23998f030638f3f15" + }, + { + 16, + "3000", + "f944dcd635f9801f7ac90a407fbc479964dec024" + }, + { + 24, + "42749e", + "a444319e9b6cc1e8464c511ec0969c37d6bb2619" + }, + { + 32, + "9fc3fe08", + "16a0ff84fcc156fd5d3ca3a744f20a232d172253" + }, + { + 40, + "b5c1c6f1af", + "fec9deebfcdedaf66dda525e1be43597a73a1f93" + }, + { + 48, + "e47571e5022e", + "8ce051181f0ed5e9d0c498f6bc4caf448d20deb5" + }, + { + 56, + "3e1b28839fb758", + "67da53837d89e03bf652ef09c369a3415937cfd3" + }, + { + 64, + "a81350cbb224cb90", + "305e4ff9888ad855a78573cddf4c5640cce7e946" + }, + { + 72, "c243d167923dec3ce1", + "5902b77b3265f023f9bbc396ba1a93fa3509bde7" + }, + { + 80, + "50ac18c59d6a37a29bf4", + "fcade5f5d156bf6f9af97bdfa9c19bccfb4ff6ab" + }, + { + 88, + "98e2b611ad3b1cccf634f6", + "1d20fbe00533c10e3cbd6b27088a5de0c632c4b5" + }, + { + 96, + "73fe9afb68e1e8712e5d4eec", + "7e1b7e0f7a8f3455a9c03e9580fd63ae205a2d93" + }, + { + 104, + "9e701ed7d412a9226a2a130e66", + "706f0677146307b20bb0e8d6311e329966884d13" + }, + { + 112, + "6d3ee90413b0a7cbf69e5e6144ca", + "a7241a703aaf0d53fe142f86bf2e849251fa8dff" + }, + { + 120, + "fae24d56514efcb530fd4802f5e71f", + "400f53546916d33ad01a5e6df66822dfbdc4e9e6" + }, + { + 128, + "c5a22dd6eda3fe2bdc4ddb3ce6b35fd1", + "fac8ab93c1ae6c16f0311872b984f729dc928ccd" + }, + { + 136, + "d98cded2adabf08fda356445c781802d95", + "fba6d750c18da58f6e2aab10112b9a5ef3301b3b" + }, + { + 144, + "bcc6d7087a84f00103ccb32e5f5487a751a2", + "29d27c2d44c205c8107f0351b05753ac708226b6" + }, + { + 152, + "36ecacb1055434190dbbc556c48bafcb0feb0d", + "b971bfc1ebd6f359e8d74cb7ecfe7f898d0ba845" + }, + { + 160, + "5ff9edb69e8f6bbd498eb4537580b7fba7ad31d0", + "96d08c430094b9fcc164ad2fb6f72d0a24268f68" + }, + { + 168, "c95b441d8270822a46a798fae5defcf7b26abace36", + "a287ea752a593d5209e287881a09c49fa3f0beb1" + }, + { + 176, + "83104c1d8a55b28f906f1b72cb53f68cbb097b44f860", + "a06c713779cbd88519ed4a585ac0cb8a5e9d612b" + }, + { + 184, + "755175528d55c39c56493d697b790f099a5ce741f7754b", + "bff7d52c13a3688132a1d407b1ab40f5b5ace298" + }, + { + 192, + "088fc38128bbdb9fd7d65228b3184b3faac6c8715f07272f", + "c7566b91d7b6f56bdfcaa9781a7b6841aacb17e9" + }, + { + 200, + "a4a586eb9245a6c87e3adf1009ac8a49f46c07e14185016895", + "ffa30c0b5c550ea4b1e34f8a60ec9295a1e06ac1" + }, + { + 208, + "8e7c555270c006092c2a3189e2a526b873e2e269f0fb28245256", + "29e66ed23e914351e872aa761df6e4f1a07f4b81" + }, + { + 216, + "a5f3bfa6bb0ba3b59f6b9cbdef8a558ec565e8aa3121f405e7f2f0", + "b28cf5e5b806a01491d41f69bd9248765c5dc292" + }, + { + 224, + "589054f0d2bd3c2c85b466bfd8ce18e6ec3e0b87d944cd093ba36469", + "60224fb72c46069652cd78bcd08029ef64da62f3" + }, + { + 232, + "a0abb12083b5bbc78128601bf1cbdbc0fdf4b862b24d899953d8da0ff3", + "b72c4a86f72608f24c05f3b9088ef92fba431df7" + }, + { + 240, + "82143f4cea6fadbf998e128a8811dc75301cf1db4f079501ea568da68eeb", + "73779ad5d6b71b9b8328ef7220ff12eb167076ac" + }, + { + 248, + "9f1231dd6df1ff7bc0b0d4f989d048672683ce35d956d2f57913046267e6f3", + "a09671d4452d7cf50015c914a1e31973d20cc1a0" + }, + { + 256, + "041c512b5eed791f80d3282f3a28df263bb1df95e1239a7650e5670fc2187919", + "e88cdcd233d99184a6fd260b8fca1b7f7687aee0" + }, + { + 264, + "17e81f6ae8c2e5579d69dafa6e070e7111461552d314b691e7a3e7a4feb3fae418", + "010def22850deb1168d525e8c84c28116cb8a269" + }, + { + 272, + "d15976b23a1d712ad28fad04d805f572026b54dd64961fda94d5355a0cc98620cf77", + "aeaa40ba1717ed5439b1e6ea901b294ba500f9ad" + }, + { + 280, + "09fce4d434f6bd32a44e04b848ff50ec9f642a8a85b37a264dc73f130f22838443328f", + "c6433791238795e34f080a5f1f1723f065463ca0" + }, + { + 288, "f17af27d776ec82a257d8d46d2b46b639462c56984cc1be9c1222eadb8b26594a25c709d", + "e21e22b89c1bb944a32932e6b2a2f20d491982c3" + }, + { + 296, + "b13ce635d6f8758143ffb114f2f601cb20b6276951416a2f94fbf4ad081779d79f4f195b22", + "575323a9661f5d28387964d2ba6ab92c17d05a8a" + }, + { + 304, + "5498793f60916ff1c918dde572cdea76da8629ba4ead6d065de3dfb48de94d234cc1c5002910", + "feb44494af72f245bfe68e86c4d7986d57c11db7" + }, + { + 312, + "498a1e0b39fa49582ae688cd715c86fbaf8a81b8b11b4d1594c49c902d197c8ba8a621fd6e3be5", + "cff2290b3648ba2831b98dde436a72f9ebf51eee" + }, + { + 320, + "3a36ae71521f9af628b3e34dcb0d4513f84c78ee49f10416a98857150b8b15cb5c83afb4b570376e", + "9b4efe9d27b965905b0c3dab67b8d7c9ebacd56c" + }, + { + 328, + "dcc76b40ae0ea3ba253e92ac50fcde791662c5b6c948538cffc2d95e9de99cac34dfca38910db2678f", + "afedb0ff156205bcd831cbdbda43db8b0588c113" + }, + { + 336, + "5b5ec6ec4fd3ad9c4906f65c747fd4233c11a1736b6b228b92e90cddabb0c7c2fcf9716d3fad261dff33", + "8deb1e858f88293a5e5e4d521a34b2a4efa70fc4" + }, + { + 344, + "df48a37b29b1d6de4e94717d60cdb4293fcf170bba388bddf7a9035a15d433f20fd697c3e4c8b8c5f590ab", + "95cbdac0f74afa69cebd0e5c7defbc6faf0cbeaf" + }, + { + 352, + "1f179b3b82250a65e1b0aee949e218e2f45c7a8dbfd6ba08de05c55acfc226b48c68d7f7057e5675cd96fcfc", + "f0307bcb92842e5ae0cd4f4f14f3df7f877fbef2" + }, + { + 360, + "ee3d72da3a44d971578972a8e6780ce64941267e0f7d0179b214fa97855e1790e888e09fbe3a70412176cb3b54", + "7b13bb0dbf14964bd63b133ac85e22100542ef55" + }, + { + 368, + "d4d4c7843d312b30f610b3682254c8be96d5f6684503f8fbfbcd15774fc1b084d3741afb8d24aaa8ab9c104f7258", + "c314d2b6cf439be678d2a74e890d96cfac1c02ed" + }, + { + 376, + "32c094944f5936a190a0877fb9178a7bf60ceae36fd530671c5b38c5dbd5e6a6c0d615c2ac8ad04b213cc589541cf6", + "4d0be361e410b47a9d67d8ce0bb6a8e01c53c078" + }, + { + 384, + "e5d3180c14bf27a5409fa12b104a8fd7e9639609bfde6ee82bbf9648be2546d29688a65e2e3f3da47a45ac14343c9c02", + "e5353431ffae097f675cbf498869f6fbb6e1c9f2" + }, + { + 392, + "e7b6e4b69f724327e41e1188a37f4fe38b1dba19cbf5a7311d6e32f1038e97ab506ee05aebebc1eed09fc0e357109818b9", + "b8720a7068a085c018ab18961de2765aa6cd9ac4" + }, + { + 400, + "bc880cb83b8ac68ef2fedc2da95e7677ce2aa18b0e2d8b322701f67af7d5e7a0d96e9e33326ccb7747cfff0852b961bfd475", + "b0732181568543ba85f2b6da602b4b065d9931aa" + }, + { + 408, + "235ea9c2ba7af25400f2e98a47a291b0bccdaad63faa2475721fda5510cc7dad814bce8dabb611790a6abe56030b798b75c944", + "9c22674cf3222c3ba921672694aafee4ce67b96b" + }, + { + 416, + "07e3e29fed63104b8410f323b975fd9fba53f636af8c4e68a53fb202ca35dd9ee07cb169ec5186292e44c27e5696a967f5e67709", + "d128335f4cecca9066cdae08958ce656ff0b4cfc" + }, + { + 424, + "65d2a1dd60a517eb27bfbf530cf6a5458f9d5f4730058bd9814379547f34241822bf67e6335a6d8b5ed06abf8841884c636a25733f", + "0b67c57ac578de88a2ae055caeaec8bb9b0085a0" + }, + { + 432, + "dcc86b3bd461615bab739d8daafac231c0f462e819ad29f9f14058f3ab5b75941d4241ea2f17ebb8a458831b37a9b16dead4a76a9b0e", + "c766f912a89d4ccda88e0cce6a713ef5f178b596" + }, + { + 440, + "4627d54f0568dc126b62a8c35fb46a9ac5024400f2995e51635636e1afc4373dbb848eb32df23914230560b82477e9c3572647a7f2bb92", + "9aa3925a9dcb177b15ccff9b78e70cf344858779" + }, + { + 448, + "ba531affd4381168ef24d8b275a84d9254c7f5cc55fded53aa8024b2c5c5c8aa7146fe1d1b83d62b70467e9a2e2cb67b3361830adbab28d7", + "4811fa30042fc076acf37c8e2274d025307e5943" + }, + { + 456, + "8764dcbcf89dcf4282eb644e3d568bdccb4b13508bfa7bfe0ffc05efd1390be22109969262992d377691eb4f77f3d59ea8466a74abf57b2ef4", + "6743018450c9730761ee2b130df9b91c1e118150" + }, + { + 464, + "497d9df9ddb554f3d17870b1a31986c1be277bc44feff713544217a9f579623d18b5ffae306c25a45521d2759a72c0459b58957255ab592f3be4", + "71ad4a19d37d92a5e6ef3694ddbeb5aa61ada645" + }, + { + 472, + "72c3c2e065aefa8d9f7a65229e818176eef05da83f835107ba90ec2e95472e73e538f783b416c04654ba8909f26a12db6e5c4e376b7615e4a25819", + "a7d9dc68dacefb7d6116186048cb355cc548e11d" + }, + { + 480, + "7cc9894454d0055ab5069a33984e2f712bef7e3124960d33559f5f3b81906bb66fe64da13c153ca7f5cabc89667314c32c01036d12ecaf5f9a78de98", + "142e429f0522ba5abf5131fa81df82d355b96909" + }, + { + 488, + "74e8404d5a453c5f4d306f2cfa338ca65501c840ddab3fb82117933483afd6913c56aaf8a0a0a6b2a342fc3d9dc7599f4a850dfa15d06c61966d74ea59", + "ef72db70dcbcab991e9637976c6faf00d22caae9" + }, + { + 496, + "46fe5ed326c8fe376fcc92dc9e2714e2240d3253b105adfbb256ff7a19bc40975c604ad7c0071c4fd78a7cb64786e1bece548fa4833c04065fe593f6fb10", + "f220a7457f4588d639dc21407c942e9843f8e26b" + }, + { + 504, + "836dfa2524d621cf07c3d2908835de859e549d35030433c796b81272fd8bc0348e8ddbc7705a5ad1fdf2155b6bc48884ac0cd376925f069a37849c089c8645", + "ddd2117b6e309c233ede85f962a0c2fc215e5c69" + }, + { + 512, + "7e3a4c325cb9c52b88387f93d01ae86d42098f5efa7f9457388b5e74b6d28b2438d42d8b64703324d4aa25ab6aad153ae30cd2b2af4d5e5c00a8a2d0220c6116", + "a3054427cdb13f164a610b348702724c808a0dcc" + } + }; + + + for (i=0; i < 65; i++) { + err = hash_test_case_add(&sha1_test_case_list, + tc[i].hex_data, + tc[i].bit_len/8, + tc[i].hex_hash, 20); + if (err) { + printf("error adding hash test case (code %d)\n", err); + return err; + } + } + + return err_status_ok; +} + + +err_status_t +sha1_validate(void) { + hash_test_case_t *test_case; + err_status_t err; + + err = sha1_add_test_cases(); + if (err) { + printf("error adding SHA1 test cases (error code %d)\n", err); + return err; + } + + if (sha1_test_case_list == NULL) + return err_status_cant_check; + + test_case = sha1_test_case_list; + while (test_case != NULL) { + err = sha1_test_case_validate(test_case); + if (err) { + printf("error validating hash test case (error code %d)\n", err); + return err; + } + test_case = test_case->next_test_case; + } + + return err_status_ok; +} + + + +int +main (void) { + err_status_t err; + + printf("sha1 test driver\n"); + + err = sha1_validate(); + if (err) { + printf("SHA1 did not pass validation testing\n"); + return 1; + } + printf("SHA1 passed validation tests\n"); + + return 0; + +} diff --git a/libs/srtp/crypto/test/stat_driver.c b/libs/srtp/crypto/test/stat_driver.c new file mode 100644 index 0000000000..09cc44a646 --- /dev/null +++ b/libs/srtp/crypto/test/stat_driver.c @@ -0,0 +1,101 @@ +/* + * stat-driver.c + * + * test driver for the stat_test functions + * + * David A. McGrew + * Cisco Systems, Inc. + */ + + +#include <stdio.h> /* for printf() */ + +#include "err.h" +#include "stat.h" + +#include "cipher.h" + +typedef struct { + void *state; +} random_source_t; + +err_status_t +random_source_alloc(void); + +void +err_check(err_status_t s) { + if (s) { + printf("error (code %d)\n", s); + exit(1); + } +} + +int +main (int argc, char *argv[]) { + uint8_t buffer[2500]; + unsigned int buf_len = 2500; + int i, j; + extern cipher_type_t aes_icm; + cipher_t *c; + uint8_t key[30] = { + 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, + 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, + 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, + 0x00, 0x01, 0x02, 0x03, 0x04, 0x05 + }; + v128_t nonce; + int num_trials = 500; + int num_fail; + + printf("statistical tests driver\n"); + + for (i=0; i < 2500; i++) + buffer[i] = 0; + + /* run tests */ + printf("running stat_tests on all-null buffer, expecting failure\n"); + printf("monobit %d\n", stat_test_monobit(buffer)); + printf("poker %d\n", stat_test_poker(buffer)); + printf("runs %d\n", stat_test_runs(buffer)); + + for (i=0; i < 2500; i++) + buffer[i] = rand(); + printf("running stat_tests on rand(), expecting success\n"); + printf("monobit %d\n", stat_test_monobit(buffer)); + printf("poker %d\n", stat_test_poker(buffer)); + printf("runs %d\n", stat_test_runs(buffer)); + + printf("running stat_tests on AES-128-ICM, expecting success\n"); + /* set buffer to cipher output */ + for (i=0; i < 2500; i++) + buffer[i] = 0; + err_check(cipher_type_alloc(&aes_icm, &c, 30)); + err_check(cipher_init(c, key, direction_encrypt)); + err_check(cipher_set_iv(c, &nonce)); + err_check(cipher_encrypt(c, buffer, &buf_len)); + /* run tests on cipher outout */ + printf("monobit %d\n", stat_test_monobit(buffer)); + printf("poker %d\n", stat_test_poker(buffer)); + printf("runs %d\n", stat_test_runs(buffer)); + + printf("runs test (please be patient): "); + fflush(stdout); + num_fail = 0; + v128_set_to_zero(&nonce); + for(j=0; j < num_trials; j++) { + for (i=0; i < 2500; i++) + buffer[i] = 0; + nonce.v32[3] = i; + err_check(cipher_set_iv(c, &nonce)); + err_check(cipher_encrypt(c, buffer, &buf_len)); + if (stat_test_runs(buffer)) { + num_fail++; + } + } + + printf("%d failures in %d tests\n", num_fail, num_trials); + printf("(nota bene: a small fraction of stat_test failures does not \n" + "indicate that the random source is invalid)\n"); + + return 0; +} diff --git a/libs/srtp/doc/.cvsignore b/libs/srtp/doc/.cvsignore new file mode 100644 index 0000000000..55a080aed7 --- /dev/null +++ b/libs/srtp/doc/.cvsignore @@ -0,0 +1,3 @@ +Makefile +header.tex +latex diff --git a/libs/srtp/doc/CVS/Entries b/libs/srtp/doc/CVS/Entries new file mode 100644 index 0000000000..2b1a9cdd49 --- /dev/null +++ b/libs/srtp/doc/CVS/Entries @@ -0,0 +1,11 @@ +/.cvsignore/1.2/Thu Sep 29 14:31:57 2005// +/Doxyfile/1.2/Tue Sep 27 15:38:26 2005// +/Makefile.in/1.2/Thu Sep 29 14:27:35 2005// +/crypto_kernel.txt/1.2/Thu Sep 29 12:36:43 2005// +/draft-irtf-cfrg-icm-00.txt/1.1.1.1/Wed Sep 21 22:51:41 2005// +/header.template/1.1/Tue Sep 27 15:38:26 2005// +/intro.txt/1.4/Fri Sep 30 17:30:26 2005// +/libsrtp.pdf/1.5/Thu Jan 12 18:30:06 2006// +/references.txt/1.1.1.1/Wed Sep 21 22:51:44 2005// +/rfc3711.txt/1.1.1.1/Wed Sep 21 22:51:45 2005// +D diff --git a/libs/srtp/doc/CVS/Repository b/libs/srtp/doc/CVS/Repository new file mode 100644 index 0000000000..f8aaa517a6 --- /dev/null +++ b/libs/srtp/doc/CVS/Repository @@ -0,0 +1 @@ +srtp/doc diff --git a/libs/srtp/doc/CVS/Root b/libs/srtp/doc/CVS/Root new file mode 100644 index 0000000000..b4b74237d9 --- /dev/null +++ b/libs/srtp/doc/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@cvs.sourceforge.net:/cvsroot/srtp diff --git a/libs/srtp/doc/Doxyfile b/libs/srtp/doc/Doxyfile new file mode 100644 index 0000000000..fc6ac95e11 --- /dev/null +++ b/libs/srtp/doc/Doxyfile @@ -0,0 +1,1042 @@ +# Doxyfile 1.3-rc3 + +# This file describes the settings to be used by the documentation system +# doxygen (www.doxygen.org) for a project +# +# All text after a hash (#) is considered a comment and will be ignored +# The format is: +# TAG = value [value, ...] +# For lists items can also be appended using: +# TAG += value [value, ...] +# Values that contain spaces should be placed between quotes (" ") + +#--------------------------------------------------------------------------- +# General configuration options +#--------------------------------------------------------------------------- + +# The PROJECT_NAME tag is a single word (or a sequence of words surrounded +# by quotes) that should identify the project. + +PROJECT_NAME = libSRTP + +# The PROJECT_NUMBER tag can be used to enter a project or revision number. +# This could be handy for archiving the generated documentation or +# if some version control system is used. + +PROJECT_NUMBER = 1.3.22 + +# The OUTPUT_DIRECTORY tag is used to specify the (relative or absolute) +# base path where the generated documentation will be put. +# If a relative path is entered, it will be relative to the location +# where doxygen was started. If left blank the current directory will be used. + +OUTPUT_DIRECTORY = + +# The OUTPUT_LANGUAGE tag is used to specify the language in which all +# documentation generated by doxygen is written. Doxygen will use this +# information to generate all constant output in the proper language. +# The default language is English, other supported languages are: +# Brazilian, Catalan, Chinese, Chinese-Traditional, Croatian, Czech, Danish, Dutch, +# Finnish, French, German, Greek, Hungarian, Italian, Japanese, Japanese-en +# (Japanese with english messages), Korean, Norwegian, Polish, Portuguese, +# Romanian, Russian, Serbian, Slovak, Slovene, Spanish, Swedish and Ukrainian. + +OUTPUT_LANGUAGE = English + +# If the EXTRACT_ALL tag is set to YES doxygen will assume all entities in +# documentation are documented, even if no documentation was available. +# Private class members and static file members will be hidden unless +# the EXTRACT_PRIVATE and EXTRACT_STATIC tags are set to YES + +EXTRACT_ALL = NO + +# If the EXTRACT_PRIVATE tag is set to YES all private members of a class +# will be included in the documentation. + +EXTRACT_PRIVATE = NO + +# If the EXTRACT_STATIC tag is set to YES all static members of a file +# will be included in the documentation. + +EXTRACT_STATIC = NO + +# If the EXTRACT_LOCAL_CLASSES tag is set to YES classes (and structs) +# defined locally in source files will be included in the documentation. +# If set to NO only classes defined in header files are included. + +EXTRACT_LOCAL_CLASSES = YES + +# If the HIDE_UNDOC_MEMBERS tag is set to YES, Doxygen will hide all +# undocumented members of documented classes, files or namespaces. +# If set to NO (the default) these members will be included in the +# various overviews, but no documentation section is generated. +# This option has no effect if EXTRACT_ALL is enabled. + +HIDE_UNDOC_MEMBERS = YES + +# If the HIDE_UNDOC_CLASSES tag is set to YES, Doxygen will hide all +# undocumented classes that are normally visible in the class hierarchy. +# If set to NO (the default) these class will be included in the various +# overviews. This option has no effect if EXTRACT_ALL is enabled. + +HIDE_UNDOC_CLASSES = NO + +# If the HIDE_FRIEND_COMPOUNDS tag is set to YES, Doxygen will hide all +# friend (class|struct|union) declarations. +# If set to NO (the default) these declarations will be included in the +# documentation. + +HIDE_FRIEND_COMPOUNDS = NO + +# If the HIDE_IN_BODY_DOCS tag is set to YES, Doxygen will hide any +# documentation blocks found inside the body of a function. +# If set to NO (the default) these blocks will be appended to the +# function's detailed documentation block. + +HIDE_IN_BODY_DOCS = NO + +# If the BRIEF_MEMBER_DESC tag is set to YES (the default) Doxygen will +# include brief member descriptions after the members that are listed in +# the file and class documentation (similar to JavaDoc). +# Set to NO to disable this. + +BRIEF_MEMBER_DESC = YES + +# If the REPEAT_BRIEF tag is set to YES (the default) Doxygen will prepend +# the brief description of a member or function before the detailed description. +# Note: if both HIDE_UNDOC_MEMBERS and BRIEF_MEMBER_DESC are set to NO, the +# brief descriptions will be completely suppressed. + +REPEAT_BRIEF = NO + +# If the ALWAYS_DETAILED_SEC and REPEAT_BRIEF tags are both set to YES then +# Doxygen will generate a detailed section even if there is only a brief +# description. + +ALWAYS_DETAILED_SEC = NO + +# If the INLINE_INHERITED_MEMB tag is set to YES, doxygen will show +# all inherited members of a class in the documentation of that class +# as if those members were ordinary class members. Constructors, +# destructors and assignment operators of the base classes will not be +# shown. + +INLINE_INHERITED_MEMB = NO + +# If the FULL_PATH_NAMES tag is set to YES then Doxygen will prepend the full +# path before files name in the file list and in the header files. If set +# to NO the shortest path that makes the file name unique will be used. + +FULL_PATH_NAMES = NO + +# If the FULL_PATH_NAMES tag is set to YES then the STRIP_FROM_PATH tag +# can be used to strip a user defined part of the path. Stripping is +# only done if one of the specified strings matches the left-hand part of +# the path. It is allowed to use relative paths in the argument list. + +STRIP_FROM_PATH = + +# The INTERNAL_DOCS tag determines if documentation +# that is typed after a \internal command is included. If the tag is set +# to NO (the default) then the documentation will be excluded. +# Set it to YES to include the internal documentation. + +INTERNAL_DOCS = NO + +# If the CASE_SENSE_NAMES tag is set to NO then Doxygen will only generate +# file names in lower case letters. If set to YES upper case letters are also +# allowed. This is useful if you have classes or files whose names only differ +# in case and if your file system supports case sensitive file names. Windows +# users are adviced to set this option to NO. + +CASE_SENSE_NAMES = YES + +# If the SHORT_NAMES tag is set to YES, doxygen will generate much shorter +# (but less readable) file names. This can be useful is your file systems +# doesn't support long names like on DOS, Mac, or CD-ROM. + +SHORT_NAMES = NO + +# If the HIDE_SCOPE_NAMES tag is set to NO (the default) then Doxygen +# will show members with their full class and namespace scopes in the +# documentation. If set to YES the scope will be hidden. + +HIDE_SCOPE_NAMES = NO + +# If the VERBATIM_HEADERS tag is set to YES (the default) then Doxygen +# will generate a verbatim copy of the header file for each class for +# which an include is specified. Set to NO to disable this. + +VERBATIM_HEADERS = YES + +# If the SHOW_INCLUDE_FILES tag is set to YES (the default) then Doxygen +# will put list of the files that are included by a file in the documentation +# of that file. + +SHOW_INCLUDE_FILES = NO + +# If the JAVADOC_AUTOBRIEF tag is set to YES then Doxygen +# will interpret the first line (until the first dot) of a JavaDoc-style +# comment as the brief description. If set to NO, the JavaDoc +# comments will behave just like the Qt-style comments (thus requiring an +# explict @brief command for a brief description. + +JAVADOC_AUTOBRIEF = NO + +# The MULTILINE_CPP_IS_BRIEF tag can be set to YES to make Doxygen +# treat a multi-line C++ special comment block (i.e. a block of //! or /// +# comments) as a brief description. This used to be the default behaviour. +# The new default is to treat a multi-line C++ comment block as a detailed +# description. Set this tag to YES if you prefer the old behaviour instead. + +MULTILINE_CPP_IS_BRIEF = NO + +# If the DETAILS_AT_TOP tag is set to YES then Doxygen +# will output the detailed description near the top, like JavaDoc. +# If set to NO, the detailed description appears after the member +# documentation. + +DETAILS_AT_TOP = NO + +# If the INHERIT_DOCS tag is set to YES (the default) then an undocumented +# member inherits the documentation from any documented member that it +# reimplements. + +INHERIT_DOCS = YES + +# If the INLINE_INFO tag is set to YES (the default) then a tag [inline] +# is inserted in the documentation for inline members. + +INLINE_INFO = YES + +# If the SORT_MEMBER_DOCS tag is set to YES (the default) then doxygen +# will sort the (detailed) documentation of file and class members +# alphabetically by member name. If set to NO the members will appear in +# declaration order. + +SORT_MEMBER_DOCS = YES + +# If member grouping is used in the documentation and the DISTRIBUTE_GROUP_DOC +# tag is set to YES, then doxygen will reuse the documentation of the first +# member in the group (if any) for the other members of the group. By default +# all members of a group must be documented explicitly. + +DISTRIBUTE_GROUP_DOC = NO + +# The TAB_SIZE tag can be used to set the number of spaces in a tab. +# Doxygen uses this value to replace tabs by spaces in code fragments. + +TAB_SIZE = 3 + +# The GENERATE_TODOLIST tag can be used to enable (YES) or +# disable (NO) the todo list. This list is created by putting \todo +# commands in the documentation. + +GENERATE_TODOLIST = YES + +# The GENERATE_TESTLIST tag can be used to enable (YES) or +# disable (NO) the test list. This list is created by putting \test +# commands in the documentation. + +GENERATE_TESTLIST = YES + +# The GENERATE_BUGLIST tag can be used to enable (YES) or +# disable (NO) the bug list. This list is created by putting \bug +# commands in the documentation. + +GENERATE_BUGLIST = YES + +# The GENERATE_DEPRECATEDLIST tag can be used to enable (YES) or +# disable (NO) the deprecated list. This list is created by putting +# \deprecated commands in the documentation. + +GENERATE_DEPRECATEDLIST= YES + +# This tag can be used to specify a number of aliases that acts +# as commands in the documentation. An alias has the form "name=value". +# For example adding "sideeffect=\par Side Effects:\n" will allow you to +# put the command \sideeffect (or @sideeffect) in the documentation, which +# will result in a user defined paragraph with heading "Side Effects:". +# You can put \n's in the value part of an alias to insert newlines. + +ALIASES = + +# The ENABLED_SECTIONS tag can be used to enable conditional +# documentation sections, marked by \if sectionname ... \endif. + +ENABLED_SECTIONS = + +# The MAX_INITIALIZER_LINES tag determines the maximum number of lines +# the initial value of a variable or define consist of for it to appear in +# the documentation. If the initializer consists of more lines than specified +# here it will be hidden. Use a value of 0 to hide initializers completely. +# The appearance of the initializer of individual variables and defines in the +# documentation can be controlled using \showinitializer or \hideinitializer +# command in the documentation regardless of this setting. + +MAX_INITIALIZER_LINES = 30 + +# Set the OPTIMIZE_OUTPUT_FOR_C tag to YES if your project consists of C sources +# only. Doxygen will then generate output that is more tailored for C. +# For instance some of the names that are used will be different. The list +# of all members will be omitted, etc. + +OPTIMIZE_OUTPUT_FOR_C = YES + +# Set the OPTIMIZE_OUTPUT_JAVA tag to YES if your project consists of Java sources +# only. Doxygen will then generate output that is more tailored for Java. +# For instance namespaces will be presented as packages, qualified scopes +# will look different, etc. + +OPTIMIZE_OUTPUT_JAVA = NO + +# Set the SHOW_USED_FILES tag to NO to disable the list of files generated +# at the bottom of the documentation of classes and structs. If set to YES the +# list will mention the files that were used to generate the documentation. + +SHOW_USED_FILES = YES + +#--------------------------------------------------------------------------- +# configuration options related to warning and progress messages +#--------------------------------------------------------------------------- + +# The QUIET tag can be used to turn on/off the messages that are generated +# by doxygen. Possible values are YES and NO. If left blank NO is used. + +QUIET = NO + +# The WARNINGS tag can be used to turn on/off the warning messages that are +# generated by doxygen. Possible values are YES and NO. If left blank +# NO is used. + +WARNINGS = YES + +# If WARN_IF_UNDOCUMENTED is set to YES, then doxygen will generate warnings +# for undocumented members. If EXTRACT_ALL is set to YES then this flag will +# automatically be disabled. + +WARN_IF_UNDOCUMENTED = YES + +# If WARN_IF_DOC_ERROR is set to YES, doxygen will generate warnings for +# potential errors in the documentation, such as not documenting some +# parameters in a documented function, or documenting parameters that +# don't exist or using markup commands wrongly. + +WARN_IF_DOC_ERROR = YES + +# The WARN_FORMAT tag determines the format of the warning messages that +# doxygen can produce. The string should contain the $file, $line, and $text +# tags, which will be replaced by the file and line number from which the +# warning originated and the warning text. + +WARN_FORMAT = "$file:$line: $text" + +# The WARN_LOGFILE tag can be used to specify a file to which warning +# and error messages should be written. If left blank the output is written +# to stderr. + +WARN_LOGFILE = + +#--------------------------------------------------------------------------- +# configuration options related to the input files +#--------------------------------------------------------------------------- + +# The INPUT tag can be used to specify the files and/or directories that contain +# documented source files. You may enter file names like "myfile.cpp" or +# directories like "/usr/src/myproject". Separate the files or directories +# with spaces. + +INPUT = intro.txt ../include/srtp.h ../crypto/include/crypto_types.h ../crypto/include/err.h ../crypto/include/crypto_kernel.h crypto_kernel.txt + +# If the value of the INPUT tag contains directories, you can use the +# FILE_PATTERNS tag to specify one or more wildcard pattern (like *.cpp +# and *.h) to filter out the source-files in the directories. If left +# blank the following patterns are tested: +# *.c *.cc *.cxx *.cpp *.c++ *.java *.ii *.ixx *.ipp *.i++ *.inl *.h *.hh *.hxx *.hpp +# *.h++ *.idl *.odl + +FILE_PATTERNS = + +# The RECURSIVE tag can be used to turn specify whether or not subdirectories +# should be searched for input files as well. Possible values are YES and NO. +# If left blank NO is used. + +RECURSIVE = NO + +# The EXCLUDE tag can be used to specify files and/or directories that should +# excluded from the INPUT source files. This way you can easily exclude a +# subdirectory from a directory tree whose root is specified with the INPUT tag. + +EXCLUDE = + +# The EXCLUDE_SYMLINKS tag can be used select whether or not files or directories +# that are symbolic links (a Unix filesystem feature) are excluded from the input. + +EXCLUDE_SYMLINKS = NO + +# If the value of the INPUT tag contains directories, you can use the +# EXCLUDE_PATTERNS tag to specify one or more wildcard patterns to exclude +# certain files from those directories. + +EXCLUDE_PATTERNS = + +# The EXAMPLE_PATH tag can be used to specify one or more files or +# directories that contain example code fragments that are included (see +# the \include command). + +EXAMPLE_PATH = + +# If the value of the EXAMPLE_PATH tag contains directories, you can use the +# EXAMPLE_PATTERNS tag to specify one or more wildcard pattern (like *.cpp +# and *.h) to filter out the source-files in the directories. If left +# blank all files are included. + +EXAMPLE_PATTERNS = + +# If the EXAMPLE_RECURSIVE tag is set to YES then subdirectories will be +# searched for input files to be used with the \include or \dontinclude +# commands irrespective of the value of the RECURSIVE tag. +# Possible values are YES and NO. If left blank NO is used. + +EXAMPLE_RECURSIVE = NO + +# The IMAGE_PATH tag can be used to specify one or more files or +# directories that contain image that are included in the documentation (see +# the \image command). + +IMAGE_PATH = + +# The INPUT_FILTER tag can be used to specify a program that doxygen should +# invoke to filter for each input file. Doxygen will invoke the filter program +# by executing (via popen()) the command <filter> <input-file>, where <filter> +# is the value of the INPUT_FILTER tag, and <input-file> is the name of an +# input file. Doxygen will then use the output that the filter program writes +# to standard output. + +INPUT_FILTER = + +# If the FILTER_SOURCE_FILES tag is set to YES, the input filter (if set using +# INPUT_FILTER) will be used to filter the input files when producing source +# files to browse (i.e. when SOURCE_BROWSER is set to YES). + +FILTER_SOURCE_FILES = NO + +#--------------------------------------------------------------------------- +# configuration options related to source browsing +#--------------------------------------------------------------------------- + +# If the SOURCE_BROWSER tag is set to YES then a list of source files will +# be generated. Documented entities will be cross-referenced with these sources. + +SOURCE_BROWSER = NO + +# Setting the INLINE_SOURCES tag to YES will include the body +# of functions and classes directly in the documentation. + +INLINE_SOURCES = NO + +# Setting the STRIP_CODE_COMMENTS tag to YES (the default) will instruct +# doxygen to hide any special comment blocks from generated source code +# fragments. Normal C and C++ comments will always remain visible. + +STRIP_CODE_COMMENTS = YES + +# If the REFERENCED_BY_RELATION tag is set to YES (the default) +# then for each documented function all documented +# functions referencing it will be listed. + +REFERENCED_BY_RELATION = YES + +# If the REFERENCES_RELATION tag is set to YES (the default) +# then for each documented function all documented entities +# called/used by that function will be listed. + +REFERENCES_RELATION = YES + +#--------------------------------------------------------------------------- +# configuration options related to the alphabetical class index +#--------------------------------------------------------------------------- + +# If the ALPHABETICAL_INDEX tag is set to YES, an alphabetical index +# of all compounds will be generated. Enable this if the project +# contains a lot of classes, structs, unions or interfaces. + +ALPHABETICAL_INDEX = NO + +# If the alphabetical index is enabled (see ALPHABETICAL_INDEX) then +# the COLS_IN_ALPHA_INDEX tag can be used to specify the number of columns +# in which this list will be split (can be a number in the range [1..20]) + +COLS_IN_ALPHA_INDEX = 5 + +# In case all classes in a project start with a common prefix, all +# classes will be put under the same header in the alphabetical index. +# The IGNORE_PREFIX tag can be used to specify one or more prefixes that +# should be ignored while generating the index headers. + +IGNORE_PREFIX = + +#--------------------------------------------------------------------------- +# configuration options related to the HTML output +#--------------------------------------------------------------------------- + +# If the GENERATE_HTML tag is set to YES (the default) Doxygen will +# generate HTML output. + +GENERATE_HTML = NO + +# The HTML_OUTPUT tag is used to specify where the HTML docs will be put. +# If a relative path is entered the value of OUTPUT_DIRECTORY will be +# put in front of it. If left blank `html' will be used as the default path. + +HTML_OUTPUT = html + +# The HTML_FILE_EXTENSION tag can be used to specify the file extension for +# each generated HTML page (for example: .htm,.php,.asp). If it is left blank +# doxygen will generate files with .html extension. + +HTML_FILE_EXTENSION = .html + +# The HTML_HEADER tag can be used to specify a personal HTML header for +# each generated HTML page. If it is left blank doxygen will generate a +# standard header. + +HTML_HEADER = + +# The HTML_FOOTER tag can be used to specify a personal HTML footer for +# each generated HTML page. If it is left blank doxygen will generate a +# standard footer. + +HTML_FOOTER = + +# The HTML_STYLESHEET tag can be used to specify a user defined cascading +# style sheet that is used by each HTML page. It can be used to +# fine-tune the look of the HTML output. If the tag is left blank doxygen +# will generate a default style sheet + +HTML_STYLESHEET = + +# If the HTML_ALIGN_MEMBERS tag is set to YES, the members of classes, +# files or namespaces will be aligned in HTML using tables. If set to +# NO a bullet list will be used. + +HTML_ALIGN_MEMBERS = YES + +# If the GENERATE_HTMLHELP tag is set to YES, additional index files +# will be generated that can be used as input for tools like the +# Microsoft HTML help workshop to generate a compressed HTML help file (.chm) +# of the generated HTML documentation. + +GENERATE_HTMLHELP = NO + +# If the GENERATE_HTMLHELP tag is set to YES, the CHM_FILE tag can +# be used to specify the file name of the resulting .chm file. You +# can add a path in front of the file if the result should not be +# written to the html output dir. + +CHM_FILE = + +# If the GENERATE_HTMLHELP tag is set to YES, the HHC_LOCATION tag can +# be used to specify the location (absolute path including file name) of +# the HTML help compiler (hhc.exe). If non empty doxygen will try to run +# the html help compiler on the generated index.hhp. + +HHC_LOCATION = + +# If the GENERATE_HTMLHELP tag is set to YES, the GENERATE_CHI flag +# controls if a separate .chi index file is generated (YES) or that +# it should be included in the master .chm file (NO). + +GENERATE_CHI = NO + +# If the GENERATE_HTMLHELP tag is set to YES, the BINARY_TOC flag +# controls whether a binary table of contents is generated (YES) or a +# normal table of contents (NO) in the .chm file. + +BINARY_TOC = NO + +# The TOC_EXPAND flag can be set to YES to add extra items for group members +# to the contents of the Html help documentation and to the tree view. + +TOC_EXPAND = NO + +# The DISABLE_INDEX tag can be used to turn on/off the condensed index at +# top of each HTML page. The value NO (the default) enables the index and +# the value YES disables it. + +DISABLE_INDEX = NO + +# This tag can be used to set the number of enum values (range [1..20]) +# that doxygen will group on one line in the generated HTML documentation. + +ENUM_VALUES_PER_LINE = 4 + +# If the GENERATE_TREEVIEW tag is set to YES, a side panel will be +# generated containing a tree-like index structure (just like the one that +# is generated for HTML Help). For this to work a browser that supports +# JavaScript, DHTML, CSS and frames is required (for instance Mozilla, +# Netscape 6.0+, Internet explorer 5.0+, or Konqueror). Windows users are +# probably better off using the HTML help feature. + +GENERATE_TREEVIEW = NO + +# If the treeview is enabled (see GENERATE_TREEVIEW) then this tag can be +# used to set the initial width (in pixels) of the frame in which the tree +# is shown. + +TREEVIEW_WIDTH = 250 + +#--------------------------------------------------------------------------- +# configuration options related to the LaTeX output +#--------------------------------------------------------------------------- + +# If the GENERATE_LATEX tag is set to YES (the default) Doxygen will +# generate Latex output. + +GENERATE_LATEX = YES + +# The LATEX_OUTPUT tag is used to specify where the LaTeX docs will be put. +# If a relative path is entered the value of OUTPUT_DIRECTORY will be +# put in front of it. If left blank `latex' will be used as the default path. + +LATEX_OUTPUT = latex + +# The LATEX_CMD_NAME tag can be used to specify the LaTeX command name to be +# invoked. If left blank `latex' will be used as the default command name. + +LATEX_CMD_NAME = latex + +# The MAKEINDEX_CMD_NAME tag can be used to specify the command name to +# generate index for LaTeX. If left blank `makeindex' will be used as the +# default command name. + +MAKEINDEX_CMD_NAME = makeindex + +# If the COMPACT_LATEX tag is set to YES Doxygen generates more compact +# LaTeX documents. This may be useful for small projects and may help to +# save some trees in general. + +COMPACT_LATEX = NO + +# The PAPER_TYPE tag can be used to set the paper type that is used +# by the printer. Possible values are: a4, a4wide, letter, legal and +# executive. If left blank a4wide will be used. + +PAPER_TYPE = letter + +# The EXTRA_PACKAGES tag can be to specify one or more names of LaTeX +# packages that should be included in the LaTeX output. + +EXTRA_PACKAGES = + +# The LATEX_HEADER tag can be used to specify a personal LaTeX header for +# the generated latex document. The header should contain everything until +# the first chapter. If it is left blank doxygen will generate a +# standard header. Notice: only use this tag if you know what you are doing! + +LATEX_HEADER = header.tex + +# If the PDF_HYPERLINKS tag is set to YES, the LaTeX that is generated +# is prepared for conversion to pdf (using ps2pdf). The pdf file will +# contain links (just like the HTML output) instead of page references +# This makes the output suitable for online browsing using a pdf viewer. + +PDF_HYPERLINKS = YES + +# If the USE_PDFLATEX tag is set to YES, pdflatex will be used instead of +# plain latex in the generated Makefile. Set this option to YES to get a +# higher quality PDF documentation. + +USE_PDFLATEX = YES + +# If the LATEX_BATCHMODE tag is set to YES, doxygen will add the \\batchmode. +# command to the generated LaTeX files. This will instruct LaTeX to keep +# running if errors occur, instead of asking the user for help. +# This option is also used when generating formulas in HTML. + +LATEX_BATCHMODE = NO + +#--------------------------------------------------------------------------- +# configuration options related to the RTF output +#--------------------------------------------------------------------------- + +# If the GENERATE_RTF tag is set to YES Doxygen will generate RTF output +# The RTF output is optimised for Word 97 and may not look very pretty with +# other RTF readers or editors. + +GENERATE_RTF = NO + +# The RTF_OUTPUT tag is used to specify where the RTF docs will be put. +# If a relative path is entered the value of OUTPUT_DIRECTORY will be +# put in front of it. If left blank `rtf' will be used as the default path. + +RTF_OUTPUT = rtf + +# If the COMPACT_RTF tag is set to YES Doxygen generates more compact +# RTF documents. This may be useful for small projects and may help to +# save some trees in general. + +COMPACT_RTF = NO + +# If the RTF_HYPERLINKS tag is set to YES, the RTF that is generated +# will contain hyperlink fields. The RTF file will +# contain links (just like the HTML output) instead of page references. +# This makes the output suitable for online browsing using WORD or other +# programs which support those fields. +# Note: wordpad (write) and others do not support links. + +RTF_HYPERLINKS = NO + +# Load stylesheet definitions from file. Syntax is similar to doxygen's +# config file, i.e. a series of assigments. You only have to provide +# replacements, missing definitions are set to their default value. + +RTF_STYLESHEET_FILE = + +# Set optional variables used in the generation of an rtf document. +# Syntax is similar to doxygen's config file. + +RTF_EXTENSIONS_FILE = + +#--------------------------------------------------------------------------- +# configuration options related to the man page output +#--------------------------------------------------------------------------- + +# If the GENERATE_MAN tag is set to YES (the default) Doxygen will +# generate man pages + +GENERATE_MAN = NO + +# The MAN_OUTPUT tag is used to specify where the man pages will be put. +# If a relative path is entered the value of OUTPUT_DIRECTORY will be +# put in front of it. If left blank `man' will be used as the default path. + +MAN_OUTPUT = man + +# The MAN_EXTENSION tag determines the extension that is added to +# the generated man pages (default is the subroutine's section .3) + +MAN_EXTENSION = .3 + +# If the MAN_LINKS tag is set to YES and Doxygen generates man output, +# then it will generate one additional man file for each entity +# documented in the real man page(s). These additional files +# only source the real man page, but without them the man command +# would be unable to find the correct page. The default is NO. + +MAN_LINKS = NO + +#--------------------------------------------------------------------------- +# configuration options related to the XML output +#--------------------------------------------------------------------------- + +# If the GENERATE_XML tag is set to YES Doxygen will +# generate an XML file that captures the structure of +# the code including all documentation. Note that this +# feature is still experimental and incomplete at the +# moment. + +GENERATE_XML = NO + +# The XML_SCHEMA tag can be used to specify an XML schema, +# which can be used by a validating XML parser to check the +# syntax of the XML files. + +XML_SCHEMA = + +# The XML_DTD tag can be used to specify an XML DTD, +# which can be used by a validating XML parser to check the +# syntax of the XML files. + +XML_DTD = + +#--------------------------------------------------------------------------- +# configuration options for the AutoGen Definitions output +#--------------------------------------------------------------------------- + +# If the GENERATE_AUTOGEN_DEF tag is set to YES Doxygen will +# generate an AutoGen Definitions (see autogen.sf.net) file +# that captures the structure of the code including all +# documentation. Note that this feature is still experimental +# and incomplete at the moment. + +GENERATE_AUTOGEN_DEF = NO + +#--------------------------------------------------------------------------- +# configuration options related to the Perl module output +#--------------------------------------------------------------------------- + +# If the GENERATE_PERLMOD tag is set to YES Doxygen will +# generate a Perl module file that captures the structure of +# the code including all documentation. Note that this +# feature is still experimental and incomplete at the +# moment. + +GENERATE_PERLMOD = NO + +# If the PERLMOD_LATEX tag is set to YES Doxygen will generate +# the necessary Makefile rules, Perl scripts and LaTeX code to be able +# to generate PDF and DVI output from the Perl module output. + +PERLMOD_LATEX = NO + +# If the PERLMOD_PRETTY tag is set to YES the Perl module output will be +# nicely formatted so it can be parsed by a human reader. This is useful +# if you want to understand what is going on. On the other hand, if this +# tag is set to NO the size of the Perl module output will be much smaller +# and Perl will parse it just the same. + +PERLMOD_PRETTY = YES + +# The names of the make variables in the generated doxyrules.make file +# are prefixed with the string contained in PERLMOD_MAKEVAR_PREFIX. +# This is useful so different doxyrules.make files included by the same +# Makefile don't overwrite each other's variables. + +PERLMOD_MAKEVAR_PREFIX = + +#--------------------------------------------------------------------------- +# Configuration options related to the preprocessor +#--------------------------------------------------------------------------- + +# If the ENABLE_PREPROCESSING tag is set to YES (the default) Doxygen will +# evaluate all C-preprocessor directives found in the sources and include +# files. + +ENABLE_PREPROCESSING = YES + +# If the MACRO_EXPANSION tag is set to YES Doxygen will expand all macro +# names in the source code. If set to NO (the default) only conditional +# compilation will be performed. Macro expansion can be done in a controlled +# way by setting EXPAND_ONLY_PREDEF to YES. + +MACRO_EXPANSION = NO + +# If the EXPAND_ONLY_PREDEF and MACRO_EXPANSION tags are both set to YES +# then the macro expansion is limited to the macros specified with the +# PREDEFINED and EXPAND_AS_PREDEFINED tags. + +EXPAND_ONLY_PREDEF = NO + +# If the SEARCH_INCLUDES tag is set to YES (the default) the includes files +# in the INCLUDE_PATH (see below) will be search if a #include is found. + +SEARCH_INCLUDES = YES + +# The INCLUDE_PATH tag can be used to specify one or more directories that +# contain include files that are not input files but should be processed by +# the preprocessor. + +INCLUDE_PATH = + +# You can use the INCLUDE_FILE_PATTERNS tag to specify one or more wildcard +# patterns (like *.h and *.hpp) to filter out the header-files in the +# directories. If left blank, the patterns specified with FILE_PATTERNS will +# be used. + +INCLUDE_FILE_PATTERNS = + +# The PREDEFINED tag can be used to specify one or more macro names that +# are defined before the preprocessor is started (similar to the -D option of +# gcc). The argument of the tag is a list of macros of the form: name +# or name=definition (no spaces). If the definition and the = are +# omitted =1 is assumed. + +PREDEFINED = + +# If the MACRO_EXPANSION and EXPAND_ONLY_PREDEF tags are set to YES then +# this tag can be used to specify a list of macro names that should be expanded. +# The macro definition that is found in the sources will be used. +# Use the PREDEFINED tag if you want to use a different macro definition. + +EXPAND_AS_DEFINED = + +# If the SKIP_FUNCTION_MACROS tag is set to YES (the default) then +# doxygen's preprocessor will remove all function-like macros that are alone +# on a line, have an all uppercase name, and do not end with a semicolon. Such +# function macros are typically used for boiler-plate code, and will confuse the +# parser if not removed. + +SKIP_FUNCTION_MACROS = YES + +#--------------------------------------------------------------------------- +# Configuration::addtions related to external references +#--------------------------------------------------------------------------- + +# The TAGFILES tag can be used to specify one or more tagfiles. + +TAGFILES = + +# When a file name is specified after GENERATE_TAGFILE, doxygen will create +# a tag file that is based on the input files it reads. + +GENERATE_TAGFILE = + +# If the ALLEXTERNALS tag is set to YES all external classes will be listed +# in the class index. If set to NO only the inherited external classes +# will be listed. + +ALLEXTERNALS = NO + +# If the EXTERNAL_GROUPS tag is set to YES all external groups will be listed +# in the modules index. If set to NO, only the current project's groups will +# be listed. + +EXTERNAL_GROUPS = YES + +# The PERL_PATH should be the absolute path and name of the perl script +# interpreter (i.e. the result of `which perl'). + +PERL_PATH = /usr/bin/perl + +#--------------------------------------------------------------------------- +# Configuration options related to the dot tool +#--------------------------------------------------------------------------- + +# If the CLASS_DIAGRAMS tag is set to YES (the default) Doxygen will +# generate a inheritance diagram (in Html, RTF and LaTeX) for classes with base or +# super classes. Setting the tag to NO turns the diagrams off. Note that this +# option is superceded by the HAVE_DOT option below. This is only a fallback. It is +# recommended to install and use dot, since it yield more powerful graphs. + +CLASS_DIAGRAMS = YES + +# If set to YES, the inheritance and collaboration graphs will hide +# inheritance and usage relations if the target is undocumented +# or is not a class. + +HIDE_UNDOC_RELATIONS = YES + +# If you set the HAVE_DOT tag to YES then doxygen will assume the dot tool is +# available from the path. This tool is part of Graphviz, a graph visualization +# toolkit from AT&T and Lucent Bell Labs. The other options in this section +# have no effect if this option is set to NO (the default) + +HAVE_DOT = NO + +# If the CLASS_GRAPH and HAVE_DOT tags are set to YES then doxygen +# will generate a graph for each documented class showing the direct and +# indirect inheritance relations. Setting this tag to YES will force the +# the CLASS_DIAGRAMS tag to NO. + +CLASS_GRAPH = YES + +# If the COLLABORATION_GRAPH and HAVE_DOT tags are set to YES then doxygen +# will generate a graph for each documented class showing the direct and +# indirect implementation dependencies (inheritance, containment, and +# class references variables) of the class with other documented classes. + +COLLABORATION_GRAPH = YES + +# If set to YES, the inheritance and collaboration graphs will show the +# relations between templates and their instances. + +TEMPLATE_RELATIONS = YES + +# If the ENABLE_PREPROCESSING, SEARCH_INCLUDES, INCLUDE_GRAPH, and HAVE_DOT +# tags are set to YES then doxygen will generate a graph for each documented +# file showing the direct and indirect include dependencies of the file with +# other documented files. + +INCLUDE_GRAPH = YES + +# If the ENABLE_PREPROCESSING, SEARCH_INCLUDES, INCLUDED_BY_GRAPH, and +# HAVE_DOT tags are set to YES then doxygen will generate a graph for each +# documented header file showing the documented files that directly or +# indirectly include this file. + +INCLUDED_BY_GRAPH = YES + +# If the GRAPHICAL_HIERARCHY and HAVE_DOT tags are set to YES then doxygen +# will graphical hierarchy of all classes instead of a textual one. + +GRAPHICAL_HIERARCHY = YES + +# The DOT_IMAGE_FORMAT tag can be used to set the image format of the images +# generated by dot. Possible values are png, jpg, or gif +# If left blank png will be used. + +DOT_IMAGE_FORMAT = png + +# The tag DOT_PATH can be used to specify the path where the dot tool can be +# found. If left blank, it is assumed the dot tool can be found on the path. + +DOT_PATH = + +# The DOTFILE_DIRS tag can be used to specify one or more directories that +# contain dot files that are included in the documentation (see the +# \dotfile command). + +DOTFILE_DIRS = + +# The MAX_DOT_GRAPH_WIDTH tag can be used to set the maximum allowed width +# (in pixels) of the graphs generated by dot. If a graph becomes larger than +# this value, doxygen will try to truncate the graph, so that it fits within +# the specified constraint. Beware that most browsers cannot cope with very +# large images. + +MAX_DOT_GRAPH_WIDTH = 1024 + +# The MAX_DOT_GRAPH_HEIGHT tag can be used to set the maximum allows height +# (in pixels) of the graphs generated by dot. If a graph becomes larger than +# this value, doxygen will try to truncate the graph, so that it fits within +# the specified constraint. Beware that most browsers cannot cope with very +# large images. + +MAX_DOT_GRAPH_HEIGHT = 1024 + +# If the GENERATE_LEGEND tag is set to YES (the default) Doxygen will +# generate a legend page explaining the meaning of the various boxes and +# arrows in the dot generated graphs. + +GENERATE_LEGEND = YES + +# If the DOT_CLEANUP tag is set to YES (the default) Doxygen will +# remove the intermedate dot files that are used to generate +# the various graphs. + +DOT_CLEANUP = YES + +#--------------------------------------------------------------------------- +# Configuration::addtions related to the search engine +#--------------------------------------------------------------------------- + +# The SEARCHENGINE tag specifies whether or not a search engine should be +# used. If set to NO the values of all tags below this one will be ignored. + +SEARCHENGINE = NO + +# The CGI_NAME tag should be the name of the CGI script that +# starts the search engine (doxysearch) with the correct parameters. +# A script with this name will be generated by doxygen. + +CGI_NAME = search.cgi + +# The CGI_URL tag should be the absolute URL to the directory where the +# cgi binaries are located. See the documentation of your http daemon for +# details. + +CGI_URL = + +# The DOC_URL tag should be the absolute URL to the directory where the +# documentation is located. If left blank the absolute path to the +# documentation, with file:// prepended to it, will be used. + +DOC_URL = + +# The DOC_ABSPATH tag should be the absolute path to the directory where the +# documentation is located. If left blank the directory on the local machine +# will be used. + +DOC_ABSPATH = + +# The BIN_ABSPATH tag must point to the directory where the doxysearch binary +# is installed. + +BIN_ABSPATH = /usr/local/bin/ + +# The EXT_DOC_PATHS tag can be used to specify one or more paths to +# documentation generated for other projects. This allows doxysearch to search +# the documentation for these projects as well. + +EXT_DOC_PATHS = diff --git a/libs/srtp/doc/Makefile b/libs/srtp/doc/Makefile new file mode 100644 index 0000000000..d4c0845e62 --- /dev/null +++ b/libs/srtp/doc/Makefile @@ -0,0 +1,44 @@ +# Makefile for libSRTP documentation +# +# David A. McGrew +# Cisco Systems, Inc. +# +# This makefile does not use the autoconf system; we don't really need +# it. We just run doxygen then latex. If you don't have either of +# these, then there is no way that you can make your own +# documentation. Of course, you can just go online at pick up the +# documentation from http://srtp.sourceforge.net. + +srcdir = . +top_srcdir = .. +top_builddir = ../ + + +# Determine the version of the library + +version = $(shell cat $(top_srcdir)/VERSION) + + +.PHONY: libsrtpdoc cryptodoc clean +libsrtpdoc: + @if test ! -e Doxyfile; then \ + echo "*** Sorry, can't build doc outside source dir"; exit 1; \ + fi + sed 's/LIBSRTPVERSION/$(version)/' header.template > header.tex + doxygen + sed 's/\subsection/\section/' latex/index.tex > latex/index.tmp + mv latex/index.tmp latex/index.tex + cd latex; make + cp latex/refman.pdf libsrtp.pdf + + +cryptodoc: clean + doxygen crypto.dox + cd latex; make + cp latex/refman.pdf crypto.pdf + +clean: + rm -rf latex/ header.tex + for a in * ; do \ + if [ -f "$$a~" ] ; then rm -f $$a~; fi; \ + done; diff --git a/libs/srtp/doc/Makefile.in b/libs/srtp/doc/Makefile.in new file mode 100644 index 0000000000..5b4b6bd174 --- /dev/null +++ b/libs/srtp/doc/Makefile.in @@ -0,0 +1,44 @@ +# Makefile for libSRTP documentation +# +# David A. McGrew +# Cisco Systems, Inc. +# +# This makefile does not use the autoconf system; we don't really need +# it. We just run doxygen then latex. If you don't have either of +# these, then there is no way that you can make your own +# documentation. Of course, you can just go online at pick up the +# documentation from http://srtp.sourceforge.net. + +srcdir = @srcdir@ +top_srcdir = @top_srcdir@ +top_builddir = @top_builddir@ +VPATH = @srcdir@ + +# Determine the version of the library + +version = $(shell cat $(top_srcdir)/VERSION) + + +.PHONY: libsrtpdoc cryptodoc clean +libsrtpdoc: + @if test ! -e Doxyfile; then \ + echo "*** Sorry, can't build doc outside source dir"; exit 1; \ + fi + sed 's/LIBSRTPVERSION/$(version)/' header.template > header.tex + doxygen + sed 's/\subsection/\section/' latex/index.tex > latex/index.tmp + mv latex/index.tmp latex/index.tex + cd latex; make + cp latex/refman.pdf libsrtp.pdf + + +cryptodoc: clean + doxygen crypto.dox + cd latex; make + cp latex/refman.pdf crypto.pdf + +clean: + rm -rf latex/ header.tex + for a in * ; do \ + if [ -f "$$a~" ] ; then rm -f $$a~; fi; \ + done; diff --git a/libs/srtp/doc/crypto_kernel.txt b/libs/srtp/doc/crypto_kernel.txt new file mode 100644 index 0000000000..b0d033ff0a --- /dev/null +++ b/libs/srtp/doc/crypto_kernel.txt @@ -0,0 +1,76 @@ +/** + +@defgroup CryptoKernel Cryptographic Kernel + +All of the cryptographic functions are contained in a kernel. + +*/ + +/** + +@defgroup CipherImplementations Ciphers +@ingroup CryptoKernel + +@brief A generic cipher type enables cipher agility, that is, the +ability to write code that runs with multiple cipher types. +Ciphers can be used through the crypto kernel, or can be accessed +directly, if need be. + +@{ + +*/ + +/** + * @brief Allocates a cipher of a particular type. + * @warning May be implemented as a macro. + */ +err_status_t +cipher_type_alloc(cipher_type_t *ctype, cipher_t **cipher, + unsigned key_len); + +/** + * @brief Initialized a cipher to use a particular key. May + * be invoked more than once on the same cipher. + * @warning May be implemented as a macro. + */ + +err_status_t +cipher_init(cipher_t *cipher, const uint8_t *key); + +/** + * @brief Sets the initialization vector of a given cipher. + * @warning May be implemented as a macro. + */ + +err_status_t +cipher_set_iv(cipher_t *cipher, void *iv); + +/** + * @brief Encrypts a buffer with a given cipher. + * @warning May be implemented as a macro. + */ + +err_status_t +cipher_encrypt(cipher_t *cipher, void *buf, unsigned int *len); + +/** + * @brief Sets a buffer to the keystream generated by the cipher. + * @warning May be implemented as a macro. + */ +err_status_t +cipher_output(cipher_t *c, uint8_t *buffer, int num_octets_to_output); + +/** + * @brief Deallocates a cipher. + * @warning May be implemented as a macro. + */ +err_status_t +cipher_dealloc(cipher_t *cipher); + + + +/** + * @} + */ + + */ \ No newline at end of file diff --git a/libs/srtp/doc/draft-irtf-cfrg-icm-00.txt b/libs/srtp/doc/draft-irtf-cfrg-icm-00.txt new file mode 100644 index 0000000000..ddfce33807 --- /dev/null +++ b/libs/srtp/doc/draft-irtf-cfrg-icm-00.txt @@ -0,0 +1 @@ + Crypto Forum Research Group David A. McGrew Internet Draft Cisco Systems, Inc. Expires April, 2003 October, 2002 Integer Counter Mode <draft-irtf-cfrg-icm-00.txt> Status of this Memo This document is an Internet Draft and is in full conformance with all provisions of Section 10 of RFC-2026. Internet Drafts are working documents of the Internet Engineering Task Force (IETF), its areas, and working groups. Note that other groups may also distribute working documents as Internet Drafts. Internet Drafts are draft documents valid for a maximum of six months and may be updated, replaced, or obsoleted by other documents at any time. It is inappropriate to use Internet Drafts as reference material or to cite them other than as "work in progress." The list of current Internet-Drafts can be accessed at http://www.ietf.org/ietf/1id-abstracts.txt The list of Internet-Draft Shadow Directories can be accessed at http://www.ietf.org/shadow.html. 1. Abstract This document specifies Integer Counter Mode (ICM), a mode of operation of a block cipher which defines an indexed keystream generator (which generates a keystream segment given an index). This mode is efficient, parallelizable, and has been proven secure given realistic assumptions about the block cipher. Test vectors are provided for AES. Counter Mode admits many variations. The variant specified in this document is secure and flexible, yet it enables a single implementation of a keystream generator to suffice in different application domains. McGrew [Page 1] Internet Draft Integer Counter Mode October, 2002 2. Notational Conventions The key words "MUST", "MUST NOT", "REQUIRED", "SHALL", "SHALL NOT", "SHOULD", "SHOULD NOT", "RECOMMENDED", "MAY", and "OPTIONAL" in this document are to be interpreted as described in RFC-2119 [B97]. 3. Introduction Counter Mode is a way to define a pseudorandom keystream generator using a block cipher [CTR]. The keystream can be used for additive encryption, key derivation, or any other application requiring pseudorandom data. In ICM, the keystream is logically broken into segments. Each segment is identified with a segment index, and the segments have equal lengths. This segmentation makes ICM especially appropriate for securing packet-based protocols. 4. ICM In this section, ICM keystream generation and encryption are defined. 4.1. ICM Parameters The following parameters are used in ICM. These parameters MUST remain fixed for any given use of a key. Parameter Meaning ----------------------------------------------------------------- BLOCK_LENGTH the number of octets in the cipher block KEY_LENGTH the number of octets in the cipher key OFFSET_LENGTH the number of octets in the offset SEGMENT_INDEX_LENGTH the number of octets in the segment index BLOCK_INDEX_LENGTH the number of octets in the block index 4.2. Keystream Segments Conceptually, ICM is a keystream generator that takes a secret key and a segment index as an input and then outputs a keystream segment. The segmentation lends itself to packet encryption, as each keystream segment can be used to encrypt a distinct packet. A counter is a value containing BLOCK_LENGTH octets which is McGrew [Page 2] Internet Draft Integer Counter Mode October, 2002 incremented using an increment function based on integer addition, to produce a sequence of distinct values which are used as inputs to the block cipher. (In the context of this specification, an integer is an octet string, the most significant of which is the first.) The output blocks of the cipher are concatenated to form the keystream segment. The first octet of the segment is the first octet of the first output block, and so on. A schematic of this process is shown in Figure 1. Figure 1. The generation of a keystream segment given a segment index and a block cipher key K. Here C[i] and S[i] denote the ith counter and keystream block, respectively. segment index | v C[0] -----> C[1] -----> C[2] -----> ... | | | v v v +---+ +---+ +---+ K->| E | K->| E | K->| E | ... +---+ +---+ +---+ | | | v v v S[0] S[1] S[2] ... The ith counter C[i] of the keystream segment with segment index s is defined as C[i] = (i + s * (256^BLOCK_INDEX_LENGTH)) (+) r where r denotes the shifted Offset, which is defined as the Offset times 256^(BLOCK_LENGTH - OFFSET_LENGTH). (This multiplication left-shifts the Offset so that it is aligned with the leftmost edge of the block.) Here ^ denotes exponentiation and (+) denotes the bitwise exclusive-or operation. The number of blocks in any segment MUST NOT exceed 256^BLOCK_INDEX_LENGTH. The number of segments MUST NOT exceed 256^SEGMENT_INDEX_LENGTH. These restrictions ensure the uniqueness of each block cipher input. They also imply that each segment contains no more than (256^BLOCK_INDEX_LENGTH)*BLOCK_LENGTH octets. The sum of SEGMENT_INDEX_LENGTH and BLOCK_INDEX_LENGTH MUST NOT exceed BLOCK_LENGTH / 2. This requirement protects the ICM keystream generator from potentially failing to be pseudorandom (see McGrew [Page 3] Internet Draft Integer Counter Mode October, 2002 the rationale). Figure 2. An illustration of the structure of a counter with BLOCK_LENGTH = 8, SEGMENT_INDEX_LENGTH = 2, and BLOCK_INDEX_LENGTH = 2. The field marked `null' is not part of either the block or segment indices. 0 1 2 3 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | null | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | segment index | block index | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ 4.3. ICM Encryption Unless otherwise specified, ICM encryption consists of bitwise exclusive-oring the keystream into the plaintext to produce the ciphertext. 4.4 ICM KEY An ICM key consists of the block cipher key and an Offset. The Offset is an integer with OFFSET_LENGTH octets, which is used to `randomize' the logical starting point of keystream. The Offset is crucial to providing security; see the rationale. The value of OFFSET_LENGTH SHOULD be at least half that of BLOCK_LENGTH. For the purposes of transporting an ICM key, e.g. in a signaling protocol, that key SHOULD be considered a sequence of octets in which the block cipher key precedes the Offset. 5. Implementation Considerations Implementation of the `add one modulo 2^m' operation is simple. For example, with BLOCK_LENGTH = 8 (m=64), it can be implemented in C as if (!++x) ++y; where x and y are 32-bit unsigned integers in network byte order. The implementation of general purpose addition modulo 2^m is slightly more complicated. The fact that the Offset is left-aligned enables an implementation McGrew [Page 4] Internet Draft Integer Counter Mode October, 2002 to avoid propagating carry values outside of the block index and/or the segment index. Choosing an OFFSET_LENGTH value equal to half that of BLOCK_LENGTH avoids all of these carries, since the Offset is then shifted so that it occupies the most significant octets of the block, while the block and segment indices occupy the least significant ones. 6. Parameters and Test Vectors for AES This section provides ICM parameters and test vectors for AES with a 128 bit block size and 128 bit key (that is, with a BLOCK_LENGTH and KEY_LENGTH of 16). All integers are expressed in hexadecimal. Each consecutive pair of hex digits corresponds to an octet, so that the integer 000102030405060708090A0B0C0D0E0F corresponds to the octet sequence { 00, 01, 02, 02 ... }. BLOCK_LENGTH 16 KEY_LENGTH 16 OFFSET_LENGTH 14 SEGMENT_INDEX_LENGTH 6 BLOCK_INDEX_LENGTH 2 Block Cipher Key: 2b7e151628aed2a6abf7158809cf4f3c Offset: f0f1f2f3f4f5f6f7f8f9fafbfcfd Segment Index: 000000000000 Keystream: e03ead0935c95e80e166b16dd92b4eb4 d23513162b02d0f72a43a2fe4a5f97ab ... The counter values that correspond to the keystream blocks are outlined below. Counter Keystream f0f1f2f3f4f5f6f7f8f9fafbfcfd0000 e03ead0935c95e80e166b16dd92b4eb4 f0f1f2f3f4f5f6f7f8f9fafbfcfd0001 d23513162b02d0f72a43a2fe4a5f97ab f0f1f2f3f4f5f6f7f8f9fafbfcfd0002 41e95b3bb0a2e8dd477901e4fca894c0 ... ... 7. Security Considerations Each block cipher input is distinct for any segment and any block index. To see this fact, subtract any two counter values with distinct segment or block indices; the result is non-zero. McGrew [Page 5] Internet Draft Integer Counter Mode October, 2002 The limitation on the number of segments which can be generated ensures that the probability with which an adversary can distinguish the keystream generator from random is negligible. For a theoretical justification of this fact, see Bellare et. al. [BR98]. Their analysis shows that if the block cipher cannot be distinguished from a random permutation, then the keystream generated by ICM cannot be distinguished from keystream generated by a truly random process, as long as the length of keystream which is generated is kept below some threshold. The threshold defined in Section 4.2 is sufficient for most uses of ICM for encryption. This specification refrains from dictating a lower threshold in order to refrain from dictating a particular policy, and to avoid a complicated digression. The use of the Offset, a key-dependent value which randomizes the starting position of the keystream, is essential for security. The omission of this mechanism leaves the door open for practical attacks, such as the key collision attack and Hellman's time-memory tradeoff attack; see McGrew and Fluhrer [MF00] for a description of these attacks which is applicable to ICM. Several counter mode proposals do not include an offset, and are thus vulnerable to these attacks. 8. Rationale This speficiation includes input from implementation experience with several counter mode variants. The goals of ICM are to provide: o a secure keystream generator and cipher, and o a definition flexible enough that a single implementation can be used for a variety of applications (e.g., Secure RTP [SRTP], IPsec ESP [KA96]). The Offset slightly increases the key management overhead, but this minor disadvantage is well outweighed by other savings. The Offset is no larger than a CBC mode IV, and ICM enables the use of an explicit IV (as is commonly used with CBC [MD98]) to be avoided. 9. History This draft is based on draft-mcgrew-saag-icm-00.txt, which was submitted to SAAG on November, 2001 and which expired in May, 2002. The current definition of ICM has changed from the earlier one; the counter formation is different and the specifications are McGrew [Page 6] Internet Draft Integer Counter Mode October, 2002 unfortunately not interoperable. This change was motivated by a considerable amount of feedback on the desirability of admitting optimizations of the sort described in Section 5, in which the carry operations of counter addition need not be propagated across a large register. The current definition of ICM is interoperable with that defined in Secure RTP [SRTP]. 10. Acknowledgements Thanks are due to Helger Lipmaa, Jerome Etienne, Scott Fluhrer and Mats Naslund for their helpful discussion and comments. 11. Contact Information Questions and comments on this draft SHOULD be sent to: David A. McGrew Cisco Systems, Inc. mcgrew@cisco.com and copied to the Crypto Forum Research Group at: cfrg@ietf.org. 12. References [BR98] M. Bellare, A. Desai, E. Lokipii and P. Rogaway, A Concrete Security Treatment of Symmetric Encryption: Analysis of DES Modes of Operation, Proceedings of the 38th Symposium on Foundations of Computer Science, IEEE, 1997. [B97] S. Bradner, Key words for use in RFCs to Indicate Requirement Levels, RFC 2119, March 1997. [AES] The Advanced Encryption Standard, United States National Institute for Standards and Technology (NIST), http://www.nist.gov/aes/. [CTR] M. Dworkin, NIST Special Publication 800-38A, "Recommendation for Block Cipher Modes of Operation: Methods and Techniques", 2001. Online at McGrew [Page 7] Internet Draft Integer Counter Mode October, 2002 http://csrc.nist.gov/publications/nistpubs/800-38a/sp800- 38a.pdf. [MD98] Madson, C., and Doraswamy, N., "The ESP DES-CBC Cipher Algorithm With Explicit IV", RFC 2405, November 1998. [MF00] D. McGrew and S. Fluhrer, Attacks on Additive Encryption and Implications on Internet Security, Selected Areas in Cryptography 2000. [SRTP] The Secure Real-time Transport Protocol, Baugher et. al., Internet Draft, draft-ietf-avt-srtp-05.txt. McGrew [Page 8] \ No newline at end of file diff --git a/libs/srtp/doc/header.template b/libs/srtp/doc/header.template new file mode 100644 index 0000000000..bb1200e92b --- /dev/null +++ b/libs/srtp/doc/header.template @@ -0,0 +1,112 @@ +% header.tex +% +% header file for the libSRTP documentation - based on the header +% file generated by doxygen, with the initial chapters of the +% original libSRTP documentation tacked on +% +\documentclass[letterpaper]{book} +\usepackage{makeidx} +\usepackage{fancyhdr} +\usepackage{graphicx} +\usepackage{multicol} +\usepackage{float} +\usepackage{textcomp} +\usepackage{alltt} +\usepackage{times} +\usepackage{graphicx} +\ifx\pdfoutput\undefined +\usepackage[ps2pdf, + pagebackref=true, + colorlinks=true, + linkcolor=blue + ]{hyperref} +\else +\usepackage[pdftex, + pagebackref=true, + colorlinks=true, + linkcolor=blue + ]{hyperref} +\fi +\usepackage{doxygen} +\makeindex +\setcounter{tocdepth}{1} +\renewcommand{\footrulewidth}{0.4pt} + +% these lengths are from DAM +\textwidth = 6.5 in +%\textheight = 9 in +\oddsidemargin = 0.0 in +\evensidemargin = 0.0 in +\topmargin = 0.0 in +\headheight = 0.0 in +%\headsep = 0.0 in +\parskip = 0.2in +\parindent = 0.0in + +% these header and footer definitions from DAM +\lhead{libSRTP} +\chead{} +\rhead{\rightmark} +%\rhead{\slshape } +\lfoot{} +\cfoot{ \thepage } +\rfoot{} +%\fancyhead[LE,RO]{\rightmark } +%\fancyhead[LO,RE]{\slshape } + +% let's use the palatino font +\fontfamily{ppl} +\selectfont + + +\begin{document} +\begin{titlepage} +\vspace*{4cm} +%\begin{center} +{\Huge +libSRTP LIBSRTPVERSION Overview and Reference Manual\\ + \hrulefill +}\\ +\vspace*{0cm} +\begin{flushright} +{\Large David A. McGrew \\ \texttt{mcgrew@cisco.com} }\\ +\vspace*{0.5cm} +\end{flushright} +%\end{center} + +%\includegraphics[scale=.8]{phone} + +\end{titlepage} + + +\clearemptydoublepage +\vspace*{3cm} +{\LARGE Preface} +\vspace{1cm} + +The original implementation and documentation of libSRTP was written +by David McGrew of Cisco Systems, Inc. in order to promote the use, +understanding, and interoperability of Secure RTP. Randell Jesup +contributed a working SRTCP implementation and other fixes. Alex +Vanzella and Will Clark contributed changes so that the AES ICM +implementation can be used for ISMA media encryption. Steve Underwood +contributed x86\_64 portability changes. We also give thanks to Brian +Weis, Mark Baugher, Jeff Chan, Bill Simon, Douglas Smith, Bill May, +Richard Preistley, Joe Tardo and others for contributions, comments, +and corrections. + +This reference material in this documenation was generated using the +\texttt{doxygen} utility for automatic documentation of source code. + +\copyright 2001-2005 by David A. McGrew, Cisco Systems, Inc. +\thispagestyle{empty} + +\clearemptydoublepage +\pagenumbering{roman} +\tableofcontents +%\clearemptydoublepage + +\clearemptydoublepage +\pagenumbering{arabic} + + diff --git a/libs/srtp/doc/intro.txt b/libs/srtp/doc/intro.txt new file mode 100644 index 0000000000..f315599255 --- /dev/null +++ b/libs/srtp/doc/intro.txt @@ -0,0 +1,395 @@ +/** + +@mainpage Introduction to libSRTP + +This document describes libSRTP, the Open Source Secure RTP library +from Cisco Systems, Inc. RTP is the Real-time Transport Protocol, an +IETF standard for the transport of real-time data such as telephony, +audio, and video, defined by RFC1889. Secure RTP (SRTP) is an RTP +profile for providing confidentiality to RTP data and authentication +to the RTP header and payload. SRTP is an IETF Proposed Standard, and +is defined in RFC 3711, and was developed in the IETF Audio/Video +Transport (AVT) Working Group. This library supports all of the +mandatory features of SRTP, but not all of the optional features. See +the @ref Features section for more detailed information. + +This document is organized as follows. The first chapter provides +background material on SRTP and overview of libSRTP. The following +chapters provide a detailed reference to the libSRTP API and related +functions. The reference material is created automatically (using the +doxygen utility) from comments embedded in some of the C header +files. The documentation is organized into modules in order to improve +its clarity. These modules do not directly correspond to files. An +underlying cryptographic kernel provides much of the basic +functionality of libSRTP, but is mostly undocumented because it does +its work behind the scenes. + +@section LICENSE License and Disclaimer + +libSRTP is distributed under the following license, which is included +in the source code distribution. It is reproduced in the manual in +case you got the library from another source. + +@latexonly +\begin{quote} +Copyright (c) 2001-2005 Cisco Systems, Inc. All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions +are met: +\begin{itemize} +\item Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. +\item Redistributions in binary form must reproduce the above + copyright notice, this list of conditions and the following + disclaimer in the documentation and/or other materials provided + with the distribution. +\item Neither the name of the Cisco Systems, Inc. nor the names of its + contributors may be used to endorse or promote products derived + from this software without specific prior written permission. +\end{itemize} +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS +FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE +COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, +INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES +(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR +SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) +HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, +STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) +ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED +OF THE POSSIBILITY OF SUCH DAMAGE. +\end{quote} +@endlatexonly + +@section Features Supported Features + +This library supports all of the mandatory-to-implement features of +SRTP (as defined by the most recent Internet Draft). Some of these +features can be selected (or de-selected) at run time by setting an +appropriate policy; this is done using the structure srtp_policy_t. +Some other behaviors of the protocol can be adapted by defining an +approriate event handler for the exceptional events; see the @ref +SRTPevents section. + +Some options that are not included in the specification are supported. +Most notably, the TMMH authentication function is included, though it +was removed from the SRTP Internet Draft during the summer of 2002. + + +@latexonly +Some options that are described in the SRTP specification are not +supported. This includes +\begin{itemize} +\item the Master Key Index (MKI), +\item key derivation rates other than zero, +\item the cipher F8, +\item anti-replay lists with sizes other than 128, +\item the use of the packet index to select between master keys. +\end{itemize} +@endlatexonly + +The user should be aware that it is possible to misuse this libary, +and that the result may be that the security level it provides is +inadequate. If you are implementing a feature using this library, you +will want to read the Security Considerations section of the Internet +Draft. In addition, it is important that you read and understand the +terms outlined in the @ref LICENSE section. + + +@section Installing Installing and Building libSRTP + +@latexonly + +To install libSRTP, download the latest release of the distribution +from \texttt{srtp.sourceforge.net}. The format of the names of the +distributions are \texttt{srtp-A.B.C.tgz}, where \texttt{A} is the +version number, \texttt{B} is the major release number, \texttt{C} is +the minor release number, and \texttt{tgz} is the file +extension\footnote{The extension \texttt{.tgz} is identical to +\texttt{tar.gz}, and indicates a compressed tar file.} You probably +want to get the most recent release. Unpack the distribution and +extract the source files; the directory into which the soruce files +will go is named \texttt{srtp}. + +libSRTP uses the GNU \texttt{autoconf} and \texttt{make} +utilities\footnote{BSD make will not work; if both versions of make +are on your platform, you can invoke GNU make as \texttt{gmake}.}. In +the \texttt{srtp} directory, run the configure script and then make: +\begin{verbatim} + ./configure [ options ] + make +\end{verbatim} +The configure script accepts the following options: +\begin{quote} +\begin{description} +\item[--help] provides a usage summary. +\item[--disable-debug] compiles libSRTP without the runtime + dynamic debugging system. +\item[--enable-generic-aesicm] compile in changes for ismacryp +\item[--enable-syslog] use syslog for error reporting. +\item[--disable-stdout] diables stdout for error reporting. +\item[--enable-console] use \texttt{/dev/console} for error reporting +\item[--gdoi] use GDOI key management (disabled at present). +\end{description} +\end{quote} + +By default, dynamic debbuging is enabled and stdout is used for +debugging. You can use the configure options to have the debugging +output sent to syslog or the system console. Alternatively, you can +define ERR\_REPORTING\_FILE in \texttt{include/conf.h} to be any other +file that can be opened by libSRTP, and debug messages will be sent to +it. + +This package has been tested on the following platforms: Mac OS X +(powerpc-apple-darwin1.4), Cygwin (i686-pc-cygwin), Solaris +(sparc-sun-solaris2.6), RedHat Linux 7.1 and 9 (i686-pc-linux), and +OpenBSD (sparc-unknown-openbsd2.7). + + +@endlatexonly + +@section Applications Applications + +@latexonly + +Several test drivers and a simple and portable srtp application are +included in the \texttt{test/} subdirectory. + +\begin{center} +\begin{tabular}{ll} +\hline +Test driver & Function tested \\ +\hline +kernel\_driver & crypto kernel (ciphers, auth funcs, rng) \\ +srtp\_driver & srtp in-memory tests (does not use the network) \\ +rdbx\_driver & rdbx (extended replay database) \\ +roc\_driver & extended sequence number functions \\ +replay\_driver & replay database \\ +cipher\_driver & ciphers \\ +auth\_driver & hash functions \\ +\hline +\end{tabular} +\end{center} + +The app rtpw is a simple rtp application which reads words from +/usr/dict/words and then sends them out one at a time using [s]rtp. +Manual srtp keying uses the -k option; automated key management +using gdoi will be added later. + +The usage for rtpw is + +\texttt{rtpw [[-d $<$debug$>$]* [-k $<$key$>$ [-a][-e]] [-s | -r] dest\_ip +dest\_port][-l]} + +Either the -s (sender) or -r (receiver) option must be chosen. The +values dest\_ip, dest\_port are the IP address and UDP port to which +the dictionary will be sent, respectively. The options are: +\begin{center} +\begin{tabular}{ll} + -s & (S)RTP sender - causes app to send words \\ + -r & (S)RTP receive - causes app to receve words \\ + -k $<$key$>$ & use SRTP master key $<$key$>$, where the + key is a hexadecimal value (without the + leading "0x") \\ + -e & encrypt/decrypt (for data confidentiality) + (requires use of -k option as well)\\ + -a & message authentication + (requires use of -k option as well) \\ + -l & list the avaliable debug modules \\ + -d $<$debug$>$ & turn on debugging for module $<$debug$>$ \\ +\end{tabular} +\end{center} + +In order to get a random 30-byte value for use as a key/salt pair, you +can use the \texttt{rand\_gen} utility in the \texttt{test/} +subdirectory. + +An example of an SRTP session using two rtpw programs follows: + +\begin{verbatim} +[sh1] set k=`test/rand_gen -n 30` +[sh1] echo $k +c1eec3717da76195bb878578790af71c4ee9f859e197a414a78d5abc7451 +[sh1]$ test/rtpw -s -k $k -ea 0.0.0.0 9999 +Security services: confidentiality message authentication +set master key/salt to C1EEC3717DA76195BB878578790AF71C/4EE9F859E197A414A78D5ABC7451 +setting SSRC to 2078917053 +sending word: A +sending word: a +sending word: aa +sending word: aal +sending word: aalii +sending word: aam +sending word: Aani +sending word: aardvark +... + +[sh2] set k=c1eec3717da76195bb878578790af71c4ee9f859e197a414a78d5abc7451 +[sh2]$ test/rtpw -r -k $k -ea 0.0.0.0 9999 +security services: confidentiality message authentication +set master key/salt to C1EEC3717DA76195BB878578790AF71C/4EE9F859E197A414A78D5ABC7451 +19 octets received from SSRC 2078917053 word: A +19 octets received from SSRC 2078917053 word: a +20 octets received from SSRC 2078917053 word: aa +21 octets received from SSRC 2078917053 word: aal +... +\end{verbatim} + + +@endlatexonly + + +@section Review Secure RTP Background + +In this section we review SRTP and introduce some terms that are used +in libSRTP. An RTP session is defined by a pair of destination +transport addresses, that is, a network address plus a pair of UDP +ports for RTP and RTCP. RTCP, the RTP control protocol, is used to +coordinate between the participants in an RTP session, e.g. to provide +feedback from receivers to senders. An @e SRTP @e session is +similarly defined; it is just an RTP session for which the SRTP +profile is being used. An SRTP session consists of the traffic sent +to the SRTP or SRTCP destination transport addresses. Each +participant in a session is identified by a synchronization source +(SSRC) identifier. Some participants may not send any SRTP traffic; +they are called receivers, even though they send out SRTCP traffic, +such as receiver reports. + +RTP allows multiple sources to send RTP and RTCP traffic during the +same session. The synchronization source identifier (SSRC) is used to +distinguish these sources. In libSRTP, we call the SRTP and SRTCP +traffic from a particular source a @e stream. Each stream has its own +SSRC, sequence number, rollover counter, and other data. A particular +choice of options, cryptographic mechanisms, and keys is called a @e +policy. Each stream within a session can have a distinct policy +applied to it. A session policy is a collection of stream policies. + +A single policy can be used for all of the streams in a given session, +though the case in which a single @e key is shared across multiple +streams requires care. When key sharing is used, the SSRC values that +identify the streams @b must be distinct. This requirement can be +enforced by using the convention that each SRTP and SRTCP key is used +for encryption by only a single sender. In other words, the key is +shared only across streams that originate from a particular device (of +course, other SRTP participants will need to use the key for +decryption). libSRTP supports this enforcement by detecting the case +in which a key is used for both inbound and outbound data. + + +@section Overview libSRTP Overview + +libSRTP provides functions for protecting RTP and RTCP. RTP packets +can be encrypted and authenticated (using the srtp_protect() +function), turning them into SRTP packets. Similarly, SRTP packets +can be decrypted and have their authentication verified (using the +srtp_unprotect() function), turning them into RTP packets. Similar +functions apply security to RTCP packets. + +The typedef srtp_stream_t points to a structure holding all of the +state associated with an SRTP stream, including the keys and +parameters for cipher and message authentication functions and the +anti-replay data. A particular srtp_stream_t holds the information +needed to protect a particular RTP and RTCP stream. This datatype +is intentionally opaque in order to better seperate the libSRTP +API from its implementation. + +Within an SRTP session, there can be multiple streams, each +originating from a particular sender. Each source uses a distinct +stream context to protect the RTP and RTCP stream that it is +originating. The typedef srtp_t points to a structure holding all of +the state associated with an SRTP session. There can be multiple +stream contexts associated with a single srtp_t. A stream context +cannot exist indepent from an srtp_t, though of course an srtp_t can +be created that contains only a single stream context. A device +participating in an SRTP session must have a stream context for each +source in that session, so that it can process the data that it +receives from each sender. + + +In libSRTP, a session is created using the function srtp_create(). +The policy to be implemented in the session is passed into this +function as an srtp_policy_t structure. A single one of these +structures describes the policy of a single stream. These structures +can also be linked together to form an entire session policy. A linked +list of srtp_policy_t structures is equivalent to a session policy. +In such a policy, we refer to a single srtp_policy_t as an @e element. + +An srtp_policy_t strucutre contains two crypto_policy_t structures +that describe the cryptograhic policies for RTP and RTCP, as well as +the SRTP master key and the SSRC value. The SSRC describes what to +protect (e.g. which stream), and the crypto_policy_t structures +describe how to protect it. The key is contained in a policy element +because it simplifies the interface to the library. In many cases, it +is desirable to use the same cryptographic policies across all of the +streams in a session, but to use a distinct key for each stream. A +crypto_policy_t structure can be initialized by using either the +crypto_policy_set_rtp_default() or crypto_policy_set_rtcp_default() +functions, which set a crypto policy structure to the default policies +for RTP and RTCP protection, respectively. + +@section Example Example Code + +This section provides a simple example of how to use libSRTP. The +example code lacks error checking, but is functional. Here we assume +that the value ssrc is already set to describe the SSRC of the stream +that we are sending, and that the functions get_rtp_packet() and +send_srtp_packet() are available to us. The former puts an RTP packet +into the buffer and returns the number of octets written to that +buffer. The latter sends the RTP packet in the buffer, given the +length as its second argument. + +@verbatim + srtp_t session; + srtp_policy_t policy; + uint8_t key[30]; + + // initialize libSRTP + srtp_init(); + + // set policy to describe a policy for an SRTP stream + crypto_policy_set_rtp_default(&policy.rtp); + crypto_policy_set_rtcp_default(&policy.rtcp); + policy.ssrc = ssrc; + policy.key = key; + policy.next = NULL; + + // set key to random value + crypto_get_random(key, 30); + + // allocate and initialize the SRTP session + srtp_create(&session, policy); + + // main loop: get rtp packets, send srtp packets + while (1) { + char rtp_buffer[2048]; + unsigned len; + + len = get_rtp_packet(rtp_buffer); + srtp_protect(session, rtp_buffer, &len); + send_srtp_packet(rtp_buffer, len); + } +@endverbatim + +@section ISMAcryp ISMA Encryption Support + +The Internet Streaming Media Alliance (ISMA) specifies a way +to pre-encrypt a media file prior to streaming. This method +is an alternative to SRTP encryption, which is potentially +useful when a particular media file will be streamed +multiple times. The specification is available online +at http://www.isma.tv/specreq.nsf/SpecRequest. + +libSRTP provides the encryption and decryption functions needed for ISMAcryp +in the library @t libaesicm.a, which is included in the default +Makefile target. This library is used by the MPEG4IP project; see +http://mpeg4ip.sourceforge.net/. + +Note that ISMAcryp does not provide authentication for +RTP nor RTCP, nor confidentiality for RTCP. +ISMAcryp RECOMMENDS the use of SRTP message authentication for ISMAcryp +streams while using ISMAcryp encryption to protect the media itself. + + + */ diff --git a/libs/srtp/doc/libsrtp.pdf b/libs/srtp/doc/libsrtp.pdf new file mode 100644 index 0000000000000000000000000000000000000000..0538ab4551f293e0fc497a8b98237731d4f18d5c GIT binary patch literal 355874 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 ar4YV2o}H_{?*ZiC)D?Q@p$4rQ7W#ijmF~O% literal 0 HcmV?d00001 diff --git a/libs/srtp/doc/references.txt b/libs/srtp/doc/references.txt new file mode 100644 index 0000000000..2ec9d43782 --- /dev/null +++ b/libs/srtp/doc/references.txt @@ -0,0 +1,21 @@ +SRTP and ICM References +September, 2005 + +This document provides references for the various cryptographic +functions used in libSRTP and libaesicm. + +Secure RTP is defined in RFC 3711, which is included in this +distribution for convenience. The counter mode definition is in +Section 4.1.1 of the SRTP draft. + +SHA-1 is defined in FIPS-180-1, available online at the NIST website. + +HMAC is defined in RFC2104, and HMAC-SHA1 test vectors are available +in RFC2202, which are available online at http://www.ietf.org/rfc/ + +ICM is defined by draft-irtf-cfrg-icm-00.txt, and its application in +ISMAcryp (the Internet Streaming Media Alliance 1.0 Encryption and +Authentication) is defined in that specification. It is available +from http://www.isma.tv/. + + diff --git a/libs/srtp/doc/rfc3711.txt b/libs/srtp/doc/rfc3711.txt new file mode 100644 index 0000000000..ecc0648aca --- /dev/null +++ b/libs/srtp/doc/rfc3711.txt @@ -0,0 +1,3139 @@ + + + + + + +Network Working Group M. Baugher +Request for Comments: 3711 D. McGrew +Category: Standards Track Cisco Systems, Inc. + M. Naslund + E. Carrara + K. Norrman + Ericsson Research + March 2004 + + + The Secure Real-time Transport Protocol (SRTP) + +Status of this Memo + + This document specifies an Internet standards track protocol for the + Internet community, and requests discussion and suggestions for + improvements. Please refer to the current edition of the "Internet + Official Protocol Standards" (STD 1) for the standardization state + and status of this protocol. Distribution of this memo is unlimited. + +Copyright Notice + + Copyright (C) The Internet Society (2004). All Rights Reserved. + +Abstract + + This document describes the Secure Real-time Transport Protocol + (SRTP), a profile of the Real-time Transport Protocol (RTP), which + can provide confidentiality, message authentication, and replay + protection to the RTP traffic and to the control traffic for RTP, the + Real-time Transport Control Protocol (RTCP). + +Table of Contents + + 1. Introduction . . . . . . . . . . . . . . . . . . . . . . . . . 3 + 1.1. Notational Conventions . . . . . . . . . . . . . . . . . 3 + 2. Goals and Features . . . . . . . . . . . . . . . . . . . . . . 4 + 2.1. Features . . . . . . . . . . . . . . . . . . . . . . . . 5 + 3. SRTP Framework . . . . . . . . . . . . . . . . . . . . . . . . 5 + 3.1. Secure RTP . . . . . . . . . . . . . . . . . . . . . . . 6 + 3.2. SRTP Cryptographic Contexts. . . . . . . . . . . . . . . 7 + 3.2.1. Transform-independent parameters . . . . . . . . 8 + 3.2.2. Transform-dependent parameters . . . . . . . . . 10 + 3.2.3. Mapping SRTP Packets to Cryptographic Contexts . 10 + 3.3. SRTP Packet Processing . . . . . . . . . . . . . . . . . 11 + 3.3.1. Packet Index Determination, and ROC, s_l Update. 13 + 3.3.2. Replay Protection. . . . . . . . . . . . . . . . 15 + 3.4. Secure RTCP . . . . . . . . . . . . . . . . . . . . . . . 15 + + + +Baugher, et al. Standards Track [Page 1] + +RFC 3711 SRTP March 2004 + + + 4. Pre-Defined Cryptographic Transforms . . . . . . . . . . . . . 19 + 4.1. Encryption . . . . . . . . . . . . . . . . . . . . . . . 19 + 4.1.1. AES in Counter Mode. . . . . . . . . . . . . . . 21 + 4.1.2. AES in f8-mode . . . . . . . . . . . . . . . . . 22 + 4.1.3. NULL Cipher. . . . . . . . . . . . . . . . . . . 25 + 4.2. Message Authentication and Integrity . . . . . . . . . . 25 + 4.2.1. HMAC-SHA1. . . . . . . . . . . . . . . . . . . . 25 + 4.3. Key Derivation . . . . . . . . . . . . . . . . . . . . . 26 + 4.3.1. Key Derivation Algorithm . . . . . . . . . . . . 26 + 4.3.2. SRTCP Key Derivation . . . . . . . . . . . . . . 28 + 4.3.3. AES-CM PRF . . . . . . . . . . . . . . . . . . . 28 + 5. Default and mandatory-to-implement Transforms. . . . . . . . . 28 + 5.1. Encryption: AES-CM and NULL. . . . . . . . . . . . . . . 29 + 5.2. Message Authentication/Integrity: HMAC-SHA1. . . . . . . 29 + 5.3. Key Derivation: AES-CM PRF . . . . . . . . . . . . . . . 29 + 6. Adding SRTP Transforms . . . . . . . . . . . . . . . . . . . . 29 + 7. Rationale. . . . . . . . . . . . . . . . . . . . . . . . . . . 30 + 7.1. Key derivation . . . . . . . . . . . . . . . . . . . . . 30 + 7.2. Salting key. . . . . . . . . . . . . . . . . . . . . . . 30 + 7.3. Message Integrity from Universal Hashing . . . . . . . . 31 + 7.4. Data Origin Authentication Considerations. . . . . . . . 31 + 7.5. Short and Zero-length Message Authentication . . . . . . 32 + 8. Key Management Considerations. . . . . . . . . . . . . . . . . 33 + 8.1. Re-keying . . . . . . . . . . . . . . . . . . . . . . . 34 + 8.1.1. Use of the <From, To> for re-keying. . . . . . . 34 + 8.2. Key Management parameters. . . . . . . . . . . . . . . . 35 + 9. Security Considerations. . . . . . . . . . . . . . . . . . . . 37 + 9.1. SSRC collision and two-time pad. . . . . . . . . . . . . 37 + 9.2. Key Usage. . . . . . . . . . . . . . . . . . . . . . . . 38 + 9.3. Confidentiality of the RTP Payload . . . . . . . . . . . 39 + 9.4. Confidentiality of the RTP Header. . . . . . . . . . . . 40 + 9.5. Integrity of the RTP payload and header. . . . . . . . . 40 + 9.5.1. Risks of Weak or Null Message Authentication. . . 42 + 9.5.2. Implicit Header Authentication . . . . . . . . . 43 + 10. Interaction with Forward Error Correction mechanisms. . . . . 43 + 11. Scenarios . . . . . . . . . . . . . . . . . . . . . . . . . . 43 + 11.1. Unicast. . . . . . . . . . . . . . . . . . . . . . . . . 43 + 11.2. Multicast (one sender) . . . . . . . . . . . . . . . . . 44 + 11.3. Re-keying and access control . . . . . . . . . . . . . . 45 + 11.4. Summary of basic scenarios . . . . . . . . . . . . . . . 46 + 12. IANA Considerations. . . . . . . . . . . . . . . . . . . . . . 46 + 13. Acknowledgements . . . . . . . . . . . . . . . . . . . . . . . 47 + 14. References . . . . . . . . . . . . . . . . . . . . . . . . . . 47 + 14.1. Normative References . . . . . . . . . . . . . . . . . . 47 + 14.2. Informative References . . . . . . . . . . . . . . . . . 48 + Appendix A: Pseudocode for Index Determination . . . . . . . . . . 51 + Appendix B: Test Vectors . . . . . . . . . . . . . . . . . . . . . 51 + B.1. AES-f8 Test Vectors. . . . . . . . . . . . . . . . . . . 51 + + + +Baugher, et al. Standards Track [Page 2] + +RFC 3711 SRTP March 2004 + + + B.2. AES-CM Test Vectors. . . . . . . . . . . . . . . . . . . 52 + B.3. Key Derivation Test Vectors. . . . . . . . . . . . . . . 53 + Authors' Addresses . . . . . . . . . . . . . . . . . . . . . . . . 55 + Full Copyright Statement . . . . . . . . . . . . . . . . . . . . . 56 + +1. Introduction + + This document describes the Secure Real-time Transport Protocol + (SRTP), a profile of the Real-time Transport Protocol (RTP), which + can provide confidentiality, message authentication, and replay + protection to the RTP traffic and to the control traffic for RTP, + RTCP (the Real-time Transport Control Protocol) [RFC3350]. + + SRTP provides a framework for encryption and message authentication + of RTP and RTCP streams (Section 3). SRTP defines a set of default + cryptographic transforms (Sections 4 and 5), and it allows new + transforms to be introduced in the future (Section 6). With + appropriate key management (Sections 7 and 8), SRTP is secure + (Sections 9) for unicast and multicast RTP applications (Section 11). + + SRTP can achieve high throughput and low packet expansion. SRTP + proves to be a suitable protection for heterogeneous environments + (mix of wired and wireless networks). To get such features, default + transforms are described, based on an additive stream cipher for + encryption, a keyed-hash based function for message authentication, + and an "implicit" index for sequencing/synchronization based on the + RTP sequence number for SRTP and an index number for Secure RTCP + (SRTCP). + +1.1. Notational Conventions + + The keywords "MUST", "MUST NOT", "REQUIRED", "SHALL", "SHALL NOT", + "SHOULD", "SHOULD NOT", "RECOMMENDED", "MAY", and "OPTIONAL" in this + document are to be interpreted as described in [RFC2119]. The + terminology conforms to [RFC2828] with the following exception. For + simplicity we use the term "random" throughout the document to denote + randomly or pseudo-randomly generated values. Large amounts of + random bits may be difficult to obtain, and for the security of SRTP, + pseudo-randomness is sufficient [RFC1750]. + + By convention, the adopted representation is the network byte order, + i.e., the left most bit (octet) is the most significant one. By XOR + we mean bitwise addition modulo 2 of binary strings, and || denotes + concatenation. In other words, if C = A || B, then the most + significant bits of C are the bits of A, and the least significant + bits of C equal the bits of B. Hexadecimal numbers are prefixed by + 0x. + + + + +Baugher, et al. Standards Track [Page 3] + +RFC 3711 SRTP March 2004 + + + The word "encryption" includes also use of the NULL algorithm (which + in practice does leave the data in the clear). + + With slight abuse of notation, we use the terms "message + authentication" and "authentication tag" as is common practice, even + though in some circumstances, e.g., group communication, the service + provided is actually only integrity protection and not data origin + authentication. + +2. Goals and Features + + The security goals for SRTP are to ensure: + + * the confidentiality of the RTP and RTCP payloads, and + + * the integrity of the entire RTP and RTCP packets, together with + protection against replayed packets. + + These security services are optional and independent from each other, + except that SRTCP integrity protection is mandatory (malicious or + erroneous alteration of RTCP messages could otherwise disrupt the + processing of the RTP stream). + + Other, functional, goals for the protocol are: + + * a framework that permits upgrading with new cryptographic + transforms, + + * low bandwidth cost, i.e., a framework preserving RTP header + compression efficiency, + + and, asserted by the pre-defined transforms: + + * a low computational cost, + + * a small footprint (i.e., small code size and data memory for + keying information and replay lists), + + * limited packet expansion to support the bandwidth economy goal, + + * independence from the underlying transport, network, and physical + layers used by RTP, in particular high tolerance to packet loss + and re-ordering. + + These properties ensure that SRTP is a suitable protection scheme for + RTP/RTCP in both wired and wireless scenarios. + + + + + +Baugher, et al. Standards Track [Page 4] + +RFC 3711 SRTP March 2004 + + +2.1. Features + + Besides the above mentioned direct goals, SRTP provides for some + additional features. They have been introduced to lighten the burden + on key management and to further increase security. They include: + + * A single "master key" can provide keying material for + confidentiality and integrity protection, both for the SRTP stream + and the corresponding SRTCP stream. This is achieved with a key + derivation function (see Section 4.3), providing "session keys" + for the respective security primitive, securely derived from the + master key. + + * In addition, the key derivation can be configured to periodically + refresh the session keys, which limits the amount of ciphertext + produced by a fixed key, available for an adversary to + cryptanalyze. + + * "Salting keys" are used to protect against pre-computation and + time-memory tradeoff attacks [MF00] [BS00]. + + Detailed rationale for these features can be found in Section 7. + +3. SRTP Framework + + RTP is the Real-time Transport Protocol [RFC3550]. We define SRTP as + a profile of RTP. This profile is an extension to the RTP + Audio/Video Profile [RFC3551]. Except where explicitly noted, all + aspects of that profile apply, with the addition of the SRTP security + features. Conceptually, we consider SRTP to be a "bump in the stack" + implementation which resides between the RTP application and the + transport layer. SRTP intercepts RTP packets and then forwards an + equivalent SRTP packet on the sending side, and intercepts SRTP + packets and passes an equivalent RTP packet up the stack on the + receiving side. + + Secure RTCP (SRTCP) provides the same security services to RTCP as + SRTP does to RTP. SRTCP message authentication is MANDATORY and + thereby protects the RTCP fields to keep track of membership, provide + feedback to RTP senders, or maintain packet sequence counters. SRTCP + is described in Section 3.4. + + + + + + + + + + +Baugher, et al. Standards Track [Page 5] + +RFC 3711 SRTP March 2004 + + +3.1. Secure RTP + + The format of an SRTP packet is illustrated in Figure 1. + + 0 1 2 3 + 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 + +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+<+ + |V=2|P|X| CC |M| PT | sequence number | | + +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | + | timestamp | | + +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | + | synchronization source (SSRC) identifier | | + +=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+ | + | contributing source (CSRC) identifiers | | + | .... | | + +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | + | RTP extension (OPTIONAL) | | + +>+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | + | | payload ... | | + | | +-------------------------------+ | + | | | RTP padding | RTP pad count | | + +>+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+<+ + | ~ SRTP MKI (OPTIONAL) ~ | + | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | + | : authentication tag (RECOMMENDED) : | + | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | + | | + +- Encrypted Portion* Authenticated Portion ---+ + + Figure 1. The format of an SRTP packet. *Encrypted Portion is the + same size as the plaintext for the Section 4 pre-defined transforms. + + The "Encrypted Portion" of an SRTP packet consists of the encryption + of the RTP payload (including RTP padding when present) of the + equivalent RTP packet. The Encrypted Portion MAY be the exact size + of the plaintext or MAY be larger. Figure 1 shows the RTP payload + including any possible padding for RTP [RFC3550]. + + None of the pre-defined encryption transforms uses any padding; for + these, the RTP and SRTP payload sizes match exactly. New transforms + added to SRTP (following Section 6) may require padding, and may + hence produce larger payloads. RTP provides its own padding format + (as seen in Fig. 1), which due to the padding indicator in the RTP + header has merits in terms of compactness relative to paddings using + prefix-free codes. This RTP padding SHALL be the default method for + transforms requiring padding. Transforms MAY specify other padding + methods, and MUST then specify the amount, format, and processing of + their padding. It is important to note that encryption transforms + + + +Baugher, et al. Standards Track [Page 6] + +RFC 3711 SRTP March 2004 + + + that use padding are vulnerable to subtle attacks, especially when + message authentication is not used [V02]. Each specification for a + new encryption transform needs to carefully consider and describe the + security implications of the padding that it uses. Message + authentication codes define their own padding, so this default does + not apply to authentication transforms. + + The OPTIONAL MKI and the RECOMMENDED authentication tag are the only + fields defined by SRTP that are not in RTP. Only 8-bit alignment is + assumed. + + MKI (Master Key Identifier): configurable length, OPTIONAL. The + MKI is defined, signaled, and used by key management. The + MKI identifies the master key from which the session + key(s) were derived that authenticate and/or encrypt the + particular packet. Note that the MKI SHALL NOT identify + the SRTP cryptographic context, which is identified + according to Section 3.2.3. The MKI MAY be used by key + management for the purposes of re-keying, identifying a + particular master key within the cryptographic context + (Section 3.2.1). + + Authentication tag: configurable length, RECOMMENDED. The + authentication tag is used to carry message authentication + data. The Authenticated Portion of an SRTP packet + consists of the RTP header followed by the Encrypted + Portion of the SRTP packet. Thus, if both encryption and + authentication are applied, encryption SHALL be applied + before authentication on the sender side and conversely on + the receiver side. The authentication tag provides + authentication of the RTP header and payload, and it + indirectly provides replay protection by authenticating + the sequence number. Note that the MKI is not integrity + protected as this does not provide any extra protection. + +3.2. SRTP Cryptographic Contexts + + Each SRTP stream requires the sender and receiver to maintain + cryptographic state information. This information is called the + "cryptographic context". + + SRTP uses two types of keys: session keys and master keys. By a + "session key", we mean a key which is used directly in a + cryptographic transform (e.g., encryption or message authentication), + and by a "master key", we mean a random bit string (given by the key + management protocol) from which session keys are derived in a + + + + + +Baugher, et al. Standards Track [Page 7] + +RFC 3711 SRTP March 2004 + + + cryptographically secure way. The master key(s) and other parameters + in the cryptographic context are provided by key management + mechanisms external to SRTP, see Section 8. + +3.2.1. Transform-independent parameters + + Transform-independent parameters are present in the cryptographic + context independently of the particular encryption or authentication + transforms that are used. The transform-independent parameters of + the cryptographic context for SRTP consist of: + + * a 32-bit unsigned rollover counter (ROC), which records how many + times the 16-bit RTP sequence number has been reset to zero after + passing through 65,535. Unlike the sequence number (SEQ), which + SRTP extracts from the RTP packet header, the ROC is maintained by + SRTP as described in Section 3.3.1. + + We define the index of the SRTP packet corresponding to a given + ROC and RTP sequence number to be the 48-bit quantity + + i = 2^16 * ROC + SEQ. + + * for the receiver only, a 16-bit sequence number s_l, which can be + thought of as the highest received RTP sequence number (see + Section 3.3.1 for its handling), which SHOULD be authenticated + since message authentication is RECOMMENDED, + + * an identifier for the encryption algorithm, i.e., the cipher and + its mode of operation, + + * an identifier for the message authentication algorithm, + + * a replay list, maintained by the receiver only (when + authentication and replay protection are provided), containing + indices of recently received and authenticated SRTP packets, + + * an MKI indicator (0/1) as to whether an MKI is present in SRTP and + SRTCP packets, + + * if the MKI indicator is set to one, the length (in octets) of the + MKI field, and (for the sender) the actual value of the currently + active MKI (the value of the MKI indicator and length MUST be kept + fixed for the lifetime of the context), + + * the master key(s), which MUST be random and kept secret, + + + + + + +Baugher, et al. Standards Track [Page 8] + +RFC 3711 SRTP March 2004 + + + * for each master key, there is a counter of the number of SRTP + packets that have been processed (sent) with that master key + (essential for security, see Sections 3.3.1 and 9), + + * non-negative integers n_e, and n_a, determining the length of the + session keys for encryption, and message authentication. + + In addition, for each master key, an SRTP stream MAY use the + following associated values: + + * a master salt, to be used in the key derivation of session keys. + This value, when used, MUST be random, but MAY be public. Use of + master salt is strongly RECOMMENDED, see Section 9.2. A "NULL" + salt is treated as 00...0. + + * an integer in the set {1,2,4,...,2^24}, the "key_derivation_rate", + where an unspecified value is treated as zero. The constraint to + be a power of 2 simplifies the session-key derivation + implementation, see Section 4.3. + + * an MKI value, + + * <From, To> values, specifying the lifetime for a master key, + expressed in terms of the two 48-bit index values inside whose + range (including the range end-points) the master key is valid. + For the use of <From, To>, see Section 8.1.1. <From, To> is an + alternative to the MKI and assumes that a master key is in one- + to-one correspondence with the SRTP session key on which the + <From, To> range is defined. + + SRTCP SHALL by default share the crypto context with SRTP, except: + + * no rollover counter and s_l-value need to be maintained as the + RTCP index is explicitly carried in each SRTCP packet, + + * a separate replay list is maintained (when replay protection is + provided), + + * SRTCP maintains a separate counter for its master key (even if the + master key is the same as that for SRTP, see below), as a means to + maintain a count of the number of SRTCP packets that have been + processed with that key. + + Note in particular that the master key(s) MAY be shared between SRTP + and the corresponding SRTCP, if the pre-defined transforms (including + the key derivation) are used but the session key(s) MUST NOT be so + shared. + + + + +Baugher, et al. Standards Track [Page 9] + +RFC 3711 SRTP March 2004 + + + In addition, there can be cases (see Sections 8 and 9.1) where + several SRTP streams within a given RTP session, identified by their + synchronization source (SSRCs, which is part of the RTP header), + share most of the crypto context parameters (including possibly + master and session keys). In such cases, just as in the normal + SRTP/SRTCP parameter sharing above, separate replay lists and packet + counters for each stream (SSRC) MUST still be maintained. Also, + separate SRTP indices MUST then be maintained. + + A summary of parameters, pre-defined transforms, and default values + for the above parameters (and other SRTP parameters) can be found in + Sections 5 and 8.2. + +3.2.2. Transform-dependent parameters + + All encryption, authentication/integrity, and key derivation + parameters are defined in the transforms section (Section 4). + Typical examples of such parameters are block size of ciphers, + session keys, data for the Initialization Vector (IV) formation, etc. + Future SRTP transform specifications MUST include a section to list + the additional cryptographic context's parameters for that transform, + if any. + +3.2.3. Mapping SRTP Packets to Cryptographic Contexts + + Recall that an RTP session for each participant is defined [RFC3550] + by a pair of destination transport addresses (one network address + plus a port pair for RTP and RTCP), and that a multimedia session is + defined as a collection of RTP sessions. For example, a particular + multimedia session could include an audio RTP session, a video RTP + session, and a text RTP session. + + A cryptographic context SHALL be uniquely identified by the triplet + context identifier: + + context id = <SSRC, destination network address, destination + transport port number> + + where the destination network address and the destination transport + port are the ones in the SRTP packet. It is assumed that, when + presented with this information, the key management returns a context + with the information as described in Section 3.2. + + As noted above, SRTP and SRTCP by default share the bulk of the + parameters in the cryptographic context. Thus, retrieving the crypto + context parameters for an SRTCP stream in practice may imply a + binding to the correspondent SRTP crypto context. It is up to the + implementation to assure such binding, since the RTCP port may not be + + + +Baugher, et al. Standards Track [Page 10] + +RFC 3711 SRTP March 2004 + + + directly deducible from the RTP port only. Alternatively, the key + management may choose to provide separate SRTP- and SRTCP- contexts, + duplicating the common parameters (such as master key(s)). The + latter approach then also enables SRTP and SRTCP to use, e.g., + distinct transforms, if so desired. Similar considerations arise + when multiple SRTP streams, forming part of one single RTP session, + share keys and other parameters. + + If no valid context can be found for a packet corresponding to a + certain context identifier, that packet MUST be discarded. + +3.3. SRTP Packet Processing + + The following applies to SRTP. SRTCP is described in Section 3.4. + + Assuming initialization of the cryptographic context(s) has taken + place via key management, the sender SHALL do the following to + construct an SRTP packet: + + 1. Determine which cryptographic context to use as described in + Section 3.2.3. + + 2. Determine the index of the SRTP packet using the rollover counter, + the highest sequence number in the cryptographic context, and the + sequence number in the RTP packet, as described in Section 3.3.1. + + 3. Determine the master key and master salt. This is done using the + index determined in the previous step or the current MKI in the + cryptographic context, according to Section 8.1. + + 4. Determine the session keys and session salt (if they are used by + the transform) as described in Section 4.3, using master key, + master salt, key_derivation_rate, and session key-lengths in the + cryptographic context with the index, determined in Steps 2 and 3. + + 5. Encrypt the RTP payload to produce the Encrypted Portion of the + packet (see Section 4.1, for the defined ciphers). This step uses + the encryption algorithm indicated in the cryptographic context, + the session encryption key and the session salt (if used) found in + Step 4 together with the index found in Step 2. + + 6. If the MKI indicator is set to one, append the MKI to the packet. + + 7. For message authentication, compute the authentication tag for the + Authenticated Portion of the packet, as described in Section 4.2. + This step uses the current rollover counter, the authentication + + + + + +Baugher, et al. Standards Track [Page 11] + +RFC 3711 SRTP March 2004 + + + algorithm indicated in the cryptographic context, and the session + authentication key found in Step 4. Append the authentication tag + to the packet. + + 8. If necessary, update the ROC as in Section 3.3.1, using the packet + index determined in Step 2. + + To authenticate and decrypt an SRTP packet, the receiver SHALL do the + following: + + 1. Determine which cryptographic context to use as described in + Section 3.2.3. + + 2. Run the algorithm in Section 3.3.1 to get the index of the SRTP + packet. The algorithm uses the rollover counter and highest + sequence number in the cryptographic context with the sequence + number in the SRTP packet, as described in Section 3.3.1. + + 3. Determine the master key and master salt. If the MKI indicator in + the context is set to one, use the MKI in the SRTP packet, + otherwise use the index from the previous step, according to + Section 8.1. + + 4. Determine the session keys, and session salt (if used by the + transform) as described in Section 4.3, using master key, master + salt, key_derivation_rate and session key-lengths in the + cryptographic context with the index, determined in Steps 2 and 3. + + 5. For message authentication and replay protection, first check if + the packet has been replayed (Section 3.3.2), using the Replay + List and the index as determined in Step 2. If the packet is + judged to be replayed, then the packet MUST be discarded, and the + event SHOULD be logged. + + Next, perform verification of the authentication tag, using the + rollover counter from Step 2, the authentication algorithm + indicated in the cryptographic context, and the session + authentication key from Step 4. If the result is "AUTHENTICATION + FAILURE" (see Section 4.2), the packet MUST be discarded from + further processing and the event SHOULD be logged. + + 6. Decrypt the Encrypted Portion of the packet (see Section 4.1, for + the defined ciphers), using the decryption algorithm indicated in + the cryptographic context, the session encryption key and salt (if + used) found in Step 4 with the index from Step 2. + + + + + + +Baugher, et al. Standards Track [Page 12] + +RFC 3711 SRTP March 2004 + + + 7. Update the rollover counter and highest sequence number, s_l, in + the cryptographic context as in Section 3.3.1, using the packet + index estimated in Step 2. If replay protection is provided, also + update the Replay List as described in Section 3.3.2. + + 8. When present, remove the MKI and authentication tag fields from + the packet. + +3.3.1. Packet Index Determination, and ROC, s_l Update + + SRTP implementations use an "implicit" packet index for sequencing, + i.e., not all of the index is explicitly carried in the SRTP packet. + For the pre-defined transforms, the index i is used in replay + protection (Section 3.3.2), encryption (Section 4.1), message + authentication (Section 4.2), and for the key derivation (Section + 4.3). + + When the session starts, the sender side MUST set the rollover + counter, ROC, to zero. Each time the RTP sequence number, SEQ, wraps + modulo 2^16, the sender side MUST increment ROC by one, modulo 2^32 + (see security aspects below). The sender's packet index is then + defined as + + i = 2^16 * ROC + SEQ. + + Receiver-side implementations use the RTP sequence number to + determine the correct index of a packet, which is the location of the + packet in the sequence of all SRTP packets. A robust approach for + the proper use of a rollover counter requires its handling and use to + be well defined. In particular, out-of-order RTP packets with + sequence numbers close to 2^16 or zero must be properly handled. + + The index estimate is based on the receiver's locally maintained ROC + and s_l values. At the setup of the session, the ROC MUST be set to + zero. Receivers joining an on-going session MUST be given the + current ROC value using out-of-band signaling such as key-management + signaling. Furthermore, the receiver SHALL initialize s_l to the RTP + sequence number (SEQ) of the first observed SRTP packet (unless the + initial value is provided by out of band signaling such as key + management). + + On consecutive SRTP packets, the receiver SHOULD estimate the index + as + i = 2^16 * v + SEQ, + + where v is chosen from the set { ROC-1, ROC, ROC+1 } (modulo 2^32) + such that i is closest (in modulo 2^48 sense) to the value 2^16 * ROC + + s_l (see Appendix A for pseudocode). + + + +Baugher, et al. Standards Track [Page 13] + +RFC 3711 SRTP March 2004 + + + After the packet has been processed and authenticated (when enabled + for SRTP packets for the session), the receiver MUST use v to + conditionally update its s_l and ROC variables as follows. If + v=(ROC-1) mod 2^32, then there is no update to s_l or ROC. If v=ROC, + then s_l is set to SEQ if and only if SEQ is larger than the current + s_l; there is no change to ROC. If v=(ROC+1) mod 2^32, then s_l is + set to SEQ and ROC is set to v. + + After a re-keying occurs (changing to a new master key), the rollover + counter always maintains its sequence of values, i.e., it MUST NOT be + reset to zero. + + As the rollover counter is 32 bits long and the sequence number is 16 + bits long, the maximum number of packets belonging to a given SRTP + stream that can be secured with the same key is 2^48 using the pre- + defined transforms. After that number of SRTP packets have been sent + with a given (master or session) key, the sender MUST NOT send any + more packets with that key. (There exists a similar limit for SRTCP, + which in practice may be more restrictive, see Section 9.2.) This + limitation enforces a security benefit by providing an upper bound on + the amount of traffic that can pass before cryptographic keys are + changed. Re-keying (see Section 8.1) MUST be triggered, before this + amount of traffic, and MAY be triggered earlier, e.g., for increased + security and access control to media. Recurring key derivation by + means of a non-zero key_derivation_rate (see Section 4.3), also gives + stronger security but does not change the above absolute maximum + value. + + On the receiver side, there is a caveat to updating s_l and ROC: if + message authentication is not present, neither the initialization of + s_l, nor the ROC update can be made completely robust. The + receiver's "implicit index" approach works for the pre-defined + transforms as long as the reorder and loss of the packets are not too + great and bit-errors do not occur in unfortunate ways. In + particular, 2^15 packets would need to be lost, or a packet would + need to be 2^15 packets out of sequence before synchronization is + lost. Such drastic loss or reorder is likely to disrupt the RTP + application itself. + + The algorithm for the index estimate and ROC update is a matter of + implementation, and should take into consideration the environment + (e.g., packet loss rate) and the cases when synchronization is likely + to be lost, e.g., when the initial sequence number (randomly chosen + by RTP) is not known in advance (not sent in the key management + protocol) but may be near to wrap modulo 2^16. + + + + + + +Baugher, et al. Standards Track [Page 14] + +RFC 3711 SRTP March 2004 + + + A more elaborate and more robust scheme than the one given above is + the handling of RTP's own "rollover counter", see Appendix A.1 of + [RFC3550]. + +3.3.2. Replay Protection + + Secure replay protection is only possible when integrity protection + is present. It is RECOMMENDED to use replay protection, both for RTP + and RTCP, as integrity protection alone cannot assure security + against replay attacks. + + A packet is "replayed" when it is stored by an adversary, and then + re-injected into the network. When message authentication is + provided, SRTP protects against such attacks through a Replay List. + Each SRTP receiver maintains a Replay List, which conceptually + contains the indices of all of the packets which have been received + and authenticated. In practice, the list can use a "sliding window" + approach, so that a fixed amount of storage suffices for replay + protection. Packet indices which lag behind the packet index in the + context by more than SRTP-WINDOW-SIZE can be assumed to have been + received, where SRTP-WINDOW-SIZE is a receiver-side, implementation- + dependent parameter and MUST be at least 64, but which MAY be set to + a higher value. + + The receiver checks the index of an incoming packet against the + replay list and the window. Only packets with index ahead of the + window, or, inside the window but not already received, SHALL be + accepted. + + After the packet has been authenticated (if necessary the window is + first moved ahead), the replay list SHALL be updated with the new + index. + + The Replay List can be efficiently implemented by using a bitmap to + represent which packets have been received, as described in the + Security Architecture for IP [RFC2401]. + +3.4. Secure RTCP + + Secure RTCP follows the definition of Secure RTP. SRTCP adds three + mandatory new fields (the SRTCP index, an "encrypt-flag", and the + authentication tag) and one optional field (the MKI) to the RTCP + packet definition. The three mandatory fields MUST be appended to an + RTCP packet in order to form an equivalent SRTCP packet. The added + fields follow any other profile-specific extensions. + + + + + + +Baugher, et al. Standards Track [Page 15] + +RFC 3711 SRTP March 2004 + + + According to Section 6.1 of [RFC3550], there is a REQUIRED packet + format for compound packets. SRTCP MUST be given packets according + to that requirement in the sense that the first part MUST be a sender + report or a receiver report. However, the RTCP encryption prefix (a + random 32-bit quantity) specified in that Section MUST NOT be used + since, as is stated there, it is only applicable to the encryption + method specified in [RFC3550] and is not needed by the cryptographic + mechanisms used in SRTP. + + 0 1 2 3 + 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 + +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+<+ + |V=2|P| RC | PT=SR or RR | length | | + +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | + | SSRC of sender | | + +>+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+ | + | ~ sender info ~ | + | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | + | ~ report block 1 ~ | + | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | + | ~ report block 2 ~ | + | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | + | ~ ... ~ | + | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | + | |V=2|P| SC | PT=SDES=202 | length | | + | +=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+ | + | | SSRC/CSRC_1 | | + | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | + | ~ SDES items ~ | + | +=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+ | + | ~ ... ~ | + +>+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+ | + | |E| SRTCP index | | + | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+<+ + | ~ SRTCP MKI (OPTIONAL) ~ | + | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | + | : authentication tag : | + | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | + | | + +-- Encrypted Portion Authenticated Portion -----+ + + + Figure 2. An example of the format of a Secure RTCP packet, + consisting of an underlying RTCP compound packet with a Sender Report + and SDES packet. + + + + + + +Baugher, et al. Standards Track [Page 16] + +RFC 3711 SRTP March 2004 + + + The Encrypted Portion of an SRTCP packet consists of the encryption + (Section 4.1) of the RTCP payload of the equivalent compound RTCP + packet, from the first RTCP packet, i.e., from the ninth (9) octet to + the end of the compound packet. The Authenticated Portion of an + SRTCP packet consists of the entire equivalent (eventually compound) + RTCP packet, the E flag, and the SRTCP index (after any encryption + has been applied to the payload). + + The added fields are: + + E-flag: 1 bit, REQUIRED + The E-flag indicates if the current SRTCP packet is + encrypted or unencrypted. Section 9.1 of [RFC3550] allows + the split of a compound RTCP packet into two lower-layer + packets, one to be encrypted and one to be sent in the + clear. The E bit set to "1" indicates encrypted packet, and + "0" indicates non-encrypted packet. + + SRTCP index: 31 bits, REQUIRED + The SRTCP index is a 31-bit counter for the SRTCP packet. + The index is explicitly included in each packet, in contrast + to the "implicit" index approach used for SRTP. The SRTCP + index MUST be set to zero before the first SRTCP packet is + sent, and MUST be incremented by one, modulo 2^31, after + each SRTCP packet is sent. In particular, after a re-key, + the SRTCP index MUST NOT be reset to zero again. + + Authentication Tag: configurable length, REQUIRED + The authentication tag is used to carry message + authentication data. + + MKI: configurable length, OPTIONAL + The MKI is the Master Key Indicator, and functions according + to the MKI definition in Section 3. + + SRTCP uses the cryptographic context parameters and packet processing + of SRTP by default, with the following changes: + + * The receiver does not need to "estimate" the index, as it is + explicitly signaled in the packet. + + * Pre-defined SRTCP encryption is as specified in Section 4.1, but + using the definition of the SRTCP Encrypted Portion given in this + section, and using the SRTCP index as the index i. The encryption + transform and related parameters SHALL by default be the same + selected for the protection of the associated SRTP stream(s), + while the NULL algorithm SHALL be applied to the RTCP packets not + to be encrypted. SRTCP may have a different encryption transform + + + +Baugher, et al. Standards Track [Page 17] + +RFC 3711 SRTP March 2004 + + + than the one used by the corresponding SRTP. The expected use for + this feature is when the former has NULL-encryption and the latter + has a non NULL-encryption. + + The E-flag is assigned a value by the sender depending on whether the + packet was encrypted or not. + + * SRTCP decryption is performed as in Section 4, but only if the E + flag is equal to 1. If so, the Encrypted Portion is decrypted, + using the SRTCP index as the index i. In case the E-flag is 0, + the payload is simply left unmodified. + + * SRTCP replay protection is as defined in Section 3.3.2, but using + the SRTCP index as the index i and a separate Replay List that is + specific to SRTCP. + + * The pre-defined SRTCP authentication tag is specified as in + Section 4.2, but with the Authenticated Portion of the SRTCP + packet given in this section (which includes the index). The + authentication transform and related parameters (e.g., key size) + SHALL by default be the same as selected for the protection of the + associated SRTP stream(s). + + * In the last step of the processing, only the sender needs to + update the value of the SRTCP index by incrementing it modulo 2^31 + and for security reasons the sender MUST also check the number of + SRTCP packets processed, see Section 9.2. + + Message authentication for RTCP is REQUIRED, as it is the control + protocol (e.g., it has a BYE packet) for RTP. + + Precautions must be taken so that the packet expansion in SRTCP (due + to the added fields) does not cause SRTCP messages to use more than + their share of RTCP bandwidth. To avoid this, the following two + measures MUST be taken: + + 1. When initializing the RTCP variable "avg_rtcp_size" defined in + chapter 6.3 of [RFC3550], it MUST include the size of the fields + that will be added by SRTCP (index, E-bit, authentication tag, and + when present, the MKI). + + 2. When updating the "avg_rtcp_size" using the variable "packet_size" + (section 6.3.3 of [RFC3550]), the value of "packet_size" MUST + include the size of the additional fields added by SRTCP. + + + + + + + +Baugher, et al. Standards Track [Page 18] + +RFC 3711 SRTP March 2004 + + + With these measures in place the SRTCP messages will not use more + than the allotted bandwidth. The effect of the size of the added + fields on the SRTCP traffic will be that messages will be sent with + longer packet intervals. The increase in the intervals will be + directly proportional to size of the added fields. For the pre- + defined transforms, the size of the added fields will be at least 14 + octets, and upper bounded depending on MKI and the authentication tag + sizes. + +4. Pre-Defined Cryptographic Transforms + + While there are numerous encryption and message authentication + algorithms that can be used in SRTP, below we define default + algorithms in order to avoid the complexity of specifying the + encodings for the signaling of algorithm and parameter identifiers. + The defined algorithms have been chosen as they fulfill the goals + listed in Section 2. Recommendations on how to extend SRTP with new + transforms are given in Section 6. + +4.1. Encryption + + The following parameters are common to both pre-defined, non-NULL, + encryption transforms specified in this section. + + * BLOCK_CIPHER-MODE indicates the block cipher used and its mode of + operation + * n_b is the bit-size of the block for the block cipher + * k_e is the session encryption key + * n_e is the bit-length of k_e + * k_s is the session salting key + * n_s is the bit-length of k_s + * SRTP_PREFIX_LENGTH is the octet length of the keystream prefix, a + non-negative integer, specified by the message authentication code + in use. + + The distinct session keys and salts for SRTP/SRTCP are by default + derived as specified in Section 4.3. + + The encryption transforms defined in SRTP map the SRTP packet index + and secret key into a pseudo-random keystream segment. Each + keystream segment encrypts a single RTP packet. The process of + encrypting a packet consists of generating the keystream segment + corresponding to the packet, and then bitwise exclusive-oring that + keystream segment onto the payload of the RTP packet to produce the + Encrypted Portion of the SRTP packet. In case the payload size is + not an integer multiple of n_b bits, the excess (least significant) + bits of the keystream are simply discarded. Decryption is done the + same way, but swapping the roles of the plaintext and ciphertext. + + + +Baugher, et al. Standards Track [Page 19] + +RFC 3711 SRTP March 2004 + + + +----+ +------------------+---------------------------------+ + | KG |-->| Keystream Prefix | Keystream Suffix |---+ + +----+ +------------------+---------------------------------+ | + | + +---------------------------------+ v + | Payload of RTP Packet |->(*) + +---------------------------------+ | + | + +---------------------------------+ | + | Encrypted Portion of SRTP Packet|<--+ + +---------------------------------+ + + Figure 3: Default SRTP Encryption Processing. Here KG denotes the + keystream generator, and (*) denotes bitwise exclusive-or. + + The definition of how the keystream is generated, given the index, + depends on the cipher and its mode of operation. Below, two such + keystream generators are defined. The NULL cipher is also defined, + to be used when encryption of RTP is not required. + + The SRTP definition of the keystream is illustrated in Figure 3. The + initial octets of each keystream segment MAY be reserved for use in a + message authentication code, in which case the keystream used for + encryption starts immediately after the last reserved octet. The + initial reserved octets are called the "keystream prefix" (not to be + confused with the "encryption prefix" of [RFC3550, Section 6.1]), and + the remaining octets are called the "keystream suffix". The + keystream prefix MUST NOT be used for encryption. The process is + illustrated in Figure 3. + + The number of octets in the keystream prefix is denoted as + SRTP_PREFIX_LENGTH. The keystream prefix is indicated by a positive, + non-zero value of SRTP_PREFIX_LENGTH. This means that, even if + confidentiality is not to be provided, the keystream generator output + may still need to be computed for packet authentication, in which + case the default keystream generator (mode) SHALL be used. + + The default cipher is the Advanced Encryption Standard (AES) [AES], + and we define two modes of running AES, (1) Segmented Integer Counter + Mode AES and (2) AES in f8-mode. In the remainder of this section, + let E(k,x) be AES applied to key k and input block x. + + + + + + + + + + +Baugher, et al. Standards Track [Page 20] + +RFC 3711 SRTP March 2004 + + +4.1.1. AES in Counter Mode + + Conceptually, counter mode [AES-CTR] consists of encrypting + successive integers. The actual definition is somewhat more + complicated, in order to randomize the starting point of the integer + sequence. Each packet is encrypted with a distinct keystream + segment, which SHALL be computed as follows. + + A keystream segment SHALL be the concatenation of the 128-bit output + blocks of the AES cipher in the encrypt direction, using key k = k_e, + in which the block indices are in increasing order. Symbolically, + each keystream segment looks like + + E(k, IV) || E(k, IV + 1 mod 2^128) || E(k, IV + 2 mod 2^128) ... + + where the 128-bit integer value IV SHALL be defined by the SSRC, the + SRTP packet index i, and the SRTP session salting key k_s, as below. + + IV = (k_s * 2^16) XOR (SSRC * 2^64) XOR (i * 2^16) + + Each of the three terms in the XOR-sum above is padded with as many + leading zeros as needed to make the operation well-defined, + considered as a 128-bit value. + + The inclusion of the SSRC allows the use of the same key to protect + distinct SRTP streams within the same RTP session, see the security + caveats in Section 9.1. + + In the case of SRTCP, the SSRC of the first header of the compound + packet MUST be used, i SHALL be the 31-bit SRTCP index and k_e, k_s + SHALL be replaced by the SRTCP encryption session key and salt. + + Note that the initial value, IV, is fixed for each packet and is + formed by "reserving" 16 zeros in the least significant bits for the + purpose of the counter. The number of blocks of keystream generated + for any fixed value of IV MUST NOT exceed 2^16 to avoid keystream + re-use, see below. The AES has a block size of 128 bits, so 2^16 + output blocks are sufficient to generate the 2^23 bits of keystream + needed to encrypt the largest possible RTP packet (except for IPv6 + "jumbograms" [RFC2675], which are not likely to be used for RTP-based + multimedia traffic). This restriction on the maximum bit-size of the + packet that can be encrypted ensures the security of the encryption + method by limiting the effectiveness of probabilistic attacks [BDJR]. + + For a particular Counter Mode key, each IV value used as an input + MUST be distinct, in order to avoid the security exposure of a two- + time pad situation (Section 9.1). To satisfy this constraint, an + implementation MUST ensure that the combination of the SRTP packet + + + +Baugher, et al. Standards Track [Page 21] + +RFC 3711 SRTP March 2004 + + + index of ROC || SEQ, and the SSRC used in the construction of the IV + are distinct for any particular key. The failure to ensure this + uniqueness could be catastrophic for Secure RTP. This is in contrast + to the situation for RTP itself, which may be able to tolerate such + failures. It is RECOMMENDED that, if a dedicated security module is + present, the RTP sequence numbers and SSRC either be generated or + checked by that module (i.e., sequence-number and SSRC processing in + an SRTP system needs to be protected as well as the key). + +4.1.2. AES in f8-mode + + To encrypt UMTS (Universal Mobile Telecommunications System, as 3G + networks) data, a solution (see [f8-a] [f8-b]) known as the f8- + algorithm has been developed. On a high level, the proposed scheme + is a variant of Output Feedback Mode (OFB) [HAC], with a more + elaborate initialization and feedback function. As in normal OFB, + the core consists of a block cipher. We also define here the use of + AES as a block cipher to be used in what we shall call "f8-mode of + operation" RTP encryption. The AES f8-mode SHALL use the same + default sizes for session key and salt as AES counter mode. + + Figure 4 shows the structure of block cipher, E, running in f8-mode. + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Baugher, et al. Standards Track [Page 22] + +RFC 3711 SRTP March 2004 + + + IV + | + v + +------+ + | | + +--->| E | + | +------+ + | | + m -> (*) +-----------+-------------+-- ... ------+ + | IV' | | | | + | | j=1 -> (*) j=2 -> (*) ... j=L-1 ->(*) + | | | | | + | | +-> (*) +-> (*) ... +-> (*) + | | | | | | | | + | v | v | v | v + | +------+ | +------+ | +------+ | +------+ + k_e ---+--->| E | | | E | | | E | | | E | + | | | | | | | | | | | + +------+ | +------+ | +------+ | +------+ + | | | | | | | + +------+ +--------+ +-- ... ----+ | + | | | | + v v v v + S(0) S(1) S(2) . . . S(L-1) + + Figure 4. f8-mode of operation (asterisk, (*), denotes bitwise XOR). + The figure represents the KG in Figure 3, when AES-f8 is used. + +4.1.2.1. f8 Keystream Generation + + The Initialization Vector (IV) SHALL be determined as described in + Section 4.1.2.2 (and in Section 4.1.2.3 for SRTCP). + + Let IV', S(j), and m denote n_b-bit blocks. The keystream, + S(0) ||... || S(L-1), for an N-bit message SHALL be defined by + setting IV' = E(k_e XOR m, IV), and S(-1) = 00..0. For + j = 0,1,..,L-1 where L = N/n_b (rounded up to nearest integer if it + is not already an integer) compute + + S(j) = E(k_e, IV' XOR j XOR S(j-1)) + + Notice that the IV is not used directly. Instead it is fed through E + under another key to produce an internal, "masked" value (denoted + IV') to prevent an attacker from gaining known input/output pairs. + + + + + + + +Baugher, et al. Standards Track [Page 23] + +RFC 3711 SRTP March 2004 + + + The role of the internal counter, j, is to prevent short keystream + cycles. The value of the key mask m SHALL be + + m = k_s || 0x555..5, + + i.e., the session salting key, appended by the binary pattern 0101.. + to fill out the entire desired key size, n_e. + + The sender SHOULD NOT generate more than 2^32 blocks, which is + sufficient to generate 2^39 bits of keystream. Unlike counter mode, + there is no absolute threshold above (below) which f8 is guaranteed + to be insecure (secure). The above bound has been chosen to limit, + with sufficient security margin, the probability of degenerative + behavior in the f8 keystream generation. + +4.1.2.2. f8 SRTP IV Formation + + The purpose of the following IV formation is to provide a feature + which we call implicit header authentication (IHA), see Section 9.5. + + The SRTP IV for 128-bit block AES-f8 SHALL be formed in the following + way: + + IV = 0x00 || M || PT || SEQ || TS || SSRC || ROC + + M, PT, SEQ, TS, SSRC SHALL be taken from the RTP header; ROC is from + the cryptographic context. + + The presence of the SSRC as part of the IV allows AES-f8 to be used + when a master key is shared between multiple streams within the same + RTP session, see Section 9.1. + +4.1.2.3. f8 SRTCP IV Formation + + The SRTCP IV for 128-bit block AES-f8 SHALL be formed in the + following way: + + IV= 0..0 || E || SRTCP index || V || P || RC || PT || length || SSRC + + where V, P, RC, PT, length, SSRC SHALL be taken from the first header + in the RTCP compound packet. E and SRTCP index are the 1-bit and + 31-bit fields added to the packet. + + + + + + + + + +Baugher, et al. Standards Track [Page 24] + +RFC 3711 SRTP March 2004 + + +4.1.3. NULL Cipher + + The NULL cipher is used when no confidentiality for RTP/RTCP is + requested. The keystream can be thought of as "000..0", i.e., the + encryption SHALL simply copy the plaintext input into the ciphertext + output. + +4.2. Message Authentication and Integrity + + Throughout this section, M will denote data to be integrity + protected. In the case of SRTP, M SHALL consist of the Authenticated + Portion of the packet (as specified in Figure 1) concatenated with + the ROC, M = Authenticated Portion || ROC; in the case of SRTCP, M + SHALL consist of the Authenticated Portion (as specified in Figure 2) + only. + + Common parameters: + + * AUTH_ALG is the authentication algorithm + * k_a is the session message authentication key + * n_a is the bit-length of the authentication key + * n_tag is the bit-length of the output authentication tag + * SRTP_PREFIX_LENGTH is the octet length of the keystream prefix as + defined above, a parameter of AUTH_ALG + + The distinct session authentication keys for SRTP/SRTCP are by + default derived as specified in Section 4.3. + + The values of n_a, n_tag, and SRTP_PREFIX_LENGTH MUST be fixed for + any particular fixed value of the key. + + We describe the process of computing authentication tags as follows. + The sender computes the tag of M and appends it to the packet. The + SRTP receiver verifies a message/authentication tag pair by computing + a new authentication tag over M using the selected algorithm and key, + and then compares it to the tag associated with the received message. + If the two tags are equal, then the message/tag pair is valid; + otherwise, it is invalid and the error audit message "AUTHENTICATION + FAILURE" MUST be returned. + +4.2.1. HMAC-SHA1 + + The pre-defined authentication transform for SRTP is HMAC-SHA1 + [RFC2104]. With HMAC-SHA1, the SRTP_PREFIX_LENGTH (Figure 3) SHALL + be 0. For SRTP (respectively SRTCP), the HMAC SHALL be applied to + the session authentication key and M as specified above, i.e., + HMAC(k_a, M). The HMAC output SHALL then be truncated to the n_tag + left-most bits. + + + +Baugher, et al. Standards Track [Page 25] + +RFC 3711 SRTP March 2004 + + +4.3. Key Derivation + +4.3.1. Key Derivation Algorithm + + Regardless of the encryption or message authentication transform that + is employed (it may be an SRTP pre-defined transform or newly + introduced according to Section 6), interoperable SRTP + implementations MUST use the SRTP key derivation to generate session + keys. Once the key derivation rate is properly signaled at the start + of the session, there is no need for extra communication between the + parties that use SRTP key derivation. + + packet index ---+ + | + v + +-----------+ master +--------+ session encr_key + | ext | key | |----------> + | key mgmt |-------->| key | session auth_key + | (optional | | deriv |----------> + | rekey) |-------->| | session salt_key + | | master | |----------> + +-----------+ salt +--------+ + + Figure 5: SRTP key derivation. + + At least one initial key derivation SHALL be performed by SRTP, i.e., + the first key derivation is REQUIRED. Further applications of the + key derivation MAY be performed, according to the + "key_derivation_rate" value in the cryptographic context. The key + derivation function SHALL initially be invoked before the first + packet and then, when r > 0, a key derivation is performed whenever + index mod r equals zero. This can be thought of as "refreshing" the + session keys. The value of "key_derivation_rate" MUST be kept fixed + for the lifetime of the associated master key. + + Interoperable SRTP implementations MAY also derive session salting + keys for encryption transforms, as is done in both of the pre- + defined transforms. + + Let m and n be positive integers. A pseudo-random function family is + a set of keyed functions {PRF_n(k,x)} such that for the (secret) + random key k, given m-bit x, PRF_n(k,x) is an n-bit string, + computationally indistinguishable from random n-bit strings, see + [HAC]. For the purpose of key derivation in SRTP, a secure PRF with + m = 128 (or more) MUST be used, and a default PRF transform is + defined in Section 4.3.3. + + + + + +Baugher, et al. Standards Track [Page 26] + +RFC 3711 SRTP March 2004 + + + Let "a DIV t" denote integer division of a by t, rounded down, and + with the convention that "a DIV 0 = 0" for all a. We also make the + convention of treating "a DIV t" as a bit string of the same length + as a, and thus "a DIV t" will in general have leading zeros. + + Key derivation SHALL be defined as follows in terms of <label>, an + 8-bit constant (see below), master_salt and key_derivation_rate, as + determined in the cryptographic context, and index, the packet index + (i.e., the 48-bit ROC || SEQ for SRTP): + + * Let r = index DIV key_derivation_rate (with DIV as defined above). + + * Let key_id = <label> || r. + + * Let x = key_id XOR master_salt, where key_id and master_salt are + aligned so that their least significant bits agree (right- + alignment). + + <label> MUST be unique for each type of key to be derived. We + currently define <label> 0x00 to 0x05 (see below), and future + extensions MAY specify new values in the range 0x06 to 0xff for other + purposes. The n-bit SRTP key (or salt) for this packet SHALL then be + derived from the master key, k_master as follows: + + PRF_n(k_master, x). + + (The PRF may internally specify additional formatting and padding of + x, see e.g., Section 4.3.3 for the default PRF.) + + The session keys and salt SHALL now be derived using: + + - k_e (SRTP encryption): <label> = 0x00, n = n_e. + + - k_a (SRTP message authentication): <label> = 0x01, n = n_a. + + - k_s (SRTP salting key): <label> = 0x02, n = n_s. + + where n_e, n_s, and n_a are from the cryptographic context. + + The master key and master salt MUST be random, but the master salt + MAY be public. + + Note that for a key_derivation_rate of 0, the application of the key + derivation SHALL take place exactly once. + + The definition of DIV above is purely for notational convenience. + For a non-zero t among the set of allowed key derivation rates, "a + DIV t" can be implemented as a right-shift by the base-2 logarithm of + + + +Baugher, et al. Standards Track [Page 27] + +RFC 3711 SRTP March 2004 + + + t. The derivation operation is further facilitated if the rates are + chosen to be powers of 256, but that granularity was considered too + coarse to be a requirement of this specification. + + The upper limit on the number of packets that can be secured using + the same master key (see Section 9.2) is independent of the key + derivation. + +4.3.2. SRTCP Key Derivation + + SRTCP SHALL by default use the same master key (and master salt) as + SRTP. To do this securely, the following changes SHALL be done to + the definitions in Section 4.3.1 when applying session key derivation + for SRTCP. + + Replace the SRTP index by the 32-bit quantity: 0 || SRTCP index + (i.e., excluding the E-bit, replacing it with a fixed 0-bit), and use + <label> = 0x03 for the SRTCP encryption key, <label> = 0x04 for the + SRTCP authentication key, and, <label> = 0x05 for the SRTCP salting + key. + +4.3.3. AES-CM PRF + + The currently defined PRF, keyed by 128, 192, or 256 bit master key, + has input block size m = 128 and can produce n-bit outputs for n up + to 2^23. PRF_n(k_master,x) SHALL be AES in Counter Mode as described + in Section 4.1.1, applied to key k_master, and IV equal to (x*2^16), + and with the output keystream truncated to the n first (left-most) + bits. (Requiring n/128, rounded up, applications of AES.) + +5. Default and mandatory-to-implement Transforms + + The default transforms also are mandatory-to-implement transforms in + SRTP. Of course, "mandatory-to-implement" does not imply + "mandatory-to-use". Table 1 summarizes the pre-defined transforms. + The default values below are valid for the pre-defined transforms. + + mandatory-to-impl. optional default + + encryption AES-CM, NULL AES-f8 AES-CM + message integrity HMAC-SHA1 - HMAC-SHA1 + key derivation (PRF) AES-CM - AES-CM + + Table 1: Mandatory-to-implement, optional and default transforms in + SRTP and SRTCP. + + + + + + +Baugher, et al. Standards Track [Page 28] + +RFC 3711 SRTP March 2004 + + +5.1. Encryption: AES-CM and NULL + + AES running in Segmented Integer Counter Mode, as defined in Section + 4.1.1, SHALL be the default encryption algorithm. The default key + lengths SHALL be 128-bit for the session encryption key (n_e). The + default session salt key-length (n_s) SHALL be 112 bits. + + The NULL cipher SHALL also be mandatory-to-implement. + +5.2. Message Authentication/Integrity: HMAC-SHA1 + + HMAC-SHA1, as defined in Section 4.2.1, SHALL be the default message + authentication code. The default session authentication key-length + (n_a) SHALL be 160 bits, the default authentication tag length + (n_tag) SHALL be 80 bits, and the SRTP_PREFIX_LENGTH SHALL be zero + for HMAC-SHA1. In addition, for SRTCP, the pre-defined HMAC-SHA1 + MUST NOT be applied with a value of n_tag, nor n_a, that are smaller + than these defaults. For SRTP, smaller values are NOT RECOMMENDED, + but MAY be used after careful consideration of the issues in Section + 7.5 and 9.5. + +5.3. Key Derivation: AES-CM PRF + + The AES Counter Mode based key derivation and PRF defined in Sections + 4.3.1 to 4.3.3, using a 128-bit master key, SHALL be the default + method for generating session keys. The default master salt length + SHALL be 112 bits and the default key-derivation rate SHALL be zero. + +6. Adding SRTP Transforms + + Section 4 provides examples of the level of detail needed for + defining transforms. Whenever a new transform is to be added to + SRTP, a companion standard track RFC MUST be written to exactly + define how the new transform can be used with SRTP (and SRTCP). Such + a companion RFC SHOULD avoid overlap with the SRTP protocol document. + Note however, that it MAY be necessary to extend the SRTP or SRTCP + cryptographic context definition with new parameters (including fixed + or default values), add steps to the packet processing, or even add + fields to the SRTP/SRTCP packets. The companion RFC SHALL explain + any known issues regarding interactions between the transform and + other aspects of SRTP. + + Each new transform document SHOULD specify its key attributes, e.g., + size of keys (minimum, maximum, recommended), format of keys, + recommended/required processing of input keying material, + requirements/recommendations on key lifetime, re-keying and key + derivation, whether sharing of keys between SRTP and SRTCP is allowed + or not, etc. + + + +Baugher, et al. Standards Track [Page 29] + +RFC 3711 SRTP March 2004 + + + An added message integrity transform SHOULD define a minimum + acceptable key/tag size for SRTCP, equivalent in strength to the + minimum values as defined in Section 5.2. + +7. Rationale + + This section explains the rationale behind several important features + of SRTP. + +7.1. Key derivation + + Key derivation reduces the burden on the key establishment. As many + as six different keys are needed per crypto context (SRTP and SRTCP + encryption keys and salts, SRTP and SRTCP authentication keys), but + these are derived from a single master key in a cryptographically + secure way. Thus, the key management protocol needs to exchange only + one master key (plus master salt when required), and then SRTP itself + derives all the necessary session keys (via the first, mandatory + application of the key derivation function). + + Multiple applications of the key derivation function are optional, + but will give security benefits when enabled. They prevent an + attacker from obtaining large amounts of ciphertext produced by a + single fixed session key. If the attacker was able to collect a + large amount of ciphertext for a certain session key, he might be + helped in mounting certain attacks. + + Multiple applications of the key derivation function provide + backwards and forward security in the sense that a compromised + session key does not compromise other session keys derived from the + same master key. This means that the attacker who is able to recover + a certain session key, is anyway not able to have access to messages + secured under previous and later session keys (derived from the same + master key). (Note that, of course, a leaked master key reveals all + the session keys derived from it.) + + Considerations arise with high-rate key refresh, especially in large + multicast settings, see Section 11. + +7.2. Salting key + + The master salt guarantees security against off-line key-collision + attacks on the key derivation that might otherwise reduce the + effective key size [MF00]. + + + + + + + +Baugher, et al. Standards Track [Page 30] + +RFC 3711 SRTP March 2004 + + + The derived session salting key used in the encryption, has been + introduced to protect against some attacks on additive stream + ciphers, see Section 9.2. The explicit inclusion method of the salt + in the IV has been selected for ease of hardware implementation. + +7.3. Message Integrity from Universal Hashing + + The particular definition of the keystream given in Section 4.1 (the + keystream prefix) is to give provision for particular universal hash + functions, suitable for message authentication in the Wegman-Carter + paradigm [WC81]. Such functions are provably secure, simple, quick, + and especially appropriate for Digital Signal Processors and other + processors with a fast multiply operation. + + No authentication transforms are currently provided in SRTP other + than HMAC-SHA1. Future transforms, like the above mentioned + universal hash functions, MAY be added following the guidelines in + Section 6. + +7.4. Data Origin Authentication Considerations + + Note that in pair-wise communications, integrity and data origin + authentication are provided together. However, in group scenarios + where the keys are shared between members, the MAC tag only proves + that a member of the group sent the packet, but does not prevent + against a member impersonating another. Data origin authentication + (DOA) for multicast and group RTP sessions is a hard problem that + needs a solution; while some promising proposals are being + investigated [PCST1] [PCST2], more work is needed to rigorously + specify these technologies. Thus SRTP data origin authentication in + groups is for further study. + + DOA can be done otherwise using signatures. However, this has high + impact in terms of bandwidth and processing time, therefore we do not + offer this form of authentication in the pre-defined packet-integrity + transform. + + The presence of mixers and translators does not allow data origin + authentication in case the RTP payload and/or the RTP header are + manipulated. Note that these types of middle entities also disrupt + end-to-end confidentiality (as the IV formation depends e.g., on the + RTP header preservation). A certain trust model may choose to trust + the mixers/translators to decrypt/re-encrypt the media (this would + imply breaking the end-to-end security, with related security + implications). + + + + + + +Baugher, et al. Standards Track [Page 31] + +RFC 3711 SRTP March 2004 + + +7.5. Short and Zero-length Message Authentication + + As shown in Figure 1, the authentication tag is RECOMMENDED in SRTP. + A full 80-bit authentication-tag SHOULD be used, but a shorter tag or + even a zero-length tag (i.e., no message authentication) MAY be used + under certain conditions to support either of the following two + application environments. + + 1. Strong authentication can be impractical in environments where + bandwidth preservation is imperative. An important special + case is wireless communication systems, in which bandwidth is a + scarce and expensive resource. Studies have shown that for + certain applications and link technologies, additional bytes + may result in a significant decrease in spectrum efficiency + [SWO]. Considerable effort has been made to design IP header + compression techniques to improve spectrum efficiency + [RFC3095]. A typical voice application produces 20 byte + samples, and the RTP, UDP and IP headers need to be jointly + compressed to one or two bytes on average in order to obtain + acceptable wireless bandwidth economy [RFC3095]. In this case, + strong authentication would impose nearly fifty percent + overhead. + + 2. Authentication is impractical for applications that use data + links with fixed-width fields that cannot accommodate the + expansion due to the authentication tag. This is the case for + some important existing wireless channels. For example, zero- + byte header compression is used to adapt EVRC/SMV voice with + the legacy IS-95 bearer channel in CDMA2000 VoIP services. It + was found that not a single additional octet could be added to + the data, which motivated the creation of a zero-byte profile + for ROHC [RFC3242]. + + A short tag is secure for a restricted set of applications. Consider + a voice telephony application, for example, such as a G.729 audio + codec with a 20-millisecond packetization interval, protected by a + 32-bit message authentication tag. The likelihood of any given + packet being successfully forged is only one in 2^32. Thus an + adversary can control no more than 20 milliseconds of audio output + during a 994-day period, on average. In contrast, the effect of a + single forged packet can be much larger if the application is + stateful. A codec that uses relative or predictive compression + across packets will propagate the maliciously generated state, + affecting a longer duration of output. + + + + + + + +Baugher, et al. Standards Track [Page 32] + +RFC 3711 SRTP March 2004 + + + Certainly not all SRTP or telephony applications meet the criteria + for short or zero-length authentication tags. Section 9.5.1 + discusses the risks of weak or no message authentication, and section + 9.5 describes the circumstances when it is acceptable and when it is + unacceptable. + +8. Key Management Considerations + + There are emerging key management standards [MIKEY] [KEYMGT] [SDMS] + for establishing an SRTP cryptographic context (e.g., an SRTP master + key). Both proprietary and open-standard key management methods are + likely to be used for telephony applications [MIKEY] [KINK] and + multicast applications [GDOI]. This section provides guidance for + key management systems that service SRTP session. + + For initialization, an interoperable SRTP implementation SHOULD be + given the SSRC and MAY be given the initial RTP sequence number for + the RTP stream by key management (thus, key management has a + dependency on RTP operational parameters). Sending the RTP sequence + number in the key management may be useful e.g., when the initial + sequence number is close to wrapping (to avoid synchronization + problems), and to communicate the current sequence number to a + joining endpoint (to properly initialize its replay list). + + If the pre-defined transforms are used, SRTP allows sharing of the + same master key between SRTP/SRTCP streams belonging to the same RTP + session. + + First, sharing between SRTP streams belonging to the same RTP session + is secure if the design of the synchronization mechanism, i.e., the + IV, avoids keystream re-use (the two-time pad, Section 9.1). This is + taken care of by the fact that RTP provides for unique SSRCs for + streams belonging to the same RTP session. See Section 9.1 for + further discussion. + + Second, sharing between SRTP and the corresponding SRTCP is secure. + The fact that an SRTP stream and its associated SRTCP stream both + carry the same SSRC does not constitute a problem for the two-time + pad due to the key derivation. Thus, SRTP and SRTCP corresponding to + one RTP session MAY share master keys (as they do by default). + + Note that message authentication also has a dependency on SSRC + uniqueness that is unrelated to the problem of keystream reuse: SRTP + streams authenticated under the same key MUST have a distinct SSRC in + order to identify the sender of the message. This requirement is + needed because the SSRC is the cryptographically authenticated field + + + + + +Baugher, et al. Standards Track [Page 33] + +RFC 3711 SRTP March 2004 + + + used to distinguish between different SRTP streams. Were two streams + to use identical SSRC values, then an adversary could substitute + messages from one stream into the other without detection. + + SRTP/SRTCP MUST NOT share master keys under any other circumstances + than the ones given above, i.e., between SRTP and its corresponding + SRTCP, and, between streams belonging to the same RTP session. + +8.1. Re-keying + + The recommended way for a particular key management system to provide + re-key within SRTP is by associating a master key in a crypto context + with an MKI. + + This provides for easy master key retrieval (see Scenarios in Section + 11), but has the disadvantage of adding extra bits to each packet. + As noted in Section 7.5, some wireless links do not cater for added + bits, therefore SRTP also defines a more economic way of triggering + re-keying, via use of <From, To>, which works in some specific, + simple scenarios (see Section 8.1.1). + + SRTP senders SHALL count the amount of SRTP and SRTCP traffic being + used for a master key and invoke key management to re-key if needed + (Section 9.2). These interactions are defined by the key management + interface to SRTP and are not defined by this protocol specification. + +8.1.1. Use of the <From, To> for re-keying + + In addition to the use of the MKI, SRTP defines another optional + mechanism for master key retrieval, the <From, To>. The <From, To> + specifies the range of SRTP indices (a pair of sequence number and + ROC) within which a certain master key is valid, and is (when used) + part of the crypto context. By looking at the 48-bit SRTP index of + the current SRTP packet, the corresponding master key can be found by + determining which From-To interval it belongs to. For SRTCP, the + most recently observed/used SRTP index (which can be obtained from + the cryptographic context) is used for this purpose, even though + SRTCP has its own (31-bit) index (see caveat below). + + This method, compared to the MKI, has the advantage of identifying + the master key and defining its lifetime without adding extra bits to + each packet. This could be useful, as already noted, for some + wireless links that do not cater for added bits. However, its use + SHOULD be limited to specific, very simple scenarios. We recommend + to limit its use when the RTP session is a simple unidirectional or + bi-directional stream. This is because in case of multiple streams, + it is difficult to trigger the re-key based on the <From, To> of a + single RTP stream. For example, if several streams share a master + + + +Baugher, et al. Standards Track [Page 34] + +RFC 3711 SRTP March 2004 + + + key, there is no simple one-to-one correspondence between the index + sequence space of a certain stream, and the index sequence space on + which the <From, To> values are based. Consequently, when a master + key is shared between streams, one of these streams MUST be + designated by key management as the one whose index space defines the + re-keying points. Also, the re-key triggering on SRTCP is based on + the correspondent SRTP stream, i.e., when the SRTP stream changes the + master key, so does the correspondent SRTCP. This becomes obviously + more and more complex with multiple streams. + + The default values for the <From, To> are "from the first observed + packet" and "until further notice". However, the maximum limit of + SRTP/SRTCP packets that are sent under each given master/session key + (Section 9.2) MUST NOT be exceeded. + + In case the <From, To> is used as key retrieval, then the MKI is not + inserted in the packet (and its indicator in the crypto context is + zero). However, using the MKI does not exclude using <From, To> key + lifetime simultaneously. This can for instance be useful to signal + at the sender side at which point in time an MKI is to be made + active. + +8.2. Key Management parameters + + The table below lists all SRTP parameters that key management can + supply. For reference, it also provides a summary of the default and + mandatory-to-support values for an SRTP implementation as described + in Section 5. + + + + + + + + + + + + + + + + + + + + + + + +Baugher, et al. Standards Track [Page 35] + +RFC 3711 SRTP March 2004 + + + Parameter Mandatory-to-support Default + --------- -------------------- ------- + + SRTP and SRTCP encr transf. AES_CM, NULL AES_CM + (Other possible values: AES_f8) + + SRTP and SRTCP auth transf. HMAC-SHA1 HMAC-SHA1 + + SRTP and SRTCP auth params: + n_tag (tag length) 80 80 + SRTP prefix_length 0 0 + + Key derivation PRF AES_CM AES_CM + + Key material params + (for each master key): + master key length 128 128 + n_e (encr session key length) 128 128 + n_a (auth session key length) 160 160 + master salt key + length of the master salt 112 112 + n_s (session salt key length) 112 112 + key derivation rate 0 0 + + key lifetime + SRTP-packets-max-lifetime 2^48 2^48 + SRTCP-packets-max-lifetime 2^31 2^31 + from-to-lifetime <From, To> + MKI indicator 0 0 + length of the MKI 0 0 + value of the MKI + + Crypto context index params: + SSRC value + ROC + SEQ + SRTCP Index + Transport address + Port number + + Relation to other RTP profiles: + sender's order between FEC and SRTP FEC-SRTP FEC-SRTP + (see Section 10) + + + + + + + + +Baugher, et al. Standards Track [Page 36] + +RFC 3711 SRTP March 2004 + + +9. Security Considerations + +9.1. SSRC collision and two-time pad + + Any fixed keystream output, generated from the same key and index + MUST only be used to encrypt once. Re-using such keystream (jokingly + called a "two-time pad" system by cryptographers), can seriously + compromise security. The NSA's VENONA project [C99] provides a + historical example of such a compromise. It is REQUIRED that + automatic key management be used for establishing and maintaining + SRTP and SRTCP keying material; this requirement is to avoid + keystream reuse, which is more likely to occur with manual key + management. Furthermore, in SRTP, a "two-time pad" is avoided by + requiring the key, or some other parameter of cryptographic + significance, to be unique per RTP/RTCP stream and packet. The pre- + defined SRTP transforms accomplish packet-uniqueness by including the + packet index and stream-uniqueness by inclusion of the SSRC. + + The pre-defined transforms (AES-CM and AES-f8) allow master keys to + be shared across streams belonging to the same RTP session by the + inclusion of the SSRC in the IV. A master key MUST NOT be shared + among different RTP sessions. + + Thus, the SSRC MUST be unique between all the RTP streams within the + same RTP session that share the same master key. RTP itself provides + an algorithm for detecting SSRC collisions within the same RTP + session. Thus, temporary collisions could lead to temporary two-time + pad, in the unfortunate event that SSRCs collide at a point in time + when the streams also have identical sequence numbers (occurring with + probability roughly 2^(-48)). Therefore, the key management SHOULD + take care of avoiding such SSRC collisions by including the SSRCs to + be used in the session as negotiation parameters, proactively + assuring their uniqueness. This is a strong requirements in + scenarios where for example, there are multiple senders that can + start to transmit simultaneously, before SSRC collision are detected + at the RTP level. + + Note also that even with distinct SSRCs, extensive use of the same + key might improve chances of probabilistic collision and time- + memory-tradeoff attacks succeeding. + + As described, master keys MAY be shared between streams belonging to + the same RTP session, but it is RECOMMENDED that each SSRC have its + own master key. When master keys are shared among SSRC participants + and SSRCs are managed by a key management module as recommended + above, the RECOMMENDED policy for an SSRC collision error is for the + participant to leave the SRTP session as it is a sign of malfunction. + + + + +Baugher, et al. Standards Track [Page 37] + +RFC 3711 SRTP March 2004 + + +9.2. Key Usage + + The effective key size is determined (upper bounded) by the size of + the master key and, for encryption, the size of the salting key. Any + additive stream cipher is vulnerable to attacks that use statistical + knowledge about the plaintext source to enable key collision and + time-memory tradeoff attacks [MF00] [H80] [BS00]. These attacks take + advantage of commonalities among plaintexts, and provide a way for a + cryptanalyst to amortize the computational effort of decryption over + many keys, or over many bytes of output, thus reducing the effective + key size of the cipher. A detailed analysis of these attacks and + their applicability to the encryption of Internet traffic is provided + in [MF00]. In summary, the effective key size of SRTP when used in a + security system in which m distinct keys are used, is equal to the + key size of the cipher less the logarithm (base two) of m. + Protection against such attacks can be provided simply by increasing + the size of the keys used, which here can be accomplished by the use + of the salting key. Note that the salting key MUST be random but MAY + be public. A salt size of (the suggested) size 112 bits protects + against attacks in scenarios where at most 2^112 keys are in use. + This is sufficient for all practical purposes. + + Implementations SHOULD use keys that are as large as possible. + Please note that in many cases increasing the key size of a cipher + does not affect the throughput of that cipher. + + The use of the SRTP and SRTCP indices in the pre-defined transforms + fixes the maximum number of packets that can be secured with the same + key. This limit is fixed to 2^48 SRTP packets for an SRTP stream, + and 2^31 SRTCP packets, when SRTP and SRTCP are considered + independently. Due to for example re-keying, reaching this limit may + or may not coincide with wrapping of the indices, and thus the sender + MUST keep packet counts. However, when the session keys for related + SRTP and SRTCP streams are derived from the same master key (the + default behavior, Section 4.3), the upper bound that has to be + considered is in practice the minimum of the two quantities. That + is, when 2^48 SRTP packets or 2^31 SRTCP packets have been secured + with the same key (whichever occurs before), the key management MUST + be called to provide new master key(s) (previously stored and used + keys MUST NOT be used again), or the session MUST be terminated. If + a sender of RTCP discovers that the sender of SRTP (or SRTCP) has not + updated the master or session key prior to sending 2^48 SRTP (or 2^31 + SRTCP) packets belonging to the same SRTP (SRTCP) stream, it is up to + the security policy of the RTCP sender how to behave, e.g., whether + an RTCP BYE-packet should be sent and/or if the event should be + logged. + + + + + +Baugher, et al. Standards Track [Page 38] + +RFC 3711 SRTP March 2004 + + + Note: in most typical applications (assuming at least one RTCP packet + for every 128,000 RTP packets), it will be the SRTCP index that first + reaches the upper limit, although the time until this occurs is very + long: even at 200 SRTCP packets/sec, the 2^31 index space of SRTCP is + enough to secure approximately 4 months of communication. + + Note that if the master key is to be shared between SRTP streams + within the same RTP session (Section 9.1), although the above bounds + are on a per stream (i.e., per SSRC) basis, the sender MUST base re- + key decision on the stream whose sequence number space is the first + to be exhausted. + + Key derivation limits the amount of plaintext that is encrypted with + a fixed session key, and made available to an attacker for analysis, + but key derivation does not extend the master key's lifetime. To see + this, simply consider our requirements to avoid two-time pad: two + distinct packets MUST either be processed with distinct IVs, or with + distinct session keys, and both the distinctness of IV and of the + session keys are (for the pre-defined transforms) dependent on the + distinctness of the packet indices. + + Note that with the key derivation, the effective key size is at most + that of the master key, even if the derived session key is + considerably longer. With the pre-defined authentication transform, + the session authentication key is 160 bits, but the master key by + default is only 128 bits. This design choice was made to comply with + certain recommendations in [RFC2104] so that an existing HMAC + implementation can be plugged into SRTP without problems. Since the + default tag size is 80 bits, it is, for the applications in mind, + also considered acceptable from security point of view. Users having + concerns about this are RECOMMENDED to instead use a 192 bit master + key in the key derivation. It was, however, chosen not to mandate + 192-bit keys since existing AES implementations to be used in the + key-derivation may not always support key-lengths other than 128 + bits. Since AES is not defined (or properly analyzed) for use with + 160 bit keys it is NOT RECOMMENDED that ad-hoc key-padding schemes + are used to pad shorter keys to 192 or 256 bits. + +9.3. Confidentiality of the RTP Payload + + SRTP's pre-defined ciphers are "seekable" stream ciphers, i.e., + ciphers able to efficiently seek to arbitrary locations in their + keystream (so that the encryption or decryption of one packet does + not depend on preceding packets). By using seekable stream ciphers, + SRTP avoids the denial of service attacks that are possible on stream + ciphers that lack this property. It is important to be aware that, + as with any stream cipher, the exact length of the payload is + revealed by the encryption. This means that it may be possible to + + + +Baugher, et al. Standards Track [Page 39] + +RFC 3711 SRTP March 2004 + + + deduce certain "formatting bits" of the payload, as the length of the + codec output might vary due to certain parameter settings etc. This, + in turn, implies that the corresponding bit of the keystream can be + deduced. However, if the stream cipher is secure (counter mode and + f8 are provably secure under certain assumptions [BDJR] [KSYH] [IK]), + knowledge of a few bits of the keystream will not aid an attacker in + predicting subsequent keystream bits. Thus, the payload length (and + information deducible from this) will leak, but nothing else. + + As some RTP packet could contain highly predictable data, e.g., SID, + it is important to use a cipher designed to resist known plaintext + attacks (which is the current practice). + +9.4. Confidentiality of the RTP Header + + In SRTP, RTP headers are sent in the clear to allow for header + compression. This means that data such as payload type, + synchronization source identifier, and timestamp are available to an + eavesdropper. Moreover, since RTP allows for future extensions of + headers, we cannot foresee what kind of possibly sensitive + information might also be "leaked". + + SRTP is a low-cost method, which allows header compression to reduce + bandwidth. It is up to the endpoints' policies to decide about the + security protocol to employ. If one really needs to protect headers, + and is allowed to do so by the surrounding environment, then one + should also look at alternatives, e.g., IPsec [RFC2401]. + +9.5. Integrity of the RTP payload and header + + SRTP messages are subject to attacks on their integrity and source + identification, and these risks are discussed in Section 9.5.1. To + protect against these attacks, each SRTP stream SHOULD be protected + by HMAC-SHA1 [RFC2104] with an 80-bit output tag and a 160-bit key, + or a message authentication code with equivalent strength. Secure + RTP SHOULD NOT be used without message authentication, except under + the circumstances described in this section. It is important to note + that encryption algorithms, including AES Counter Mode and f8, do not + provide message authentication. SRTCP MUST NOT be used with weak (or + NULL) authentication. + + SRTP MAY be used with weak authentication (e.g., a 32-bit + authentication tag), or with no authentication (the NULL + authentication algorithm). These options allow SRTP to be used to + provide confidentiality in situations where + + * weak or null authentication is an acceptable security risk, and + * it is impractical to provide strong message authentication. + + + +Baugher, et al. Standards Track [Page 40] + +RFC 3711 SRTP March 2004 + + + These conditions are described below and in Section 7.5. Note that + both conditions MUST hold in order for weak or null authentication to + be used. The risks associated with exercising the weak or null + authentication options need to be considered by a security audit + prior to their use for a particular application or environment given + the risks, which are discussed in Section 9.5.1. + + Weak authentication is acceptable when the RTP application is such + that the effect of a small fraction of successful forgeries is + negligible. If the application is stateless, then the effect of a + single forged RTP packet is limited to the decoding of that + particular packet. Under this condition, the size of the + authentication tag MUST ensure that only a negligible fraction of the + packets passed to the RTP application by the SRTP receiver can be + forgeries. This fraction is negligible when an adversary, if given + control of the forged packets, is not able to make a significant + impact on the output of the RTP application (see the example of + Section 7.5). + + Weak or null authentication MAY be acceptable when it is unlikely + that an adversary can modify ciphertext so that it decrypts to an + intelligible value. One important case is when it is difficult for + an adversary to acquire the RTP plaintext data, since for many + codecs, an adversary that does not know the input signal cannot + manipulate the output signal in a controlled way. In many cases it + may be difficult for the adversary to determine the actual value of + the plaintext. For example, a hidden snooping device might be + required in order to know a live audio or video signal. The + adversary's signal must have a quality equivalent to or greater than + that of the signal under attack, since otherwise the adversary would + not have enough information to encode that signal with the codec used + by the victim. Plaintext prediction may also be especially difficult + for an interactive application such as a telephone call. + + Weak or null authentication MUST NOT be used when the RTP application + makes data forwarding or access control decisions based on the RTP + data. In such a case, an attacker may be able to subvert + confidentiality by causing the receiver to forward data to an + attacker. See Section 3 of [B96] for a real-life example of such + attacks. + + Null authentication MUST NOT be used when a replay attack, in which + an adversary stores packets then replays them later in the session, + could have a non-negligible impact on the receiver. An example of a + successful replay attack is the storing of the output of a + surveillance camera for a period of time, later followed by the + + + + + +Baugher, et al. Standards Track [Page 41] + +RFC 3711 SRTP March 2004 + + + injection of that output to the monitoring station to avoid + surveillance. Encryption does not protect against this attack, and + non-null authentication is REQUIRED in order to defeat it. + + If existential message forgery is an issue, i.e., when the accuracy + of the received data is of non-negligible importance, null + authentication MUST NOT be used. + +9.5.1. Risks of Weak or Null Message Authentication + + During a security audit considering the use of weak or null + authentication, it is important to keep in mind the following attacks + which are possible when no message authentication algorithm is used. + + An attacker who cannot predict the plaintext is still always able to + modify the message sent between the sender and the receiver so that + it decrypts to a random plaintext value, or to send a stream of bogus + packets to the receiver that will decrypt to random plaintext values. + This attack is essentially a denial of service attack, though in the + absence of message authentication, the RTP application will have + inputs that are bit-wise correlated with the true value. Some + multimedia codecs and common operating systems will crash when such + data are accepted as valid video data. This denial of service attack + may be a much larger threat than that due to an attacker dropping, + delaying, or re-ordering packets. + + An attacker who cannot predict the plaintext can still replay a + previous message with certainty that the receiver will accept it. + Applications with stateless codecs might be robust against this type + of attack, but for other, more complex applications these attacks may + be far more grave. + + An attacker who can predict the plaintext can modify the ciphertext + so that it will decrypt to any value of her choosing. With an + additive stream cipher, an attacker will always be able to change + individual bits. + + An attacker may be able to subvert confidentiality due to the lack of + authentication when a data forwarding or access control decision is + made on decrypted but unauthenticated plaintext. This is because the + receiver may be fooled into forwarding data to an attacker, leading + to an indirect breach of confidentiality (see Section 3 of [B96]). + This is because data-forwarding decisions are made on the decrypted + plaintext; information in the plaintext will determine to what subnet + (or process) the plaintext is forwarded in ESP [RFC2401] tunnel mode + (respectively, transport mode). When Secure RTP is used without + + + + + +Baugher, et al. Standards Track [Page 42] + +RFC 3711 SRTP March 2004 + + + message authentication, it should be verified that the application + does not make data forwarding or access control decisions based on + the decrypted plaintext. + + Some cipher modes of operation that require padding, e.g., standard + cipher block chaining (CBC) are very sensitive to attacks on + confidentiality if certain padding types are used in the absence of + integrity. The attack [V02] shows that this is indeed the case for + the standard RTP padding as discussed in reference to Figure 1, when + used together with CBC mode. Later transform additions to SRTP MUST + therefore carefully consider the risk of using this padding without + proper integrity protection. + +9.5.2. Implicit Header Authentication + + The IV formation of the f8-mode gives implicit authentication (IHA) + of the RTP header, even when message authentication is not used. + When IHA is used, an attacker that modifies the value of the RTP + header will cause the decryption process at the receiver to produce + random plaintext values. While this protection is not equivalent to + message authentication, it may be useful for some applications. + +10. Interaction with Forward Error Correction mechanisms + + The default processing when using Forward Error Correction (e.g., RFC + 2733) processing with SRTP SHALL be to perform FEC processing prior + to SRTP processing on the sender side and to perform SRTP processing + prior to FEC processing on the receiver side. Any change to this + ordering (reversing it, or, placing FEC between SRTP encryption and + SRTP authentication) SHALL be signaled out of band. + +11. Scenarios + + SRTP can be used as security protocol for the RTP/RTCP traffic in + many different scenarios. SRTP has a number of configuration + options, in particular regarding key usage, and can have impact on + the total performance of the application according to the way it is + used. Hence, the use of SRTP is dependent on the kind of scenario + and application it is used with. In the following, we briefly + illustrate some use cases for SRTP, and give some guidelines for + recommended setting of its options. + +11.1. Unicast + + A typical example would be a voice call or video-on-demand + application. + + + + + +Baugher, et al. Standards Track [Page 43] + +RFC 3711 SRTP March 2004 + + + Consider one bi-directional RTP stream, as one RTP session. It is + possible for the two parties to share the same master key in the two + directions according to the principles of Section 9.1. The first + round of the key derivation splits the master key into any or all of + the following session keys (according to the provided security + functions): + + SRTP_encr_key, SRTP_auth_key, SRTCP_encr_key, and SRTCP_auth key. + + (For simplicity, we omit discussion of the salts, which are also + derived.) In this scenario, it will in most cases suffice to have a + single master key with the default lifetime. This guarantees + sufficiently long lifetime of the keys and a minimum set of keys in + place for most practical purposes. Also, in this case RTCP + protection can be applied smoothly. Under these assumptions, use of + the MKI can be omitted. As the key-derivation in combination with + large difference in the packet rate in the respective directions may + require simultaneous storage of several session keys, if storage is + an issue, we recommended to use low-rate key derivation. + + The same considerations can be extended to the unicast scenario with + multiple RTP sessions, where each session would have a distinct + master key. + +11.2. Multicast (one sender) + + Just as with (unprotected) RTP, a scalability issue arises in big + groups due to the possibly very large amount of SRTCP Receiver + Reports that the sender might need to process. In SRTP, the sender + may have to keep state (the cryptographic context) for each receiver, + or more precisely, for the SRTCP used to protect Receiver Reports. + The overhead increases proportionally to the size of the group. In + particular, re-keying requires special concern, see below. + + Consider first a small group of receivers. There are a few possible + setups with the distribution of master keys among the receivers. + Given a single RTP session, one possibility is that the receivers + share the same master key as per Section 9.1 to secure all their + respective RTCP traffic. This shared master key could then be the + same one used by the sender to protect its outbound SRTP traffic. + Alternatively, it could be a master key shared only among the + receivers and used solely for their SRTCP traffic. Both alternatives + require the receivers to trust each other. + + Considering SRTCP and key storage, it is recommended to use low-rate + (or zero) key_derivation (except the mandatory initial one), so that + the sender does not need to store too many session keys (each SRTCP + stream might otherwise have a different session key at a given point + + + +Baugher, et al. Standards Track [Page 44] + +RFC 3711 SRTP March 2004 + + + in time, as the SRTCP sources send at different times). Thus, in + case key derivation is wanted for SRTP, the cryptographic context for + SRTP can be kept separate from the SRTCP crypto context, so that it + is possible to have a key_derivation_rate of 0 for SRTCP and a non- + zero value for SRTP. + + Use of the MKI for re-keying is RECOMMENDED for most applications + (see Section 8.1). + + If there are more than one SRTP/SRTCP stream (within the same RTP + session) that share the master key, the upper limit of 2^48 SRTP + packets / 2^31 SRTCP packets means that, before one of the streams + reaches its maximum number of packets, re-keying MUST be triggered on + ALL streams sharing the master key. (From strict security point of + view, only the stream reaching the maximum would need to be re-keyed, + but then the streams would no longer be sharing master key, which is + the intention.) A local policy at the sender side should force + rekeying in a way that the maximum packet limit is not reached on any + of the streams. Use of the MKI for re-keying is RECOMMENDED. + + In large multicast with one sender, the same considerations as for + the small group multicast hold. The biggest issue in this scenario + is the additional load placed at the sender side, due to the state + (cryptographic contexts) that has to be maintained for each receiver, + sending back RTCP Receiver Reports. At minimum, a replay window + might need to be maintained for each RTCP source. + +11.3. Re-keying and access control + + Re-keying may occur due to access control (e.g., when a member is + removed during a multicast RTP session), or for pure cryptographic + reasons (e.g., the key is at the end of its lifetime). When using + SRTP default transforms, the master key MUST be replaced before any + of the index spaces are exhausted for any of the streams protected by + one and the same master key. + + How key management re-keys SRTP implementations is out of scope, but + it is clear that there are straightforward ways to manage keys for a + multicast group. In one-sender multicast, for example, it is + typically the responsibility of the sender to determine when a new + key is needed. The sender is the one entity that can keep track of + when the maximum number of packets has been sent, as receivers may + join and leave the session at any time, there may be packet loss and + delay etc. In scenarios other than one-sender multicast, other + methods can be used. Here, one must take into consideration that key + exchange can be a costly operation, taking several seconds for a + single exchange. Hence, some time before the master key is + exhausted/expires, out-of-band key management is initiated, resulting + + + +Baugher, et al. Standards Track [Page 45] + +RFC 3711 SRTP March 2004 + + + in a new master key that is shared with the receiver(s). In any + event, to maintain synchronization when switching to the new key, + group policy might choose between using the MKI and the <From, To>, + as described in Section 8.1. + + For access control purposes, the <From, To> periods are set at the + desired granularity, dependent on the packet rate. High rate re- + keying can be problematic for SRTCP in some large-group scenarios. + As mentioned, there are potential problems in using the SRTP index, + rather than the SRTCP index, for determining the master key. In + particular, for short periods during switching of master keys, it may + be the case that SRTCP packets are not under the current master key + of the correspondent SRTP. Therefore, using the MKI for re-keying in + such scenarios will produce better results. + +11.4. Summary of basic scenarios + + The description of these scenarios highlights some recommendations on + the use of SRTP, mainly related to re-keying and large scale + multicast: + + - Do not use fast re-keying with the <From, To> feature. It may, in + particular, give problems in retrieving the correct SRTCP key, if + an SRTCP packet arrives close to the re-keying time. The MKI + SHOULD be used in this case. + + - If multiple SRTP streams in the same RTP session share the same + master key, also moderate rate re-keying MAY have the same + problems, and the MKI SHOULD be used. + + - Though offering increased security, a non-zero key_derivation_rate + is NOT RECOMMENDED when trying to minimize the number of keys in + use with multiple streams. + +12. IANA Considerations + + The RTP specification establishes a registry of profile names for use + by higher-level control protocols, such as the Session Description + Protocol (SDP), to refer to transport methods. This profile + registers the name "RTP/SAVP". + + SRTP uses cryptographic transforms which a key management protocol + signals. It is the task of each particular key management protocol + to register the cryptographic transforms or suites of transforms with + IANA. The key management protocol conveys these protocol numbers, + not SRTP, and each key management protocol chooses the numbering + scheme and syntax that it requires. + + + + +Baugher, et al. Standards Track [Page 46] + +RFC 3711 SRTP March 2004 + + + Specification of a key management protocol for SRTP is out of scope + here. Section 8.2, however, provides guidance on the parameters that + need to be defined for the default and mandatory transforms. + +13. Acknowledgements + + David Oran (Cisco) and Rolf Blom (Ericsson) are co-authors of this + document but their valuable contributions are acknowledged here to + keep the length of the author list down. + + The authors would in addition like to thank Magnus Westerlund, Brian + Weis, Ghyslain Pelletier, Morgan Lindqvist, Robert Fairlie- + Cuninghame, Adrian Perrig, the AVT WG and in particular the chairmen + Colin Perkins and Stephen Casner, the Transport and Security Area + Directors, and Eric Rescorla for their reviews and support. + +14. References + +14.1. Normative References + + [AES] NIST, "Advanced Encryption Standard (AES)", FIPS PUB 197, + http://www.nist.gov/aes/ + + [RFC2104] Krawczyk, H., Bellare, M. and R. Canetti, "HMAC: Keyed- + Hashing for Message Authentication", RFC 2104, February + 1997. + + [RFC2119] Bradner, S., "Key words for use in RFCs to Indicate + Requirement Levels", BCP 14, RFC 2119, March 1997. + + [RFC2401] Kent, S. and R. Atkinson, "Security Architecture for + Internet Protocol", RFC 2401, November 1998. + + [RFC2828] Shirey, R., "Internet Security Glossary", FYI 36, RFC 2828, + May 2000. + + [RFC3550] Schulzrinne, H., Casner, S., Frederick, R. and V. Jacobson, + "RTP: A Transport Protocol for Real-time Applications", RFC + 3550, July 2003. + + [RFC3551] Schulzrinne, H. and S. Casner, "RTP Profile for Audio and + Video Conferences with Minimal Control", RFC 3551, July + 2003. + + + + + + + + +Baugher, et al. Standards Track [Page 47] + +RFC 3711 SRTP March 2004 + + +14.2. Informative References + + [AES-CTR] Lipmaa, H., Rogaway, P. and D. Wagner, "CTR-Mode + Encryption", NIST, http://csrc.nist.gov/encryption/modes/ + workshop1/papers/lipmaa-ctr.pdf + + [B96] Bellovin, S., "Problem Areas for the IP Security + Protocols," in Proceedings of the Sixth Usenix Unix + Security Symposium, pp. 1-16, San Jose, CA, July 1996 + (http://www.research.att.com/~smb/papers/index.html). + + [BDJR] Bellare, M., Desai, A., Jokipii, E. and P. Rogaway, "A + Concrete Treatment of Symmetric Encryption: Analysis of DES + Modes of Operation", Proceedings 38th IEEE FOCS, pp. 394- + 403, 1997. + + [BS00] Biryukov, A. and A. Shamir, "Cryptanalytic Time/Memory/Data + Tradeoffs for Stream Ciphers", Proceedings, ASIACRYPT 2000, + LNCS 1976, pp. 1-13, Springer Verlag. + + [C99] Crowell, W. P., "Introduction to the VENONA Project", + http://www.nsa.gov:8080/docs/venona/index.html. + + [CTR] Dworkin, M., NIST Special Publication 800-38A, + "Recommendation for Block Cipher Modes of Operation: + Methods and Techniques", 2001. + http://csrc.nist.gov/publications/nistpubs/800-38a/sp800- + 38a.pdf. + + [f8-a] 3GPP TS 35.201 V4.1.0 (2001-12) Technical Specification 3rd + Generation Partnership Project; Technical Specification + Group Services and System Aspects; 3G Security; + Specification of the 3GPP Confidentiality and Integrity + Algorithms; Document 1: f8 and f9 Specification (Release + 4). + + [f8-b] 3GPP TR 33.908 V4.0.0 (2001-09) Technical Report 3rd + Generation Partnership Project; Technical Specification + Group Services and System Aspects; 3G Security; General + Report on the Design, Specification and Evaluation of 3GPP + Standard Confidentiality and Integrity Algorithms (Release + 4). + + [GDOI] Baugher, M., Weis, B., Hardjono, T. and H. Harney, "The + Group Domain of Interpretation, RFC 3547, July 2003. + + + + + + +Baugher, et al. Standards Track [Page 48] + +RFC 3711 SRTP March 2004 + + + [HAC] Menezes, A., Van Oorschot, P. and S. Vanstone, "Handbook + of Applied Cryptography", CRC Press, 1997, ISBN 0-8493- + 8523-7. + + [H80] Hellman, M. E., "A cryptanalytic time-memory trade-off", + IEEE Transactions on Information Theory, July 1980, pp. + 401-406. + + [IK] T. Iwata and T. Kohno: "New Security Proofs for the 3GPP + Confidentiality and Integrity Algorithms", Proceedings of + FSE 2004. + + [KINK] Thomas, M. and J. Vilhuber, "Kerberized Internet + Negotiation of Keys (KINK)", Work in Progress. + + [KEYMGT] Arrko, J., et al., "Key Management Extensions for Session + Description Protocol (SDP) and Real Time Streaming Protocol + (RTSP)", Work in Progress. + + [KSYH] Kang, J-S., Shin, S-U., Hong, D. and O. Yi, "Provable + Security of KASUMI and 3GPP Encryption Mode f8", + Proceedings Asiacrypt 2001, Springer Verlag LNCS 2248, pp. + 255-271, 2001. + + [MIKEY] Arrko, J., et. al., "MIKEY: Multimedia Internet KEYing", + Work in Progress. + + [MF00] McGrew, D. and S. Fluhrer, "Attacks on Encryption of + Redundant Plaintext and Implications on Internet Security", + the Proceedings of the Seventh Annual Workshop on Selected + Areas in Cryptography (SAC 2000), Springer-Verlag. + + [PCST1] Perrig, A., Canetti, R., Tygar, D. and D. Song, "Efficient + and Secure Source Authentication for Multicast", in Proc. + of Network and Distributed System Security Symposium NDSS + 2001, pp. 35-46, 2001. + + [PCST2] Perrig, A., Canetti, R., Tygar, D. and D. Song, "Efficient + Authentication and Signing of Multicast Streams over Lossy + Channels", in Proc. of IEEE Security and Privacy Symposium + S&P2000, pp. 56-73, 2000. + + [RFC1750] Eastlake, D., Crocker, S. and J. Schiller, "Randomness + Recommendations for Security", RFC 1750, December 1994. + + [RFC2675] Borman, D., Deering, S. and R. Hinden, "IPv6 Jumbograms", + RFC 2675, August 1999. + + + + +Baugher, et al. Standards Track [Page 49] + +RFC 3711 SRTP March 2004 + + + [RFC3095] Bormann, C., Burmeister, C., Degermark, M., Fukuhsima, H., + Hannu, H., Jonsson, L-E., Hakenberg, R., Koren, T., Le, K., + Liu, Z., Martensson, A., Miyazaki, A., Svanbro, K., Wiebke, + T., Yoshimura, T. and H. Zheng, "RObust Header Compression: + Framework and Four Profiles: RTP, UDP, ESP, and + uncompressed (ROHC)", RFC 3095, July 2001. + + [RFC3242] Jonsson, L-E. and G. Pelletier, "RObust Header Compression + (ROHC): A Link-Layer Assisted Profile for IP/UDP/RTP ", RFC + 3242, April 2002. + + [SDMS] Andreasen, F., Baugher, M. and D. Wing, "Session + Description Protocol Security Descriptions for Media + Streams", Work in Progress. + + [SWO] Svanbro, K., Wiorek, J. and B. Olin, "Voice-over-IP-over- + wireless", Proc. PIMRC 2000, London, Sept. 2000. + + [V02] Vaudenay, S., "Security Flaws Induced by CBC Padding - + Application to SSL, IPsec, WTLS...", Advances in + Cryptology, EUROCRYPT'02, LNCS 2332, pp. 534-545. + + [WC81] Wegman, M. N., and J.L. Carter, "New Hash Functions and + Their Use in Authentication and Set Equality", JCSS 22, + 265-279, 1981. + + + + + + + + + + + + + + + + + + + + + + + + + + +Baugher, et al. Standards Track [Page 50] + +RFC 3711 SRTP March 2004 + + +Appendix A: Pseudocode for Index Determination + + The following is an example of pseudo-code for the algorithm to + determine the index i of an SRTP packet with sequence number SEQ. In + the following, signed arithmetic is assumed. + + if (s_l < 32,768) + if (SEQ - s_l > 32,768) + set v to (ROC-1) mod 2^32 + else + set v to ROC + endif + else + if (s_l - 32,768 > SEQ) + set v to (ROC+1) mod 2^32 + else + set v to ROC + endif + endif + return SEQ + v*65,536 + +Appendix B: Test Vectors + + All values are in hexadecimal. + +B.1. AES-f8 Test Vectors + + SRTP PREFIX LENGTH : 0 + + RTP packet header : 806e5cba50681de55c621599 + + RTP packet payload : 70736575646f72616e646f6d6e657373 + 20697320746865206e65787420626573 + 74207468696e67 + + ROC : d462564a + key : 234829008467be186c3de14aae72d62c + salt key : 32f2870d + key-mask (m) : 32f2870d555555555555555555555555 + key XOR key-mask : 11baae0dd132eb4d3968b41ffb278379 + + IV : 006e5cba50681de55c621599d462564a + IV' : 595b699bbd3bc0df26062093c1ad8f73 + + + + + + + + +Baugher, et al. Standards Track [Page 51] + +RFC 3711 SRTP March 2004 + + + j = 0 + IV' xor j : 595b699bbd3bc0df26062093c1ad8f73 + S(-1) : 00000000000000000000000000000000 + IV' xor S(-1) xor j : 595b699bbd3bc0df26062093c1ad8f73 + S(0) : 71ef82d70a172660240709c7fbb19d8e + plaintext : 70736575646f72616e646f6d6e657373 + ciphertext : 019ce7a26e7854014a6366aa95d4eefd + + j = 1 + IV' xor j : 595b699bbd3bc0df26062093c1ad8f72 + S(0) : 71ef82d70a172660240709c7fbb19d8e + IV' xor S(0) xor j : 28b4eb4cb72ce6bf020129543a1c12fc + S(1) : 3abd640a60919fd43bd289a09649b5fc + plaintext : 20697320746865206e65787420626573 + ciphertext : 1ad4172a14f9faf455b7f1d4b62bd08f + + j = 2 + IV' xor j : 595b699bbd3bc0df26062093c1ad8f71 + S(1) : 3abd640a60919fd43bd289a09649b5fc + IV' xor S(1) xor j : 63e60d91ddaa5f0b1dd4a93357e43a8d + S(2) : 220c7a8715266565b09ecc8a2a62b11b + plaintext : 74207468696e67 + ciphertext : 562c0eef7c4802 + +B.2. AES-CM Test Vectors + + Keystream segment length: 1044512 octets (65282 AES blocks) + Session Key: 2B7E151628AED2A6ABF7158809CF4F3C + Rollover Counter: 00000000 + Sequence Number: 0000 + SSRC: 00000000 + Session Salt: F0F1F2F3F4F5F6F7F8F9FAFBFCFD0000 (already shifted) + Offset: F0F1F2F3F4F5F6F7F8F9FAFBFCFD0000 + + Counter Keystream + + F0F1F2F3F4F5F6F7F8F9FAFBFCFD0000 E03EAD0935C95E80E166B16DD92B4EB4 + F0F1F2F3F4F5F6F7F8F9FAFBFCFD0001 D23513162B02D0F72A43A2FE4A5F97AB + F0F1F2F3F4F5F6F7F8F9FAFBFCFD0002 41E95B3BB0A2E8DD477901E4FCA894C0 + ... ... + F0F1F2F3F4F5F6F7F8F9FAFBFCFDFEFF EC8CDF7398607CB0F2D21675EA9EA1E4 + F0F1F2F3F4F5F6F7F8F9FAFBFCFDFF00 362B7C3C6773516318A077D7FC5073AE + F0F1F2F3F4F5F6F7F8F9FAFBFCFDFF01 6A2CC3787889374FBEB4C81B17BA6C44 + + Nota Bene: this test case is contrived so that the latter part of the + keystream segment coincides with the test case in Section F.5.1 of + [CTR]. + + + + +Baugher, et al. Standards Track [Page 52] + +RFC 3711 SRTP March 2004 + + +B.3. Key Derivation Test Vectors + + This section provides test data for the default key derivation + function, which uses AES-128 in Counter Mode. In the following, we + walk through the initial key derivation for the AES-128 Counter Mode + cipher, which requires a 16 octet session encryption key and a 14 + octet session salt, and an authentication function which requires a + 94-octet session authentication key. These values are called the + cipher key, the cipher salt, and the auth key in the following. + Since this is the initial key derivation and the key derivation rate + is equal to zero, the value of (index DIV key_derivation_rate) is + zero (actually, a six-octet string of zeros). In the following, we + shorten key_derivation_rate to kdr. + + The inputs to the key derivation function are the 16 octet master key + and the 14 octet master salt: + + master key: E1F97A0D3E018BE0D64FA32C06DE4139 + master salt: 0EC675AD498AFEEBB6960B3AABE6 + + We first show how the cipher key is generated. The input block for + AES-CM is generated by exclusive-oring the master salt with the + concatenation of the encryption key label 0x00 with (index DIV kdr), + then padding on the right with two null octets (which implements the + multiply-by-2^16 operation, see Section 4.3.3). The resulting value + is then AES-CM- encrypted using the master key to get the cipher key. + + index DIV kdr: 000000000000 + label: 00 + master salt: 0EC675AD498AFEEBB6960B3AABE6 + ----------------------------------------------- + xor: 0EC675AD498AFEEBB6960B3AABE6 (x, PRF input) + + x*2^16: 0EC675AD498AFEEBB6960B3AABE60000 (AES-CM input) + + cipher key: C61E7A93744F39EE10734AFE3FF7A087 (AES-CM output) + + + + + + + + + + + + + + + +Baugher, et al. Standards Track [Page 53] + +RFC 3711 SRTP March 2004 + + + Next, we show how the cipher salt is generated. The input block for + AES-CM is generated by exclusive-oring the master salt with the + concatenation of the encryption salt label. That value is padded and + encrypted as above. + + index DIV kdr: 000000000000 + label: 02 + master salt: 0EC675AD498AFEEBB6960B3AABE6 + + ---------------------------------------------- + xor: 0EC675AD498AFEE9B6960B3AABE6 (x, PRF input) + + x*2^16: 0EC675AD498AFEE9B6960B3AABE60000 (AES-CM input) + + 30CBBC08863D8C85D49DB34A9AE17AC6 (AES-CM ouptut) + + cipher salt: 30CBBC08863D8C85D49DB34A9AE1 + + We now show how the auth key is generated. The input block for AES- + CM is generated as above, but using the authentication key label. + + index DIV kdr: 000000000000 + label: 01 + master salt: 0EC675AD498AFEEBB6960B3AABE6 + ----------------------------------------------- + xor: 0EC675AD498AFEEAB6960B3AABE6 (x, PRF input) + + x*2^16: 0EC675AD498AFEEAB6960B3AABE60000 (AES-CM input) + + Below, the auth key is shown on the left, while the corresponding AES + input blocks are shown on the right. + + auth key AES input blocks + CEBE321F6FF7716B6FD4AB49AF256A15 0EC675AD498AFEEAB6960B3AABE60000 + 6D38BAA48F0A0ACF3C34E2359E6CDBCE 0EC675AD498AFEEAB6960B3AABE60001 + E049646C43D9327AD175578EF7227098 0EC675AD498AFEEAB6960B3AABE60002 + 6371C10C9A369AC2F94A8C5FBCDDDC25 0EC675AD498AFEEAB6960B3AABE60003 + 6D6E919A48B610EF17C2041E47403576 0EC675AD498AFEEAB6960B3AABE60004 + 6B68642C59BBFC2F34DB60DBDFB2 0EC675AD498AFEEAB6960B3AABE60005 + + + + + + + + + + + + +Baugher, et al. Standards Track [Page 54] + +RFC 3711 SRTP March 2004 + + +Authors' Addresses + + Questions and comments should be directed to the authors and + avt@ietf.org: + + Mark Baugher + Cisco Systems, Inc. + 5510 SW Orchid Street + Portland, OR 97219 USA + + Phone: +1 408-853-4418 + EMail: mbaugher@cisco.com + + + Elisabetta Carrara + Ericsson Research + SE-16480 Stockholm + Sweden + + Phone: +46 8 50877040 + EMail: elisabetta.carrara@ericsson.com + + + David A. McGrew + Cisco Systems, Inc. + San Jose, CA 95134-1706 + USA + + Phone: +1 301-349-5815 + EMail: mcgrew@cisco.com + + + Mats Naslund + Ericsson Research + SE-16480 Stockholm + Sweden + + Phone: +46 8 58533739 + EMail: mats.naslund@ericsson.com + + + Karl Norrman + Ericsson Research + SE-16480 Stockholm + Sweden + + Phone: +46 8 4044502 + EMail: karl.norrman@ericsson.com + + + +Baugher, et al. Standards Track [Page 55] + +RFC 3711 SRTP March 2004 + + +Full Copyright Statement + + Copyright (C) The Internet Society (2004). This document is subject + to the rights, licenses and restrictions contained in BCP 78 and + except as set forth therein, the authors retain all their rights. + + This document and the information contained herein are provided on an + "AS IS" basis and THE CONTRIBUTOR, THE ORGANIZATION HE/SHE REPRESENTS + OR IS SPONSORED BY (IF ANY), THE INTERNET SOCIETY AND THE INTERNET + ENGINEERING TASK FORCE DISCLAIM ALL WARRANTIES, EXPRESS OR IMPLIED, + INCLUDING BUT NOT LIMITED TO ANY WARRANTY THAT THE USE OF THE + INFORMATION HEREIN WILL NOT INFRINGE ANY RIGHTS OR ANY IMPLIED + WARRANTIES OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE. + +Intellectual Property + + The IETF takes no position regarding the validity or scope of any + Intellectual Property Rights or other rights that might be claimed to + pertain to the implementation or use of the technology described in + this document or the extent to which any license under such rights + might or might not be available; nor does it represent that it has + made any independent effort to identify any such rights. Information + on the procedures with respect to rights in RFC documents can be + found in BCP 78 and BCP 79. + + Copies of IPR disclosures made to the IETF Secretariat and any + assurances of licenses to be made available, or the result of an + attempt made to obtain a general license or permission for the use of + such proprietary rights by implementers or users of this + specification can be obtained from the IETF on-line IPR repository at + http://www.ietf.org/ipr. + + The IETF invites any interested party to bring to its attention any + copyrights, patents or patent applications, or other proprietary + rights that may cover technology that may be required to implement + this standard. Please address the information to the IETF at ietf- + ipr@ietf.org. + +Acknowledgement + + Funding for the RFC Editor function is currently provided by the + Internet Society. + + + + + + + + + +Baugher, et al. Standards Track [Page 56] + diff --git a/libs/srtp/include/CVS/Entries b/libs/srtp/include/CVS/Entries new file mode 100644 index 0000000000..4fe7cd9b83 --- /dev/null +++ b/libs/srtp/include/CVS/Entries @@ -0,0 +1,4 @@ +/rtp.h/1.2/Sun Oct 2 20:16:47 2005// +/srtp.h/1.9/Wed Mar 8 20:06:12 2006// +/ut_sim.h/1.3/Thu Mar 16 19:13:33 2006// +D diff --git a/libs/srtp/include/CVS/Repository b/libs/srtp/include/CVS/Repository new file mode 100644 index 0000000000..29fe7cdd44 --- /dev/null +++ b/libs/srtp/include/CVS/Repository @@ -0,0 +1 @@ +srtp/include diff --git a/libs/srtp/include/CVS/Root b/libs/srtp/include/CVS/Root new file mode 100644 index 0000000000..b4b74237d9 --- /dev/null +++ b/libs/srtp/include/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@cvs.sourceforge.net:/cvsroot/srtp diff --git a/libs/srtp/include/rtp.h b/libs/srtp/include/rtp.h new file mode 100644 index 0000000000..65975ff35b --- /dev/null +++ b/libs/srtp/include/rtp.h @@ -0,0 +1,125 @@ +/* + * rtp.h + * + * rtp interface for srtp reference implementation + * + * David A. McGrew + * Cisco Systems, Inc. + * + * data types: + * + * rtp_msg_t an rtp message (the data that goes on the wire) + * rtp_sender_t sender side socket and rtp info + * rtp_receiver_t receiver side socket and rtp info + * + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#ifndef RTP_H +#define RTP_H + +#include "srtp.h" + +#ifdef HAVE_NETINET_IN_H +# include <netinet/in.h> +#elif defined HAVE_WINSOCK2_H +# include <winsock2.h> +#endif + +#define rtp_header_len 12 + +typedef srtp_hdr_t rtp_hdr_t; + +#define RTP_MAX_BUF_LEN 16384 + +typedef struct { + srtp_hdr_t header; + char body[RTP_MAX_BUF_LEN]; +} rtp_msg_t; + +typedef struct { + rtp_msg_t message; + int socket; + srtp_ctx_t *srtp_ctx; + struct sockaddr_in addr; /* reciever's address */ +} rtp_sender_t; + +typedef struct { + rtp_msg_t message; + int socket; + srtp_ctx_t *srtp_ctx; + struct sockaddr_in addr; /* receiver's address */ +} rtp_receiver_t; + + +ssize_t +rtp_sendto(rtp_sender_t *sender, const void* msg, int len); + +ssize_t +rtp_recvfrom(rtp_receiver_t *receiver, void *msg, int *len); + +int +rtp_receiver_init(rtp_receiver_t *rcvr, int socket, + struct sockaddr_in addr, uint32_t ssrc); + +int +rtp_sender_init(rtp_sender_t *sender, int socket, + struct sockaddr_in addr, uint32_t ssrc); + +/* + * srtp_sender_init(...) initializes an rtp_sender_t + * + */ + +int +srtp_sender_init(rtp_sender_t *rtp_ctx, /* structure to be init'ed */ + struct sockaddr_in name, /* socket name */ + sec_serv_t security_services, /* sec. servs. to be used */ + unsigned char *input_key /* master key/salt in hex */ + ); + +int +srtp_receiver_init(rtp_receiver_t *rtp_ctx, /* structure to be init'ed */ + struct sockaddr_in name, /* socket name */ + sec_serv_t security_services, /* sec. servs. to be used */ + unsigned char *input_key /* master key/salt in hex */ + ); + + +#endif /* RTP_H */ diff --git a/libs/srtp/include/srtp.h b/libs/srtp/include/srtp.h new file mode 100644 index 0000000000..ade423a1fb --- /dev/null +++ b/libs/srtp/include/srtp.h @@ -0,0 +1,993 @@ +/* + * srtp.h + * + * interface to libsrtp + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#ifndef SRTP_H +#define SRTP_H + +#ifdef __cplusplus +extern "C" { +#endif + +#include "crypto_kernel.h" +#include "rdbx.h" +#include "rdb.h" +#include "integers.h" + +/** + * @defgroup SRTP Secure RTP + * + * @brief libSRTP provides functions for protecting RTP and RTCP. See + * Section @ref Overview for an introduction to the use of the library. + * + * @{ + */ + +/* + * SRTP_MASTER_KEY_LEN is the nominal master key length supported by libSRTP + */ + +#define SRTP_MASTER_KEY_LEN 30 + +/* + * SRTP_MAX_KEY_LEN is the maximum key length supported by libSRTP + */ +#define SRTP_MAX_KEY_LEN 64 + +/* + * SRTP_MAX_TAG_LEN is the maximum tag length supported by libSRTP + */ + +#define SRTP_MAX_TAG_LEN 12 + +/** + * SRTP_MAX_TRAILER_LEN is the maximum length of the SRTP trailer + * (authentication tag and MKI) supported by libSRTP. This value is + * the maximum number of octets that will be added to an RTP packet by + * srtp_protect(). + * + * @brief the maximum number of octets added by srtp_protect(). + */ +#define SRTP_MAX_TRAILER_LEN SRTP_MAX_TAG_LEN + +/* + * nota bene: since libSRTP doesn't support the use of the MKI, the + * SRTP_MAX_TRAILER_LEN value is just the maximum tag length + */ + +/** + * @brief sec_serv_t describes a set of security services. + * + * A sec_serv_t enumeration is used to describe the particular + * security services that will be applied by a particular crypto + * policy (or other mechanism). + */ + +typedef enum { + sec_serv_none = 0, /**< no services */ + sec_serv_conf = 1, /**< confidentiality */ + sec_serv_auth = 2, /**< authentication */ + sec_serv_conf_and_auth = 3 /**< confidentiality and authentication */ +} sec_serv_t; + +/** + * @brief crypto_policy_t describes a particular crypto policy that + * can be applied to an SRTP stream. + * + * A crypto_policy_t describes a particular cryptographic policy that + * can be applied to an SRTP or SRTCP stream. An SRTP session policy + * consists of a list of these policies, one for each SRTP stream + * in the session. + */ + +typedef struct crypto_policy_t { + cipher_type_id_t cipher_type; /**< An integer representing + * the type of cipher. */ + int cipher_key_len; /**< The length of the cipher key + * in octets. */ + auth_type_id_t auth_type; /**< An integer representing the + * authentication function. */ + int auth_key_len; /**< The length of the authentication + * function key in octets. */ + int auth_tag_len; /**< The length of the authentication + * tag in octets. */ + sec_serv_t sec_serv; /**< The flag indicating the security + * services to be applied. */ +} crypto_policy_t; + + +/** + * @brief ssrc_type_t describes the type of an SSRC. + * + * An ssrc_type_t enumeration is used to indicate a type of SSRC. See + * @ref srtp_policy_t for more informataion. + */ + +typedef enum { + ssrc_undefined = 0, /**< Indicates an undefined SSRC type. */ + ssrc_specific = 1, /**< Indicates a specific SSRC value */ + ssrc_any_inbound = 2, /**< Indicates any inbound SSRC value + (i.e. a value that is used in the + function srtp_unprotect()) */ + ssrc_any_outbound = 3 /**< Indicates any outbound SSRC value + (i.e. a value that is used in the + function srtp_protect()) */ +} ssrc_type_t; + +/** + * @brief An ssrc_t represents a particular SSRC value, or a `wildcard' SSRC. + * + * An ssrc_t represents a particular SSRC value (if its type is + * ssrc_specific), or a wildcard SSRC value that will match all + * outbound SSRCs (if its type is ssrc_any_outbound) or all inbound + * SSRCs (if its type is ssrc_any_inbound). + * + */ + +typedef struct { + ssrc_type_t type; /**< The type of this particular SSRC */ + uint32_t value; /**< The value of this SSRC, if it is not a wildcard */ +} ssrc_t; + + +/** + * @brief represents the policy for an SRTP session. + * + * A single srtp_policy_t struct represents the policy for a single + * SRTP stream, and a linked list of these elements represents the + * policy for an entire SRTP session. Each element contains the SRTP + * and SRTCP crypto policies for that stream, a pointer to the SRTP + * master key for that stream, the SSRC describing that stream, or a + * flag indicating a `wildcard' SSRC value, and a `next' field that + * holds a pointer to the next element in the list of policy elements, + * or NULL if it is the last element. + * + * The wildcard value SSRC_ANY_INBOUND matches any SSRC from an + * inbound stream that for which there is no explicit SSRC entry in + * another policy element. Similarly, the value SSRC_ANY_OUTBOUND + * will matches any SSRC from an outbound stream that does not appear + * in another policy element. Note that wildcard SSRCs &b cannot be + * used to match both inbound and outbound traffic. This restriction + * is intentional, and it allows libSRTP to ensure that no security + * lapses result from accidental re-use of SSRC values during key + * sharing. + * + * + * @warning The final element of the list @b must have its `next' pointer + * set to NULL. + */ + +typedef struct srtp_policy_t { + ssrc_t ssrc; /**< The SSRC value of stream, or the + * flags SSRC_ANY_INBOUND or + * SSRC_ANY_OUTBOUND if key sharing + * is used for this policy element. + */ + crypto_policy_t rtp; /**< SRTP crypto policy. */ + crypto_policy_t rtcp; /**< SRTCP crypto policy. */ + uint8_t *key; /**< Pointer to the SRTP master key for + * this stream. */ + struct srtp_policy_t *next; /**< Pointer to next stream policy. */ +} srtp_policy_t; + + + + +/** + * @brief An srtp_t points to an SRTP session structure. + * + * The typedef srtp_t is a pointer to a structure that represents + * an SRTP session. This datatype is intentially opaque in + * order to separate the interface from the implementation. + * + * An SRTP session consists of all of the traffic sent to the RTP and + * RTCP destination transport addresses, using the RTP/SAVP (Secure + * Audio/Video Profile). A session can be viewed as a set of SRTP + * streams, each of which originates with a different participant. + */ + +typedef struct srtp_ctx_t *srtp_t; + + +/** + * @brief An srtp_stream_t points to an SRTP stream structure. + * + * The typedef srtp_stream_t is a pointer to a structure that + * represents an SRTP stream. This datatype is intentionally + * opaque in order to separate the interface from the implementation. + * + * An SRTP stream consists of all of the traffic sent to an SRTP + * session by a single participant. A session can be viewed as + * a set of streams. + * + */ +typedef struct srtp_stream_ctx_t *srtp_stream_t; + + + +/** + * @brief srtp_init() initializes the srtp library. + * + * @warning This function @b must be called before any other srtp + * functions. + */ + +err_status_t +srtp_init(void); + +/** + * @brief srtp_protect() is the Secure RTP sender-side packet processing + * function. + * + * The function call srtp_protect(ctx, rtp_hdr, len_ptr) applies SRTP + * protection to the RTP packet rtp_hdr (which has length *len_ptr) using + * the SRTP context ctx. If err_status_ok is returned, then rtp_hdr + * points to the resulting SRTP packet and *len_ptr is the number of + * octets in that packet; otherwise, no assumptions should be made + * about the value of either data elements. + * + * The sequence numbers of the RTP packets presented to this function + * need not be consecutive, but they @b must be out of order by less + * than 2^15 = 32,768 packets. + * + * @warning This function assumes that it can write the authentication + * tag into the location in memory immediately following the RTP + * packet, and assumes that the RTP packet is aligned on a 32-bit + * boundary. + * + * @param ctx is the SRTP context to use in processing the packet. + * + * @param rtp_hdr is a pointer to the RTP packet (before the call); after + * the function returns, it points to the srtp packet. + * + * @param len_ptr is a pointer to the length in octets of the complete + * RTP packet (header and body) before the function call, and of the + * complete SRTP packet after the call, if err_status_ok was returned. + * Otherwise, the value of the data to which it points is undefined. + * + * @return + * - err_status_ok no problems + * - err_status_replay_fail rtp sequence number was non-increasing + * - @e other failure in cryptographic mechanisms + */ + +err_status_t +srtp_protect(srtp_t ctx, void *rtp_hdr, int *len_ptr); + +/** + * @brief srtp_unprotect() is the Secure RTP receiver-side packet + * processing function. + * + * The function call srtp_unprotect(ctx, srtp_hdr, len_ptr) verifies + * the Secure RTP protection of the SRTP packet pointed to by srtp_hdr + * (which has length *len_ptr), using the SRTP context ctx. If + * err_status_ok is returned, then srtp_hdr points to the resulting + * RTP packet and *len_ptr is the number of octets in that packet; + * otherwise, no assumptions should be made about the value of either + * data elements. + * + * The sequence numbers of the RTP packets presented to this function + * need not be consecutive, but they @b must be out of order by less + * than 2^15 = 32,768 packets. + * + * @warning This function assumes that the SRTP packet is aligned on a + * 32-bit boundary. + * + * @param ctx is a pointer to the srtp_t which applies to the + * particular packet. + * + * @param srtp_hdr is a pointer to the header of the SRTP packet + * (before the call). after the function returns, it points to the + * rtp packet if err_status_ok was returned; otherwise, the value of + * the data to which it points is undefined. + * + * @param len_ptr is a pointer to the length in octets of the complete + * srtp packet (header and body) before the function call, and of the + * complete rtp packet after the call, if err_status_ok was returned. + * Otherwise, the value of the data to which it points is undefined. + * + * @return + * - err_status_ok if the RTP packet is valid. + * - err_status_auth_fail if the SRTP packet failed the message + * authentication check. + * - err_status_replay_fail if the SRTP packet is a replay (e.g. packet has + * already been processed and accepted). + * - [other] if there has been an error in the cryptographic mechanisms. + * + */ + +err_status_t +srtp_unprotect(srtp_t ctx, void *srtp_hdr, int *len_ptr); + + +/** + * @brief srtp_create() allocates and initializes an SRTP session. + + * The function call srtp_create(session, policy, key) allocates and + * initializes an SRTP session context, applying the given policy and + * key. + * + * @param session is the SRTP session to which the policy is to be added. + * + * @param policy is the srtp_policy_t struct that describes the policy + * for the session. The struct may be a single element, or it may be + * the head of a list, in which case each element of the list is + * processed. It may also be NULL, in which case streams should be added + * later using srtp_add_stream(). The final element of the list @b must + * have its `next' field set to NULL. + * + * @return + * - err_status_ok if creation succeded. + * - err_status_alloc_fail if allocation failed. + * - err_status_init_fail if initialization failed. + */ + +err_status_t +srtp_create(srtp_t *session, const srtp_policy_t *policy); + + +/** + * @brief srtp_add_stream() allocates and initializes an SRTP stream + * within a given SRTP session. + * + * The function call srtp_add_stream(session, policy) allocates and + * initializes a new SRTP stream within a given, previously created + * session, applying the policy given as the other argument to that + * stream. + * + * @return values: + * - err_status_ok if stream creation succeded. + * - err_status_alloc_fail if stream allocation failed + * - err_status_init_fail if stream initialization failed. + */ + +err_status_t +srtp_add_stream(srtp_t session, + const srtp_policy_t *policy); + + +/** + * @brief srtp_remove_stream() deallocates an SRTP stream. + * + * The function call srtp_remove_stream(session, ssrc) removes + * the SRTP stream with the SSRC value ssrc from the SRTP session + * context given by the argument session. + * + * @param session is the SRTP session from which the stream + * will be removed. + * + * @param ssrc is the SSRC value of the stream to be removed. + * + * @warning Wildcard SSRC values cannot be removed from a + * session. + * + * @return + * - err_status_ok if the stream deallocation succeded. + * - [other] otherwise. + * + */ + +err_status_t +srtp_remove_stream(srtp_t session, uint32_t ssrc); + +/** + * @brief crypto_policy_set_rtp_default() sets a crypto policy + * structure to the SRTP default policy for RTP protection. + * + * @param p is a pointer to the policy strucutre to be set to the + * default policy. + * + * The function call crypto_policy_set_rtp_default(&p) sets the + * crypto_policy_t at location p to the SRTP default policy for RTP + * protection, as defined in the specification. This function is a + * convenience that helps to avoid dealing directly with the policy + * data structure. You are encouraged to initialize policy elements + * with this function call. Doing so may allow your code to be + * forward compatible with later versions of libSRTP that include more + * elements in the crypto_policy_t datatype. + * + * @return void. + * + */ + +void +crypto_policy_set_rtp_default(crypto_policy_t *p); + +/** + * @brief crypto_policy_set_rtcp_default() sets a crypto policy + * structure to the SRTP default policy for RTCP protection. + * + * @param p is a pointer to the policy strucutre to be set to the + * default policy. + * + * The function call crypto_policy_set_rtcp_default(&p) sets the + * crypto_policy_t at location p to the SRTP default policy for RTCP + * protection, as defined in the specification. This function is a + * convenience that helps to avoid dealing directly with the policy + * data structure. You are encouraged to initialize policy elements + * with this function call. Doing so may allow your code to be + * forward compatible with later versions of libSRTP that include more + * elements in the crypto_policy_t datatype. + * + * @return void. + * + */ + +void +crypto_policy_set_rtcp_default(crypto_policy_t *p); + +/** + * @brief crypto_policy_set_aes_cm_128_hmac_sha1_80() sets a crypto + * policy structure to the SRTP default policy for RTP protection. + * + * @param p is a pointer to the policy strucutre to be set to the + * default policy. + * + * The function crypto_policy_set_aes_cm_128_hmac_sha1_80() is a + * synonym for crypto_policy_set_rtp_default(). It conforms to the + * naming convention used in + * http://www.ietf.org/internet-drafts/draft-ietf-mmusic-sdescriptions-12.txt + * + * @return void. + * + */ + +#define crypto_policy_set_aes_cm_128_hmac_sha1_80(p) crypto_policy_set_rtp_default(p) + + +/** + * @brief crypto_policy_set_aes_cm_128_hmac_sha1_32() sets a crypto + * policy structure to a short-authentication tag policy + * + * @param p is a pointer to the policy strucutre to be set to the + * default policy. + * + * The function call crypto_policy_set_aes_cm_128_hmac_sha1_32(&p) + * sets the crypto_policy_t at location p to use policy + * AES_CM_128_HMAC_SHA1_32 as defined in + * draft-ietf-mmusic-sdescriptions-12.txt. This policy uses AES-128 + * Counter Mode encryption and HMAC-SHA1 authentication, with an + * authentication tag that is only 32 bits long. This length is + * considered adequate only for protecting audio and video media that + * use a stateless playback function. See Section 7.5 of RFC 3711 + * (http://www.ietf.org/rfc/rfc3711.txt). + * + * This function is a convenience that helps to avoid dealing directly + * with the policy data structure. You are encouraged to initialize + * policy elements with this function call. Doing so may allow your + * code to be forward compatible with later versions of libSRTP that + * include more elements in the crypto_policy_t datatype. + * + * @warning This crypto policy is intended for use in SRTP, but not in + * SRTCP. It is recommended that a policy that uses longer + * authentication tags be used for SRTCP. See Section 7.5 of RFC 3711 + * (http://www.ietf.org/rfc/rfc3711.txt). + * + * @return void. + * + */ + +void +crypto_policy_set_aes_cm_128_hmac_sha1_32(crypto_policy_t *p); + + + +/** + * @brief crypto_policy_set_aes_cm_128_null_auth() sets a crypto + * policy structure to an encryption-only policy + * + * @param p is a pointer to the policy strucutre to be set to the + * default policy. + * + * The function call crypto_policy_set_aes_cm_128_null_auth(&p) sets + * the crypto_policy_t at location p to use the SRTP default cipher + * (AES-128 Counter Mode), but to use no authentication method. This + * policy is NOT RECOMMENDED unless it is unavoidable; see Section 7.5 + * of RFC 3711 (http://www.ietf.org/rfc/rfc3711.txt). + * + * This function is a convenience that helps to avoid dealing directly + * with the policy data structure. You are encouraged to initialize + * policy elements with this function call. Doing so may allow your + * code to be forward compatible with later versions of libSRTP that + * include more elements in the crypto_policy_t datatype. + * + * @warning This policy is NOT RECOMMENDED for SRTP unless it is + * unavoidable, and it is NOT RECOMMENDED at all for SRTCP; see + * Section 7.5 of RFC 3711 (http://www.ietf.org/rfc/rfc3711.txt). + * + * @return void. + * + */ + +void +crypto_policy_set_aes_cm_128_null_auth(crypto_policy_t *p); + + +/** + * @brief crypto_policy_set_null_cipher_hmac_sha1_80() sets a crypto + * policy structure to an authentication-only policy + * + * @param p is a pointer to the policy strucutre to be set to the + * default policy. + * + * The function call crypto_policy_set_null_cipher_hmac_sha1_80(&p) + * sets the crypto_policy_t at location p to use HMAC-SHA1 with an 80 + * bit authentication tag to provide message authentication, but to + * use no encryption. This policy is NOT RECOMMENDED for SRTP unless + * there is a requirement to forego encryption. + * + * This function is a convenience that helps to avoid dealing directly + * with the policy data structure. You are encouraged to initialize + * policy elements with this function call. Doing so may allow your + * code to be forward compatible with later versions of libSRTP that + * include more elements in the crypto_policy_t datatype. + * + * @warning This policy is NOT RECOMMENDED for SRTP unless there is a + * requirement to forego encryption. + * + * @return void. + * + */ + +void +crypto_policy_set_null_cipher_hmac_sha1_80(crypto_policy_t *p); + +/** + * @brief srtp_dealloc() deallocates storage for an SRTP session + * context. + * + * The function call srtp_dealloc(s) deallocates storage for the + * SRTP session context s. This function should be called no more + * than one time for each of the contexts allocated by the function + * srtp_create(). + * + * @param s is the srtp_t for the session to be deallocated. + * + * @return + * - err_status_ok if there no problems. + * - err_status_dealloc_fail a memory deallocation failure occured. + */ + +err_status_t +srtp_dealloc(srtp_t s); + + + +/** + * @} + */ + + +/* + * the following declarations are libSRTP internal functions + */ + +/* + * srtp_get_stream(ssrc) returns a pointer to the stream corresponding + * to ssrc, or NULL if no stream exists for that ssrc + */ + +srtp_stream_t +srtp_get_stream(srtp_t srtp, uint32_t ssrc); + + + +/* + * libsrtp internal datatypes + */ + +typedef enum direction_t { + dir_unknown = 0, + dir_srtp_sender = 1, + dir_srtp_receiver = 2 +} direction_t; + +/* + * an srtp_stream_t has its own SSRC, encryption key, authentication + * key, sequence number, and replay database + * + * note that the keys might not actually be unique, in which case the + * cipher_t and auth_t pointers will point to the same structures + */ + +typedef struct srtp_stream_ctx_t { + uint32_t ssrc; + cipher_t *rtp_cipher; + auth_t *rtp_auth; + rdbx_t rtp_rdbx; + sec_serv_t rtp_services; + cipher_t *rtcp_cipher; + auth_t *rtcp_auth; + rdb_t rtcp_rdb; + sec_serv_t rtcp_services; + key_limit_ctx_t *limit; + direction_t direction; + struct srtp_stream_ctx_t *next; /* linked list of streams */ +} srtp_stream_ctx_t; + + +/* + * an srtp_ctx_t holds a stream list and a service description + */ + +typedef struct srtp_ctx_t { + srtp_stream_ctx_t *stream_list; /* linked list of streams */ + srtp_stream_ctx_t *stream_template; /* act as template for other streams */ +} srtp_ctx_t; + + + +/** + * @defgroup SRTCP Secure RTCP + * @ingroup SRTP + * + * @brief Secure RTCP functions are used to protect RTCP traffic. + * + * RTCP is the control protocol for RTP. libSRTP protects RTCP + * traffic in much the same way as it does RTP traffic. The function + * srtp_protect_rtcp() applies cryptographic protections to outbound + * RTCP packets, and srtp_unprotect_rtcp() verifies the protections on + * inbound RTCP packets. + * + * A note on the naming convention: srtp_protect_rtcp() has an srtp_t + * as its first argument, and thus has `srtp_' as its prefix. The + * trailing `_rtcp' indicates the protocol on which it acts. + * + * @{ + */ + +/** + * @brief srtp_protect_rtcp() is the Secure RTCP sender-side packet + * processing function. + * + * The function call srtp_protect_rtcp(ctx, rtp_hdr, len_ptr) applies + * SRTCP protection to the RTCP packet rtcp_hdr (which has length + * *len_ptr) using the SRTP session context ctx. If err_status_ok is + * returned, then rtp_hdr points to the resulting SRTCP packet and + * *len_ptr is the number of octets in that packet; otherwise, no + * assumptions should be made about the value of either data elements. + * + * @warning This function assumes that it can write the authentication + * tag into the location in memory immediately following the RTCP + * packet, and assumes that the RTCP packet is aligned on a 32-bit + * boundary. + * + * @param ctx is the SRTP context to use in processing the packet. + * + * @param rtcp_hdr is a pointer to the RTCP packet (before the call); after + * the function returns, it points to the srtp packet. + * + * @param pkt_octet_len is a pointer to the length in octets of the + * complete RTCP packet (header and body) before the function call, + * and of the complete SRTCP packet after the call, if err_status_ok + * was returned. Otherwise, the value of the data to which it points + * is undefined. + * + * @return + * - err_status_ok if there were no problems. + * - [other] if there was a failure in + * the cryptographic mechanisms. + */ + + +err_status_t +srtp_protect_rtcp(srtp_t ctx, void *rtcp_hdr, int *pkt_octet_len); + +/** + * @brief srtp_unprotect_rtcp() is the Secure RTCP receiver-side packet + * processing function. + * + * The function call srtp_unprotect_rtcp(ctx, srtp_hdr, len_ptr) + * verifies the Secure RTCP protection of the SRTCP packet pointed to + * by srtcp_hdr (which has length *len_ptr), using the SRTP session + * context ctx. If err_status_ok is returned, then srtcp_hdr points + * to the resulting RTCP packet and *len_ptr is the number of octets + * in that packet; otherwise, no assumptions should be made about the + * value of either data elements. + * + * @warning This function assumes that the SRTCP packet is aligned on a + * 32-bit boundary. + * + * @param ctx is a pointer to the srtp_t which applies to the + * particular packet. + * + * @param srtcp_hdr is a pointer to the header of the SRTCP packet + * (before the call). After the function returns, it points to the + * rtp packet if err_status_ok was returned; otherwise, the value of + * the data to which it points is undefined. + * + * @param pkt_octet_len is a pointer to the length in octets of the + * complete SRTCP packet (header and body) before the function call, + * and of the complete rtp packet after the call, if err_status_ok was + * returned. Otherwise, the value of the data to which it points is + * undefined. + * + * @return + * - err_status_ok if the RTCP packet is valid. + * - err_status_auth_fail if the SRTCP packet failed the message + * authentication check. + * - err_status_replay_fail if the SRTCP packet is a replay (e.g. has + * already been processed and accepted). + * - [other] if there has been an error in the cryptographic mechanisms. + * + */ + +err_status_t +srtp_unprotect_rtcp(srtp_t ctx, void *srtcp_hdr, int *pkt_octet_len); + +/** + * @} + */ + +/** + * @defgroup SRTPevents SRTP events and callbacks + * @ingroup SRTP + * + * @brief libSRTP can use a user-provided callback function to + * handle events. + * + * + * libSRTP allows a user to provide a callback function to handle + * events that need to be dealt with outside of the data plane (see + * the enum srtp_event_t for a description of these events). Dealing + * with these events is not a strict necessity; they are not + * security-critical, but the application may suffer if they are not + * handled. The function srtp_set_event_handler() is used to provide + * the callback function. + * + * A default event handler that merely reports on the events as they + * happen is included. It is also possible to set the event handler + * function to NULL, in which case all events will just be silently + * ignored. + * + * @{ + */ + +/** + * @brief srtp_event_t defines events that need to be handled + * + * The enum srtp_event_t defines events that need to be handled + * outside the `data plane', such as SSRC collisions and + * key expirations. + * + * When a key expires or the maximum number of packets has been + * reached, an SRTP stream will enter an `expired' state in which no + * more packets can be protected or unprotected. When this happens, + * it is likely that you will want to either deallocate the stream + * (using srtp_stream_dealloc()), and possibly allocate a new one. + * + * When an SRTP stream expires, the other streams in the same session + * are unaffected, unless key sharing is used by that stream. In the + * latter case, all of the streams in the session will expire. + */ + +typedef enum { + event_ssrc_collision, /**< + * An SSRC collision occured. + */ + event_key_soft_limit, /**< An SRTP stream reached the soft key + * usage limit and will expire soon. + */ + event_key_hard_limit, /**< An SRTP stream reached the hard + * key usage limit and has expired. + */ + event_packet_index_limit /**< An SRTP stream reached the hard + * packet limit (2^48 packets). + */ +} srtp_event_t; + +/** + * @brief srtp_event_data_t is the structure passed as a callback to + * the event handler function + * + * The struct srtp_event_data_t holds the data passed to the event + * handler function. + */ + +typedef struct srtp_event_data_t { + srtp_t session; /**< The session in which the event happend. */ + srtp_stream_t stream; /**< The stream in which the event happend. */ + srtp_event_t event; /**< An enum indicating the type of event. */ +} srtp_event_data_t; + +/** + * @brief srtp_event_handler_func_t is the function prototype for + * the event handler. + * + * The typedef srtp_event_handler_func_t is the prototype for the + * event handler function. It has as its only argument an + * srtp_event_data_t which describes the event that needs to be handled. + * There can only be a single, global handler for all events in + * libSRTP. + */ + +typedef void (srtp_event_handler_func_t)(srtp_event_data_t *data); + +/** + * @brief sets the event handler to the function supplied by the caller. + * + * The function call srtp_install_event_handler(func) sets the event + * handler function to the value func. The value NULL is acceptable + * as an argument; in this case, events will be ignored rather than + * handled. + * + * @param func is a pointer to a fuction that takes an srtp_event_data_t + * pointer as an argument and returns void. This function + * will be used by libSRTP to handle events. + */ + +err_status_t +srtp_install_event_handler(srtp_event_handler_func_t func); + +/** + * @} + */ + +/* + * srtp_handle_event(srtp, srtm, evnt) calls the event handling + * function, if there is one. + * + * This macro is not included in the documentation as it is + * an internal-only function. + */ + +#define srtp_handle_event(srtp, strm, evnt) \ + if(srtp_event_handler) { \ + srtp_event_data_t data; \ + data.session = srtp; \ + data.stream = strm; \ + data.event = evnt; \ + srtp_event_handler(&data); \ +} + +/* + * an srtp_hdr_t represents the srtp header + * + * in this implementation, an srtp_hdr_t is assumed to be 32-bit aligned + * + * (note that this definition follows that of RFC 1889 Appendix A, but + * is not identical) + */ + +#ifndef WORDS_BIGENDIAN + +typedef struct { + unsigned cc:4; /* CSRC count */ + unsigned x:1; /* header extension flag */ + unsigned p:1; /* padding flag */ + unsigned version:2; /* protocol version */ + unsigned pt:7; /* payload type */ + unsigned m:1; /* marker bit */ + uint16_t seq; /* sequence number */ + uint32_t ts; /* timestamp */ + uint32_t ssrc; /* synchronization source */ +} srtp_hdr_t; + +#else /* BIG_ENDIAN */ + +typedef struct { + unsigned version:2; /* protocol version */ + unsigned p:1; /* padding flag */ + unsigned x:1; /* header extension flag */ + unsigned cc:4; /* CSRC count */ + unsigned m:1; /* marker bit */ + unsigned pt:7; /* payload type */ + uint16_t seq; /* sequence number */ + uint32_t ts; /* timestamp */ + uint32_t ssrc; /* synchronization source */ +} srtp_hdr_t; + +#endif + +typedef struct { + uint16_t profile_specific; /* profile-specific info */ + uint16_t length; /* number of 32-bit words in extension */ +} srtp_hdr_xtnd_t; + + +/* + * srtcp_hdr_t represents a secure rtcp header + * + * in this implementation, an srtcp header is assumed to be 32-bit + * alinged + */ + +#ifndef WORDS_BIGENDIAN + +typedef struct { + unsigned char rc:5; /* reception report count */ + unsigned char p:1; /* padding flag */ + unsigned char version:2; /* protocol version */ + unsigned char pt:8; /* payload type */ + uint16_t len; /* length */ + uint32_t ssrc; /* synchronization source */ +} srtcp_hdr_t; + +typedef struct { + unsigned int index:31; /* srtcp packet index in network order! */ + unsigned int e:1; /* encrypted? 1=yes */ + /* optional mikey/etc go here */ + /* and then the variable-length auth tag */ +} srtcp_trailer_t; + + +#else /* BIG_ENDIAN */ + +typedef struct { + unsigned char version:2; /* protocol version */ + unsigned char p:1; /* padding flag */ + unsigned char rc:5; /* reception report count */ + unsigned char pt:8; /* payload type */ + uint16_t len; /* length */ + uint32_t ssrc; /* synchronization source */ +} srtcp_hdr_t; + +typedef struct { + unsigned int version:2; /* protocol version */ + unsigned int p:1; /* padding flag */ + unsigned int count:5; /* varies by packet type */ + unsigned int pt:8; /* payload type */ + uint16_t length; /* len of uint32s of packet less header */ +} rtcp_common_t; + +typedef struct { + unsigned int e:1; /* encrypted? 1=yes */ + unsigned int index:31; /* srtcp packet index */ + /* optional mikey/etc go here */ + /* and then the variable-length auth tag */ +} srtcp_trailer_t; + +#endif + +/* in host order, so outside the #if */ +#define SRTCP_E_BIT 0x80000000 +/* for byte-access */ +#define SRTCP_E_BYTE_BIT 0x80 +#define SRTCP_INDEX_MASK 0x7fffffff + +#ifdef __cplusplus +} +#endif + +#endif /* SRTP_H */ diff --git a/libs/srtp/include/ut_sim.h b/libs/srtp/include/ut_sim.h new file mode 100644 index 0000000000..deed553b54 --- /dev/null +++ b/libs/srtp/include/ut_sim.h @@ -0,0 +1,80 @@ +/* + * ut-sim.h + * + * an unreliable transport simulator + * (for testing replay databases and suchlike) + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + + +#ifndef UT_SIM_H +#define UT_SIM_H + +#include "integers.h" /* for uint32_t */ + +#define UT_BUF 160 /* maximum amount of packet reorder */ + +typedef struct { + uint32_t index; + uint32_t buffer[UT_BUF]; +} ut_connection; + +/* + * ut_init(&u) initializes the ut_connection + * + * this function should always be the first one called on a new + * ut_connection + */ + +void +ut_init(ut_connection *utc); + +/* + * ut_next_index(&u) returns the next index from the simulated + * unreliable connection + */ + +uint32_t +ut_next_index(ut_connection *utc); + + +#endif /* UT_SIM_H */ diff --git a/libs/srtp/install-sh b/libs/srtp/install-sh new file mode 100755 index 0000000000..e9de23842d --- /dev/null +++ b/libs/srtp/install-sh @@ -0,0 +1,251 @@ +#!/bin/sh +# +# install - install a program, script, or datafile +# This comes from X11R5 (mit/util/scripts/install.sh). +# +# Copyright 1991 by the Massachusetts Institute of Technology +# +# Permission to use, copy, modify, distribute, and sell this software and its +# documentation for any purpose is hereby granted without fee, provided that +# the above copyright notice appear in all copies and that both that +# copyright notice and this permission notice appear in supporting +# documentation, and that the name of M.I.T. not be used in advertising or +# publicity pertaining to distribution of the software without specific, +# written prior permission. M.I.T. makes no representations about the +# suitability of this software for any purpose. It is provided "as is" +# without express or implied warranty. +# +# Calling this script install-sh is preferred over install.sh, to prevent +# `make' implicit rules from creating a file called install from it +# when there is no Makefile. +# +# This script is compatible with the BSD install script, but was written +# from scratch. It can only install one file at a time, a restriction +# shared with many OS's install programs. + + +# set DOITPROG to echo to test this script + +# Don't use :- since 4.3BSD and earlier shells don't like it. +doit="${DOITPROG-}" + + +# put in absolute paths if you don't have them in your path; or use env. vars. + +mvprog="${MVPROG-mv}" +cpprog="${CPPROG-cp}" +chmodprog="${CHMODPROG-chmod}" +chownprog="${CHOWNPROG-chown}" +chgrpprog="${CHGRPPROG-chgrp}" +stripprog="${STRIPPROG-strip}" +rmprog="${RMPROG-rm}" +mkdirprog="${MKDIRPROG-mkdir}" + +transformbasename="" +transform_arg="" +instcmd="$mvprog" +chmodcmd="$chmodprog 0755" +chowncmd="" +chgrpcmd="" +stripcmd="" +rmcmd="$rmprog -f" +mvcmd="$mvprog" +src="" +dst="" +dir_arg="" + +while [ x"$1" != x ]; do + case $1 in + -c) instcmd="$cpprog" + shift + continue;; + + -d) dir_arg=true + shift + continue;; + + -m) chmodcmd="$chmodprog $2" + shift + shift + continue;; + + -o) chowncmd="$chownprog $2" + shift + shift + continue;; + + -g) chgrpcmd="$chgrpprog $2" + shift + shift + continue;; + + -s) stripcmd="$stripprog" + shift + continue;; + + -t=*) transformarg=`echo $1 | sed 's/-t=//'` + shift + continue;; + + -b=*) transformbasename=`echo $1 | sed 's/-b=//'` + shift + continue;; + + *) if [ x"$src" = x ] + then + src=$1 + else + # this colon is to work around a 386BSD /bin/sh bug + : + dst=$1 + fi + shift + continue;; + esac +done + +if [ x"$src" = x ] +then + echo "install: no input file specified" + exit 1 +else + true +fi + +if [ x"$dir_arg" != x ]; then + dst=$src + src="" + + if [ -d $dst ]; then + instcmd=: + chmodcmd="" + else + instcmd=mkdir + fi +else + +# Waiting for this to be detected by the "$instcmd $src $dsttmp" command +# might cause directories to be created, which would be especially bad +# if $src (and thus $dsttmp) contains '*'. + + if [ -f $src -o -d $src ] + then + true + else + echo "install: $src does not exist" + exit 1 + fi + + if [ x"$dst" = x ] + then + echo "install: no destination specified" + exit 1 + else + true + fi + +# If destination is a directory, append the input filename; if your system +# does not like double slashes in filenames, you may need to add some logic + + if [ -d $dst ] + then + dst="$dst"/`basename $src` + else + true + fi +fi + +## this sed command emulates the dirname command +dstdir=`echo $dst | sed -e 's,[^/]*$,,;s,/$,,;s,^$,.,'` + +# Make sure that the destination directory exists. +# this part is taken from Noah Friedman's mkinstalldirs script + +# Skip lots of stat calls in the usual case. +if [ ! -d "$dstdir" ]; then +defaultIFS=' +' +IFS="${IFS-${defaultIFS}}" + +oIFS="${IFS}" +# Some sh's can't handle IFS=/ for some reason. +IFS='%' +set - `echo ${dstdir} | sed -e 's@/@%@g' -e 's@^%@/@'` +IFS="${oIFS}" + +pathcomp='' + +while [ $# -ne 0 ] ; do + pathcomp="${pathcomp}${1}" + shift + + if [ ! -d "${pathcomp}" ] ; + then + $mkdirprog "${pathcomp}" + else + true + fi + + pathcomp="${pathcomp}/" +done +fi + +if [ x"$dir_arg" != x ] +then + $doit $instcmd $dst && + + if [ x"$chowncmd" != x ]; then $doit $chowncmd $dst; else true ; fi && + if [ x"$chgrpcmd" != x ]; then $doit $chgrpcmd $dst; else true ; fi && + if [ x"$stripcmd" != x ]; then $doit $stripcmd $dst; else true ; fi && + if [ x"$chmodcmd" != x ]; then $doit $chmodcmd $dst; else true ; fi +else + +# If we're going to rename the final executable, determine the name now. + + if [ x"$transformarg" = x ] + then + dstfile=`basename $dst` + else + dstfile=`basename $dst $transformbasename | + sed $transformarg`$transformbasename + fi + +# don't allow the sed command to completely eliminate the filename + + if [ x"$dstfile" = x ] + then + dstfile=`basename $dst` + else + true + fi + +# Make a temp file name in the proper directory. + + dsttmp=$dstdir/#inst.$$# + +# Move or copy the file name to the temp name + + $doit $instcmd $src $dsttmp && + + trap "rm -f ${dsttmp}" 0 && + +# and set any options; do chmod last to preserve setuid bits + +# If any of these fail, we abort the whole thing. If we want to +# ignore errors from any of these, just make sure not to ignore +# errors from the above "$doit $instcmd $src $dsttmp" command. + + if [ x"$chowncmd" != x ]; then $doit $chowncmd $dsttmp; else true;fi && + if [ x"$chgrpcmd" != x ]; then $doit $chgrpcmd $dsttmp; else true;fi && + if [ x"$stripcmd" != x ]; then $doit $stripcmd $dsttmp; else true;fi && + if [ x"$chmodcmd" != x ]; then $doit $chmodcmd $dsttmp; else true;fi && + +# Now rename the file to the real destination. + + $doit $rmcmd -f $dstdir/$dstfile && + $doit $mvcmd $dsttmp $dstdir/$dstfile + +fi && + + +exit 0 diff --git a/libs/srtp/srtp/CVS/Entries b/libs/srtp/srtp/CVS/Entries new file mode 100644 index 0000000000..8f7789bab1 --- /dev/null +++ b/libs/srtp/srtp/CVS/Entries @@ -0,0 +1,2 @@ +/srtp.c/1.15/Thu Mar 9 21:17:00 2006// +D diff --git a/libs/srtp/srtp/CVS/Repository b/libs/srtp/srtp/CVS/Repository new file mode 100644 index 0000000000..a3933d5df8 --- /dev/null +++ b/libs/srtp/srtp/CVS/Repository @@ -0,0 +1 @@ +srtp/srtp diff --git a/libs/srtp/srtp/CVS/Root b/libs/srtp/srtp/CVS/Root new file mode 100644 index 0000000000..b4b74237d9 --- /dev/null +++ b/libs/srtp/srtp/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@cvs.sourceforge.net:/cvsroot/srtp diff --git a/libs/srtp/srtp/srtp.c b/libs/srtp/srtp/srtp.c new file mode 100644 index 0000000000..c1cd88c2ce --- /dev/null +++ b/libs/srtp/srtp/srtp.c @@ -0,0 +1,1779 @@ +/* + * srtp.c + * + * the secure real-time transport protocol + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#include "srtp.h" +#include "aes_icm.h" /* aes_icm is used in the KDF */ +#include "alloc.h" /* for crypto_alloc() */ + +#ifndef SRTP_KERNEL +# include <limits.h> +# ifdef HAVE_NETINET_IN_H +# include <netinet/in.h> +# elif defined(HAVE_WINSOCK2_H) +# include <winsock2.h> +# endif +#endif /* ! SRTP_KERNEL */ + + +extern cipher_type_t aes_icm; +extern auth_type_t tmmhv2; + +/* the debug module for srtp */ + +debug_module_t mod_srtp = { + 0, /* debugging is off by default */ + "srtp" /* printable name for module */ +}; + +#define octets_in_rtp_header 12 +#define uint32s_in_rtp_header 3 +#define octets_in_rtcp_header 8 +#define uint32s_in_rtcp_header 2 + + +err_status_t +srtp_stream_alloc(srtp_stream_ctx_t **str_ptr, + const srtp_policy_t *p) { + srtp_stream_ctx_t *str; + err_status_t stat; + + /* + * This function allocates the stream context, rtp and rtcp ciphers + * and auth functions, and key limit structure. If there is a + * failure during allocation, we free all previously allocated + * memory and return a failure code. The code could probably + * be improved, but it works and should be clear. + */ + + /* allocate srtp stream and set str_ptr */ + str = (srtp_stream_ctx_t *) crypto_alloc(sizeof(srtp_stream_ctx_t)); + if (str == NULL) + return err_status_alloc_fail; + *str_ptr = str; + + /* allocate cipher */ + stat = crypto_kernel_alloc_cipher(p->rtp.cipher_type, + &str->rtp_cipher, + p->rtp.cipher_key_len); + if (stat) { + crypto_free(str); + return stat; + } + + /* allocate auth function */ + stat = crypto_kernel_alloc_auth(p->rtp.auth_type, + &str->rtp_auth, + p->rtp.auth_key_len, + p->rtp.auth_tag_len); + if (stat) { + cipher_dealloc(str->rtp_cipher); + crypto_free(str); + return stat; + } + + /* allocate key limit structure */ + str->limit = crypto_alloc(sizeof(key_limit_ctx_t)); + if (str->limit == NULL) { + auth_dealloc(str->rtp_auth); + cipher_dealloc(str->rtp_cipher); + crypto_free(str); + return err_status_alloc_fail; + } + + /* + * ...and now the RTCP-specific initialization - first, allocate + * the cipher + */ + stat = crypto_kernel_alloc_cipher(p->rtcp.cipher_type, + &str->rtcp_cipher, + p->rtcp.cipher_key_len); + if (stat) { + auth_dealloc(str->rtp_auth); + cipher_dealloc(str->rtp_cipher); + crypto_free(str->limit); + crypto_free(str); + return stat; + } + + /* allocate auth function */ + stat = crypto_kernel_alloc_auth(p->rtcp.auth_type, + &str->rtcp_auth, + p->rtcp.auth_key_len, + p->rtcp.auth_tag_len); + if (stat) { + cipher_dealloc(str->rtcp_cipher); + auth_dealloc(str->rtp_auth); + cipher_dealloc(str->rtp_cipher); + crypto_free(str->limit); + crypto_free(str); + return stat; + } + + return err_status_ok; +} + +err_status_t +srtp_stream_dealloc(srtp_t session, srtp_stream_ctx_t *stream) { + err_status_t status; + + /* + * we use a conservative deallocation strategy - if any deallocation + * fails, then we report that fact without trying to deallocate + * anything else + */ + + /* deallocate cipher, if it is not the same as that in template */ + if (session->stream_template + && stream->rtp_cipher == session->stream_template->rtp_cipher) { + /* do nothing */ + } else { + status = cipher_dealloc(stream->rtp_cipher); + if (status) + return status; + } + + /* deallocate auth function, if it is not the same as that in template */ + if (session->stream_template + && stream->rtp_auth == session->stream_template->rtp_auth) { + /* do nothing */ + } else { + status = auth_dealloc(stream->rtp_auth); + if (status) + return status; + } + + /* deallocate key usage limit, if it is not the same as that in template */ + if (session->stream_template + && stream->limit == session->stream_template->limit) { + /* do nothing */ + } else { + crypto_free(stream->limit); + } + + /* + * deallocate rtcp cipher, if it is not the same as that in + * template + */ + if (session->stream_template + && stream->rtcp_cipher == session->stream_template->rtcp_cipher) { + /* do nothing */ + } else { + status = cipher_dealloc(stream->rtcp_cipher); + if (status) + return status; + } + + /* + * deallocate rtcp auth function, if it is not the same as that in + * template + */ + if (session->stream_template + && stream->rtcp_auth == session->stream_template->rtcp_auth) { + /* do nothing */ + } else { + status = auth_dealloc(stream->rtcp_auth); + if (status) + return status; + } + + /* deallocate srtp stream context */ + crypto_free(stream); + + return err_status_ok; +} + + +/* + * srtp_stream_clone(stream_template, new) allocates a new stream and + * initializes it using the cipher and auth of the stream_template + * + * the only unique data in a cloned stream is the replay database and + * the SSRC + */ + +err_status_t +srtp_stream_clone(const srtp_stream_ctx_t *stream_template, + uint32_t ssrc, + srtp_stream_ctx_t **str_ptr) { + err_status_t status; + srtp_stream_ctx_t *str; + + debug_print(mod_srtp, "cloning stream (SSRC: 0x%08x)", ssrc); + + /* allocate srtp stream and set str_ptr */ + str = (srtp_stream_ctx_t *) crypto_alloc(sizeof(srtp_stream_ctx_t)); + if (str == NULL) + return err_status_alloc_fail; + *str_ptr = str; + + /* set cipher and auth pointers to those of the template */ + str->rtp_cipher = stream_template->rtp_cipher; + str->rtp_auth = stream_template->rtp_auth; + str->rtcp_cipher = stream_template->rtcp_cipher; + str->rtcp_auth = stream_template->rtcp_auth; + + /* set key limit to point to that of the template */ + status = key_limit_clone(stream_template->limit, &str->limit); + if (status) + return status; + + /* initialize replay databases */ + rdbx_init(&str->rtp_rdbx); + rdb_init(&str->rtcp_rdb); + + /* set ssrc to that provided */ + str->ssrc = ssrc; + + /* set direction and security services */ + str->direction = stream_template->direction; + str->rtp_services = stream_template->rtp_services; + str->rtcp_services = stream_template->rtcp_services; + + /* defensive coding */ + str->next = NULL; + + return err_status_ok; +} + + +/* + * key derivation functions, internal to libSRTP + * + * srtp_kdf_t is a key derivation context + * + * srtp_kdf_init(&kdf, k) initializes kdf with the key k + * + * srtp_kdf_generate(&kdf, l, kl, keylen) derives the key + * corresponding to label l and puts it into kl; the length + * of the key in octets is provided as keylen. this function + * should be called once for each subkey that is derived. + * + * srtp_kdf_clear(&kdf) zeroizes the kdf state + */ + +typedef enum { + label_rtp_encryption = 0x00, + label_rtp_msg_auth = 0x01, + label_rtp_salt = 0x02, + label_rtcp_encryption = 0x03, + label_rtcp_msg_auth = 0x04, + label_rtcp_salt = 0x05 +} srtp_prf_label; + + +/* + * srtp_kdf_t represents a key derivation function. The SRTP + * default KDF is the only one implemented at present. + */ + +typedef struct { + aes_icm_ctx_t c; /* cipher used for key derivation */ +} srtp_kdf_t; + +err_status_t +srtp_kdf_init(srtp_kdf_t *kdf, const uint8_t key[30]) { + + aes_icm_context_init(&kdf->c, key); + + return err_status_ok; +} + +err_status_t +srtp_kdf_generate(srtp_kdf_t *kdf, srtp_prf_label label, + uint8_t *key, int length) { + + v128_t nonce; + + /* set eigth octet of nonce to <label>, set the rest of it to zero */ + v128_set_to_zero(&nonce); + nonce.v8[7] = label; + + aes_icm_set_iv(&kdf->c, &nonce); + + /* generate keystream output */ + aes_icm_output(&kdf->c, key, length); + + return err_status_ok; +} + +err_status_t +srtp_kdf_clear(srtp_kdf_t *kdf) { + + /* zeroize aes context */ + octet_string_set_to_zero((uint8_t *)kdf, sizeof(srtp_kdf_t)); + + return err_status_ok; +} + +/* + * end of key derivation functions + */ + +#define MAX_SRTP_KEY_LEN 256 + + + +err_status_t +srtp_stream_init(srtp_stream_ctx_t *srtp, + const srtp_policy_t *p) { + err_status_t stat; + srtp_kdf_t kdf; + uint8_t tmp_key[MAX_SRTP_KEY_LEN]; + uint8_t *key = p->key; + + debug_print(mod_srtp, "initializing stream (SSRC: 0x%08x)", + p->ssrc.value); + + /* initialize replay database */ + rdbx_init(&srtp->rtp_rdbx); + + /* initialize key limit to maximum value */ +#ifdef NO_64BIT_MATH +{ + uint64_t temp; + temp = make64(UINT_MAX,UINT_MAX); + key_limit_set(srtp->limit, temp); +} +#else + key_limit_set(srtp->limit, 0xffffffffffffLL); +#endif + + /* set the SSRC value */ + srtp->ssrc = htonl(p->ssrc.value); + + /* set the security service flags */ + srtp->rtp_services = p->rtp.sec_serv; + srtp->rtcp_services = p->rtcp.sec_serv; + + /* + * set direction to unknown - this flag gets checked in srtp_protect(), + * srtp_unprotect(), srtp_protect_rtcp(), and srtp_unprotect_rtcp(), and + * gets set appropriately if it is set to unknown. + */ + srtp->direction = dir_unknown; + + /* initialize KDF state */ + srtp_kdf_init(&kdf, key); + + /* generate encryption key */ + srtp_kdf_generate(&kdf, label_rtp_encryption, + tmp_key, cipher_get_key_length(srtp->rtp_cipher)); + /* + * if the cipher in the srtp context is aes_icm, then we need + * to generate the salt value + */ + if (srtp->rtp_cipher->type == &aes_icm) { + /* FIX!!! this is really the cipher key length; rest is salt */ + int base_key_len = 16; + int salt_len = cipher_get_key_length(srtp->rtp_cipher) - base_key_len; + + debug_print(mod_srtp, "found aes_icm, generating salt", NULL); + + /* generate encryption salt, put after encryption key */ + srtp_kdf_generate(&kdf, label_rtp_salt, + tmp_key + base_key_len, salt_len); + } + debug_print(mod_srtp, "cipher key: %s", + octet_string_hex_string(tmp_key, + cipher_get_key_length(srtp->rtp_cipher))); + + /* initialize cipher */ + stat = cipher_init(srtp->rtp_cipher, tmp_key, direction_any); + if (stat) { + /* zeroize temp buffer */ + octet_string_set_to_zero(tmp_key, MAX_SRTP_KEY_LEN); + return err_status_init_fail; + } + + /* generate authentication key */ + srtp_kdf_generate(&kdf, label_rtp_msg_auth, + tmp_key, auth_get_key_length(srtp->rtp_auth)); + debug_print(mod_srtp, "auth key: %s", + octet_string_hex_string(tmp_key, + auth_get_key_length(srtp->rtp_auth))); + + /* initialize auth function */ + stat = auth_init(srtp->rtp_auth, tmp_key); + if (stat) { + /* zeroize temp buffer */ + octet_string_set_to_zero(tmp_key, MAX_SRTP_KEY_LEN); + return err_status_init_fail; + } + + /* + * ...now initialize RTCP-specific structures + */ + + /* initialize replay database */ + rdb_init(&srtp->rtcp_rdb); + + /* DAM - no RTCP key limit at present */ + + /* generate encryption key */ + srtp_kdf_generate(&kdf, label_rtcp_encryption, + tmp_key, cipher_get_key_length(srtp->rtcp_cipher)); + /* + * if the cipher in the srtp context is aes_icm, then we need + * to generate the salt value + */ + if (srtp->rtcp_cipher->type == &aes_icm) { + /* FIX!!! this is really the cipher key length; rest is salt */ + int base_key_len = 16; + int salt_len = cipher_get_key_length(srtp->rtcp_cipher) - base_key_len; + + debug_print(mod_srtp, "found aes_icm, generating rtcp salt", NULL); + + /* generate encryption salt, put after encryption key */ + srtp_kdf_generate(&kdf, label_rtcp_salt, + tmp_key + base_key_len, salt_len); + } + debug_print(mod_srtp, "rtcp cipher key: %s", + octet_string_hex_string(tmp_key, + cipher_get_key_length(srtp->rtcp_cipher))); + + /* initialize cipher */ + stat = cipher_init(srtp->rtcp_cipher, tmp_key, direction_any); + if (stat) { + /* zeroize temp buffer */ + octet_string_set_to_zero(tmp_key, MAX_SRTP_KEY_LEN); + return err_status_init_fail; + } + + /* generate authentication key */ + srtp_kdf_generate(&kdf, label_rtcp_msg_auth, + tmp_key, auth_get_key_length(srtp->rtcp_auth)); + debug_print(mod_srtp, "rtcp auth key: %s", + octet_string_hex_string(tmp_key, + auth_get_key_length(srtp->rtcp_auth))); + + /* initialize auth function */ + stat = auth_init(srtp->rtcp_auth, tmp_key); + if (stat) { + /* zeroize temp buffer */ + octet_string_set_to_zero(tmp_key, MAX_SRTP_KEY_LEN); + return err_status_init_fail; + } + + /* clear memory then return */ + srtp_kdf_clear(&kdf); + octet_string_set_to_zero(tmp_key, MAX_SRTP_KEY_LEN); + return err_status_ok; + } + + + /* + * srtp_event_reporter is an event handler function that merely + * reports the events that are reported by the callbacks + */ + + void + srtp_event_reporter(srtp_event_data_t *data) { + + err_report(err_level_warning, "srtp: in stream 0x%x: ", + data->stream->ssrc); + + switch(data->event) { + case event_ssrc_collision: + err_report(err_level_warning, "\tSSRC collision\n"); + break; + case event_key_soft_limit: + err_report(err_level_warning, "\tkey usage soft limit reached\n"); + break; + case event_key_hard_limit: + err_report(err_level_warning, "\tkey usage hard limit reached\n"); + break; + case event_packet_index_limit: + err_report(err_level_warning, "\tpacket index limit reached\n"); + break; + default: + err_report(err_level_warning, "\tunknown event reported to handler\n"); + } + } + + /* + * srtp_event_handler is a global variable holding a pointer to the + * event handler function; this function is called for any unexpected + * event that needs to be handled out of the SRTP data path. see + * srtp_event_t in srtp.h for more info + * + * it is okay to set srtp_event_handler to NULL, but we set + * it to the srtp_event_reporter. + */ + + static srtp_event_handler_func_t *srtp_event_handler = srtp_event_reporter; + + err_status_t + srtp_install_event_handler(srtp_event_handler_func_t func) { + + /* + * note that we accept NULL arguments intentionally - calling this + * function with a NULL arguments removes an event handler that's + * been previously installed + */ + + /* set global event handling function */ + srtp_event_handler = func; + return err_status_ok; + } + + err_status_t + srtp_protect(srtp_ctx_t *ctx, void *rtp_hdr, int *pkt_octet_len) { + srtp_hdr_t *hdr = rtp_hdr; + uint32_t *enc_start; /* pointer to start of encrypted portion */ + uint32_t *auth_start; /* pointer to start of auth. portion */ + unsigned enc_octet_len = 0; /* number of octets in encrypted portion */ + xtd_seq_num_t est; /* estimated xtd_seq_num_t of *hdr */ + int delta; /* delta of local pkt idx and that in hdr */ + uint8_t *auth_tag = NULL; /* location of auth_tag within packet */ + err_status_t status; + int tag_len; + srtp_stream_ctx_t *stream; + int prefix_len; + + debug_print(mod_srtp, "function srtp_protect", NULL); + + /* we assume the hdr is 32-bit aligned to start */ + + /* check the packet length - it must at least contain a full header */ + if (*pkt_octet_len < octets_in_rtp_header) + return err_status_bad_param; + + /* + * look up ssrc in srtp_stream list, and process the packet with + * the appropriate stream. if we haven't seen this stream before, + * there's a template key for this srtp_session, and the cipher + * supports key-sharing, then we assume that a new stream using + * that key has just started up + */ + stream = srtp_get_stream(ctx, hdr->ssrc); + if (stream == NULL) { + if (ctx->stream_template != NULL) { + srtp_stream_ctx_t *new_stream; + + /* allocate and initialize a new stream */ + status = srtp_stream_clone(ctx->stream_template, + hdr->ssrc, &new_stream); + if (status) + return status; + + /* add new stream to the head of the stream_list */ + new_stream->next = ctx->stream_list; + ctx->stream_list = new_stream; + + /* set direction to outbound */ + new_stream->direction = dir_srtp_sender; + + /* set stream (the pointer used in this function) */ + stream = new_stream; + } else { + /* no template stream, so we return an error */ + return err_status_no_ctx; + } + } + + /* + * verify that stream is for sending traffic - this check will + * detect SSRC collisions, since a stream that appears in both + * srtp_protect() and srtp_unprotect() will fail this test in one of + * those functions. + */ + if (stream->direction != dir_srtp_sender) { + if (stream->direction == dir_unknown) { + stream->direction = dir_srtp_sender; + } else { + srtp_handle_event(ctx, stream, event_ssrc_collision); + } + } + + /* + * update the key usage limit, and check it to make sure that we + * didn't just hit either the soft limit or the hard limit, and call + * the event handler if we hit either. + */ + switch(key_limit_update(stream->limit)) { + case key_event_normal: + break; + case key_event_soft_limit: + srtp_handle_event(ctx, stream, event_key_soft_limit); + break; + case key_event_hard_limit: + srtp_handle_event(ctx, stream, event_key_hard_limit); + return err_status_key_expired; + default: + break; + } + + /* get tag length from stream */ + tag_len = auth_get_tag_length(stream->rtp_auth); + + /* + * find starting point for encryption and length of data to be + * encrypted - the encrypted portion starts after the rtp header + * extension, if present; otherwise, it starts after the last csrc, + * if any are present + * + * if we're not providing confidentiality, set enc_start to NULL + */ + if (stream->rtp_services & sec_serv_conf) { + enc_start = (uint32_t *)hdr + uint32s_in_rtp_header + hdr->cc; + if (hdr->x == 1) { + srtp_hdr_xtnd_t *xtn_hdr = (srtp_hdr_xtnd_t *)enc_start; + enc_start += (ntohs(xtn_hdr->length) + 1); + } + enc_octet_len = *pkt_octet_len - ((enc_start - (uint32_t *)hdr) << 2); + } else { + enc_start = NULL; + } + + /* + * if we're providing authentication, set the auth_start and auth_tag + * pointers to the proper locations; otherwise, set auth_start to NULL + * to indicate that no authentication is needed + */ + if (stream->rtp_services & sec_serv_auth) { + auth_start = (uint32_t *)hdr; + auth_tag = (uint8_t *)hdr + *pkt_octet_len; + } else { + auth_start = NULL; + auth_tag = NULL; + } + + /* + * estimate the packet index using the start of the replay window + * and the sequence number from the header + */ + delta = rdbx_estimate_index(&stream->rtp_rdbx, &est, ntohs(hdr->seq)); + status = rdbx_check(&stream->rtp_rdbx, delta); + if (status) + return status; /* we've been asked to reuse an index */ + rdbx_add_index(&stream->rtp_rdbx, delta); + +#ifdef NO_64BIT_MATH + debug_print2(mod_srtp, "estimated packet index: %08x%08x", + high32(est),low32(est)); +#else + debug_print(mod_srtp, "estimated packet index: %016llx", est); +#endif + + /* + * if we're using rindael counter mode, set nonce and seq + */ + if (stream->rtp_cipher->type == &aes_icm) { + v128_t iv; + + iv.v32[0] = 0; + iv.v32[1] = hdr->ssrc; +#ifdef NO_64BIT_MATH + iv.v64[1] = be64_to_cpu(make64((high32(est) << 16) | (low32(est) >> 16), + low32(est) << 16)); +#else + iv.v64[1] = be64_to_cpu(est << 16); +#endif + status = cipher_set_iv(stream->rtp_cipher, &iv); + + } else { + v128_t iv; + + /* otherwise, set the index to est */ +#ifdef NO_64BIT_MATH + iv.v32[0] = 0; + iv.v32[1] = 0; +#else + iv.v64[0] = 0; +#endif + iv.v64[1] = be64_to_cpu(est); + status = cipher_set_iv(stream->rtp_cipher, &iv); + } + if (status) + return err_status_cipher_fail; + + /* shift est, put into network byte order */ +#ifdef NO_64BIT_MATH + est = be64_to_cpu(make64((high32(est) << 16) | + (low32(est) >> 16), + low32(est) << 16)); +#else + est = be64_to_cpu(est << 16); +#endif + + /* + * if we're authenticating using a universal hash, put the keystream + * prefix into the authentication tag + */ + if (auth_start) { + + prefix_len = auth_get_prefix_length(stream->rtp_auth); + if (prefix_len) { + status = cipher_output(stream->rtp_cipher, auth_tag, prefix_len); + if (status) + return err_status_cipher_fail; + debug_print(mod_srtp, "keystream prefix: %s", + octet_string_hex_string(auth_tag, prefix_len)); + } + } + + /* if we're encrypting, exor keystream into the message */ + if (enc_start) { + status = cipher_encrypt(stream->rtp_cipher, + (uint8_t *)enc_start, &enc_octet_len); + if (status) + return err_status_cipher_fail; + } + + /* + * if we're authenticating, run authentication function and put result + * into the auth_tag + */ + if (auth_start) { + + /* initialize auth func context */ + status = auth_start(stream->rtp_auth); + if (status) return status; + + /* run auth func over packet */ + status = auth_update(stream->rtp_auth, + (uint8_t *)auth_start, *pkt_octet_len); + if (status) return status; + + /* run auth func over ROC, put result into auth_tag */ + debug_print(mod_srtp, "estimated packet index: %016llx", est); + status = auth_compute(stream->rtp_auth, (uint8_t *)&est, 4, auth_tag); + debug_print(mod_srtp, "srtp auth tag: %s", + octet_string_hex_string(auth_tag, tag_len)); + if (status) + return err_status_auth_fail; + + } + + if (auth_tag) { + + /* increase the packet length by the length of the auth tag */ + *pkt_octet_len += tag_len; + } + + return err_status_ok; +} + + +err_status_t +srtp_unprotect(srtp_ctx_t *ctx, void *srtp_hdr, int *pkt_octet_len) { + srtp_hdr_t *hdr = srtp_hdr; + uint32_t *enc_start; /* pointer to start of encrypted portion */ + uint32_t *auth_start; /* pointer to start of auth. portion */ + uint32_t enc_octet_len = 0;/* number of octets in encrypted portion */ + uint8_t *auth_tag = NULL; /* location of auth_tag within packet */ + xtd_seq_num_t est; /* estimated xtd_seq_num_t of *hdr */ + int delta; /* delta of local pkt idx and that in hdr */ + v128_t iv; + err_status_t status; + srtp_stream_ctx_t *stream; + uint8_t tmp_tag[SRTP_MAX_TAG_LEN]; + int tag_len, prefix_len; + + debug_print(mod_srtp, "function srtp_unprotect", NULL); + + /* we assume the hdr is 32-bit aligned to start */ + + /* check the packet length - it must at least contain a full header */ + if (*pkt_octet_len < octets_in_rtp_header) + return err_status_bad_param; + + /* + * look up ssrc in srtp_stream list, and process the packet with + * the appropriate stream. if we haven't seen this stream before, + * there's only one key for this srtp_session, and the cipher + * supports key-sharing, then we assume that a new stream using + * that key has just started up + */ + stream = srtp_get_stream(ctx, hdr->ssrc); + if (stream == NULL) { + if (ctx->stream_template != NULL) { + stream = ctx->stream_template; + debug_print(mod_srtp, "using provisional stream (SSRC: 0x%08x)", + hdr->ssrc); + + /* + * set estimated packet index to sequence number from header, + * and set delta equal to the same value + */ +#ifdef NO_64BIT_MATH + est = (xtd_seq_num_t) make64(0,ntohs(hdr->seq)); + delta = low32(est); +#else + est = (xtd_seq_num_t) ntohs(hdr->seq); + delta = est; +#endif + } else { + + /* + * no stream corresponding to SSRC found, and we don't do + * key-sharing, so return an error + */ + return err_status_no_ctx; + } + } else { + + /* estimate packet index from seq. num. in header */ + delta = rdbx_estimate_index(&stream->rtp_rdbx, &est, ntohs(hdr->seq)); + + /* check replay database */ + status = rdbx_check(&stream->rtp_rdbx, delta); + if (status) + return status; + } + +#ifdef NO_64BIT_MATH + debug_print2(mod_srtp, "estimated u_packet index: %08x%08x", high32(est),low32(est)); +#else + debug_print(mod_srtp, "estimated u_packet index: %016llx", est); +#endif + + /* get tag length from stream */ + tag_len = auth_get_tag_length(stream->rtp_auth); + + /* + * set the cipher's IV properly, depending on whatever cipher we + * happen to be using + */ + if (stream->rtp_cipher->type == &aes_icm) { + + /* aes counter mode */ + iv.v32[0] = 0; + iv.v32[1] = hdr->ssrc; /* still in network order */ +#ifdef NO_64BIT_MATH + iv.v64[1] = be64_to_cpu(make64((high32(est) << 16) | (low32(est) >> 16), + low32(est) << 16)); +#else + iv.v64[1] = be64_to_cpu(est << 16); +#endif + status = aes_icm_set_iv(stream->rtp_cipher->state, &iv); + } else { + + /* no particular format - set the iv to the pakcet index */ +#ifdef NO_64BIT_MATH + iv.v32[0] = 0; + iv.v32[1] = 0; +#else + iv.v64[0] = 0; +#endif + iv.v64[1] = be64_to_cpu(est); + status = cipher_set_iv(stream->rtp_cipher, &iv); + } + if (status) + return err_status_cipher_fail; + + /* shift est, put into network byte order */ +#ifdef NO_64BIT_MATH + est = be64_to_cpu(make64((high32(est) << 16) | + (low32(est) >> 16), + low32(est) << 16)); +#else + est = be64_to_cpu(est << 16); +#endif + + /* + * find starting point for decryption and length of data to be + * decrypted - the encrypted portion starts after the rtp header + * extension, if present; otherwise, it starts after the last csrc, + * if any are present + * + * if we're not providing confidentiality, set enc_start to NULL + */ + if (stream->rtp_services & sec_serv_conf) { + enc_start = (uint32_t *)hdr + uint32s_in_rtp_header + hdr->cc; + if (hdr->x == 1) { + srtp_hdr_xtnd_t *xtn_hdr = (srtp_hdr_xtnd_t *)enc_start; + enc_start += (ntohs(xtn_hdr->length) + 1); + } + enc_octet_len = *pkt_octet_len - tag_len + - ((enc_start - (uint32_t *)hdr) << 2); + } else { + enc_start = NULL; + } + + /* + * if we're providing authentication, set the auth_start and auth_tag + * pointers to the proper locations; otherwise, set auth_start to NULL + * to indicate that no authentication is needed + */ + if (stream->rtp_services & sec_serv_auth) { + auth_start = (uint32_t *)hdr; + auth_tag = (uint8_t *)hdr + *pkt_octet_len - tag_len; + } else { + auth_start = NULL; + auth_tag = NULL; + } + + /* + * if we expect message authentication, run the authentication + * function and compare the result with the value of the auth_tag + */ + if (auth_start) { + + /* + * if we're using a universal hash, then we need to compute the + * keystream prefix for encrypting the universal hash output + * + * if the keystream prefix length is zero, then we know that + * the authenticator isn't using a universal hash function + */ + if (stream->rtp_auth->prefix_len != 0) { + + prefix_len = auth_get_prefix_length(stream->rtp_auth); + status = cipher_output(stream->rtp_cipher, tmp_tag, prefix_len); + debug_print(mod_srtp, "keystream prefix: %s", + octet_string_hex_string(tmp_tag, prefix_len)); + if (status) + return err_status_cipher_fail; + } + + /* initialize auth func context */ + status = auth_start(stream->rtp_auth); + if (status) return status; + + /* now compute auth function over packet */ + status = auth_update(stream->rtp_auth, (uint8_t *)auth_start, + *pkt_octet_len - tag_len); + + /* run auth func over ROC, then write tmp tag */ + status = auth_compute(stream->rtp_auth, (uint8_t *)&est, 4, tmp_tag); + + debug_print(mod_srtp, "computed auth tag: %s", + octet_string_hex_string(tmp_tag, tag_len)); + debug_print(mod_srtp, "packet auth tag: %s", + octet_string_hex_string(auth_tag, tag_len)); + if (status) + return err_status_auth_fail; + + if (octet_string_is_eq(tmp_tag, auth_tag, tag_len)) + return err_status_auth_fail; + } + + /* + * update the key usage limit, and check it to make sure that we + * didn't just hit either the soft limit or the hard limit, and call + * the event handler if we hit either. + */ + switch(key_limit_update(stream->limit)) { + case key_event_normal: + break; + case key_event_soft_limit: + srtp_handle_event(ctx, stream, event_key_soft_limit); + break; + case key_event_hard_limit: + srtp_handle_event(ctx, stream, event_key_hard_limit); + return err_status_key_expired; + default: + break; + } + + /* if we're encrypting, add keystream into ciphertext */ + if (enc_start) { + status = cipher_encrypt(stream->rtp_cipher, + (uint8_t *)enc_start, &enc_octet_len); + if (status) + return err_status_cipher_fail; + } + + /* + * verify that stream is for received traffic - this check will + * detect SSRC collisions, since a stream that appears in both + * srtp_protect() and srtp_unprotect() will fail this test in one of + * those functions. + * + * we do this check *after* the authentication check, so that the + * latter check will catch any attempts to fool us into thinking + * that we've got a collision + */ + if (stream->direction != dir_srtp_receiver) { + if (stream->direction == dir_unknown) { + stream->direction = dir_srtp_receiver; + } else { + srtp_handle_event(ctx, stream, event_ssrc_collision); + } + } + + /* + * if the stream is a 'provisional' one, in which the template context + * is used, then we need to allocate a new stream at this point, since + * the authentication passed + */ + if (stream == ctx->stream_template) { + srtp_stream_ctx_t *new_stream; + + /* + * allocate and initialize a new stream + * + * note that we indicate failure if we can't allocate the new + * stream, and some implementations will want to not return + * failure here + */ + status = srtp_stream_clone(ctx->stream_template, hdr->ssrc, &new_stream); + if (status) + return status; + + /* add new stream to the head of the stream_list */ + new_stream->next = ctx->stream_list; + ctx->stream_list = new_stream; + + /* set stream (the pointer used in this function) */ + stream = new_stream; + } + + /* + * the message authentication function passed, so add the packet + * index into the replay database + */ + rdbx_add_index(&stream->rtp_rdbx, delta); + + /* decrease the packet length by the length of the auth tag */ + *pkt_octet_len -= tag_len; + + return err_status_ok; +} + +err_status_t +srtp_init() { + err_status_t status; + + /* initialize crypto kernel */ + status = crypto_kernel_init(); + if (status) + return status; + + /* load srtp debug module into the kernel */ + status = crypto_kernel_load_debug_module(&mod_srtp); + if (status) + return status; + + return err_status_ok; +} + +/* + * The following code is under consideration for removal. See + * SRTP_MAX_TRAILER_LEN + */ +#if 0 + +/* + * srtp_get_trailer_length(&a) returns the number of octets that will + * be added to an RTP packet by the SRTP processing. This value + * is constant for a given srtp_stream_t (i.e. between initializations). + */ + +int +srtp_get_trailer_length(const srtp_stream_t s) { + return auth_get_tag_length(s->rtp_auth); +} + +#endif + +/* + * srtp_get_stream(ssrc) returns a pointer to the stream corresponding + * to ssrc, or NULL if no stream exists for that ssrc + * + * this is an internal function + */ + +srtp_stream_ctx_t * +srtp_get_stream(srtp_t srtp, uint32_t ssrc) { + srtp_stream_ctx_t *stream; + + /* walk down list until ssrc is found */ + stream = srtp->stream_list; + while (stream != NULL) { + if (stream->ssrc == ssrc) + return stream; + stream = stream->next; + } + + /* we haven't found our ssrc, so return a null */ + return NULL; +} + +err_status_t +srtp_dealloc(srtp_t session) { + srtp_stream_ctx_t *stream; + err_status_t status; + + /* + * we take a conservative deallocation strategy - if we encounter an + * error deallocating a stream, then we stop trying to deallocate + * memory and just return an error + */ + + /* walk list of streams, deallocating as we go */ + stream = session->stream_list; + while (stream != NULL) { + srtp_stream_t next = stream->next; + status = srtp_stream_dealloc(session, stream); + if (status) + return status; + stream = next; + } + + /* deallocate stream template, if there is one */ + if (session->stream_template != NULL) { + status = auth_dealloc(session->stream_template->rtcp_auth); + if (status) + return status; + status = cipher_dealloc(session->stream_template->rtcp_cipher); + if (status) + return status; + crypto_free(session->stream_template->limit); + status = cipher_dealloc(session->stream_template->rtp_cipher); + if (status) + return status; + status = auth_dealloc(session->stream_template->rtp_auth); + if (status) + return status; + crypto_free(session->stream_template); + } + + /* deallocate session context */ + crypto_free(session); + + return err_status_ok; +} + + +err_status_t +srtp_add_stream(srtp_t session, + const srtp_policy_t *policy) { + err_status_t status; + srtp_stream_t tmp; + + /* sanity check arguments */ + if ((session == NULL) || (policy == NULL) || (policy->key == NULL)) + return err_status_bad_param; + + /* allocate stream */ + status = srtp_stream_alloc(&tmp, policy); + if (status) { + return status; + } + + /* initialize stream */ + status = srtp_stream_init(tmp, policy); + if (status) { + crypto_free(tmp); + return status; + } + + /* + * set the head of the stream list or the template to point to the + * stream that we've just alloced and init'ed, depending on whether + * or not it has a wildcard SSRC value or not + * + * if the template stream has already been set, then the policy is + * inconsistent, so we return a bad_param error code + */ + switch (policy->ssrc.type) { + case (ssrc_any_outbound): + if (session->stream_template) { + return err_status_bad_param; + } + session->stream_template = tmp; + session->stream_template->direction = dir_srtp_sender; + break; + case (ssrc_any_inbound): + if (session->stream_template) { + return err_status_bad_param; + } + session->stream_template = tmp; + session->stream_template->direction = dir_srtp_receiver; + break; + case (ssrc_specific): + tmp->next = session->stream_list; + session->stream_list = tmp; + break; + case (ssrc_undefined): + default: + crypto_free(tmp); + return err_status_bad_param; + } + + return err_status_ok; +} + + +err_status_t +srtp_create(srtp_t *session, /* handle for session */ + const srtp_policy_t *policy) { /* SRTP policy (list) */ + err_status_t stat; + srtp_ctx_t *ctx; + + /* sanity check arguments */ + if (session == NULL) + return err_status_bad_param; + + /* allocate srtp context and set ctx_ptr */ + ctx = (srtp_ctx_t *) crypto_alloc(sizeof(srtp_ctx_t)); + if (ctx == NULL) + return err_status_alloc_fail; + *session = ctx; + + /* + * loop over elements in the policy list, allocating and + * initializing a stream for each element + */ + ctx->stream_template = NULL; + ctx->stream_list = NULL; + while (policy != NULL) { + + stat = srtp_add_stream(ctx, policy); + if (stat) { + /* clean up everything */ + srtp_dealloc(*session); + return stat; + } + + /* set policy to next item in list */ + policy = policy->next; + } + + return err_status_ok; +} + + +err_status_t +srtp_remove_stream(srtp_t session, uint32_t ssrc) { + srtp_stream_ctx_t *stream, *last_stream; + err_status_t status; + + /* sanity check arguments */ + if (session == NULL) + return err_status_bad_param; + + /* find stream in list; complain if not found */ + last_stream = stream = session->stream_list; + while ((stream != NULL) && (ssrc != stream->ssrc)) { + last_stream = stream; + stream = stream->next; + } + if (stream == NULL) + return err_status_no_ctx; + + /* remove stream from the list */ + last_stream->next = stream->next; + + /* deallocate the stream */ + status = srtp_stream_dealloc(session, stream); + if (status) + return status; + + return err_status_ok; +} + + +/* + * the default policy - provides a convenient way for callers to use + * the default security policy + * + * this policy is that defined in the current SRTP internet draft. + * + */ + +/* + * NOTE: cipher_key_len is really key len (128 bits) plus salt len + * (112 bits) + */ +/* There are hard-coded 16's for base_key_len in the key generation code */ + +void +crypto_policy_set_rtp_default(crypto_policy_t *p) { + + p->cipher_type = AES_128_ICM; + p->cipher_key_len = 30; /* default 128 bits per RFC 3711 */ + p->auth_type = HMAC_SHA1; + p->auth_key_len = 20; /* default 160 bits per RFC 3711 */ + p->auth_tag_len = 10; /* default 80 bits per RFC 3711 */ + p->sec_serv = sec_serv_conf_and_auth; + +} + +void +crypto_policy_set_rtcp_default(crypto_policy_t *p) { + + p->cipher_type = AES_128_ICM; + p->cipher_key_len = 30; /* default 128 bits per RFC 3711 */ + p->auth_type = HMAC_SHA1; + p->auth_key_len = 20; /* default 160 bits per RFC 3711 */ + p->auth_tag_len = 10; /* default 80 bits per RFC 3711 */ + p->sec_serv = sec_serv_conf_and_auth; + +} + +void +crypto_policy_set_aes_cm_128_hmac_sha1_32(crypto_policy_t *p) { + + /* + * corresponds to draft-ietf-mmusic-sdescriptions-12.txt + * + * note that this crypto policy is intended for SRTP, but not SRTCP + */ + + p->cipher_type = AES_128_ICM; + p->cipher_key_len = 30; /* 128 bit key, 112 bit salt */ + p->auth_type = HMAC_SHA1; + p->auth_key_len = 20; /* 160 bit key */ + p->auth_tag_len = 4; /* 32 bit tag */ + p->sec_serv = sec_serv_conf_and_auth; + +} + + +void +crypto_policy_set_aes_cm_128_null_auth(crypto_policy_t *p) { + + /* + * corresponds to draft-ietf-mmusic-sdescriptions-12.txt + * + * note that this crypto policy is intended for SRTP, but not SRTCP + */ + + p->cipher_type = AES_128_ICM; + p->cipher_key_len = 30; /* 128 bit key, 112 bit salt */ + p->auth_type = NULL_AUTH; + p->auth_key_len = 0; + p->auth_tag_len = 0; + p->sec_serv = sec_serv_conf; + +} + + +void +crypto_policy_set_null_cipher_hmac_sha1_80(crypto_policy_t *p) { + + /* + * corresponds to draft-ietf-mmusic-sdescriptions-12.txt + */ + + p->cipher_type = NULL_CIPHER; + p->cipher_key_len = 0; + p->auth_type = HMAC_SHA1; + p->auth_key_len = 20; + p->auth_tag_len = 10; + p->sec_serv = sec_serv_auth; + +} + + +/* + * secure rtcp functions + */ + +err_status_t +srtp_protect_rtcp(srtp_t ctx, void *rtcp_hdr, int *pkt_octet_len) { + srtcp_hdr_t *hdr = rtcp_hdr; + uint32_t *enc_start; /* pointer to start of encrypted portion */ + uint32_t *auth_start; /* pointer to start of auth. portion */ + uint32_t *trailer; /* pointer to start of trailer */ + uint32_t enc_octet_len = 0;/* number of octets in encrypted portion */ + uint8_t *auth_tag = NULL; /* location of auth_tag within packet */ + err_status_t status; + int tag_len; + srtp_stream_ctx_t *stream; + int prefix_len; + uint32_t seq_num; + + /* we assume the hdr is 32-bit aligned to start */ + /* + * look up ssrc in srtp_stream list, and process the packet with + * the appropriate stream. if we haven't seen this stream before, + * there's only one key for this srtp_session, and the cipher + * supports key-sharing, then we assume that a new stream using + * that key has just started up + */ + stream = srtp_get_stream(ctx, hdr->ssrc); + if (stream == NULL) { + if (ctx->stream_template != NULL) { + srtp_stream_ctx_t *new_stream; + + /* allocate and initialize a new stream */ + status = srtp_stream_clone(ctx->stream_template, + hdr->ssrc, &new_stream); + if (status) + return status; + + /* add new stream to the head of the stream_list */ + new_stream->next = ctx->stream_list; + ctx->stream_list = new_stream; + + /* set stream (the pointer used in this function) */ + stream = new_stream; + } else { + /* no template stream, so we return an error */ + return err_status_no_ctx; + } + } + + /* + * verify that stream is for sending traffic - this check will + * detect SSRC collisions, since a stream that appears in both + * srtp_protect() and srtp_unprotect() will fail this test in one of + * those functions. + */ + if (stream->direction != dir_srtp_sender) { + if (stream->direction == dir_unknown) { + stream->direction = dir_srtp_sender; + } else { + srtp_handle_event(ctx, stream, event_ssrc_collision); + } + } + + /* get tag length from stream context */ + tag_len = auth_get_tag_length(stream->rtcp_auth); + + /* + * set encryption start and encryption length - if we're not + * providing confidentiality, set enc_start to NULL + */ + enc_start = (uint32_t *)hdr + uint32s_in_rtcp_header; + enc_octet_len = *pkt_octet_len - octets_in_rtcp_header; + + /* all of the packet, except the header, gets encrypted */ + /* NOTE: hdr->length is not usable - it refers to only the first + RTCP report in the compound packet! */ + /* NOTE: trailer is 32-bit aligned because RTCP 'packets' are always + multiples of 32-bits (RFC 3550 6.1) */ + trailer = (uint32_t *) ((char *)enc_start + enc_octet_len); + + if (stream->rtcp_services & sec_serv_conf) { + *trailer = htonl(SRTCP_E_BIT); /* set encrypt bit */ + } else { + enc_start = NULL; + enc_octet_len = 0; + /* 0 is network-order independant */ + *trailer = 0x00000000; /* set encrypt bit */ + } + + /* + * set the auth_start and auth_tag pointers to the proper locations + * (note that srtpc *always* provides authentication, unlike srtp) + */ + /* Note: This would need to change for optional mikey data */ + auth_start = (uint32_t *)hdr; + auth_tag = (uint8_t *)hdr + *pkt_octet_len + sizeof(srtcp_trailer_t); + + /* + * check sequence number for overruns, and copy it into the packet + * if its value isn't too big + */ + status = rdb_increment(&stream->rtcp_rdb); + if (status) + return status; + seq_num = rdb_get_value(&stream->rtcp_rdb); + *trailer |= htonl(seq_num); + debug_print(mod_srtp, "srtcp index: %x", seq_num); + + /* + * if we're using rindael counter mode, set nonce and seq + */ + if (stream->rtcp_cipher->type == &aes_icm) { + v128_t iv; + + iv.v32[0] = 0; + iv.v32[1] = hdr->ssrc; /* still in network order! */ + iv.v32[2] = htonl(seq_num >> 16); + iv.v32[3] = htonl(seq_num << 16); + status = aes_icm_set_iv(stream->rtcp_cipher->state, &iv); + + } else { + v128_t iv; + + /* otherwise, just set the index to seq_num */ + iv.v32[0] = 0; + iv.v32[1] = 0; + iv.v32[2] = 0; + iv.v32[3] = htonl(seq_num); + status = cipher_set_iv(stream->rtcp_cipher, &iv); + } + if (status) + return err_status_cipher_fail; + + /* + * if we're authenticating using a universal hash, put the keystream + * prefix into the authentication tag + */ + + /* if auth_start is non-null, then put keystream into tag */ + if (auth_start) { + + /* put keystream prefix into auth_tag */ + prefix_len = auth_get_prefix_length(stream->rtcp_auth); + status = cipher_output(stream->rtcp_cipher, auth_tag, prefix_len); + + debug_print(mod_srtp, "keystream prefix: %s", + octet_string_hex_string(auth_tag, prefix_len)); + + if (status) + return err_status_cipher_fail; + } + + /* if we're encrypting, exor keystream into the message */ + if (enc_start) { + status = cipher_encrypt(stream->rtcp_cipher, + (uint8_t *)enc_start, &enc_octet_len); + if (status) + return err_status_cipher_fail; + } + + /* initialize auth func context */ + auth_start(stream->rtcp_auth); + + /* per spec, do auth after encryption */ + /* run auth func over packet, put result into auth_tag */ + status = auth_compute(stream->rtcp_auth, + (uint8_t *)auth_start, *pkt_octet_len, auth_tag); + debug_print(mod_srtp, "srtcp auth tag: %s", + octet_string_hex_string(auth_tag, tag_len)); + if (status) + return err_status_auth_fail; + + /* increase the packet length by the length of the auth tag and seq_num*/ + *pkt_octet_len += (tag_len + sizeof(srtcp_trailer_t)); + + return err_status_ok; +} + + +err_status_t +srtp_unprotect_rtcp(srtp_t ctx, void *srtcp_hdr, int *pkt_octet_len) { + srtcp_hdr_t *hdr = srtcp_hdr; + uint32_t *enc_start; /* pointer to start of encrypted portion */ + uint32_t *auth_start; /* pointer to start of auth. portion */ + uint32_t *trailer; /* pointer to start of trailer */ + uint32_t enc_octet_len = 0;/* number of octets in encrypted portion */ + uint8_t *auth_tag = NULL; /* location of auth_tag within packet */ + uint8_t tmp_tag[SRTP_MAX_TAG_LEN]; + err_status_t status; + int tag_len; + srtp_stream_ctx_t *stream; + int prefix_len; + uint32_t seq_num; + + /* we assume the hdr is 32-bit aligned to start */ + /* + * look up ssrc in srtp_stream list, and process the packet with + * the appropriate stream. if we haven't seen this stream before, + * there's only one key for this srtp_session, and the cipher + * supports key-sharing, then we assume that a new stream using + * that key has just started up + */ + stream = srtp_get_stream(ctx, hdr->ssrc); + if (stream == NULL) { + if (ctx->stream_template != NULL) { + stream = ctx->stream_template; + debug_print(mod_srtp, "srtcp using provisional stream (SSRC: 0x%08x)", + hdr->ssrc); + } else { + /* no template stream, so we return an error */ + return err_status_no_ctx; + } + } + + /* get tag length from stream context */ + tag_len = auth_get_tag_length(stream->rtcp_auth); + + /* + * set encryption start, encryption length, and trailer + */ + enc_octet_len = *pkt_octet_len - + (octets_in_rtcp_header + tag_len + sizeof(srtcp_trailer_t)); + /* index & E (encryption) bit follow normal data. hdr->len + is the number of words (32-bit) in the normal packet minus 1 */ + /* This should point trailer to the word past the end of the + normal data. */ + /* This would need to be modified for optional mikey data */ + /* + * NOTE: trailer is 32-bit aligned because RTCP 'packets' are always + * multiples of 32-bits (RFC 3550 6.1) + */ + trailer = (uint32_t *) ((char *) hdr + + *pkt_octet_len -(tag_len + sizeof(srtcp_trailer_t))); + if (*((unsigned char *) trailer) & SRTCP_E_BYTE_BIT) { + enc_start = (uint32_t *)hdr + uint32s_in_rtcp_header; + } else { + enc_octet_len = 0; + enc_start = NULL; /* this indicates that there's no encryption */ + } + + /* + * set the auth_start and auth_tag pointers to the proper locations + * (note that srtcp *always* uses authentication, unlike srtp) + */ + auth_start = (uint32_t *)hdr; + auth_tag = (uint8_t *)hdr + *pkt_octet_len - tag_len; + + /* + * check the sequence number for replays + */ + /* this is easier than dealing with bitfield access */ + seq_num = ntohl(*trailer) & SRTCP_INDEX_MASK; + status = rdb_check(&stream->rtcp_rdb, seq_num); + if (status) + return status; + debug_print(mod_srtp, "srtcp index: %x", seq_num); + + /* + * if we're using aes counter mode, set nonce and seq + */ + if (stream->rtcp_cipher->type == &aes_icm) { + v128_t iv; + + iv.v32[0] = 0; + iv.v32[1] = hdr->ssrc; /* still in network order! */ + iv.v32[2] = htonl(seq_num >> 16); + iv.v32[3] = htonl(seq_num << 16); + status = aes_icm_set_iv(stream->rtcp_cipher->state, &iv); + + } else { + v128_t iv; + + /* otherwise, just set the index to seq_num */ + iv.v32[0] = 0; + iv.v32[1] = 0; + iv.v32[2] = 0; + iv.v32[3] = htonl(seq_num); + status = cipher_set_iv(stream->rtcp_cipher, &iv); + + } + if (status) + return err_status_cipher_fail; + + /* initialize auth func context */ + auth_start(stream->rtcp_auth); + + /* run auth func over packet, put result into tmp_tag */ + status = auth_compute(stream->rtcp_auth, (uint8_t *)auth_start, + *pkt_octet_len - (tag_len + sizeof(srtcp_trailer_t)), + tmp_tag); + debug_print(mod_srtp, "srtcp computed tag: %s", + octet_string_hex_string(tmp_tag, tag_len)); + if (status) + return err_status_auth_fail; + + /* compare the tag just computed with the one in the packet */ + debug_print(mod_srtp, "srtcp tag from packet: %s", + octet_string_hex_string(auth_tag, tag_len)); + if (octet_string_is_eq(tmp_tag, auth_tag, tag_len)) + return err_status_auth_fail; + + /* + * if we're authenticating using a universal hash, put the keystream + * prefix into the authentication tag + */ + prefix_len = auth_get_prefix_length(stream->rtcp_auth); + if (prefix_len) { + status = cipher_output(stream->rtcp_cipher, auth_tag, prefix_len); + debug_print(mod_srtp, "keystream prefix: %s", + octet_string_hex_string(auth_tag, prefix_len)); + if (status) + return err_status_cipher_fail; + } + + /* if we're decrypting, exor keystream into the message */ + if (enc_start) { + status = cipher_encrypt(stream->rtcp_cipher, + (uint8_t *)enc_start, &enc_octet_len); + if (status) + return err_status_cipher_fail; + } + + /* decrease the packet length by the length of the auth tag and seq_num*/ + *pkt_octet_len -= (tag_len + sizeof(srtcp_trailer_t)); + + /* + * verify that stream is for received traffic - this check will + * detect SSRC collisions, since a stream that appears in both + * srtp_protect() and srtp_unprotect() will fail this test in one of + * those functions. + * + * we do this check *after* the authentication check, so that the + * latter check will catch any attempts to fool us into thinking + * that we've got a collision + */ + if (stream->direction != dir_srtp_receiver) { + if (stream->direction == dir_unknown) { + stream->direction = dir_srtp_receiver; + } else { + srtp_handle_event(ctx, stream, event_ssrc_collision); + } + } + + /* + * if the stream is a 'provisional' one, in which the template context + * is used, then we need to allocate a new stream at this point, since + * the authentication passed + */ + if (stream == ctx->stream_template) { + srtp_stream_ctx_t *new_stream; + + /* + * allocate and initialize a new stream + * + * note that we indicate failure if we can't allocate the new + * stream, and some implementations will want to not return + * failure here + */ + status = srtp_stream_clone(ctx->stream_template, hdr->ssrc, &new_stream); + if (status) + return status; + + /* add new stream to the head of the stream_list */ + new_stream->next = ctx->stream_list; + ctx->stream_list = new_stream; + + /* set stream (the pointer used in this function) */ + stream = new_stream; + } + + /* we've passed the authentication check, so add seq_num to the rdb */ + rdb_add_index(&stream->rtcp_rdb, seq_num); + + + return err_status_ok; +} diff --git a/libs/srtp/tables/.cvsignore b/libs/srtp/tables/.cvsignore new file mode 100644 index 0000000000..45492b4a3d --- /dev/null +++ b/libs/srtp/tables/.cvsignore @@ -0,0 +1 @@ +aes_tables diff --git a/libs/srtp/tables/CVS/Entries b/libs/srtp/tables/CVS/Entries new file mode 100644 index 0000000000..0e5d5f5697 --- /dev/null +++ b/libs/srtp/tables/CVS/Entries @@ -0,0 +1,3 @@ +/.cvsignore/1.1/Thu Sep 29 12:06:41 2005// +/aes_tables.c/1.6/Sat Oct 8 16:38:06 2005// +D diff --git a/libs/srtp/tables/CVS/Repository b/libs/srtp/tables/CVS/Repository new file mode 100644 index 0000000000..9cf6c91951 --- /dev/null +++ b/libs/srtp/tables/CVS/Repository @@ -0,0 +1 @@ +srtp/tables diff --git a/libs/srtp/tables/CVS/Root b/libs/srtp/tables/CVS/Root new file mode 100644 index 0000000000..b4b74237d9 --- /dev/null +++ b/libs/srtp/tables/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@cvs.sourceforge.net:/cvsroot/srtp diff --git a/libs/srtp/tables/aes_tables.c b/libs/srtp/tables/aes_tables.c new file mode 100644 index 0000000000..2beedfd563 --- /dev/null +++ b/libs/srtp/tables/aes_tables.c @@ -0,0 +1,346 @@ +/* + * aes_tables.c + * + * generate tables for the AES cipher + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright(c) 2001-2005 Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include <stdio.h> +#include "gf2_8.h" +#include "crypto_math.h" + + +unsigned char aes_sbox[256]; + +unsigned char aes_inv_sbox[256]; + +uint32_t T0[256], T1[256], T2[256], T3[256], T4[256]; + + +#define AES_INVERSE_TEST 0 /* set to 1 to test forward/backwards aes */ + +/* functions for precomputing AES values */ + +/* + * A[] is the 8 x 8 binary matrix (represented as an array of columns, + * where each column is an octet) which defines the affine + * transformation used in the AES substitution table (Section + * 4.2.1 of the spec). + */ + +uint8_t A[8] = { 31, 62, 124, 248, 241, 227, 199, 143 }; + +/* + * b is the 8 bit vector (represented as an octet) used in the affine + * transform described above. + */ + +uint8_t b = 99; + + +void +aes_init_sbox(void) { + unsigned int i; + uint8_t x; + + for (i=0; i < 256; i++) { + x = gf2_8_compute_inverse((gf2_8)i); + x = A_times_x_plus_b(A, x, b); + aes_sbox[i] = x; + aes_inv_sbox[x] = i; + } +} + +void +aes_compute_tables(void) { + int i; + uint32_t x1, x2, x3; + v32_t tmp; + + /* initialize substitution table */ + aes_init_sbox(); + + /* combine sbox with linear operations to form 8-bit to 32-bit tables */ + for (i=0; i < 256; i++) { + x1 = aes_sbox[i]; + x2 = gf2_8_shift(x1); + x3 = x2 ^ x1; + + tmp.v8[0] = x2; + tmp.v8[1] = x1; + tmp.v8[2] = x1; + tmp.v8[3] = x3; + T0[i] = tmp.value; + + tmp.v8[0] = x3; + tmp.v8[1] = x2; + tmp.v8[2] = x1; + tmp.v8[3] = x1; + T1[i] = tmp.value; + + tmp.v8[0] = x1; + tmp.v8[1] = x3; + tmp.v8[2] = x2; + tmp.v8[3] = x1; + T2[i] = tmp.value; + + tmp.v8[0] = x1; + tmp.v8[1] = x1; + tmp.v8[2] = x3; + tmp.v8[3] = x2; + T3[i] = tmp.value; + + } +} + + +/* + * the tables U0, U1, U2, U3 implement the aes operations invSubBytes, + * invMixColumns, and invShiftRows + */ + +uint32_t U0[256], U1[256], U2[256], U3[256], U4[256]; + +extern uint8_t aes_inv_sbox[256]; + +void +aes_compute_inv_tables(void) { + int i; + uint8_t x, xe, x9, xd, xb; + v32_t tmp; + + /* combine sbox with linear operations to form 8-bit to 32-bit tables */ + for (i=0; i < 256; i++) { + x = aes_inv_sbox[i]; + + xe = gf2_8_multiply(0x0e, x); + x9 = gf2_8_multiply(0x09, x); + xd = gf2_8_multiply(0x0d, x); + xb = gf2_8_multiply(0x0b, x); + + tmp.v8[0] = xe; + tmp.v8[1] = x9; + tmp.v8[2] = xd; + tmp.v8[3] = xb; + U0[i] = tmp.value; + + tmp.v8[0] = xb; + tmp.v8[1] = xe; + tmp.v8[2] = x9; + tmp.v8[3] = xd; + U1[i] = tmp.value; + + tmp.v8[0] = xd; + tmp.v8[1] = xb; + tmp.v8[2] = xe; + tmp.v8[3] = x9; + U2[i] = tmp.value; + + tmp.v8[0] = x9; + tmp.v8[1] = xd; + tmp.v8[2] = xb; + tmp.v8[3] = xe; + U3[i] = tmp.value; + + tmp.v8[0] = tmp.v8[1] = tmp.v8[2] = tmp.v8[3] = x; + U4[i] = tmp.value; + } +} + + +/* + * aes_test_inverse() returns err_status_ok if aes + * encryption and decryption are true inverses of each other, and + * returns err_status_algo_fail otherwise + */ + +#include "err.h" + +err_status_t +aes_test_inverse(void); + +#define TABLES_32BIT 1 + +int +main(void) { + int i; + + aes_init_sbox(); + aes_compute_inv_tables(); + +#if TABLES_32BIT + printf("uint32_t U0 = {"); + for (i=0; i < 256; i++) { + if ((i % 4) == 0) + printf("\n"); + printf("0x%0x, ", U0[i]); + } + printf("\n}\n"); + + printf("uint32_t U1 = {"); + for (i=0; i < 256; i++) { + if ((i % 4) == 0) + printf("\n"); + printf("0x%x, ", U1[i]); + } + printf("\n}\n"); + + printf("uint32_t U2 = {"); + for (i=0; i < 256; i++) { + if ((i % 4) == 0) + printf("\n"); + printf("0x%x, ", U2[i]); + } + printf("\n}\n"); + + printf("uint32_t U3 = {"); + for (i=0; i < 256; i++) { + if ((i % 4) == 0) + printf("\n"); + printf("0x%x, ", U3[i]); + } + printf("\n}\n"); + + printf("uint32_t U4 = {"); + for (i=0; i < 256; i++) { + if ((i % 4) == 0) + printf("\n"); + printf("0x%x, ", U4[i]); + } + printf("\n}\n"); + +#else + + printf("uint32_t U0 = {"); + for (i=0; i < 256; i++) { + if ((i % 4) == 0) + printf("\n"); + printf("0x%lx, ", U0[i]); + } + printf("\n}\n"); + + printf("uint32_t U1 = {"); + for (i=0; i < 256; i++) { + if ((i % 4) == 0) + printf("\n"); + printf("0x%lx, ", U1[i]); + } + printf("\n}\n"); + + printf("uint32_t U2 = {"); + for (i=0; i < 256; i++) { + if ((i % 4) == 0) + printf("\n"); + printf("0x%lx, ", U2[i]); + } + printf("\n}\n"); + + printf("uint32_t U3 = {"); + for (i=0; i < 256; i++) { + if ((i % 4) == 0) + printf("\n"); + printf("0x%lx, ", U3[i]); + } + printf("\n}\n"); + + printf("uint32_t U4 = {"); + for (i=0; i < 256; i++) { + if ((i % 4) == 0) + printf("\n"); + printf("0x%lx, ", U4[i]); + } + printf("\n}\n"); + + +#endif /* TABLES_32BIT */ + + +#if AES_INVERSE_TEST + /* + * test that aes_encrypt and aes_decrypt are actually + * inverses of each other + */ + + printf("aes inverse test: "); + if (aes_test_inverse() == err_status_ok) + printf("passed\n"); + else { + printf("failed\n"); + exit(1); + } +#endif + + return 0; +} + +#if AES_INVERSE_TEST + +err_status_t +aes_test_inverse(void) { + v128_t x, y; + aes_expanded_key_t expanded_key, decrypt_key; + uint8_t plaintext[16] = { + 0x00, 0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77, + 0x88, 0x99, 0xaa, 0xbb, 0xcc, 0xdd, 0xee, 0xff + }; + uint8_t key[16] = { + 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, + 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f + }; + v128_t k; + v128_set_to_zero(&x); + + v128_copy_octet_string(&k, key); + v128_copy_octet_string(&x, plaintext); + aes_expand_encryption_key(k, expanded_key); + aes_expand_decryption_key(k, decrypt_key); + aes_encrypt(&x, expanded_key); + aes_decrypt(&x, decrypt_key); + + /* compare to expected value then report */ + v128_copy_octet_string(&y, plaintext); + + if (v128_is_eq(&x, &y)) + return err_status_ok; + return err_status_algo_fail; + +} + +#endif diff --git a/libs/srtp/test/.cvsignore b/libs/srtp/test/.cvsignore new file mode 100644 index 0000000000..2b9f0e785a --- /dev/null +++ b/libs/srtp/test/.cvsignore @@ -0,0 +1,12 @@ +aes_calc +cipher_driver +datatypes_driver +kernel_driver +rand_gen +rdbx_driver +replay_driver +roc_driver +rtpw +sha1_driver +srtp_driver +stat_driver diff --git a/libs/srtp/test/CVS/Entries b/libs/srtp/test/CVS/Entries new file mode 100644 index 0000000000..5a4a045f77 --- /dev/null +++ b/libs/srtp/test/CVS/Entries @@ -0,0 +1,9 @@ +/.cvsignore/1.1/Thu Sep 29 11:59:01 2005// +/lfsr.c/1.1.1.1/Wed Sep 21 22:51:46 2005// +/rdbx_driver.c/1.2/Sun Oct 2 20:46:24 2005// +/replay_driver.c/1.3/Thu Sep 29 12:48:42 2005// +/roc_driver.c/1.2/Thu Sep 29 12:48:42 2005// +/rtp.c/1.5/Thu Oct 6 14:28:00 2005// +/rtpw.c/1.8/Thu Jan 12 17:56:02 2006// +/srtp_driver.c/1.6/Fri Feb 3 19:43:59 2006// +D diff --git a/libs/srtp/test/CVS/Repository b/libs/srtp/test/CVS/Repository new file mode 100644 index 0000000000..ef6b44c731 --- /dev/null +++ b/libs/srtp/test/CVS/Repository @@ -0,0 +1 @@ +srtp/test diff --git a/libs/srtp/test/CVS/Root b/libs/srtp/test/CVS/Root new file mode 100644 index 0000000000..b4b74237d9 --- /dev/null +++ b/libs/srtp/test/CVS/Root @@ -0,0 +1 @@ +:pserver:anonymous@cvs.sourceforge.net:/cvsroot/srtp diff --git a/libs/srtp/test/lfsr.c b/libs/srtp/test/lfsr.c new file mode 100644 index 0000000000..28ea02eb5b --- /dev/null +++ b/libs/srtp/test/lfsr.c @@ -0,0 +1,310 @@ +/* + * lfsr.c + * + */ + + +#include <stdio.h> +#include "datatypes.h" + +uint32_t +parity(uint32_t x) { + + x ^= (x >> 16); + x ^= (x >> 8); + x ^= (x >> 4); + x ^= (x >> 2); + x ^= (x >> 1); + + return x & 1; +} + + +/* typedef struct { */ +/* uint32_t register[8]; */ +/* } lfsr_t; */ + +void +compute_period(uint32_t feedback_polynomial) { + int i; + v32_t lfsr; + v32_t mask; + + mask.value = feedback_polynomial; + lfsr.value = 1; + + printf("polynomial: %s\t", v32_bit_string(mask)); + + for (i=0; i < 256; i++) { +/* printf("%s\n", v32_bit_string(lfsr)); */ + if (parity(mask.value & lfsr.value)) + lfsr.value = ((lfsr.value << 1) | 1) & 0xff; + else + lfsr.value = (lfsr.value << 1) & 0xff; + + /* now halt if we're back at the initial state */ + if (lfsr.value == 1) { + printf("period: %d\n", i); + break; + } + } +} + +uint32_t poly0 = 223; + + +uint32_t polynomials[39] = { +31, +47, +55, +59, +61, +79, +87, +91, +103, +107, +109, +115, +117, +121, +143, +151, +157, +167, +171, +173, +179, +181, +185, +199, +203, +205, +211, +213, +227, +229, +233, +241, +127, +191, +223, +239, +247, +251, +253 +}; + +char binary_string[32]; + +char * +u32_bit_string(uint32_t x, unsigned int length) { + unsigned int mask; + int index; + + mask = 1 << length; + index = 0; + for (; mask > 0; mask >>= 1) + if ((x & mask) == 0) + binary_string[index++] = '0'; + else + binary_string[index++] = '1'; + + binary_string[index++] = 0; /* NULL terminate string */ + return binary_string; +} + +extern int octet_weight[256]; + +unsigned int +weight(uint32_t poly) { + int wt = 0; + + /* note: endian-ness makes no difference */ + wt += octet_weight[poly & 0xff]; + wt += octet_weight[(poly >> 8) & 0xff]; + wt += octet_weight[(poly >> 16) & 0xff]; + wt += octet_weight[(poly >> 24)]; + + return wt; +} + +#define MAX_PERIOD 65535 + +#define debug_print 0 + +int +period(uint32_t poly) { + int i; + uint32_t x; + + + /* set lfsr to 1 */ + x = 1; +#if debug_print + printf("%d:\t%s\n", 0, u32_bit_string(x,8)); +#endif + for (i=1; i < MAX_PERIOD; i++) { + if (x & 1) + x = (x >> 1) ^ poly; + else + x = (x >> 1); + +#if debug_print + /* print for a sanity check */ + printf("%d:\t%s\n", i, u32_bit_string(x,8)); +#endif + + /* check for return to original value */ + if (x == 1) + return i; + } + return i; +} + +/* + * weight distribution computes the weight distribution of the + * code generated by the polynomial poly + */ + +#define MAX_LEN 8 +#define MAX_WEIGHT (1 << MAX_LEN) + +int A[MAX_WEIGHT+1]; + +void +weight_distribution2(uint32_t poly, int *A) { + int i; + uint32_t x; + + /* zeroize array */ + for (i=0; i < MAX_WEIGHT+1; i++) + A[i] = 0; + + /* loop over all input sequences */ + + + /* set lfsr to 1 */ + x = 1; +#if debug_print + printf("%d:\t%s\n", 0, u32_bit_string(x,8)); +#endif + for (i=1; i < MAX_PERIOD; i++) { + if (x & 1) + x = (x >> 1) ^ poly; + else + x = (x >> 1); + +#if debug_print + /* print for a sanity check */ + printf("%d:\t%s\n", i, u32_bit_string(x,8)); +#endif + + /* increment weight */ + wt += (x & 1); + + /* check for return to original value */ + if (x == 1) + break; + } + + /* set zero */ + A[0] = 0; +} + + +void +weight_distribution(uint32_t poly, int *A) { + int i; + uint32_t x; + + /* zeroize array */ + for (i=0; i < MAX_WEIGHT+1; i++) + A[i] = 0; + + /* set lfsr to 1 */ + x = 1; +#if debug_print + printf("%d:\t%s\n", 0, u32_bit_string(x,8)); +#endif + for (i=1; i < MAX_PERIOD; i++) { + if (x & 1) + x = (x >> 1) ^ poly; + else + x = (x >> 1); + +#if debug_print + /* print for a sanity check */ + printf("%d:\t%s\n", i, u32_bit_string(x,8)); +#endif + + /* compute weight, increment proper element */ + A[weight(x)]++; + + /* check for return to original value */ + if (x == 1) + break; + } + + /* set zero */ + A[0] = 0; +} + + + + +int +main () { + + int i,j; + v32_t x; + v32_t p; + + /* originally 0xaf */ + p.value = 0x9; + + printf("polynomial: %s\tperiod: %d\n", + u32_bit_string(p.value,8), period(p.value)); + + /* compute weight distribution */ + weight_distribution(p.value, A); + + /* print weight distribution */ + for (i=0; i <= 8; i++) { + printf("A[%d]: %d\n", i, A[i]); + } + +#if 0 + for (i=0; i < 39; i++) { + printf("polynomial: %s\tperiod: %d\n", + u32_bit_string(polynomials[i],8), period(polynomials[i])); + + /* compute weight distribution */ + weight_distribution(p.value, A); + + /* print weight distribution */ + for (j=0; j <= 8; j++) { + printf("A[%d]: %d\n", j, A[j]); + } + } +#endif + + { + int bits = 8; + uint32_t y; + for (y=0; y < (1 << bits); y++) { + printf("polynomial: %s\tweight: %d\tperiod: %d\n", + u32_bit_string(y,bits), weight(y), period(y)); + + /* compute weight distribution */ + weight_distribution(y, A); + + /* print weight distribution */ + for (j=0; j <= 8; j++) { + printf("A[%d]: %d\n", j, A[j]); + } + } + } + + return 0; +} diff --git a/libs/srtp/test/rdbx_driver.c b/libs/srtp/test/rdbx_driver.c new file mode 100644 index 0000000000..5107ecf229 --- /dev/null +++ b/libs/srtp/test/rdbx_driver.c @@ -0,0 +1,306 @@ +/* + * rdbx_driver.c + * + * driver for the rdbx implementation (replay database with extended range) + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include <stdio.h> /* for printf() */ +#include <unistd.h> /* for getopt() */ + +#include "rdbx.h" + +#ifdef ROC_TEST +#error "rdbx_t won't work with ROC_TEST - bitmask same size as seq_median" +#endif + +#include "ut_sim.h" + +err_status_t +test_replay_dbx(int num_trials); + +double +rdbx_check_adds_per_second(int num_trials); + +void +usage(char *prog_name) { + printf("usage: %s [ -t | -v ]\n", prog_name); + exit(255); +} + +int +main (int argc, char *argv[]) { + double rate; + err_status_t status; + char q; + unsigned do_timing_test = 0; + unsigned do_validation = 0; + + /* process input arguments */ + while (1) { + q = getopt(argc, argv, "tv"); + if (q == -1) + break; + switch (q) { + case 't': + do_timing_test = 1; + break; + case 'v': + do_validation = 1; + break; + default: + usage(argv[0]); + } + } + + printf("rdbx (replay database w/ extended range) test driver\n" + "David A. McGrew\n" + "Cisco Systems, Inc.\n"); + + if (!do_validation && !do_timing_test) + usage(argv[0]); + + if (do_validation) { + printf("testing rdbx_t...\n"); + + status = test_replay_dbx(1 << 12); + if (status) { + printf("failed\n"); + exit(1); + } + printf("passed\n"); + } + + if (do_timing_test) { + rate = rdbx_check_adds_per_second(1 << 18); + printf("rdbx_check/replay_adds per second: %e\n", rate); + } + + return 0; +} + +void +print_rdbx(rdbx_t *rdbx) { + printf("rdbx: {%llu, %s}\n", + (unsigned long long)(rdbx->index), v128_bit_string(&rdbx->bitmask)); +} + + +/* + * rdbx_check_add(rdbx, idx) checks a known-to-be-good idx against + * rdbx, then adds it. if a failure is detected (i.e., the check + * indicates that the value is already in rdbx) then + * err_status_algo_fail is returned. + * + */ + +err_status_t +rdbx_check_add(rdbx_t *rdbx, uint32_t idx) { + int delta; + xtd_seq_num_t est; + + delta = index_guess(&rdbx->index, &est, idx); + + if (rdbx_check(rdbx, delta) != err_status_ok) { + printf("replay_check failed at index %u\n", idx); + return err_status_algo_fail; + } + + /* + * in practice, we'd authenticate the packet containing idx, using + * the estimated value est, at this point + */ + + if (rdbx_add_index(rdbx, delta) != err_status_ok) { + printf("rdbx_add_index failed at index %u\n", idx); + return err_status_algo_fail; + } + + return err_status_ok; +} + +/* + * rdbx_check_expect_failure(rdbx_t *rdbx, uint32_t idx) + * + * checks that a sequence number idx is in the replay database + * and thus will be rejected + */ + +err_status_t +rdbx_check_expect_failure(rdbx_t *rdbx, uint32_t idx) { + int delta; + xtd_seq_num_t est; + err_status_t status; + + delta = index_guess(&rdbx->index, &est, idx); + + status = rdbx_check(rdbx, delta); + if (status == err_status_ok) { + printf("delta: %d ", delta); + printf("replay_check failed at index %u (false positive)\n", idx); + return err_status_algo_fail; + } + + return err_status_ok; +} + +err_status_t +rdbx_check_unordered(rdbx_t *rdbx, uint32_t idx) { + err_status_t rstat; + + rstat = rdbx_check(rdbx, idx); + if ((rstat != err_status_ok) && (rstat != err_status_replay_old)) { + printf("replay_check_unordered failed at index %u\n", idx); + return err_status_algo_fail; + } + return err_status_ok; +} + +#define MAX_IDX 160 + +err_status_t +test_replay_dbx(int num_trials) { + rdbx_t rdbx; + uint32_t idx, ircvd; + ut_connection utc; + err_status_t status; + int num_fp_trials; + + status = rdbx_init(&rdbx); + if (status) { + printf("replay_init failed with error code %d\n", status); + exit(1); + } + + /* + * test sequential insertion + */ + printf("\ttesting sequential insertion..."); + for (idx=0; idx < num_trials; idx++) { + status = rdbx_check_add(&rdbx, idx); + if (status) + return status; + } + printf("passed\n"); + + /* + * test for false positives by checking all of the index + * values which we've just added + * + * note that we limit the number of trials here, since allowing the + * rollover counter to roll over would defeat this test + */ + num_fp_trials = num_trials % 0x10000; + if (num_fp_trials == 0) { + printf("warning: no false positive tests performed\n"); + } + printf("\ttesting for false positives..."); + for (idx=0; idx < num_fp_trials; idx++) { + status = rdbx_check_expect_failure(&rdbx, idx); + if (status) + return status; + } + printf("passed\n"); + + /* re-initialize */ + if (rdbx_init(&rdbx) != err_status_ok) { + printf("replay_init failed\n"); + return err_status_init_fail; + } + + /* + * test non-sequential insertion + * + * this test covers only fase negatives, since the values returned + * by ut_next_index(...) are distinct + */ + ut_init(&utc); + + printf("\ttesting non-sequential insertion..."); + for (idx=0; idx < num_trials; idx++) { + ircvd = ut_next_index(&utc); + status = rdbx_check_unordered(&rdbx, ircvd); + if (status) + return status; + } + printf("passed\n"); + + return err_status_ok; +} + + + +#include <time.h> /* for clock() */ +#include <stdlib.h> /* for random() */ + +double +rdbx_check_adds_per_second(int num_trials) { + uint32_t i; + int delta; + rdbx_t rdbx; + xtd_seq_num_t est; + clock_t timer; + int failures; /* count number of failures */ + + if (rdbx_init(&rdbx) != err_status_ok) { + printf("replay_init failed\n"); + exit(1); + } + + failures = 0; + timer = clock(); + for(i=0; i < num_trials; i++) { + + delta = index_guess(&rdbx.index, &est, i); + + if (rdbx_check(&rdbx, delta) != err_status_ok) + ++failures; + else + if (rdbx_add_index(&rdbx, delta) != err_status_ok) + ++failures; + } + timer = clock() - timer; + + printf("number of failures: %d \n", failures); + + return (double) CLOCKS_PER_SEC * num_trials / timer; +} + diff --git a/libs/srtp/test/replay_driver.c b/libs/srtp/test/replay_driver.c new file mode 100644 index 0000000000..4aa2b02158 --- /dev/null +++ b/libs/srtp/test/replay_driver.c @@ -0,0 +1,180 @@ +/* + * replay_driver.c + * + * A driver for the replay_database implementation + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include <stdio.h> + +#include "rdb.h" +#include "ut_sim.h" + +void +test_rdb_db(void); + +double +rdb_check_adds_per_second(void); + +int +main (void) { + + printf("testing anti-replay database (rdb_t)...\n"); + test_rdb_db(); + printf("done\n"); + + printf("rdb_check/rdb_adds per second: %e\n", + rdb_check_adds_per_second()); + + return 0; +} + + +void +print_rdb(rdb_t *rdb) { + printf("rdb: {%u, %s}\n", rdb->window_start, v128_bit_string(&rdb->bitmask)); +} + +void +rdb_check_add(rdb_t *rdb, uint32_t idx) { + + if (rdb_check(rdb, idx) != err_status_ok) { + printf("rdb_check failed at index %u\n", idx); + return; + } + if (rdb_add_index(rdb, idx) != err_status_ok) + printf("rdb_add_index failed at index %u\n", idx); + +} + +void +rdb_check_expect_failure(rdb_t *rdb, uint32_t idx) { + + if (rdb_check(rdb, idx) != err_status_fail) + printf("rdb_check failed at index %u (false positive)\n", idx); +} + +void +rdb_check_unordered(rdb_t *rdb, uint32_t idx) { + err_status_t rstat; + + rstat = rdb_check(rdb, idx); + if ((rstat != err_status_ok) && (rstat != err_status_replay_old)) + printf("rdb_check_unordered failed at index %u\n", idx); +} + + + +#define MAX_IDX 160 + +void +test_rdb_db() { + rdb_t rdb; + uint32_t idx, ircvd; + ut_connection utc; + + if (rdb_init(&rdb) != err_status_ok) { + printf("rdb_init failed\n"); + exit(1); + } + + /* test sequential insertion */ + for (idx=0; idx < MAX_IDX; idx++) { + rdb_check_add(&rdb, idx); + } + + /* test for false positives */ + for (idx=0; idx < MAX_IDX; idx++) { + rdb_check_expect_failure(&rdb, idx); + } + + /* re-initialize */ + if (rdb_init(&rdb) != err_status_ok) { + printf("rdb_init failed\n"); + exit(1); + } + + /* test non-sequential insertion */ + ut_init(&utc); + + for (idx=0; idx < MAX_IDX; idx++) { + ircvd = ut_next_index(&utc); + rdb_check_unordered(&rdb, ircvd); + } + +} + +#include <time.h> /* for clock() */ +#include <stdlib.h> /* for random() */ + +#define REPLAY_NUM_TRIALS 10000000 + +double +rdb_check_adds_per_second(void) { + uint32_t i; + rdb_t rdb; + clock_t timer; + int failures; /* count number of failures */ + + if (rdb_init(&rdb) != err_status_ok) { + printf("rdb_init failed\n"); + exit(1); + } + + timer = clock(); + for(i=0; i < REPLAY_NUM_TRIALS; i+=3) { + if (rdb_check(&rdb, i+2) != err_status_ok) + ++failures; + if (rdb_add_index(&rdb, i+2) != err_status_ok) + ++failures; + if (rdb_check(&rdb, i+1) != err_status_ok) + ++failures; + if (rdb_add_index(&rdb, i+1) != err_status_ok) + ++failures; + if (rdb_check(&rdb, i) != err_status_ok) + ++failures; + if (rdb_add_index(&rdb, i) != err_status_ok) + ++failures; + } + timer = clock() - timer; + + return (double) CLOCKS_PER_SEC * REPLAY_NUM_TRIALS / timer; +} diff --git a/libs/srtp/test/roc_driver.c b/libs/srtp/test/roc_driver.c new file mode 100644 index 0000000000..42f4185f0f --- /dev/null +++ b/libs/srtp/test/roc_driver.c @@ -0,0 +1,165 @@ +/* + * roc_driver.c + * + * test driver for rollover counter replay implementation + * + * David A. McGrew + * Cisco Systems, Inc. + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#include <stdio.h> + +/* + * defining ROC_TEST causes small datatypes to be used in + * xtd_seq_num_t - this allows the functions to be exhaustively tested. + */ +#if ROC_NEEDS_TO_BE_TESTED +#define ROC_TEST +#endif + +#include "rdbx.h" +#include "ut_sim.h" + +err_status_t +roc_test(int num_trials); + +int +main (void) { + err_status_t status; + + printf("rollover counter test driver\n" + "David A. McGrew\n" + "Cisco Systems, Inc.\n"); + + printf("testing index functions..."); + status = roc_test(1 << 18); + if (status) { + printf("failed\n"); + exit(status); + } + printf("passed\n"); + return 0; +} + + +#define ROC_VERBOSE 0 + +err_status_t +roc_test(int num_trials) { + xtd_seq_num_t local, est, ref; + ut_connection utc; + int i, num_bad_est = 0; + int delta; + uint32_t ircvd; + double failure_rate; + + index_init(&local); + index_init(&ref); + index_init(&est); + + printf("\n\ttesting sequential insertion..."); + for (i=0; i < 2048; i++) { + delta = index_guess(&local, &est, (uint16_t) ref); +#if ROC_VERBOSE + printf("%lld, %lld, %d\n", ref, est, i); +#endif + if (ref != est) { +#if ROC_VERBOSE + printf(" *bad estimate*\n"); +#endif + ++num_bad_est; + } + index_advance(&ref, 1); + } + failure_rate = (double) num_bad_est / num_trials; + if (failure_rate > 0.01) { + printf("error: failure rate too high (%d bad estimates in %d trials)\n", + num_bad_est, num_trials); + return err_status_algo_fail; + } + printf("done\n"); + + + printf("\ttesting non-sequential insertion..."); + index_init(&local); + index_init(&ref); + index_init(&est); + ut_init(&utc); + + for (i=0; i < num_trials; i++) { + + /* get next seq num from unreliable transport simulator */ + ircvd = ut_next_index(&utc); + + /* set ref to value of ircvd */ + ref = ircvd; + + /* estimate index based on low bits of ircvd */ + delta = index_guess(&local, &est, (uint16_t) ref); +#if ROC_VERBOSE + printf("ref: %lld, local: %lld, est: %lld, ircvd: %d, delta: %d\n", + ref, local, est, ircvd, delta); +#endif + + /* now update local xtd_seq_num_t as necessary */ + if (delta > 0) + index_advance(&local, delta); + + if (ref != est) { +#if ROC_VERBOSE + printf(" *bad estimate*\n"); +#endif + /* record failure event */ + ++num_bad_est; + + /* reset local value to correct value */ + local = ref; + } + } + failure_rate = (double) num_bad_est / num_trials; + if (failure_rate > 0.01) { + printf("error: failure rate too high (%d bad estimates in %d trials)\n", + num_bad_est, num_trials); + return err_status_algo_fail; + } + printf("done\n"); + + return err_status_ok; +} diff --git a/libs/srtp/test/rtp.c b/libs/srtp/test/rtp.c new file mode 100644 index 0000000000..8804238d78 --- /dev/null +++ b/libs/srtp/test/rtp.c @@ -0,0 +1,150 @@ +/* + * rtp.c + * + * library functions for the real-time transport protocol + * + * David A. McGrew + * Cisco Systems, Inc. + */ + + +#include "rtp.h" + +#include <stdio.h> +#include <string.h> + +#include <sys/types.h> +#ifdef HAVE_SYS_SOCKET_H +# include <sys/socket.h> +#endif + +#define PRINT_DEBUG 0 /* set to 1 to print out debugging data */ +#define VERBOSE_DEBUG 0 /* set to 1 to print out more data */ + +ssize_t +rtp_sendto(rtp_sender_t *sender, const void* msg, int len) { + int octets_sent; + err_status_t stat; + int pkt_len = len + rtp_header_len; + + /* marshal data */ + strncpy(sender->message.body, msg, len); + + /* update header */ + sender->message.header.seq = ntohs(sender->message.header.seq) + 1; + sender->message.header.seq = htons(sender->message.header.seq); + sender->message.header.ts = ntohl(sender->message.header.ts) + 1; + sender->message.header.ts = htonl(sender->message.header.ts); + + /* apply srtp */ + stat = srtp_protect(sender->srtp_ctx, &sender->message.header, &pkt_len); + if (stat) { +#if PRINT_DEBUG + fprintf(stderr, "error: srtp protection failed with code %d\n", stat); +#endif + return -1; + } +#if VERBOSE_DEBUG + srtp_print_packet(&sender->message.header, pkt_len); +#endif + octets_sent = sendto(sender->socket, (void*)&sender->message, + pkt_len, 0, (struct sockaddr *)&sender->addr, + sizeof (struct sockaddr_in)); + + if (octets_sent != pkt_len) { +#if PRINT_DEBUG + fprintf(stderr, "error: couldn't send message %s", (char *)msg); + perror(""); +#endif + } + + return octets_sent; +} + +ssize_t +rtp_recvfrom(rtp_receiver_t *receiver, void *msg, int *len) { + int octets_recvd; + err_status_t stat; + + octets_recvd = recvfrom(receiver->socket, (void *)&receiver->message, + *len, 0, (struct sockaddr *) NULL, 0); + + /* verify rtp header */ + if (receiver->message.header.version != 2) { + *len = 0; + return -1; + } + +#if PRINT_DEBUG + fprintf(stderr, "%d octets received from SSRC %u\n", + octets_recvd, receiver->message.header.ssrc); +#endif +#if VERBOSE_DEBUG + srtp_print_packet(&receiver->message.header, octets_recvd); +#endif + + /* apply srtp */ + stat = srtp_unprotect(receiver->srtp_ctx, + &receiver->message.header, &octets_recvd); + if (stat) { + fprintf(stderr, + "error: srtp unprotection failed with code %d%s\n", stat, + stat == err_status_replay_fail ? " (replay check failed)" : + stat == err_status_auth_fail ? " (auth check failed)" : ""); + return -1; + } + strncpy(msg, receiver->message.body, octets_recvd); + + return octets_recvd; +} + +int +rtp_sender_init(rtp_sender_t *sender, + int socket, + struct sockaddr_in addr, + uint32_t ssrc) { + + /* set header values */ + sender->message.header.ssrc = htonl(ssrc); + sender->message.header.ts = 0; + sender->message.header.seq = (uint16_t) rand(); + sender->message.header.m = 0; + sender->message.header.pt = 0x1; + sender->message.header.version = 2; + sender->message.header.p = 0; + sender->message.header.x = 0; + sender->message.header.cc = 0; + + /* set other stuff */ + sender->socket = socket; + sender->addr = addr; + + return 0; +} + +int +rtp_receiver_init(rtp_receiver_t *rcvr, + int socket, + struct sockaddr_in addr, + uint32_t ssrc) { + + /* set header values */ + rcvr->message.header.ssrc = htonl(ssrc); + rcvr->message.header.ts = 0; + rcvr->message.header.seq = 0; + rcvr->message.header.m = 0; + rcvr->message.header.pt = 0x1; + rcvr->message.header.version = 2; + rcvr->message.header.p = 0; + rcvr->message.header.x = 0; + rcvr->message.header.cc = 0; + + /* set other stuff */ + rcvr->socket = socket; + rcvr->addr = addr; + + return 0; +} + + + diff --git a/libs/srtp/test/rtpw.c b/libs/srtp/test/rtpw.c new file mode 100644 index 0000000000..906080de1e --- /dev/null +++ b/libs/srtp/test/rtpw.c @@ -0,0 +1,507 @@ +/* + * rtpw.c + * + * rtp word sender/receiver + * + * David A. McGrew + * Cisco Systems, Inc. + * + * This app is a simple RTP application intended only for testing + * libsrtp. It reads one word at a time from /usr/dict/words (or + * whatever file is specified as DICT_FILE), and sends one word out + * each USEC_RATE microseconds. Secure RTP protections can be + * applied. See the usage() function for more details. + * + */ + +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#include "datatypes.h" + +#include <stdio.h> /* for printf, fprintf */ +#include <stdlib.h> /* for atoi() */ +#include <errno.h> +#include <unistd.h> /* for close() */ +#include <string.h> /* for strncpy() */ +#include <time.h> /* for usleep() */ +#ifdef HAVE_SYS_SOCKET_H +# include <sys/socket.h> +#endif +#ifdef HAVE_NETINET_IN_H +# include <netinet/in.h> +#elif defined HAVE_WINSOCK2_H +# include <winsock2.h> +# include <ws2tcpip.h> +# define RTPW_USE_WINSOCK2 1 +#endif +#ifdef HAVE_ARPA_INET_H +# include <arpa/inet.h> +#endif + +#include "srtp.h" +#include "rtp.h" + +#ifdef RTPW_USE_WINSOCK2 +# define DICT_FILE "words.txt" +#else +# define DICT_FILE "/usr/share/dict/words" +#endif +#define USEC_RATE (5e5) +#define MAX_WORD_LEN 128 +#define ADDR_IS_MULTICAST(a) IN_MULTICAST(htonl(a)) +#define MAX_KEY_LEN 64 +#define MASTER_KEY_LEN 30 + + +#ifndef HAVE_USLEEP +# ifdef HAVE_WINDOWS_H +# define usleep(us) Sleep((us)/1000) +# else +# define usleep(us) sleep((us)/1000000) +# endif +#endif + + +/* + * the function usage() prints an error message describing how this + * program should be called, then calls exit() + */ + +void +usage(char *prog_name); + +/* + * leave_group(...) de-registers from a multicast group + */ + +void +leave_group(int sock, struct ip_mreq mreq, char *name); + + +/* + * program_type distinguishes the [s]rtp sender and receiver cases + */ + +typedef enum { sender, receiver, unknown } program_type; + +int +main (int argc, char *argv[]) { + char *dictfile = DICT_FILE; + FILE *dict; + char word[MAX_WORD_LEN]; + int sock, ret; + struct in_addr rcvr_addr; + struct sockaddr_in name; + struct ip_mreq mreq; +#if BEW + struct sockaddr_in local; +#endif + program_type prog_type = unknown; + sec_serv_t sec_servs = sec_serv_none; + unsigned char ttl = 5; + int c; + char *input_key = NULL; + char *address = NULL; + char key[MAX_KEY_LEN]; + unsigned short port = 0; + rtp_sender_t snd; + srtp_policy_t policy; + err_status_t status; + int len; + int do_list_mods = 0; + uint32_t ssrc = 0xdeadbeef; /* ssrc value hardcoded for now */ +#ifdef RTPW_USE_WINSOCK2 + WORD wVersionRequested = MAKEWORD(2, 0); + WSADATA wsaData; + + ret = WSAStartup(wVersionRequested, &wsaData); + if (ret != 0) { + fprintf(stderr, "error: WSAStartup() failed: %d\n", ret); + exit(1); + } +#endif + + /* initialize srtp library */ + status = srtp_init(); + if (status) { + printf("error: srtp initialization failed with error code %d\n", status); + exit(1); + } + + /* check args */ + while (1) { + c = getopt(argc, argv, "k:rsaeld:"); + if (c == -1) { + break; + } + switch (c) { + case 'k': + input_key = optarg; + break; + case 'e': + sec_servs |= sec_serv_conf; + break; + case 'a': + sec_servs |= sec_serv_auth; + break; + case 'r': + prog_type = receiver; + break; + case 's': + prog_type = sender; + break; + case 'd': + status = crypto_kernel_set_debug_module(optarg, 1); + if (status) { + printf("error: set debug module (%s) failed\n", optarg); + exit(1); + } + break; + case 'l': + do_list_mods = 1; + break; + default: + usage(argv[0]); + } + } + + if (prog_type == unknown) { + if (do_list_mods) { + status = crypto_kernel_list_debug_modules(); + if (status) { + printf("error: list of debug modules failed\n"); + exit(1); + } + return 0; + } else { + printf("error: neither sender [-s] nor receiver [-r] specified\n"); + usage(argv[0]); + } + } + + if ((sec_servs && !input_key) || (!sec_servs && input_key)) { + /* + * a key must be provided if and only if security services have + * been requested + */ + usage(argv[0]); + } + + if (argc != optind + 2) { + /* wrong number of arguments */ + usage(argv[0]); + } + + /* get address from arg */ + address = argv[optind++]; + + /* get port from arg */ + port = atoi(argv[optind++]); + + /* set address */ +#ifdef HAVE_INET_ATON + if (0 == inet_aton(address, &rcvr_addr)) { + fprintf(stderr, "%s: cannot parse IP v4 address %s\n", argv[0], address); + exit(1); + } + if (rcvr_addr.s_addr == INADDR_NONE) { + fprintf(stderr, "%s: address error", argv[0]); + exit(1); + } +#else + rcvr_addr.s_addr = inet_addr(address); + if (0xffffffff == rcvr_addr.s_addr) { + fprintf(stderr, "%s: cannot parse IP v4 address %s\n", argv[0], address); + exit(1); + } +#endif + + /* open socket */ + sock = socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP); + if (sock < 0) { + int err; +#ifdef RTPW_USE_WINSOCK2 + err = WSAGetLastError(); +#else + err = errno; +#endif + fprintf(stderr, "%s: couldn't open socket: %d\n", argv[0], err); + exit(1); + } + + name.sin_addr = rcvr_addr; + name.sin_family = PF_INET; + name.sin_port = htons(port); + + if (ADDR_IS_MULTICAST(rcvr_addr.s_addr)) { + if (prog_type == sender) { + ret = setsockopt(sock, IPPROTO_IP, IP_MULTICAST_TTL, &ttl, + sizeof(ttl)); + if (ret < 0) { + fprintf(stderr, "%s: Failed to set TTL for multicast group", argv[0]); + perror(""); + exit(1); + } + } + + mreq.imr_multiaddr.s_addr = rcvr_addr.s_addr; + mreq.imr_interface.s_addr = htonl(INADDR_ANY); + ret = setsockopt(sock, IPPROTO_IP, IP_ADD_MEMBERSHIP, (void*)&mreq, + sizeof(mreq)); + if (ret < 0) { + fprintf(stderr, "%s: Failed to join multicast group", argv[0]); + perror(""); + exit(1); + } + } + + /* report security services selected on the command line */ + printf("security services: "); + if (sec_servs & sec_serv_conf) + printf("confidentiality "); + if (sec_servs & sec_serv_auth) + printf("message authentication"); + if (sec_servs == sec_serv_none) + printf("none"); + printf("\n"); + + /* set up the srtp policy and master key */ + if (sec_servs) { + /* + * create policy structure, using the default mechanisms but + * with only the security services requested on the command line, + * using the right SSRC value + */ + switch (sec_servs) { + case sec_serv_conf_and_auth: + crypto_policy_set_rtp_default(&policy.rtp); + crypto_policy_set_rtcp_default(&policy.rtcp); + break; + case sec_serv_conf: + crypto_policy_set_aes_cm_128_null_auth(&policy.rtp); + crypto_policy_set_rtcp_default(&policy.rtcp); + break; + case sec_serv_auth: + crypto_policy_set_null_cipher_hmac_sha1_80(&policy.rtp); + crypto_policy_set_rtcp_default(&policy.rtcp); + break; + default: + printf("error: unknown security service requested\n"); + return -1; + } + policy.ssrc.type = ssrc_specific; + policy.ssrc.value = ssrc; + policy.key = (uint8_t *) key; + policy.next = NULL; + policy.rtp.sec_serv = sec_servs; + policy.rtcp.sec_serv = sec_serv_none; /* we don't do RTCP anyway */ + + /* + * read key from hexadecimal on command line into an octet string + */ + len = hex_string_to_octet_string(key, input_key, MASTER_KEY_LEN*2); + + /* check that hex string is the right length */ + if (len < MASTER_KEY_LEN*2) { + fprintf(stderr, + "error: too few digits in key/salt " + "(should be %d hexadecimal digits, found %d)\n", + MASTER_KEY_LEN*2, len); + exit(1); + } + if (strlen(input_key) > MASTER_KEY_LEN*2) { + fprintf(stderr, + "error: too many digits in key/salt " + "(should be %d hexadecimal digits, found %u)\n", + MASTER_KEY_LEN*2, (unsigned)strlen(input_key)); + exit(1); + } + + printf("set master key/salt to %s/", octet_string_hex_string(key, 16)); + printf("%s\n", octet_string_hex_string(key+16, 14)); + + } else { + /* + * we're not providing security services, so set the policy to the + * null policy + * + * Note that this policy does not conform to the SRTP + * specification, since RTCP authentication is required. However, + * the effect of this policy is to turn off SRTP, so that this + * application is now a vanilla-flavored RTP application. + */ + policy.key = (uint8_t *)key; + policy.ssrc.type = ssrc_specific; + policy.ssrc.value = ssrc; + policy.rtp.cipher_type = NULL_CIPHER; + policy.rtp.cipher_key_len = 0; + policy.rtp.auth_type = NULL_AUTH; + policy.rtp.auth_key_len = 0; + policy.rtp.auth_tag_len = 0; + policy.rtp.sec_serv = sec_serv_none; + policy.rtcp.cipher_type = NULL_CIPHER; + policy.rtcp.cipher_key_len = 0; + policy.rtcp.auth_type = NULL_AUTH; + policy.rtcp.auth_key_len = 0; + policy.rtcp.auth_tag_len = 0; + policy.rtcp.sec_serv = sec_serv_none; + policy.next = NULL; + } + + if (prog_type == sender) { + +#if BEW + /* bind to local socket (to match crypto policy, if need be) */ + memset(&local, 0, sizeof(struct sockaddr_in)); + local.sin_addr.s_addr = htonl(INADDR_ANY); + local.sin_port = htons(port); + ret = bind(sock, (struct sockaddr *) &local, sizeof(struct sockaddr_in)); + if (ret < 0) { + fprintf(stderr, "%s: bind failed\n", argv[0]); + perror(""); + exit(1); + } +#endif /* BEW */ + + /* initialize sender's rtp and srtp contexts */ + rtp_sender_init(&snd, sock, name, ssrc); + status = srtp_create(&snd.srtp_ctx, &policy); + if (status) { + fprintf(stderr, + "error: srtp_create() failed with code %d\n", + status); + exit(1); + } + + /* open dictionary */ + dict = fopen (dictfile, "r"); + if (dict == NULL) { + fprintf(stderr, "%s: couldn't open file %s\n", argv[0], dictfile); + if (ADDR_IS_MULTICAST(rcvr_addr.s_addr)) { + leave_group(sock, mreq, argv[0]); + } + exit(1); + } + + /* read words from dictionary, then send them off */ + while (fgets(word, MAX_WORD_LEN, dict) != NULL) { + len = strlen(word) + 1; /* plus one for null */ + + if (len > MAX_WORD_LEN) + printf("error: word %s too large to send\n", word); + else { + rtp_sendto(&snd, word, len); + printf("sending word: %s", word); + } + usleep(USEC_RATE); + } + + } else { /* prog_type == receiver */ + rtp_receiver_t rcvr; + + if (bind(sock, (struct sockaddr *)&name, sizeof(name)) < 0) { + close(sock); + fprintf(stderr, "%s: socket bind error\n", argv[0]); + perror(NULL); + if (ADDR_IS_MULTICAST(rcvr_addr.s_addr)) { + leave_group(sock, mreq, argv[0]); + } + exit(1); + } + + rtp_receiver_init(&rcvr, sock, name, ssrc); + status = srtp_create(&rcvr.srtp_ctx, &policy); + if (status) { + fprintf(stderr, + "error: srtp_create() failed with code %d\n", + status); + exit(1); + } + + /* get next word and loop */ + while (1) { + len = MAX_WORD_LEN; + if (rtp_recvfrom(&rcvr, word, &len) > -1) + printf("\tword: %s", word); + } + + } + + if (ADDR_IS_MULTICAST(rcvr_addr.s_addr)) { + leave_group(sock, mreq, argv[0]); + } + +#ifdef RTPW_USE_WINSOCK2 + WSACleanup(); +#endif + + return 0; +} + + +void +usage(char *string) { + + printf("usage: %s [-d <debug>]* [-k <key> [-a][-e]] " + "[-s | -r] dest_ip dest_port\n" + "or %s -l\n" + "where -a use message authentication\n" + " -e use encryption\n" + " -k <key> sets the srtp master key\n" + " -s act as rtp sender\n" + " -r act as rtp receiver\n" + " -l list debug modules\n" + " -d <debug> turn on debugging for module <debug>\n", + string, string); + exit(1); + +} + + +void +leave_group(int sock, struct ip_mreq mreq, char *name) { + int ret; + + ret = setsockopt(sock, IPPROTO_IP, IP_DROP_MEMBERSHIP, (void*)&mreq, + sizeof(mreq)); + if (ret < 0) { + fprintf(stderr, "%s: Failed to leave multicast group", name); + perror(""); + } +} + diff --git a/libs/srtp/test/srtp_driver.c b/libs/srtp/test/srtp_driver.c new file mode 100644 index 0000000000..363c5ed688 --- /dev/null +++ b/libs/srtp/test/srtp_driver.c @@ -0,0 +1,1274 @@ +/* + * srtp_driver.c + * + * a test driver for libSRTP + * + * David A. McGrew + * Cisco Systems, Inc. + */ +/* + * + * Copyright (c) 2001-2005, Cisco Systems, Inc. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above + * copyright notice, this list of conditions and the following + * disclaimer in the documentation and/or other materials provided + * with the distribution. + * + * Neither the name of the Cisco Systems, Inc. nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED + * OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + + +#include <string.h> /* for memcpy() */ +#include <time.h> /* for clock() */ +#include <stdlib.h> /* for malloc(), free() */ +#include <stdio.h> /* for print(), fflush() */ +#include <unistd.h> /* for getopt() */ + +#include "srtp.h" + +#ifdef HAVE_NETINET_IN_H +# include <netinet/in.h> +#elif defined HAVE_WINSOCK2_H +# include <winsock2.h> +#endif + +#define PRINT_REFERENCE_PACKET 1 + +err_status_t +srtp_validate(void); + +err_status_t +srtp_create_big_policy(srtp_policy_t **list); + +err_status_t +srtp_test_remove_stream(void); + +double +srtp_bits_per_second(int msg_len_octets, const srtp_policy_t *policy); + +double +srtp_rejections_per_second(int msg_len_octets, const srtp_policy_t *policy); + +void +srtp_do_timing(const srtp_policy_t *policy); + +void +srtp_do_rejection_timing(const srtp_policy_t *policy); + +err_status_t +srtp_test(const srtp_policy_t *policy); + +err_status_t +srtp_session_print_policy(srtp_t srtp); + +err_status_t +srtp_print_policy(const srtp_policy_t *policy); + +char * +srtp_packet_to_string(srtp_hdr_t *hdr, int packet_len); + +double +mips_estimate(int num_trials, int *ignore); + +extern uint8_t test_key[30]; + +void +usage(char *prog_name) { + printf("usage: %s [ -t ][ -c ][ -v ][-d <debug_module> ]* [ -l ]\n" + " -t run timing test\n" + " -r run rejection timing test\n" + " -c run codec timing test\n" + " -v run validation tests\n" + " -d <mod> turn on debugging module <mod>\n" + " -l list debugging modules\n", prog_name); + exit(1); +} + +/* + * The policy_array is a null-terminated array of policy structs. it + * is declared at the end of this file + */ + +extern const srtp_policy_t *policy_array[]; + + +/* the wildcard_policy is declared below; it has a wildcard ssrc */ + +extern const srtp_policy_t wildcard_policy; + +/* + * mod_driver debug module - debugging module for this test driver + * + * we use the crypto_kernel debugging system in this driver, which + * makes the interface uniform and increases portability + */ + +debug_module_t mod_driver = { + 0, /* debugging is off by default */ + "driver" /* printable name for module */ +}; + +int +main (int argc, char *argv[]) { + char q; + unsigned do_timing_test = 0; + unsigned do_rejection_test = 0; + unsigned do_codec_timing = 0; + unsigned do_validation = 0; + unsigned do_list_mods = 0; + err_status_t status; + + /* + * verify that the compiler has interpreted the header data + * structure srtp_hdr_t correctly + */ + if (sizeof(srtp_hdr_t) != 12) { + printf("error: srtp_hdr_t has incorrect size\n"); + exit(1); + } + + /* initialize srtp library */ + status = srtp_init(); + if (status) { + printf("error: srtp init failed with error code %d\n", status); + exit(1); + } + + /* load srtp_driver debug module */ + status = crypto_kernel_load_debug_module(&mod_driver); + if (status) { + printf("error: load of srtp_driver debug module failed " + "with error code %d\n", status); + exit(1); + } + + /* process input arguments */ + while (1) { + q = getopt(argc, argv, "trcvld:"); + if (q == -1) + break; + switch (q) { + case 't': + do_timing_test = 1; + break; + case 'r': + do_rejection_test = 1; + break; + case 'c': + do_codec_timing = 1; + break; + case 'v': + do_validation = 1; + break; + case 'l': + do_list_mods = 1; + break; + case 'd': + status = crypto_kernel_set_debug_module(optarg, 1); + if (status) { + printf("error: set debug module (%s) failed\n", optarg); + exit(1); + } + break; + default: + usage(argv[0]); + } + } + + if (!do_validation && !do_timing_test && !do_codec_timing + && !do_list_mods && !do_rejection_test) + usage(argv[0]); + + if (do_list_mods) { + status = crypto_kernel_list_debug_modules(); + if (status) { + printf("error: list of debug modules failed\n"); + exit(1); + } + } + + if (do_validation) { + const srtp_policy_t **policy = policy_array; + srtp_policy_t *big_policy; + + /* loop over policy array, testing srtp for each policy */ + while (*policy != NULL) { + printf("testing srtp_protect and srtp_unprotect\n"); + if (srtp_test(*policy) == err_status_ok) + printf("passed\n\n"); + else { + printf("failed\n"); + exit(1); + } + policy++; + } + + /* create a big policy list and run tests on it */ + status = srtp_create_big_policy(&big_policy); + if (status) { + printf("unexpected failure with error code %d\n", status); + exit(1); + } + printf("testing srtp_protect and srtp_unprotect with big policy\n"); + if (srtp_test(big_policy) == err_status_ok) + printf("passed\n\n"); + else { + printf("failed\n"); + exit(1); + } + + /* run test on wildcard policy */ + printf("testing srtp_protect and srtp_unprotect on " + "wildcard ssrc policy\n"); + if (srtp_test(&wildcard_policy) == err_status_ok) + printf("passed\n\n"); + else { + printf("failed\n"); + exit(1); + } + + /* + * run validation test against the reference packets - note + * that this test only covers the default policy + */ + printf("testing srtp_protect and srtp_unprotect against " + "reference packets\n"); + if (srtp_validate() == err_status_ok) + printf("passed\n\n"); + else { + printf("failed\n"); + exit(1); + } + + /* + * test the function srtp_remove_stream() + */ + printf("testing srtp_remove_stream()..."); + if (srtp_test_remove_stream() == err_status_ok) + printf("passed\n"); + else { + printf("failed\n"); + exit(1); + } + } + + if (do_timing_test) { + const srtp_policy_t **policy = policy_array; + + /* loop over policies, run timing test for each */ + while (*policy != NULL) { + srtp_print_policy(*policy); + srtp_do_timing(*policy); + policy++; + } + } + + if (do_rejection_test) { + const srtp_policy_t **policy = policy_array; + + /* loop over policies, run rejection timing test for each */ + while (*policy != NULL) { + srtp_print_policy(*policy); + srtp_do_rejection_timing(*policy); + policy++; + } + } + + if (do_codec_timing) { + srtp_policy_t policy; + int ignore; + double mips = mips_estimate(1000000000, &ignore); + + crypto_policy_set_rtp_default(&policy.rtp); + crypto_policy_set_rtcp_default(&policy.rtcp); + policy.ssrc.type = ssrc_specific; + policy.ssrc.value = 0xdecafbad; + policy.key = test_key; + policy.next = NULL; + + printf("mips estimate: %e\n", mips); + + printf("testing srtp processing time for voice codecs:\n"); + printf("codec\t\tlength (octets)\t\tsrtp instructions/second\n"); + printf("G.711\t\t%d\t\t\t%e\n", 80, + (double) mips * (80 * 8) / + srtp_bits_per_second(80, &policy) / .01 ); + printf("G.711\t\t%d\t\t\t%e\n", 160, + (double) mips * (160 * 8) / + srtp_bits_per_second(160, &policy) / .02); + printf("G.726-32\t%d\t\t\t%e\n", 40, + (double) mips * (40 * 8) / + srtp_bits_per_second(40, &policy) / .01 ); + printf("G.726-32\t%d\t\t\t%e\n", 80, + (double) mips * (80 * 8) / + srtp_bits_per_second(80, &policy) / .02); + printf("G.729\t\t%d\t\t\t%e\n", 10, + (double) mips * (10 * 8) / + srtp_bits_per_second(10, &policy) / .01 ); + printf("G.729\t\t%d\t\t\t%e\n", 20, + (double) mips * (20 * 8) / + srtp_bits_per_second(20, &policy) / .02 ); + } + + return 0; +} + + + +/* + * srtp_create_test_packet(len, ssrc) returns a pointer to a + * (malloced) example RTP packet whose data field has the length given + * by pkt_octet_len and the SSRC value ssrc. The total length of the + * packet is twelve octets longer, since the header is at the + * beginning. There is room at the end of the packet for a trailer, + * and the four octets following the packet are filled with 0xff + * values to enable testing for overwrites. + * + * note that the location of the test packet can (and should) be + * deallocated with the free() call once it is no longer needed. + */ + +srtp_hdr_t * +srtp_create_test_packet(int pkt_octet_len, uint32_t ssrc) { + int i; + uint8_t *buffer; + srtp_hdr_t *hdr; + int bytes_in_hdr = 12; + + /* allocate memory for test packet */ + hdr = malloc(pkt_octet_len + bytes_in_hdr + + SRTP_MAX_TRAILER_LEN + 4); + if (!hdr) + return NULL; + + hdr->version = 2; /* RTP version two */ + hdr->p = 0; /* no padding needed */ + hdr->x = 0; /* no header extension */ + hdr->cc = 0; /* no CSRCs */ + hdr->m = 0; /* marker bit */ + hdr->pt = 0xf; /* payload type */ + hdr->seq = htons(0x1234); /* sequence number */ + hdr->ts = htonl(0xdecafbad); /* timestamp */ + hdr->ssrc = htonl(ssrc); /* synch. source */ + + buffer = (uint8_t *)hdr; + buffer += bytes_in_hdr; + + /* set RTP data to 0xab */ + for (i=0; i < pkt_octet_len; i++) + *buffer++ = 0xab; + + /* set post-data value to 0xffff to enable overrun checking */ + for (i=0; i < SRTP_MAX_TRAILER_LEN+4; i++) + *buffer++ = 0xff; + + return hdr; +} + +void +srtp_do_timing(const srtp_policy_t *policy) { + int len; + + /* + * note: the output of this function is formatted so that it + * can be used in gnuplot. '#' indicates a comment, and "\r\n" + * terminates a record + */ + + printf("# testing srtp throughput:\r\n"); + printf("# mesg length (octets)\tthroughput (megabits per second)\r\n"); + + for (len=16; len <= 2048; len *= 2) + printf("%d\t\t\t%f\r\n", len, + srtp_bits_per_second(len, policy) / 1.0E6); + + /* these extra linefeeds let gnuplot know that a dataset is done */ + printf("\r\n\r\n"); + +} + +void +srtp_do_rejection_timing(const srtp_policy_t *policy) { + int len; + + /* + * note: the output of this function is formatted so that it + * can be used in gnuplot. '#' indicates a comment, and "\r\n" + * terminates a record + */ + + printf("# testing srtp rejection throughput:\r\n"); + printf("# mesg length (octets)\trejections per second\r\n"); + + for (len=8; len <= 2048; len *= 2) + printf("%d\t\t\t%e\r\n", len, srtp_rejections_per_second(len, policy)); + + /* these extra linefeeds let gnuplot know that a dataset is done */ + printf("\r\n\r\n"); + +} + + +#define MAX_MSG_LEN 1024 + +double +srtp_bits_per_second(int msg_len_octets, const srtp_policy_t *policy) { + srtp_t srtp; + srtp_hdr_t *mesg; + int i; + clock_t timer; + int num_trials = 100000; + int len; + uint32_t ssrc; + err_status_t status; + + /* + * allocate and initialize an srtp session + */ + status = srtp_create(&srtp, policy); + if (status) { + printf("error: srtp_create() failed with error code %d\n", status); + exit(1); + } + + /* + * if the ssrc is unspecified, use a predetermined one + */ + if (policy->ssrc.type != ssrc_specific) { + ssrc = 0xdeadbeef; + } else { + ssrc = policy->ssrc.value; + } + + /* + * create a test packet + */ + mesg = srtp_create_test_packet(msg_len_octets, ssrc); + if (mesg == NULL) + return 0.0; /* indicate failure by returning zero */ + + timer = clock(); + for (i=0; i < num_trials; i++) { + err_status_t status; + len = msg_len_octets + 12; /* add in rtp header length */ + + /* srtp protect message */ + status = srtp_protect(srtp, mesg, &len); + if (status) { + printf("error: srtp_protect() failed with error code %d\n", status); + exit(1); + } + + /* increment message number */ + mesg->seq = htons(ntohs(mesg->seq) + 1); + + } + timer = clock() - timer; + + free(mesg); + + return (double) (msg_len_octets) * 8 * + num_trials * CLOCKS_PER_SEC / timer; +} + +double +srtp_rejections_per_second(int msg_len_octets, const srtp_policy_t *policy) { + srtp_ctx_t *srtp; + srtp_hdr_t *mesg; + int i; + int len; + clock_t timer; + int num_trials = 1000000; + uint32_t ssrc = policy->ssrc.value; + err_status_t status; + + /* + * allocate and initialize an srtp session + */ + status = srtp_create(&srtp, policy); + if (status) { + printf("error: srtp_create() failed with error code %d\n", status); + exit(1); + } + + mesg = srtp_create_test_packet(msg_len_octets, ssrc); + if (mesg == NULL) + return 0.0; /* indicate failure by returning zero */ + + len = msg_len_octets; + srtp_protect(srtp, (srtp_hdr_t *)mesg, &len); + + timer = clock(); + for (i=0; i < num_trials; i++) { + len = msg_len_octets; + srtp_unprotect(srtp, (srtp_hdr_t *)mesg, &len); + } + timer = clock() - timer; + + free(mesg); + + return (double) num_trials * CLOCKS_PER_SEC / timer; +} + + +void +err_check(err_status_t s) { + if (s == err_status_ok) + return; + else + fprintf(stderr, "error: unexpected srtp failure (code %d)\n", s); + exit (1); +} + +err_status_t +srtp_test(const srtp_policy_t *policy) { + int i; + srtp_t srtp_sender; + srtp_t srtp_rcvr; + err_status_t status = err_status_ok; + srtp_hdr_t *hdr, *hdr2; + uint8_t hdr_enc[64]; + uint8_t *pkt_end; + int msg_len_octets, msg_len_enc; + int len; + int tag_length = policy->rtp.auth_tag_len; + uint32_t ssrc; + srtp_policy_t *rcvr_policy; + + err_check(srtp_create(&srtp_sender, policy)); + + /* print out policy */ + err_check(srtp_session_print_policy(srtp_sender)); + + /* + * initialize data buffer, using the ssrc in the policy unless that + * value is a wildcard, in which case we'll just use an arbitrary + * one + */ + if (policy->ssrc.type != ssrc_specific) + ssrc = 0xdecafbad; + else + ssrc = policy->ssrc.value; + msg_len_octets = 28; + hdr = srtp_create_test_packet(msg_len_octets, ssrc); + + if (hdr == NULL) + return err_status_alloc_fail; + hdr2 = srtp_create_test_packet(msg_len_octets, ssrc); + if (hdr2 == NULL) { + free(hdr); + return err_status_alloc_fail; + } + + /* set message length */ + len = msg_len_octets; + + debug_print(mod_driver, "before protection:\n%s", + srtp_packet_to_string(hdr, len)); + +#if PRINT_REFERENCE_PACKET + debug_print(mod_driver, "reference packet before protection:\n%s", + octet_string_hex_string((uint8_t *)hdr, len)); +#endif + err_check(srtp_protect(srtp_sender, hdr, &len)); + + debug_print(mod_driver, "after protection:\n%s", + srtp_packet_to_string(hdr, len)); +#if PRINT_REFERENCE_PACKET + debug_print(mod_driver, "after protection:\n%s", + octet_string_hex_string((uint8_t *)hdr, len)); +#endif + + /* save protected message and length */ + memcpy(hdr_enc, hdr, len); + msg_len_enc = len; + + /* + * check for overrun of the srtp_protect() function + * + * The packet is followed by a value of 0xfffff; if the value of the + * data following the packet is different, then we know that the + * protect function is overwriting the end of the packet. + */ + pkt_end = (uint8_t *)hdr + sizeof(srtp_hdr_t) + + msg_len_octets + tag_length; + for (i = 0; i < 4; i++) + if (pkt_end[i] != 0xff) { + fprintf(stdout, "overwrite in srtp_protect() function " + "(expected %x, found %x in trailing octet %d)\n", + 0xff, ((uint8_t *)hdr)[i], i); + free(hdr); + free(hdr2); + return err_status_algo_fail; + } + + /* + * if the policy includes confidentiality, check that ciphertext is + * different than plaintext + * + * Note that this check will give false negatives, with some small + * probability, especially if the packets are short. For that + * reason, we skip this check if the plaintext is less than four + * octets long. + */ + if ((policy->rtp.sec_serv & sec_serv_conf) && (msg_len_octets >= 4)) { + printf("testing that ciphertext is distinct from plaintext..."); + status = err_status_algo_fail; + for (i=12; i < msg_len_octets+12; i++) + if (((uint8_t *)hdr)[i] != ((uint8_t *)hdr2)[i]) { + status = err_status_ok; + } + if (status) { + printf("failed\n"); + free(hdr); + free(hdr2); + return status; + } + printf("passed\n"); + } + + /* + * if the policy uses a 'wildcard' ssrc, then we need to make a copy + * of the policy that changes the direction to inbound + * + * we always copy the policy into the rcvr_policy, since otherwise + * the compiler would fret about the constness of the policy + */ + rcvr_policy = malloc(sizeof(srtp_policy_t)); + if (rcvr_policy == NULL) + return err_status_alloc_fail; + memcpy(rcvr_policy, policy, sizeof(srtp_policy_t)); + if (policy->ssrc.type == ssrc_any_outbound) { + rcvr_policy->ssrc.type = ssrc_any_inbound; + } + + err_check(srtp_create(&srtp_rcvr, rcvr_policy)); + + err_check(srtp_unprotect(srtp_rcvr, hdr, &len)); + + debug_print(mod_driver, "after unprotection:\n%s", + srtp_packet_to_string(hdr, len)); + + /* verify that the unprotected packet matches the origial one */ + for (i=0; i < msg_len_octets; i++) + if (((uint8_t *)hdr)[i] != ((uint8_t *)hdr2)[i]) { + fprintf(stdout, "mismatch at octet %d\n", i); + status = err_status_algo_fail; + } + if (status) { + free(hdr); + free(hdr2); + return status; + } + + /* + * if the policy includes authentication, then test for false positives + */ + if (policy->rtp.sec_serv & sec_serv_auth) { + char *data = ((char *)hdr) + 12; + + printf("testing for false positives in replay check..."); + + /* set message length */ + len = msg_len_enc; + + /* unprotect a second time - should fail with a replay error */ + status = srtp_unprotect(srtp_rcvr, hdr_enc, &len); + if (status != err_status_replay_fail) { + printf("failed with error code %d\n", status); + free(hdr); + free(hdr2); + return status; + } else { + printf("passed\n"); + } + + printf("testing for false positives in auth check..."); + + /* increment sequence number in header */ + hdr->seq++; + + /* set message length */ + len = msg_len_octets; + + /* apply protection */ + err_check(srtp_protect(srtp_sender, hdr, &len)); + + /* flip bits in packet */ + data[0] ^= 0xff; + + /* unprotect, and check for authentication failure */ + status = srtp_unprotect(srtp_rcvr, hdr, &len); + if (status != err_status_auth_fail) { + printf("failed\n"); + free(hdr); + free(hdr2); + return status; + } else { + printf("passed\n"); + } + + } + + err_check(srtp_dealloc(srtp_sender)); + err_check(srtp_dealloc(srtp_rcvr)); + + free(hdr); + free(hdr2); + return err_status_ok; +} + + +err_status_t +srtp_session_print_policy(srtp_t srtp) { + char *serv_descr[4] = { + "none", + "confidentiality", + "authentication", + "confidentiality and authentication" + }; + char *direction[3] = { + "unknown", + "outbound", + "inbound" + }; + srtp_stream_t stream; + + /* sanity checking */ + if (srtp == NULL) + return err_status_fail; + + /* if there's a template stream, print it out */ + if (srtp->stream_template != NULL) { + stream = srtp->stream_template; + printf("# SSRC: any %s\r\n" + "# rtp cipher: %s\r\n" + "# rtp auth: %s\r\n" + "# rtp services: %s\r\n" + "# rtcp cipher: %s\r\n" + "# rtcp auth: %s\r\n" + "# rtcp services: %s\r\n", + direction[stream->direction], + stream->rtp_cipher->type->description, + stream->rtp_auth->type->description, + serv_descr[stream->rtp_services], + stream->rtcp_cipher->type->description, + stream->rtcp_auth->type->description, + serv_descr[stream->rtcp_services]); + } + + /* loop over streams in session, printing the policy of each */ + stream = srtp->stream_list; + while (stream != NULL) { + if (stream->rtp_services > sec_serv_conf_and_auth) + return err_status_bad_param; + + printf("# SSRC: 0x%08x\r\n" + "# rtp cipher: %s\r\n" + "# rtp auth: %s\r\n" + "# rtp services: %s\r\n" + "# rtcp cipher: %s\r\n" + "# rtcp auth: %s\r\n" + "# rtcp services: %s\r\n", + stream->ssrc, + stream->rtp_cipher->type->description, + stream->rtp_auth->type->description, + serv_descr[stream->rtp_services], + stream->rtcp_cipher->type->description, + stream->rtcp_auth->type->description, + serv_descr[stream->rtcp_services]); + + /* advance to next stream in the list */ + stream = stream->next; + } + return err_status_ok; +} + +err_status_t +srtp_print_policy(const srtp_policy_t *policy) { + err_status_t status; + srtp_t session; + + status = srtp_create(&session, policy); + if (status) + return status; + status = srtp_session_print_policy(session); + if (status) + return status; + status = srtp_dealloc(session); + if (status) + return status; + return err_status_ok; +} + +/* + * srtp_print_packet(...) is for debugging only + * it prints an RTP packet to the stdout + * + * note that this function is *not* threadsafe + */ + +#include <stdio.h> + +#define MTU 2048 + +char packet_string[MTU]; + +char * +srtp_packet_to_string(srtp_hdr_t *hdr, int pkt_octet_len) { + int octets_in_rtp_header = 12; + uint8_t *data = ((uint8_t *)hdr)+octets_in_rtp_header; + int hex_len = pkt_octet_len-octets_in_rtp_header; + + /* sanity checking */ + if ((hdr == NULL) || (pkt_octet_len > MTU)) + return NULL; + + /* write packet into string */ + sprintf(packet_string, + "(s)rtp packet: {\n" + " version:\t%d\n" + " p:\t\t%d\n" + " x:\t\t%d\n" + " cc:\t\t%d\n" + " m:\t\t%d\n" + " pt:\t\t%x\n" + " seq:\t\t%x\n" + " ts:\t\t%x\n" + " ssrc:\t%x\n" + " data:\t%s\n" + "} (%d octets in total)\n", + hdr->version, + hdr->p, + hdr->x, + hdr->cc, + hdr->m, + hdr->pt, + hdr->seq, + hdr->ts, + hdr->ssrc, + octet_string_hex_string(data, hex_len), + pkt_octet_len); + + return packet_string; +} + +/* + * mips_estimate() is a simple function to estimate the number of + * instructions per second that the host can perform. note that this + * function can be grossly wrong; you may want to have a manual sanity + * check of its output! + * + * the 'ignore' pointer is there to convince the compiler to not just + * optimize away the function + */ + +double +mips_estimate(int num_trials, int *ignore) { + clock_t t; + int i, sum; + + sum = 0; + t = clock(); + for (i=0; i<num_trials; i++) + sum += i; + t = clock() - t; + +/* printf("%d\n", sum); */ + *ignore = sum; + + return (double) num_trials * CLOCKS_PER_SEC / t; +} + + +/* + * srtp_validate() verifies the correctness of libsrtp by comparing + * some computed packets against some pre-computed reference values. + * These packets were made with the default SRTP policy. + */ + + +err_status_t +srtp_validate() { + unsigned char test_key[30] = { + 0xe1, 0xf9, 0x7a, 0x0d, 0x3e, 0x01, 0x8b, 0xe0, + 0xd6, 0x4f, 0xa3, 0x2c, 0x06, 0xde, 0x41, 0x39, + 0x0e, 0xc6, 0x75, 0xad, 0x49, 0x8a, 0xfe, 0xeb, + 0xb6, 0x96, 0x0b, 0x3a, 0xab, 0xe6 + }; + uint8_t srtp_plaintext_ref[28] = { + 0x80, 0x0f, 0x12, 0x34, 0xde, 0xca, 0xfb, 0xad, + 0xca, 0xfe, 0xba, 0xbe, 0xab, 0xab, 0xab, 0xab, + 0xab, 0xab, 0xab, 0xab, 0xab, 0xab, 0xab, 0xab, + 0xab, 0xab, 0xab, 0xab + }; + uint8_t srtp_plaintext[38] = { + 0x80, 0x0f, 0x12, 0x34, 0xde, 0xca, 0xfb, 0xad, + 0xca, 0xfe, 0xba, 0xbe, 0xab, 0xab, 0xab, 0xab, + 0xab, 0xab, 0xab, 0xab, 0xab, 0xab, 0xab, 0xab, + 0xab, 0xab, 0xab, 0xab, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 + }; + uint8_t srtp_ciphertext[38] = { + 0x80, 0x0f, 0x12, 0x34, 0xde, 0xca, 0xfb, 0xad, + 0xca, 0xfe, 0xba, 0xbe, 0x4e, 0x55, 0xdc, 0x4c, + 0xe7, 0x99, 0x78, 0xd8, 0x8c, 0xa4, 0xd2, 0x15, + 0x94, 0x9d, 0x24, 0x02, 0xb7, 0x8d, 0x6a, 0xcc, + 0x99, 0xea, 0x17, 0x9b, 0x8d, 0xbb + }; + srtp_t srtp_snd, srtp_recv; + err_status_t status; + int len; + srtp_policy_t policy; + + /* + * create a session with a single stream using the default srtp + * policy and with the SSRC value 0xcafebabe + */ + crypto_policy_set_rtp_default(&policy.rtp); + crypto_policy_set_rtcp_default(&policy.rtcp); + policy.ssrc.type = ssrc_specific; + policy.ssrc.value = 0xcafebabe; + policy.key = test_key; + policy.next = NULL; + + status = srtp_create(&srtp_snd, &policy); + if (status) + return status; + + /* + * protect plaintext, then compare with ciphertext + */ + len = 28; + status = srtp_protect(srtp_snd, srtp_plaintext, &len); + if (status || (len != 38)) + return err_status_fail; + + debug_print(mod_driver, "ciphertext:\n %s", + octet_string_hex_string(srtp_plaintext, len)); + debug_print(mod_driver, "ciphertext reference:\n %s", + octet_string_hex_string(srtp_ciphertext, len)); + + if (octet_string_is_eq(srtp_plaintext, srtp_ciphertext, len)) + return err_status_fail; + + /* + * create a receiver session context comparable to the one created + * above - we need to do this so that the replay checking doesn't + * complain + */ + status = srtp_create(&srtp_recv, &policy); + if (status) + return status; + + /* + * unprotect ciphertext, then compare with plaintext + */ + status = srtp_unprotect(srtp_recv, srtp_ciphertext, &len); + if (status || (len != 28)) + return status; + + if (octet_string_is_eq(srtp_ciphertext, srtp_plaintext_ref, len)) + return err_status_fail; + + return err_status_ok; +} + + +err_status_t +srtp_create_big_policy(srtp_policy_t **list) { + extern const srtp_policy_t *policy_array[]; + srtp_policy_t *p, *tmp; + int i = 0; + uint32_t ssrc = 0; + + /* sanity checking */ + if ((list == NULL) || (policy_array[0] == NULL)) + return err_status_bad_param; + + /* + * loop over policy list, mallocing a new list and copying values + * into it (and incrementing the SSRC value as we go along) + */ + tmp = NULL; + while (policy_array[i] != NULL) { + p = malloc(sizeof(srtp_policy_t)); + if (p == NULL) + return err_status_bad_param; + memcpy(p, policy_array[i], sizeof(srtp_policy_t)); + p->ssrc.type = ssrc_specific; + p->ssrc.value = ssrc++; + p->next = tmp; + tmp = p; + i++; + } + *list = p; + + return err_status_ok; +} + +err_status_t +srtp_test_remove_stream() { + err_status_t status; + srtp_policy_t *policy_list; + srtp_t session; + srtp_stream_t stream; + /* + * srtp_get_stream() is a libSRTP internal function that we declare + * here so that we can use it to verify the correct operation of the + * library + */ + extern srtp_stream_t srtp_get_stream(srtp_t srtp, uint32_t ssrc); + + + status = srtp_create_big_policy(&policy_list); + if (status) + return status; + + status = srtp_create(&session, policy_list); + if (status) + return status; + + /* + * check for false positives by trying to remove a stream that's not + * in the session + */ + status = srtp_remove_stream(session, htonl(0xaaaaaaaa)); + if (status != err_status_no_ctx) + return err_status_fail; + + /* + * check for false negatives by removing stream 0x1, then + * searching for streams 0x0 and 0x2 + */ + status = srtp_remove_stream(session, htonl(0x1)); + if (status != err_status_ok) + return err_status_fail; + stream = srtp_get_stream(session, htonl(0x0)); + if (stream == NULL) + return err_status_fail; + stream = srtp_get_stream(session, htonl(0x2)); + if (stream == NULL) + return err_status_fail; + + return err_status_ok; +} + +/* + * srtp policy definitions - these definitions are used above + */ + +unsigned char test_key[30] = { + 0xe1, 0xf9, 0x7a, 0x0d, 0x3e, 0x01, 0x8b, 0xe0, + 0xd6, 0x4f, 0xa3, 0x2c, 0x06, 0xde, 0x41, 0x39, + 0x0e, 0xc6, 0x75, 0xad, 0x49, 0x8a, 0xfe, 0xeb, + 0xb6, 0x96, 0x0b, 0x3a, 0xab, 0xe6 +}; + + +const srtp_policy_t default_policy = { + { ssrc_any_outbound, 0 }, /* SSRC */ + { /* SRTP policy */ + AES_128_ICM, /* cipher type */ + 30, /* cipher key length in octets */ + HMAC_SHA1, /* authentication func type */ + 16, /* auth key length in octets */ + 10, /* auth tag length in octets */ + sec_serv_conf_and_auth /* security services flag */ + }, + { /* SRTCP policy */ + AES_128_ICM, /* cipher type */ + 30, /* cipher key length in octets */ + HMAC_SHA1, /* authentication func type */ + 16, /* auth key length in octets */ + 10, /* auth tag length in octets */ + sec_serv_conf_and_auth /* security services flag */ + }, + test_key, + NULL +}; + +const srtp_policy_t aes_tmmh_policy = { + { ssrc_any_outbound, 0 }, /* SSRC */ + { + AES_128_ICM, /* cipher type */ + 30, /* cipher key length in octets */ + UST_TMMHv2, /* authentication func type */ + 94, /* auth key length in octets */ + 4, /* auth tag length in octets */ + sec_serv_conf_and_auth /* security services flag */ + }, + { + AES_128_ICM, /* cipher type */ + 30, /* cipher key length in octets */ + UST_TMMHv2, /* authentication func type */ + 94, /* auth key length in octets */ + 4, /* auth tag length in octets */ + sec_serv_conf_and_auth /* security services flag */ + }, + test_key, + NULL +}; + +const srtp_policy_t tmmh_only_policy = { + { ssrc_any_outbound, 0 }, /* SSRC */ + { + AES_128_ICM, /* cipher type */ + 30, /* cipher key length in octets */ + UST_TMMHv2, /* authentication func type */ + 94, /* auth key length in octets */ + 4, /* auth tag length in octets */ + sec_serv_auth /* security services flag */ + }, + { + AES_128_ICM, /* cipher type */ + 30, /* cipher key length in octets */ + UST_TMMHv2, /* authentication func type */ + 94, /* auth key length in octets */ + 4, /* auth tag length in octets */ + sec_serv_auth /* security services flag */ + }, + test_key, + NULL +}; + +const srtp_policy_t aes_only_policy = { + { ssrc_any_outbound, 0 }, /* SSRC */ + { + AES_128_ICM, /* cipher type */ + 30, /* cipher key length in octets */ + NULL_AUTH, /* authentication func type */ + 0, /* auth key length in octets */ + 0, /* auth tag length in octets */ + sec_serv_conf /* security services flag */ + }, + { + AES_128_ICM, /* cipher type */ + 30, /* cipher key length in octets */ + NULL_AUTH, /* authentication func type */ + 0, /* auth key length in octets */ + 0, /* auth tag length in octets */ + sec_serv_conf /* security services flag */ + }, + test_key, + NULL +}; + +const srtp_policy_t hmac_only_policy = { + { ssrc_any_outbound, 0 }, /* SSRC */ + { + NULL_CIPHER, /* cipher type */ + 0, /* cipher key length in octets */ + HMAC_SHA1, /* authentication func type */ + 20, /* auth key length in octets */ + 4, /* auth tag length in octets */ + sec_serv_auth /* security services flag */ + }, + { + NULL_CIPHER, /* cipher type */ + 0, /* cipher key length in octets */ + HMAC_SHA1, /* authentication func type */ + 20, /* auth key length in octets */ + 4, /* auth tag length in octets */ + sec_serv_auth /* security services flag */ + }, + test_key, + NULL +}; + +const srtp_policy_t null_policy = { + { ssrc_any_outbound, 0 }, /* SSRC */ + { + NULL_CIPHER, /* cipher type */ + 0, /* cipher key length in octets */ + NULL_AUTH, /* authentication func type */ + 0, /* auth key length in octets */ + 0, /* auth tag length in octets */ + sec_serv_none /* security services flag */ + }, + { + NULL_CIPHER, /* cipher type */ + 0, /* cipher key length in octets */ + NULL_AUTH, /* authentication func type */ + 0, /* auth key length in octets */ + 0, /* auth tag length in octets */ + sec_serv_none /* security services flag */ + }, + test_key, + NULL +}; + + +/* + * an array of pointers to the policies listed above + * + * This array is used to test various aspects of libSRTP for + * different cryptographic policies. The order of the elements + * matters - the timing test generates output that can be used + * in a plot (see the gnuplot script file 'timing'). If you + * add to this list, you should do it at the end. + */ + +#define USE_TMMH 0 + +const srtp_policy_t * +policy_array[] = { + &hmac_only_policy, +#if USE_TMMH + &tmmh_only_policy, +#endif + &aes_only_policy, +#if USE_TMMH + &aes_tmmh_policy, +#endif + &default_policy, + &null_policy, + NULL +}; + +const srtp_policy_t wildcard_policy = { + { ssrc_any_outbound, 0 }, /* SSRC */ + { /* SRTP policy */ + AES_128_ICM, /* cipher type */ + 30, /* cipher key length in octets */ + HMAC_SHA1, /* authentication func type */ + 16, /* auth key length in octets */ + 10, /* auth tag length in octets */ + sec_serv_conf_and_auth /* security services flag */ + }, + { /* SRTCP policy */ + AES_128_ICM, /* cipher type */ + 30, /* cipher key length in octets */ + HMAC_SHA1, /* authentication func type */ + 16, /* auth key length in octets */ + 10, /* auth tag length in octets */ + sec_serv_conf_and_auth /* security services flag */ + }, + test_key, + NULL +}; diff --git a/libs/srtp/timing b/libs/srtp/timing new file mode 100644 index 0000000000..66b00c452e --- /dev/null +++ b/libs/srtp/timing @@ -0,0 +1 @@ +# timing.plt # # gnuplot script file for plotting the output generated by srtp_driver -t # # David A. McGrew # Cisco Systems, Inc. # set xrange [0:2500] set term pict "Times-Roman" 9 # # plot authentication-only data # set title "Authentication Only" set ylabel "Megabits per second" set xlabel "Octets in packet" set yrange [0:2000] set output "plot-auth.pict" plot "timing.dat" index 0 title "HMAC SHA1" with lines, "timing.dat" index 1 title "TMMH/AES" with lines, "timing.dat" index 2 title "TMMH/SEAL" with lines # # plot encryption-only data # set title "Encryption Only" set ylabel "Megabits per second" set xlabel "Octets in packet" set output "plot-enc.pict" set yrange [0:1200] plot "timing.dat" index 3 title "SEAL" with lines, "timing.dat" index 4 title "AES ICM" with lines # # plot encryption and authentication data # set title "Encryption and Authentication" set ylabel "Megabits per second" set xlabel "Octets in packet" set yrange [0:1000] set output "plot-enc-auth.pict" plot "timing.dat" index 5 title "TMMH/SEAL" with lines, "timing.dat" index 6 title "TMMH/AES" with lines, "timing.dat" index 7 title "HMAC/AES" with lines \ No newline at end of file diff --git a/libs/srtp/undos.sh b/libs/srtp/undos.sh new file mode 100755 index 0000000000..db12064968 --- /dev/null +++ b/libs/srtp/undos.sh @@ -0,0 +1,10 @@ +#!/bin/sh +# +# usage: undos <file> +# +# strips CRs from a file - useful when moving DOS-created files +# onto UN*X machines + +cat $1 | tr -d "\r" > $1.tmp +mv $1.tmp $1 + diff --git a/libs/srtp/update.sh b/libs/srtp/update.sh new file mode 100755 index 0000000000..d116f82925 --- /dev/null +++ b/libs/srtp/update.sh @@ -0,0 +1,15 @@ +#!/bin/sh +# +# update.sh +# +# update copyright dates in files + +a=`find . -name "*.[ch]"` +for x in $a; do + sed 's/(c) 2001-2004/(c) 2001-2005/' $x > $x.tmp; + mv $x.tmp $x; +done + + + +