|
|
//************************************************************************
// Microsoft Corporation
// Copyright(c) Microsoft Corp., 1990-1992
//
//
// Revision history:
// 5/5/94 Created gurdeep
//
//************************************************************************
//#define COMP_12K
#include "wan.h"
#define __FILE_SIG__ COMPRESS_FILESIG
//#define DEBUG
CONST unsigned long lookup_array1[256] = { 0, 10276755, 20553510, 30830265, 41107020, 51383775, 61660530, 71937285, 82214040, 92490795, 102767550, 113044305, 123321060, 133597815, 143874570, 154151325, 164428080, 174704835, 184981590, 195258345, 205535100, 215811855, 226088610, 236365365, 246642120, 256918875, 267195630, 277472385, 287749140, 298025895, 308302650, 318579405, 328856160, 339132915, 349409670, 359686425, 369963180, 380239935, 390516690, 400793445, 411070200, 421346955, 431623710, 441900465, 452177220, 462453975, 472730730, 483007485, 493284240, 503560995, 513837750, 524114505, 534391260, 544668015, 554944770, 565221525, 575498280, 585775035, 596051790, 606328545, 616605300, 626882055, 637158810, 647435565, 657712320, 667989075, 678265830, 688542585, 698819340, 709096095, 719372850, 729649605, 739926360, 750203115, 760479870, 770756625, 781033380, 791310135, 801586890, 811863645, 822140400, 832417155, 842693910, 852970665, 863247420, 873524175, 883800930, 894077685, 904354440, 914631195, 924907950, 935184705, 945461460, 955738215, 966014970, 976291725, 986568480, 996845235, 1007121990, 1017398745, 1027675500, 1037952255, 1048229010, 1058505765, 1068782520, 1079059275, 1089336030, 1099612785, 1109889540, 1120166295, 1130443050, 1140719805, 1150996560, 1161273315, 1171550070, 1181826825, 1192103580, 1202380335, 1212657090, 1222933845, 1233210600, 1243487355, 1253764110, 1264040865, 1274317620, 1284594375, 1294871130, 1305147885, 1315424640, 1325701395, 1335978150, 1346254905, 1356531660, 1366808415, 1377085170, 1387361925, 1397638680, 1407915435, 1418192190, 1428468945, 1438745700, 1449022455, 1459299210, 1469575965, 1479852720, 1490129475, 1500406230, 1510682985, 1520959740, 1531236495, 1541513250, 1551790005, 1562066760, 1572343515, 1582620270, 1592897025, 1603173780, 1613450535, 1623727290, 1634004045, 1644280800, 1654557555, 1664834310, 1675111065, 1685387820, 1695664575, 1705941330, 1716218085, 1726494840, 1736771595, 1747048350, 1757325105, 1767601860, 1777878615, 1788155370, 1798432125, 1808708880, 1818985635, 1829262390, 1839539145, 1849815900, 1860092655, 1870369410, 1880646165, 1890922920, 1901199675, 1911476430, 1921753185, 1932029940, 1942306695, 1952583450, 1962860205, 1973136960, 1983413715, 1993690470, 2003967225, 2014243980, 2024520735, 2034797490, 2045074245, 2055351000, 2065627755, 2075904510, 2086181265, 2096458020, 2106734775, 2117011530, 2127288285, 2137565040, 2147841795, 2158118550, 2168395305, 2178672060, 2188948815, 2199225570, 2209502325, 2219779080, 2230055835, 2240332590, 2250609345, 2260886100, 2271162855, 2281439610, 2291716365, 2301993120, 2312269875, 2322546630, 2332823385, 2343100140, 2353376895, 2363653650, 2373930405, 2384207160, 2394483915, 2404760670, 2415037425, 2425314180, 2435590935, 2445867690, 2456144445, 2466421200, 2476697955, 2486974710, 2497251465, 2507528220, 2517804975, 2528081730, 2538358485, 2548635240, 2558911995, 2569188750, 2579465505, 2589742260, 2600019015, 2610295770, 2620572525 };
/*
for i = 0 to 255, lookup_array2[i] = lookup_array1[i] << 8; */ CONST unsigned long lookup_array2[256] = { 0, 2630849280, 966731264, 3597580544, 1933462528, 269344512, 2900193792, 1236075776, 3866925056, 2202807040, 538689024, 3169538304, 1505420288, 4136269568, 2472151552, 808033536, 3438882816, 1774764800, 110646784, 2741496064, 1077378048, 3708227328, 2044109312, 379991296, 3010840576, 1346722560, 3977571840, 2313453824, 649335808, 3280185088, 1616067072, 4246916352, 2582798336, 918680320, 3549529600, 1885411584, 221293568, 2852142848, 1188024832, 3818874112, 2154756096, 490638080, 3121487360, 1457369344, 4088218624, 2424100608, 759982592, 3390831872, 1726713856, 62595840, 2693445120, 1029327104, 3660176384, 1996058368, 331940352, 2962789632, 1298671616, 3929520896, 2265402880, 601284864, 3232134144, 1568016128, 4198865408, 2534747392, 870629376, 3501478656, 1837360640, 173242624, 2804091904, 1139973888, 3770823168, 2106705152, 442587136, 3073436416, 1409318400, 4040167680, 2376049664, 711931648, 3342780928, 1678662912, 14544896, 2645394176, 981276160, 3612125440, 1948007424, 283889408, 2914738688, 1250620672, 3881469952, 2217351936, 553233920, 3184083200, 1519965184, 4150814464, 2486696448, 822578432, 3453427712, 1789309696, 125191680, 2756040960, 1091922944, 3722772224, 2058654208, 394536192, 3025385472, 1361267456, 3992116736, 2327998720, 663880704, 3294729984, 1630611968, 4261461248, 2597343232, 933225216, 3564074496, 1899956480, 235838464, 2866687744, 1202569728, 3833419008, 2169300992, 505182976, 3136032256, 1471914240, 4102763520, 2438645504, 774527488, 3405376768, 1741258752, 77140736, 2707990016, 1043872000, 3674721280, 2010603264, 346485248, 2977334528, 1313216512, 3944065792, 2279947776, 615829760, 3246679040, 1582561024, 4213410304, 2549292288, 885174272, 3516023552, 1851905536, 187787520, 2818636800, 1154518784, 3785368064, 2121250048, 457132032, 3087981312, 1423863296, 4054712576, 2390594560, 726476544, 3357325824, 1693207808, 29089792, 2659939072, 995821056, 3626670336, 1962552320, 298434304, 2929283584, 1265165568, 3896014848, 2231896832, 567778816, 3198628096, 1534510080, 4165359360, 2501241344, 837123328, 3467972608, 1803854592, 139736576, 2770585856, 1106467840, 3737317120, 2073199104, 409081088, 3039930368, 1375812352, 4006661632, 2342543616, 678425600, 3309274880, 1645156864, 4276006144, 2611888128, 947770112, 3578619392, 1914501376, 250383360, 2881232640, 1217114624, 3847963904, 2183845888, 519727872, 3150577152, 1486459136, 4117308416, 2453190400, 789072384, 3419921664, 1755803648, 91685632, 2722534912, 1058416896, 3689266176, 2025148160, 361030144, 2991879424, 1327761408, 3958610688, 2294492672, 630374656, 3261223936, 1597105920, 4227955200, 2563837184, 899719168, 3530568448, 1866450432, 202332416, 2833181696, 1169063680, 3799912960, 2135794944, 471676928, 3102526208, 1438408192, 4069257472, 2405139456, 741021440, 3371870720, 1707752704, 43634688, 2674483968, 1010365952, 3641215232, 1977097216, 312979200, 2943828480, 1279710464, 3910559744, 2246441728, 582323712, 3213172992, 1549054976, 4179904256, 2515786240, 851668224 };
/*
for i = 0 to 255, lookup_array3[i] = lookup_array1[i] << 16; */ CONST unsigned long lookup_array3[256] = { 0, 3482517504, 2670067712, 1857617920, 1045168128, 232718336, 3715235840, 2902786048, 2090336256, 1277886464, 465436672, 3947954176, 3135504384, 2323054592, 1510604800, 698155008, 4180672512, 3368222720, 2555772928, 1743323136, 930873344, 118423552, 3600941056, 2788491264, 1976041472, 1163591680, 351141888, 3833659392, 3021209600, 2208759808, 1396310016, 583860224, 4066377728, 3253927936, 2441478144, 1629028352, 816578560, 4128768, 3486646272, 2674196480, 1861746688, 1049296896, 236847104, 3719364608, 2906914816, 2094465024, 1282015232, 469565440, 3952082944, 3139633152, 2327183360, 1514733568, 702283776, 4184801280, 3372351488, 2559901696, 1747451904, 935002112, 122552320, 3605069824, 2792620032, 1980170240, 1167720448, 355270656, 3837788160, 3025338368, 2212888576, 1400438784, 587988992, 4070506496, 3258056704, 2445606912, 1633157120, 820707328, 8257536, 3490775040, 2678325248, 1865875456, 1053425664, 240975872, 3723493376, 2911043584, 2098593792, 1286144000, 473694208, 3956211712, 3143761920, 2331312128, 1518862336, 706412544, 4188930048, 3376480256, 2564030464, 1751580672, 939130880, 126681088, 3609198592, 2796748800, 1984299008, 1171849216, 359399424, 3841916928, 3029467136, 2217017344, 1404567552, 592117760, 4074635264, 3262185472, 2449735680, 1637285888, 824836096, 12386304, 3494903808, 2682454016, 1870004224, 1057554432, 245104640, 3727622144, 2915172352, 2102722560, 1290272768, 477822976, 3960340480, 3147890688, 2335440896, 1522991104, 710541312, 4193058816, 3380609024, 2568159232, 1755709440, 943259648, 130809856, 3613327360, 2800877568, 1988427776, 1175977984, 363528192, 3846045696, 3033595904, 2221146112, 1408696320, 596246528, 4078764032, 3266314240, 2453864448, 1641414656, 828964864, 16515072, 3499032576, 2686582784, 1874132992, 1061683200, 249233408, 3731750912, 2919301120, 2106851328, 1294401536, 481951744, 3964469248, 3152019456, 2339569664, 1527119872, 714670080, 4197187584, 3384737792, 2572288000, 1759838208, 947388416, 134938624, 3617456128, 2805006336, 1992556544, 1180106752, 367656960, 3850174464, 3037724672, 2225274880, 1412825088, 600375296, 4082892800, 3270443008, 2457993216, 1645543424, 833093632, 20643840, 3503161344, 2690711552, 1878261760, 1065811968, 253362176, 3735879680, 2923429888, 2110980096, 1298530304, 486080512, 3968598016, 3156148224, 2343698432, 1531248640, 718798848, 4201316352, 3388866560, 2576416768, 1763966976, 951517184, 139067392, 3621584896, 2809135104, 1996685312, 1184235520, 371785728, 3854303232, 3041853440, 2229403648, 1416953856, 604504064, 4087021568, 3274571776, 2462121984, 1649672192, 837222400, 24772608, 3507290112, 2694840320, 1882390528, 1069940736, 257490944, 3740008448, 2927558656, 2115108864, 1302659072, 490209280, 3972726784, 3160276992, 2347827200, 1535377408, 722927616, 4205445120, 3392995328, 2580545536, 1768095744, 955645952, 143196160, 3625713664, 2813263872, 2000814080, 1188364288, 375914496, 3858432000, 3045982208, 2233532416, 1421082624, 608632832, 4091150336, 3278700544 };
/*
The key for the multiplicative hash function consists of 3 unsigned characters. They are composed (logically) by concatenating them i.e. the composed key = 2^16*c2 + 2^8*c2 + c3 and fits in 24 bits. The composed key is not actually computed here as we use the components to directly compute the hash function.
The multiplicative hash function consists of taking the higher order 12 bits (2^12 = 4096) of the lower order 24 bits of the product key * Multiplier where Multiplier = floor(A * pow(2.0, (double) w)); double A = 0.6125423371; (chosen according to Knuth) w = 24 (the key's width in bits) The algorithm for this is in Cormen/Leiserson/Rivest.
To do the multplication efficiently, the product c*Multiplier is precomputed and stored in lookup_array1 (for all 256 possible c's). lookup_array2 and lookup_array3 contain the same data as lookup_array1 but shifted left 8 and 16 bits respectively.
MultHash1 is the mult hashing function. MultHash0 contains an older (slower but less space-efficient) version of the same function. */
#define MULTHASH1(c1,c2,c3) \
((lookup_array1[c1]+ \ lookup_array2[c2]+ \ lookup_array3[c3] ) & 0x00fff000) >> 12
/*
USHORT xorlookup1 [256] = { 0x110, 0x120, 0x130, 0x140, 0x150, 0x160, 0x170, 0x180, // 0-7
0x190, 0x1a0, 0x1b0, 0x1c0, 0x1d0, 0x1e0, 0x1f0, 0x100, // 8-15
0x210, 0x220, 0x230, 0x240, 0x250, 0x260, 0x270, 0x280, // 16-23
0x290, 0x2a0, 0x2b0, 0x2c0, 0x2d0, 0x2e0, 0x2f0, 0x200, // 24-31
0x310, 0x320, 0x330, 0x340, 0x350, 0x360, 0x370, 0x380, // 32-39
0x390, 0x3a0, 0x3b0, 0x3c0, 0x3d0, 0x3e0, 0x3f0, 0x300, // 40-47
0x410, 0x420, 0x430, 0x440, 0x450, 0x460, 0x470, 0x480, // 48-55
0x490, 0x4a0, 0x4b0, 0x4c0, 0x4d0, 0x4e0, 0x4f0, 0x400, // 56-63
0x510, 0x520, 0x530, 0x540, 0x550, 0x560, 0x570, 0x580, // 64-71
0x590, 0x5a0, 0x5b0, 0x5c0, 0x5d0, 0x5e0, 0x5f0, 0x500, // 72-79
0x610, 0x620, 0x630, 0x640, 0x650, 0x660, 0x670, 0x680, // 80-87
0x690, 0x6a0, 0x6b0, 0x6c0, 0x6d0, 0x6e0, 0x6f0, 0x600, // 88-95
0x710, 0x720, 0x730, 0x740, 0x750, 0x760, 0x770, 0x780, // 96-103
0x790, 0x7a0, 0x7b0, 0x7c0, 0x7d0, 0x7e0, 0x7f0, 0x700, // 104-111
0x810, 0x820, 0x830, 0x840, 0x850, 0x860, 0x870, 0x880, // 112-119
0x890, 0x8a0, 0x8b0, 0x8c0, 0x8d0, 0x8e0, 0x8f0, 0x800, // 120-127
0x910, 0x920, 0x930, 0x940, 0x950, 0x960, 0x970, 0x980, // 128-135
0x990, 0x9a0, 0x9b0, 0x9c0, 0x9d0, 0x9e0, 0x9f0, 0x900, // 136-143
0xa10, 0xa20, 0xa30, 0xa40, 0xa50, 0xa60, 0xa70, 0xa80, // 144-151
0xa90, 0xaa0, 0xab0, 0xac0, 0xad0, 0xae0, 0xaf0, 0xa00, // 152-159
0xb10, 0xb20, 0xb30, 0xb40, 0xb50, 0xb60, 0xb70, 0xb80, // 160-167
0xb90, 0xba0, 0xbb0, 0xbc0, 0xbd0, 0xbe0, 0xbf0, 0xb00, // 168-175
0xc10, 0xc20, 0xc30, 0xc40, 0xc50, 0xc60, 0xc70, 0xc80, // 176-183
0xc90, 0xca0, 0xcb0, 0xcc0, 0xcd0, 0xce0, 0xcf0, 0xc00, // 184-191
0xd10, 0xd20, 0xd30, 0xd40, 0xd50, 0xd60, 0xd70, 0xd80, // 192-199
0xd90, 0xda0, 0xdb0, 0xdc0, 0xdd0, 0xde0, 0xdf0, 0xd00, // 200-207
0xe10, 0xe20, 0xe30, 0xe40, 0xe50, 0xe60, 0xe70, 0xe80, // 208-215
0xe90, 0xea0, 0xeb0, 0xec0, 0xed0, 0xee0, 0xef0, 0xe00, // 216-223
0xf10, 0xf20, 0xf30, 0xf40, 0xf50, 0xf60, 0xf70, 0xf80, // 224-231
0xf90, 0xfa0, 0xfb0, 0xfc0, 0xfd0, 0xfe0, 0xff0, 0xf00, // 232-239
0x010, 0x020, 0x030, 0x040, 0x050, 0x060, 0x070, 0x080, // 240-247
0x090, 0x0a0, 0x0b0, 0x0c0, 0x0d0, 0x0e0, 0x0f0, 0x000 }; // 248-255
USHORT xorlookup2 [256] = { 0x101, 0x201, 0x301, 0x401, 0x501, 0x601, 0x701, 0x801, // 0-7
0x901, 0xa01, 0xb01, 0xc01, 0xd01, 0xe01, 0xf01, 0x001, // 8-15
0x102, 0x202, 0x302, 0x402, 0x502, 0x602, 0x702, 0x802, // 16-23
0x902, 0xa02, 0xb02, 0xc02, 0xd02, 0xe02, 0xf02, 0x002, // 24-31
0x103, 0x203, 0x303, 0x403, 0x503, 0x603, 0x703, 0x803, // 32-39
0x903, 0xa03, 0xb03, 0xc03, 0xd03, 0xe03, 0xf03, 0x003, // 40-47
0x104, 0x204, 0x304, 0x404, 0x504, 0x604, 0x704, 0x804, // 48-55
0x904, 0xa04, 0xb04, 0xc04, 0xd04, 0xe04, 0xf04, 0x004, // 56-63
0x105, 0x205, 0x305, 0x405, 0x505, 0x605, 0x705, 0x805, // 64-71
0x905, 0xa05, 0xb05, 0xc05, 0xd05, 0xe05, 0xf05, 0x005, // 72-79
0x106, 0x206, 0x306, 0x406, 0x506, 0x606, 0x706, 0x806, // 80-87
0x906, 0xa06, 0xb06, 0xc06, 0xd06, 0xe06, 0xf06, 0x006, // 88-95
0x107, 0x207, 0x307, 0x407, 0x507, 0x607, 0x707, 0x807, // 96-103
0x907, 0xa07, 0xb07, 0xc07, 0xd07, 0xe07, 0xf07, 0x007, // 104-111
0x108, 0x208, 0x308, 0x408, 0x508, 0x608, 0x708, 0x808, // 112-119
0x908, 0xa08, 0xb08, 0xc08, 0xd08, 0xe08, 0xf08, 0x008, // 120-127
0x109, 0x209, 0x309, 0x409, 0x509, 0x609, 0x709, 0x809, // 128-135
0x909, 0xa09, 0xb09, 0xc09, 0xd09, 0xe09, 0xf09, 0x009, // 136-143
0x10a, 0x20a, 0x30a, 0x40a, 0x50a, 0x60a, 0x70a, 0x80a, // 144-151
0x90a, 0xa0a, 0xb0a, 0xc0a, 0xd0a, 0xe0a, 0xf0a, 0x00a, // 152-159
0x10b, 0x20b, 0x30b, 0x40b, 0x50b, 0x60b, 0x70b, 0x80b, // 160-167
0x90b, 0xa0b, 0xb0b, 0xc0b, 0xd0b, 0xe0b, 0xf0b, 0x00b, // 168-175
0x10c, 0x20c, 0x30c, 0x40c, 0x50c, 0x60c, 0x70c, 0x80c, // 176-183
0x90c, 0xa0c, 0xb0c, 0xc0c, 0xd0c, 0xe0c, 0xf0c, 0x00c, // 184-191
0x10d, 0x20d, 0x30d, 0x40d, 0x50d, 0x60d, 0x70d, 0x80d, // 192-199
0x90d, 0xa0d, 0xb0d, 0xc0d, 0xd0d, 0xe0d, 0xf0d, 0x00d, // 200-207
0x10e, 0x20e, 0x30e, 0x40e, 0x50e, 0x60e, 0x70e, 0x80e, // 208-215
0x90e, 0xa0e, 0xb0e, 0xc0e, 0xd0e, 0xe0e, 0xf0e, 0x00e, // 216-223
0x10f, 0x20f, 0x30f, 0x40f, 0x50f, 0x60f, 0x70f, 0x80f, // 224-231
0x90f, 0xa0f, 0xb0f, 0xc0f, 0xd0f, 0xe0f, 0xf0f, 0x00f, // 232-239
0x000, 0x200, 0x300, 0x400, 0x500, 0x600, 0x700, 0x800, // 240-247
0x900, 0xa00, 0xb00, 0xc00, 0xd00, 0xe00, 0xf00, 0x100 }; // 248-255
*/
/* Bitptrs point to the current byte. The current bit (i.e. next bit to be
* stored) is masked off by the bit entry. When this reaches zero, it is * reset to 0x80 and the next byte is set up. The bytes are filled MSBit * first. */
/* Starts and sets the first byte to zero for the bitptr. */ #define bitptr_init(s) pbyte = s; byte=0; bit = 16;
/* Sets up the byte part of the bitptr so that it is pointing to the byte after
* the byte which had the last bit put into it. */ #define bitptr_end() if (bit != 16) *pbyte++=(UCHAR)(byte >> 8);
/* Goes to the next bit, and byte if necessary. */ #define bitptr_next() \
if (bit < 10) { \ *pbyte++=(UCHAR)(byte >> 8); \ byte <<= 8; \ bit = 16; \ } else \ bit-- ;
/*
#define bitptr_next() \
bit--; \ if (bit < 9) { \ *pbyte++=(UCHAR)(byte >> 8); \ byte <<= 8; \ bit = 16; \ } */
/* Advances to the next bit, and byte if necessary, readjusting the bit. */ #define bitptr_advance() \
if (bit < 9) { \ *pbyte++=(UCHAR)(byte >> 8); \ bit+=8; \ byte <<= 8; \ }
/* BIT I/O FUNCTIONS *********************************************************/
/* These routines output most-significant-bit-first and the input will return
* them MSB first, too. */
/* Outputs a one bit in the bit stream. */ #define out_bit_1() bit--; byte |= (1 << bit); bitptr_advance();
#define out_bit_0() bitptr_next();
/* TestBit; output 1 if that bit is set */ //#define tb(b,w,n) if ((w) & (n)) *pbyte |= bit; bitptr_next(b);
#define out_bits_2(w) bit-=2; byte|=(w << bit); bitptr_advance();
#define out_bits_3(w) bit-=3; byte|=(w << bit); bitptr_advance();
#define out_bits_4(w) bit-=4; byte|=(w << bit); bitptr_advance();
#define out_bits_5(w) bit-=5; byte|=(w << bit); bitptr_advance();
#define out_bits_6(w) bit-=6; byte|=(w << bit); bitptr_advance();
#define out_bits_7(w) bit-=7; byte|=(w << bit); bitptr_advance();
// #define out_bits_8(w) bit-=8; byte|=(w << bit); bit+=8; *pbyte++=(UCHAR)(byte >> 8); byte <<= 8;
#define out_bits_8(w) byte|=(w << (bit-8)); *pbyte++=(UCHAR)(byte >> 8); byte <<= 8;
/*
#define out_bits_9(w) \
if (bit > 9) { \ bit-=9; byte|=(w << bit); \ *pbyte++=(UCHAR)(byte >> 8);\ bit+=8; \ byte <<= 8; \ } else { \ bit=16; byte |= w; \ *pbyte++=(UCHAR)(byte >> 8); *pbyte++=(UCHAR)(byte); byte=0; \ } */
#define out_bits_9(w) \
if (bit > 9) { \ byte|=(w << (bit-9)); \ *pbyte++=(UCHAR)(byte >> 8);\ bit--; \ byte <<= 8; \ } else { \ bit=16; byte |= w; \ *pbyte++=(UCHAR)(byte >> 8); *pbyte++=(UCHAR)(byte); byte=0; \ }
#define out_bits_10(w) \
if (bit > 10) { \ bit-=10; byte |= (w << bit); *pbyte++ = (UCHAR)(byte >> 8); bit+=8; byte <<=8; \ } else { \ out_bits_2((w >> 8)); \ out_bits_8((w & 0xFF)); \ }
//
// Weird effect - if out_bits_9 used instead of out_bits_8,
// it's faster! if (bit == 11) is faster than if (bit != 11).
//
#define out_bits_11(w) \
if (bit > 11) { \ bit-=11; byte |= (w << bit); *pbyte++ = (UCHAR)(byte >> 8); bit+=8; byte <<=8; \ } else { \ if (bit == 11) { \ bit=16; byte |= w; \ *pbyte++=(UCHAR)(byte >> 8); *pbyte++=(UCHAR)(byte); byte=0; \ } else { \ bit=11-bit; \ byte|=(w >> bit); \ *pbyte++=(UCHAR)(byte >> 8); *pbyte++=(UCHAR)(byte); \ bit=16-bit; \ byte=(w << bit); \ } \ }
#define out_bits_12(w) \
if (bit > 12) { \ bit-=12; byte |= (w << bit); *pbyte++ = (UCHAR)(byte >> 8); bit+=8; byte <<=8; \ } else { \ out_bits_4((w >> 8)); \ out_bits_8((w & 0xFF)); \ } #define out_bits_13(w) \
if (bit > 13) { \ bit-=13; byte |= (w << bit); *pbyte++ = (UCHAR)(byte >> 8); bit+=8; byte <<=8; \ } else { \ out_bits_5((w >> 8)); \ out_bits_8((w & 0xFF)); \ }
#define out_bits_14(w) \
if (bit > 14) { \ bit-=14; byte |= (w << bit); *pbyte++ = (UCHAR)(byte >> 8); bit+=8; byte <<=8; \ } else { \ out_bits_6((w >> 8)); \ out_bits_8((w & 0xFF)); \ }
#define out_reserve_4() \
bit-=4; bitptr_advance();
/* Starts the given bit pointer */ #define inbit_start(s) pbyte = s; bit = 16; byte=(*pbyte << 8) + *(pbyte+1); pbyte++;
#define inbit_end() if (bit != 16) pbyte++;
#define in_bit_next() if (bit < 9) { \
bit=16; \ byte <<=8; \ byte |= *(++pbyte); \ }
#define in_bit_advance() if (bit < 9) { \
bit+=8; \ byte <<=8; \ byte |= *(++pbyte); \ }
/* Returns non-zero in bitset if the next bit in the stream is a 1. */ #define in_bit() bit--; bitset = (byte >> bit) & 1; in_bit_next()
#define in_bits_2(w) bit-=2; w = (byte >> bit) & 0x03;\
in_bit_advance();
#define in_bits_3(w) bit-=3; w = (byte >> bit) & 0x07;\
in_bit_advance();
#define in_bits_4(w) bit-=4; w = (byte >> bit) & 0x0F;\
in_bit_advance();
#define in_bits_5(w) bit-=5; w = (byte >> bit) & 0x1F;\
in_bit_advance();
#define in_bits_6(w) bit-=6; w = (byte >> bit) & 0x3F;\
in_bit_advance();
#define in_bits_7(w) bit-=7; w = (byte >> bit) & 0x7F;\
in_bit_advance();
#define in_bits_8(w) bit-=8; w = (byte >> bit) & 0xFF;\
bit+=8; byte <<=8; byte |= *(++pbyte);
#define in_bits_9(w) bit-=9; w = (byte >> bit) & 0x1FF; \
bit+=8; byte <<=8; byte |= *(++pbyte); \ in_bit_advance();
#define in_bits_10(w) if (bit > 10) { \
bit-=10; w = (byte >> bit) & 0x3FF; \ bit+=8; byte <<=8; byte |= *(++pbyte); \ } else { \ in_bits_2(bitset); \ in_bits_8(w); \ w= w + (bitset << 8); \ }
#define in_bits_11(w) if (bit > 11) { \
bit-=11; w = (byte >> bit) & 0x7FF; \ bit+=8; byte <<=8; byte |= *(++pbyte); \ } else { \ in_bits_3(bitset); \ in_bits_8(w); \ w= w + (bitset << 8); \ }
#define in_bits_12(w) if (bit > 12) { \
bit-=12; w = (byte >> bit) & 0xFFF; \ bit+=8; byte <<=8; byte |= *(++pbyte); \ } else { \ in_bits_4(bitset); \ in_bits_8(w); \ w= w + (bitset << 8); \ }
#define in_bits_13(w)\
if (bit > 13) { \ bit-=13; w = (byte >> bit) & 0x1FFF; \ bit+=8; byte <<=8; byte |= *(++pbyte); \ } else { \ in_bits_5(bitset); \ in_bits_8(w); \ w=w + (bitset << 8); \ }
#define in_bits_14(w)\
if (bit > 14) { \ bit-=14; w = (byte >> bit) & 0x3FFF; \ bit+=8; byte <<=8; byte |= *(++pbyte); \ } else { \ in_bits_6(bitset); \ in_bits_8(w); \ w=w + (bitset << 8); \ }
UCHAR SHApad1[40] = {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, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; UCHAR SHApad2[40] = {0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2};
PUCHAR ClntSSrvR = "On the client side, this is the send key; on the server side, it is the receive key."; PUCHAR ClntRSrvS = "On the client side, this is the receive key; on the server side, it is the send key."; #define ECP_STRING_LEN strlen(ClntSSrvR)
#ifdef DEBUG
char ChPrint(UCHAR b) { if (isprint(b)) return (char)b; else return '.'; } #endif
//* compress()
//
// Function: Main compression function.
//
// Parameters:
// IN CurrentBuffer -> points to NDIS_WAN_PACKET with data to compress
// OUT CompOutBuffer -> points to NDIS_WAN_PACKET to compress data to
// IN CurrentLength -> points to Length of data to compress
// IN context -> connection compress context
//
// Returns: Nothing
//
// WARNING: CODE IS HIGHLY OPTIMIZED FOR TIME.
//
//
UCHAR compress (UCHAR *CurrentBuffer, UCHAR *CompOutBuffer, ULONG *CurrentLength, SendContext *context) { int copylen ; int bit; int byte; int backptr ; int cbMatch; int hashvalue ; int lookup1 ; UCHAR *matchptr ; UCHAR *pbyte; UCHAR *historyptr ; UCHAR *currentptr ; UCHAR *endptr ; UCHAR hashchar1; UCHAR hashchar2; UCHAR hashchar3; int literal ; UCHAR status=0; // return flags
PUCHAR currentbuf ;
// Will this packet fit at the end of the history buffer?
//
if (((context->CurrentIndex + *CurrentLength) >= (HISTORY_MAX - 1 )) || (context->CurrentIndex == 0)) { context->CurrentIndex = 0; // Index into the history
status |= PACKET_AT_FRONT; }
//
// we no longer need to save the non compressed data - tonybe 01-12-95
//
// RtlMoveMemory(context->CompressBuffer, CurrentBuffer, *CurrentLength) ;
// Start out the bit pointing output
//
bitptr_init(CompOutBuffer);
//
// We are now compressing into an output buffer - tonybe 01-12-95
//
// bitptr_init(CurrentBuffer);
historyptr = context->History + context->CurrentIndex ;
currentptr = CurrentBuffer;
//
// we are now compressing from the currentbuffer - tonybe 01-12-95
//
// currentptr = context->CompressBuffer ;
endptr = currentptr + *CurrentLength - 1;
while (currentptr < (endptr-2)) {
*historyptr++ = hashchar1 = *currentptr++ ; hashchar2 = *currentptr ; hashchar3 = *(currentptr+1) ;
// "fast" hash function
// hashvalue = (int)hashchar1 ^ xorlookup1[hashchar2] ^ xorlookup2[hashchar3];
hashvalue = MULTHASH1(hashchar1, hashchar2, hashchar3) ;
matchptr = context->History + context->HashTable[hashvalue] ;
if (matchptr != (historyptr - 1)) context->HashTable[hashvalue] = (USHORT)(historyptr - context->History) ;
if (context->ValidHistory < historyptr) context->ValidHistory = historyptr ;
if (matchptr != context->History && *(matchptr-1) == hashchar1 && *matchptr == hashchar2 && *(matchptr+1) == hashchar3 && matchptr != (historyptr - 1) && matchptr != historyptr && (matchptr+1) <= context->ValidHistory) {
backptr = ((int)(historyptr - matchptr)) & (HISTORY_SIZE - 1) ;
*historyptr++ = hashchar2 ; // copy the other 2 chars
*historyptr++ = hashchar3 ; // copy the other 2 chars
currentptr +=2 ; cbMatch = 3 ; // length of match
matchptr +=2 ; // we have already matched 3
while ((*matchptr == *currentptr) && (currentptr < endptr) && (matchptr <= context->ValidHistory)) { matchptr++ ; *historyptr++ = *currentptr++ ; cbMatch++ ; }
// First output the backpointer
//
if (backptr >= 320) { backptr -= 320 ; out_bits_8((0xc000 + backptr) >> 8) ; // 110 + 13 bits
out_bits_8((backptr)) ; } else if (backptr < 64) { // 1111 + 6 bits
backptr += 0x3c0 ; out_bits_10(backptr); } else { backptr += (0xE00 - 64); // 1110 + 8 bits
out_bits_12(backptr); }
// output the length of the match encoding
//
switch (cbMatch) { case 3: out_bit_0(); // length of 3 - most common
break; case 4: out_bits_4(8); break; case 5: out_bits_4(9); break; case 6: out_bits_4(10); break; case 7: out_bits_4(11); break; case 8: out_bits_6(48); break; case 9: out_bits_6(49); break; case 10: out_bits_6(50); break; case 11: out_bits_6(51); break; case 12: out_bits_6(52); break; case 13: out_bits_6(53); break; case 14: out_bits_6(54); break; case 15: out_bits_6(55); break; case 16: out_bits_8(0xe0); break; case 17: out_bits_8(0xe1); break; case 18: out_bits_8(0xe2); break; case 19: out_bits_8(0xe3); break; case 20: out_bits_8(0xe4); break; case 21: out_bits_8(0xe5); break; case 22: out_bits_8(0xe6); break; case 23: out_bits_8(0xe7); break; case 24: out_bits_8(0xe8); break; case 25: out_bits_8(0xe9); break; case 26: out_bits_8(0xea); break; case 27: out_bits_8(0xeb); break; case 28: out_bits_8(0xec); break; case 29: out_bits_8(0xed); break; case 30: out_bits_8(0xee); break; case 31: out_bits_8(0xef); break; default: if (cbMatch < 64) { out_bits_4(0xF) ; cbMatch -= 32 ; out_bits_6(cbMatch) ; } else if (cbMatch < 128) { out_bits_5(0x1F) ; cbMatch -= 64 ; out_bits_7(cbMatch) ; } else if (cbMatch < 256) { out_bits_6(0x3F) ; cbMatch -= 128 ; out_bits_8(cbMatch) ; } else if (cbMatch < 512) { out_bits_7(0x7F) ; cbMatch -= 256 ; out_bits_9(cbMatch) ; } else if (cbMatch < 1024) { out_bits_8(0xFF) ; cbMatch -= 512 ; out_bits_10(cbMatch) ; } else if (cbMatch < 2048) { out_bits_9(0x1FF) ; cbMatch -= 1024 ; out_bits_11(cbMatch) ; } else if (cbMatch < 4096) { out_bits_10(0x3FF) ; cbMatch -= 2048 ; out_bits_12(cbMatch) ; } else if (cbMatch < 8192) { out_bits_11(0x7FF) ; cbMatch -= 4096 ; out_bits_13(cbMatch) ; } else { // 8192 and greater
out_bits_12(0xFFF) ; cbMatch -= 8192 ; out_bits_14(cbMatch) ; } break ; }
} else { // encode a literal
// temp=literallookup[context->History[i-1]] ;
literal= hashchar1 ;
if (literal & 0x80) { literal += 0x80; out_bits_9(literal) ; } else { out_bits_8(literal) ; }
}
} // while
// get any remaining chars as literals
while (currentptr <= endptr) {
// temp=literallookup[context->History[i-1]] ;
literal=*currentptr ;
if (literal & 0x80) { literal += 0x80; out_bits_9(literal) ; } else { out_bits_8(literal) ; }
*historyptr++ = *currentptr++ ; }
bitptr_end() ;
// Check if we had expansion instead of compression
//
if ((ULONG)(pbyte - CompOutBuffer) > *CurrentLength) { // expansion.
//
// We don't need to do this copy since we can just signal the outside world
// that compression did not take place and the valid data is still in the
// current buffer
//
// RtlMoveMemory(CompOutBuffer, CurrentBuffer, *CurrentLength) ;
memset (context->History, 0, sizeof(context->History)) ; memset (context->HashTable, 0, sizeof(context->HashTable)) ; #ifdef COMP_12K
status = 0 ; #else
status = PACKET_FLUSHED; #endif
context->CurrentIndex = HISTORY_SIZE+1 ; // this forces a start over next time
} else { // compression successful
*CurrentLength = (ULONG)(pbyte - CompOutBuffer);
//
// the compressed data is now in CompOutBuffer - tonybe 01-12-95
//
// *CurrentLength = pbyte - CurrentBuffer ;
status |= PACKET_COMPRESSED ; context->CurrentIndex = (int)(historyptr - context->History) ; }
return(status); }
//* getcontextsizes()
//
// Function: Returns size of send and receive context blocks
//
// Parameters: OUT send -> sizeof(SendContext)
// OUT recv -> sizeof(RecvContext)
//
// Returns: Nothing
//
//*
void getcontextsizes (long *send, long *recv) { *send = sizeof(SendContext) ; *recv = sizeof(RecvContext) ; }
//* initsendcontext()
//
// Function: Initialize SendContext block
//
// Parameters: IN context -> connection compress context
//
// Returns: Nothing
//
//*
void initsendcontext (SendContext *context) { context->CurrentIndex = 0; // Index into the history
context->ValidHistory = 0 ; // reset valid history
memset (context->HashTable, 0, sizeof(context->HashTable)) ; memset (context->History, 0, sizeof(context->HashTable)) ; }
//* initrecvcontext()
//
// Function: Initialize RecvContext block
//
// Parameters: IN context -> connection decompress context
//
// Returns: Nothing
//
//*
void initrecvcontext (RecvContext *context) { context->CurrentPtr = context->History ;
#if DBG
context->DebugFence = DEBUG_FENCE_VALUE; #endif
memset (context->History, 0, sizeof(context->History)) ; }
//* decompress()
//
// Function: de-compression function.
//
// Parameters: IN inbuf -> points to data to be uncompressed
// IN inlen -> length of data
// IN start -> flag indicating whether to start with a clean history buffer
// OUT output-> decompressed data
// OUT outlen-> lenght of decompressed data
// IN context -> connection decompress context
//
// Returns: TRUE if decompress was successful
// FALSE if it wasnt
//
// WARNING: CODE IS HIGHLY OPTIMIZED FOR TIME.
//
//*
int decompress( UCHAR *inbuf, int inlen, int start, UCHAR **output, int *outlen, RecvContext *context) { UCHAR *inend; // When we know we're done decompressing
UCHAR *outstart; // Remember where in dbuf we started
UCHAR *current;
int backptr; // Back pointer for copy items
int length; // Where to copy from in dbuf
UCHAR *s1, *s2;
int bitset; int bit; int byte; UCHAR *pbyte; UCHAR *historyend = context->History + HISTORY_SIZE ;
inend = inbuf + inlen ;
//
// Start out looking at the first bit
//
inbit_start(inbuf);
if (start) // start over clean?
context->CurrentPtr = current = context->History ; else current = context->CurrentPtr ;
//
// Save our starting position
//
outstart = current;
//
// Decompress until we run out of input
//
while (pbyte < inend) {
//
// Jump on what to do with these three bits.
//
in_bits_3(length);
switch (length) {
case 0: in_bits_5(length) ; goto LITERAL ;
case 1: in_bits_5(length) ; length += 32 ; goto LITERAL ;
case 2: in_bits_5(length) ; length += 64 ; goto LITERAL ;
case 3: in_bits_5(length) ; length += 96 ; goto LITERAL ;
case 4: in_bits_6(length) ; length +=128 ; goto LITERAL ;
case 5: in_bits_6(length) ; length +=192 ; goto LITERAL ;
case 6: in_bits_13 (backptr) ; // 110 - 14 bit offset
backptr+=320 ; break ;
case 7: in_bit() ; if (bitset) { in_bits_6(backptr) ; } else { in_bits_8(backptr) ; backptr+=64 ; } break ; }
//
// If we reach here, it's a copy item
//
//
// Now get the length
//
in_bit() ; // 1st length bit
if (!bitset) { length = 3 ; goto DONE ; }
in_bit() ; // 2nd length bit
if (!bitset) { in_bits_2 (length) ; length += 4 ; goto DONE ; }
in_bit() ; // 3rd length bit
if (!bitset) { in_bits_3 (length) ; length += 8 ; goto DONE ; }
in_bit() ; // 4th length bit
if (!bitset) { in_bits_4 (length) ; length += 16 ; goto DONE ; }
in_bit() ; // 5th length bit
if (!bitset) { in_bits_5 (length) ; length += 32 ; goto DONE ; }
in_bit() ; // 6th length bit
if (!bitset) { in_bits_6 (length) ; length += 64 ; goto DONE ; }
in_bit() ; // 7th length bit
if (!bitset) { in_bits_7 (length) ; length += 128 ; goto DONE ; }
in_bit() ; // 8th length bit
if (!bitset) { in_bits_8 (length) ; length += 256 ; goto DONE ; }
in_bit() ; // 9th length bit
if (!bitset) { in_bits_9 (length) ; length += 512 ; goto DONE ; }
in_bit() ; // 10th length bit
if (!bitset) { in_bits_10 (length) ; length += 1024 ; goto DONE ; }
//
// length cannot be greater than max packets size which is 1500
//
#if DBG
DbgPrint("NDISWAN: RAS Decompressor problem1: Possible data corruption\n"); #endif
return FALSE ;
DONE: //
// Turn the backptr into an index location
//
#ifdef COMP_12K
s2 = current - backptr ; #else
s2 = context->History + (((current - context->History) - backptr) & (HISTORY_SIZE -1)) ; #endif
s1 = current;
current += length;
// if we are past the end of the history this is a bad sign: abort decompression
//
if (current >= historyend) { #if DBG
DbgPrint("NDISWAN: RAS Decompressor problem2: Possible data corruption\n"); #endif
return FALSE ; }
// loop unrolled to handle lenght>backptr case
//
*s1=*s2; *(s1+1)=*(s2+1); s1+=2; s2+=2; length-=2;
//
// copy all the bytes
//
while (length) { *s1++=*s2++; length--; }
//
// We have another copy item, and no literals
//
continue;
LITERAL:
//
// We have a literal
//
if (current >= historyend) { #if DBG
DbgPrint("NDISWAN: RAS Decompressor problem3: Possible data corruption\n"); #endif
return FALSE ; }
//*current++ = literallookup[length];
*current++ = (UCHAR)length;
} // while loop
// End case:
//
if (current >= historyend) { #if DBG
DbgPrint("NDISWAN: RAS Decompressor problem4: Possible data corruption\n"); #endif
return FALSE ; }
if ((bit == 16) && (pbyte == inend)) { *current++ = *(pbyte -1) ; }
#if DBG
if (context->DebugFence != DEBUG_FENCE_VALUE) { DbgPrint("Decompression Error!\n"); DbgPrint("context %p, current %p, outstart %p\n", context, current, outstart); DbgPrint("inbuf %p, inlength %d, start %p\n", inbuf, inlen, start); DbgBreakPoint(); } #endif
*outlen = (int)(current - outstart) ; // the length of decompressed data
*output = context->CurrentPtr ;
context->CurrentPtr = current ;
return TRUE ; }
//
// This function uses the 16 byte user session key and the 8 byte
// challenge to create an intial 16 byte encryption session key.
//
VOID GetStartKeyFromSHA( PCRYPTO_INFO CryptoInfo, PUCHAR Challenge ) { UCHAR Digest[A_SHA_DIGEST_LEN]; UCHAR SessionKeyChallenge[MAX_USERSESSIONKEY_SIZE + MAX_CHALLENGE_SIZE];
NdisZeroMemory(Digest, A_SHA_DIGEST_LEN);
//
// Copy the start session key
//
NdisMoveMemory(SessionKeyChallenge, CryptoInfo->StartKey, MAX_USERSESSIONKEY_SIZE);
//
// Append the challenge
//
NdisMoveMemory((PUCHAR)(SessionKeyChallenge + MAX_USERSESSIONKEY_SIZE), Challenge, MAX_CHALLENGE_SIZE);
//
// SHAInit(context)
// SHAUpdate(context, sessionkey, sessionkeylength)
// SHAUpdate(context, sessionkeychallenge, sessionkeylength + challengelength)
// SHAFinal(context, digest)
//
// Start key is the first 16 bytes of the digest.
//
A_SHAInit(CryptoInfo->Context);
A_SHAUpdate(CryptoInfo->Context, CryptoInfo->StartKey, MAX_USERSESSIONKEY_SIZE);
A_SHAUpdate(CryptoInfo->Context, SessionKeyChallenge, MAX_USERSESSIONKEY_SIZE + MAX_CHALLENGE_SIZE);
A_SHAFinal(CryptoInfo->Context, Digest);
NdisMoveMemory(CryptoInfo->StartKey, Digest, CryptoInfo->SessionKeyLength);
NdisMoveMemory(CryptoInfo->SessionKey, Digest, CryptoInfo->SessionKeyLength); }
VOID GetNewKeyFromSHA( PCRYPTO_INFO CryptoInfo ) { UCHAR Digest[A_SHA_DIGEST_LEN];
NdisZeroMemory(Digest, A_SHA_DIGEST_LEN);
A_SHAInit(CryptoInfo->Context); A_SHAUpdate(CryptoInfo->Context, CryptoInfo->StartKey, CryptoInfo->SessionKeyLength); A_SHAUpdate(CryptoInfo->Context, SHApad1, 40); A_SHAUpdate(CryptoInfo->Context, CryptoInfo->SessionKey, CryptoInfo->SessionKeyLength); A_SHAUpdate(CryptoInfo->Context, SHApad2, 40); A_SHAFinal(CryptoInfo->Context, Digest);
NdisMoveMemory(CryptoInfo->SessionKey, Digest, CryptoInfo->SessionKeyLength); }
VOID GetMasterKey( PCRYPTO_INFO CryptoInfo, PUCHAR NTResponse ) { UCHAR Digest[A_SHA_DIGEST_LEN]; PVOID Context;
NdisZeroMemory(Digest, A_SHA_DIGEST_LEN);
Context = CryptoInfo->Context;
A_SHAInit(Context);
#ifdef DEBUG_CCP
{ PUCHAR Key;
Key = CryptoInfo->StartKey;
DbgPrint("GMK-UserSessionKey: %.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x\n", Key[0],Key[1],Key[2],Key[3], Key[4],Key[5],Key[6],Key[7], Key[8],Key[9],Key[10],Key[11], Key[12],Key[13],Key[14],Key[15]);
Key = NTResponse;
DbgPrint("GMK-NTResponse: %.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x\n", Key[0],Key[1],Key[2],Key[3], Key[4],Key[5],Key[6],Key[7], Key[8],Key[9],Key[10],Key[11], Key[12],Key[13],Key[14],Key[15]);
DbgPrint(" %.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x\n", Key[16],Key[17],Key[18],Key[19], Key[20],Key[21],Key[22],Key[23]); } #endif
A_SHAUpdate(Context, CryptoInfo->StartKey, MAX_USERSESSIONKEY_SIZE);
A_SHAUpdate(Context, NTResponse, MAX_NT_RESPONSE);
A_SHAUpdate(Context, "This is the MPPE Master Key", 27);
A_SHAFinal(Context, Digest);
NdisMoveMemory(CryptoInfo->StartKey, Digest, MAX_USERSESSIONKEY_SIZE);
#ifdef DEBUG_CCP
{ PUCHAR Key;
Key = CryptoInfo->StartKey;
DbgPrint("MasterKey: %.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x\n", Key[0],Key[1],Key[2],Key[3], Key[4],Key[5],Key[6],Key[7], Key[8],Key[9],Key[10],Key[11], Key[12],Key[13],Key[14],Key[15]);
} #endif
}
VOID GetAsymetricStartKey( PCRYPTO_INFO CryptoInfo, BOOLEAN IsSend ) { UCHAR Digest[A_SHA_DIGEST_LEN]; PVOID Context; PUCHAR s;
NdisZeroMemory(Digest, A_SHA_DIGEST_LEN);
Context = CryptoInfo->Context;
if (IsSend) { if (CryptoInfo->Flags & CRYPTO_IS_SERVER) { s = ClntRSrvS; } else { s = ClntSSrvR; } } else { if (CryptoInfo->Flags & CRYPTO_IS_SERVER) { s = ClntSSrvR; } else { s = ClntRSrvS; } }
#ifdef DEBUG_CCP
{ PUCHAR Key;
Key = CryptoInfo->StartKey;
DbgPrint("GASK-StartKey: %.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x\n", Key[0],Key[1],Key[2],Key[3], Key[4],Key[5],Key[6],Key[7], Key[8],Key[9],Key[10],Key[11], Key[12],Key[13],Key[14],Key[15]);
DbgPrint("GASK-String: %s\n", s); } #endif
A_SHAInit(Context);
A_SHAUpdate(Context, CryptoInfo->StartKey, MAX_USERSESSIONKEY_SIZE);
A_SHAUpdate(Context,SHApad1,40);
A_SHAUpdate(Context,s,strlen(s));
A_SHAUpdate(Context,SHApad2,40);
A_SHAFinal(Context,Digest);
NdisMoveMemory(CryptoInfo->StartKey, Digest, CryptoInfo->SessionKeyLength);
NdisMoveMemory(CryptoInfo->SessionKey, Digest, CryptoInfo->SessionKeyLength);
#ifdef DEBUG_CCP
{ PUCHAR Key;
Key = CryptoInfo->StartKey;
DbgPrint("%s %s AsymetricKey: %.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x\n", (CryptoInfo->Flags & CRYPTO_IS_SERVER) ? "Server" : "Client", (IsSend) ? "Send" : "Recv", Key[0],Key[1],Key[2],Key[3], Key[4],Key[5],Key[6],Key[7], Key[8],Key[9],Key[10],Key[11], Key[12],Key[13],Key[14],Key[15]);
} #endif
}
/* Copyright (C) RSA Data Security, Inc. created 1993. This is an
unpublished work protected as such under copyright law. This work contains proprietary, confidential, and trade secret information of RSA Data Security, Inc. Use, disclosure or reproduction without the express written authorization of RSA Data Security, Inc. is prohibited. */
/* SHA initialization. Begins an SHA operation, writing a new context.
*/ void A_SHAInitCommon (context) A_SHA_COMM_CTX *context; { context->count[0] = context->count[1] = 0;
/* Load magic initialization constants.
*/ context->state[0] = 0x67452301; context->state[1] = 0xefcdab89; context->state[2] = 0x98badcfe; context->state[3] = 0x10325476; context->state[4] = 0xc3d2e1f0; }
/* SHA block update operation. Continues an SHA message-digest
operation, processing another message block, and updating the context. */ void A_SHAUpdateCommon (context, partIn, partInLen, Transform) A_SHA_COMM_CTX *context; unsigned char *partIn; ULONG partInLen; A_SHA_TRANSFORM *Transform; { unsigned int bufferLen;
/* Compute length of buffer */ bufferLen = (unsigned int)(context->count[1] & 0x3f);
/* Update number of bytes */ if ((context->count[1] += partInLen) < partInLen) context->count[0]++;
/* If previous input in buffer, buffer new input and transform if
possible. */ if (bufferLen > 0 && bufferLen + partInLen >= 64) { NdisMoveMemory(context->buffer+bufferLen, partIn, 64-bufferLen); partIn += (64-bufferLen); partInLen -= (64-bufferLen); (*Transform) (context->state, context->buffer); bufferLen = 0; }
/* Transform directly from input.
*/ while (partInLen >= 64) { (*Transform) (context->state, partIn); partIn += 64; partInLen -= 64; }
/* Buffer remaining input */ NdisMoveMemory((context->buffer+bufferLen), partIn, partInLen); }
/* SHA finalization. Ends an SHA message-digest operation, writing
the message digest and zeroizing the context. */ void A_SHAFinalCommon (context, digest, Transform) A_SHA_COMM_CTX *context; unsigned char digest[A_SHA_DIGEST_LEN]; A_SHA_TRANSFORM *Transform; { ULONG bitCount[2]; unsigned char pad[72]; unsigned int padLen;
/* Compute padding: 80 00 00 ... 00 00 <bit count>
*/ padLen = 64 - (unsigned int)(context->count[1] & 0x3f); if (padLen <= 8) padLen += 64; pad[0] = 0x80; NdisZeroMemory(pad+1, padLen-7); bitCount[0] = (context->count[0] << 3) | (context->count[1] >> 29); bitCount[1] = context->count[1] << 3; ByteReverse ((UNALIGNED ULONG*)(pad+padLen-8), bitCount, 2);
/* Digest padding */ A_SHAUpdateCommon (context, pad, padLen, Transform);
/* Store digest */ ByteReverse ((UNALIGNED ULONG*)digest, context->state, 5);
/* Restart the context */ A_SHAInitCommon (context); }
void A_SHAInit (A_SHA_CTX *context) { A_SHAInitCommon (&context->commonContext); }
void A_SHAUpdate (context, partIn, partInLen) A_SHA_CTX *context; unsigned char *partIn; unsigned int partInLen; { A_SHAUpdateCommon (&context->commonContext, partIn, partInLen, SHATransform); }
void A_SHAFinal (context, digest) A_SHA_CTX *context; unsigned char digest[A_SHA_DIGEST_LEN]; { A_SHAFinalCommon (&context->commonContext, digest, SHATransform); }
void SHATransform (state, block) ULONG state[5]; unsigned char block[64]; { ULONG a = state[0], b = state[1], c = state[2], d = state[3], e = state[4], x[80];
ByteReverse (x, (ULONG*)block, 16); SHAExpand (x);
/* Round 1 */ FF (a, b, c, d, e, x[ 0]); FF (e, a, b, c, d, x[ 1]); FF (d, e, a, b, c, x[ 2]); FF (c, d, e, a, b, x[ 3]); FF (b, c, d, e, a, x[ 4]); FF (a, b, c, d, e, x[ 5]); FF (e, a, b, c, d, x[ 6]); FF (d, e, a, b, c, x[ 7]); FF (c, d, e, a, b, x[ 8]); FF (b, c, d, e, a, x[ 9]); FF (a, b, c, d, e, x[10]); FF (e, a, b, c, d, x[11]); FF (d, e, a, b, c, x[12]); FF (c, d, e, a, b, x[13]); FF (b, c, d, e, a, x[14]); FF (a, b, c, d, e, x[15]); FF (e, a, b, c, d, x[16]); FF (d, e, a, b, c, x[17]); FF (c, d, e, a, b, x[18]); FF (b, c, d, e, a, x[19]);
/* Round 2 */ GG (a, b, c, d, e, x[20]); GG (e, a, b, c, d, x[21]); GG (d, e, a, b, c, x[22]); GG (c, d, e, a, b, x[23]); GG (b, c, d, e, a, x[24]); GG (a, b, c, d, e, x[25]); GG (e, a, b, c, d, x[26]); GG (d, e, a, b, c, x[27]); GG (c, d, e, a, b, x[28]); GG (b, c, d, e, a, x[29]); GG (a, b, c, d, e, x[30]); GG (e, a, b, c, d, x[31]); GG (d, e, a, b, c, x[32]); GG (c, d, e, a, b, x[33]); GG (b, c, d, e, a, x[34]); GG (a, b, c, d, e, x[35]); GG (e, a, b, c, d, x[36]); GG (d, e, a, b, c, x[37]); GG (c, d, e, a, b, x[38]); GG (b, c, d, e, a, x[39]);
/* Round 3 */ HH (a, b, c, d, e, x[40]); HH (e, a, b, c, d, x[41]); HH (d, e, a, b, c, x[42]); HH (c, d, e, a, b, x[43]); HH (b, c, d, e, a, x[44]); HH (a, b, c, d, e, x[45]); HH (e, a, b, c, d, x[46]); HH (d, e, a, b, c, x[47]); HH (c, d, e, a, b, x[48]); HH (b, c, d, e, a, x[49]); HH (a, b, c, d, e, x[50]); HH (e, a, b, c, d, x[51]); HH (d, e, a, b, c, x[52]); HH (c, d, e, a, b, x[53]); HH (b, c, d, e, a, x[54]); HH (a, b, c, d, e, x[55]); HH (e, a, b, c, d, x[56]); HH (d, e, a, b, c, x[57]); HH (c, d, e, a, b, x[58]); HH (b, c, d, e, a, x[59]);
/* Round 4 */ II (a, b, c, d, e, x[60]); II (e, a, b, c, d, x[61]); II (d, e, a, b, c, x[62]); II (c, d, e, a, b, x[63]); II (b, c, d, e, a, x[64]); II (a, b, c, d, e, x[65]); II (e, a, b, c, d, x[66]); II (d, e, a, b, c, x[67]); II (c, d, e, a, b, x[68]); II (b, c, d, e, a, x[69]); II (a, b, c, d, e, x[70]); II (e, a, b, c, d, x[71]); II (d, e, a, b, c, x[72]); II (c, d, e, a, b, x[73]); II (b, c, d, e, a, x[74]); II (a, b, c, d, e, x[75]); II (e, a, b, c, d, x[76]); II (d, e, a, b, c, x[77]); II (c, d, e, a, b, x[78]); II (b, c, d, e, a, x[79]);
state[0] += a; state[1] += b; state[2] += c; state[3] += d; state[4] += e;
/* Zeroize potentially sensitive information.
*/ NdisZeroMemory((void *)x, sizeof (x)); }
/* Expands x[0..15] into x[16..79], according to the recurrence
x[i] = x[i-3] ^ x[i-8] ^ x[i-14] ^ x[i-16]. */ void SHAExpand (x) ULONG x[80]; { unsigned int i; ULONG tmp;
for (i = 16; i < 80; i++) { tmp = x[i-3] ^ x[i-8] ^ x[i-14] ^ x[i-16]; x[i] = (tmp << 1) | (tmp >> 31); } }
VOID ByteReverse( UNALIGNED ULONG *Out, ULONG *In, ULONG Count ) { ULONG i; ULONG Value;
for (i = 0; i < Count; i++) { Value = (ULONG)(In[i] << 16) | (In[i] >> 16); Out[i] = ((Value & 0xFF00FF00L) >> 8) | ((Value & 0x00FF00FFL) << 8); } }
|