FreeCalypso > hg > freecalypso-sw
diff gsm-fw/ccd/cdc_std.c @ 648:970d6199f2c5
gsm-fw/ccd/*.[ch]: initial import from the LoCosto source
author | Michael Spacefalcon <msokolov@ivan.Harhan.ORG> |
---|---|
date | Thu, 04 Sep 2014 05:48:57 +0000 |
parents | |
children |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/gsm-fw/ccd/cdc_std.c Thu Sep 04 05:48:57 2014 +0000 @@ -0,0 +1,1204 @@ +/* ++----------------------------------------------------------------------------- +| Project : +| Modul : cdc_std.c ++----------------------------------------------------------------------------- +| Copyright 2002 Texas Instruments Berlin, AG +| All rights reserved. +| +| This file is confidential and a trade secret of Texas +| Instruments Berlin, AG +| The receipt of or possession of this file does not convey +| any rights to reproduce or disclose its contents or to +| manufacture, use, or sell anything it may describe, in +| whole, or in part, without the specific written consent of +| Texas Instruments Berlin, AG. ++----------------------------------------------------------------------------- +| Purpose : Condat Conder Decoder +| Definitions of non protocol specific encoding and decoding +| functions ++----------------------------------------------------------------------------- +*/ + +#define CCD_STD_C + +#ifdef _MSDOS +#include <dos.h> +#include <conio.h> +#endif + +/* + * standard definitions like UCHAR, ERROR etc. + */ +#include "typedefs.h" +#include "header.h" +#include <string.h> + +/* + * Prototypes of ccd (USE_DRIVER EQ undef) for prototypes only + * look at ccdapi.h. + */ +#undef USE_DRIVER +#include "ccdapi.h" + +/* + * Types and functions for bit access and manipulation + */ +#include "ccd_globs.h" +#include "bitfun.h" + +/* + * Declaration of coder/decoder tables and/or functions to access them + */ +#include "ccdtable.h" +#include "ccddata.h" + +/* + * Prototypes of ccd internal functions + */ +#include "ccd.h" + +/* + * Need memory allocation functions for dynamic arrays (pointers) + */ +#ifdef DYNAMIC_ARRAYS +#include "vsi.h" +#endif + +#ifndef RUN_FLASH +/* ++--------------------------------------------------------------------+ +| PROJECT : CCD (6144) MODULE : CCD | +| STATE : code ROUTINE : cdc_decodeElemvalue | ++--------------------------------------------------------------------+ + + PURPOSE : Performs a standard decoding for a given elem table entry. + This means for non structured elements that 1-n bits are + read from the bitstream and write to a C-Variable + in a machine dependent format. For structured elements + an (indirect) recursive call to cc_decodeComposition() + is performed. If the element is a bitbuffer with variable + size, the repeat value gives the number of bits to read + from the bitstream into the buffer. The maxBitpos + indicates the maximum valid position for the + readpointer of the bitstream while decoding this element. + If the readpointer break this boundary, this element will + not be decoded. +*/ + +void cdc_decodeElemvalue (ULONG e_ref, ULONG *repeat, T_CCD_Globs *globs) +{ + UBYTE *ActStructpos; + ULONG i; + UBYTE spareLen; + + /* + * element is a bit field of variable length + */ + if ((melem[e_ref].repType == 'b') || (melem[e_ref].repType == 's')) + { + if (*repeat > (ULONG) (globs->maxBitpos-globs->bitpos)) + { + ccd_recordFault (globs, ERR_MAX_REPEAT, CONTINUE, + (USHORT) e_ref, globs->pstruct + globs->pstructOffs); + *repeat = MINIMUM (*repeat, (ULONG) (globs->maxBitpos-globs->bitpos)); + } + if (melem[e_ref].repType == 'b') + { + if (mvar[melem[e_ref].elemRef].cType EQ 'X') + bf_readBitChunk (*repeat, globs); + else + bf_readBits (*repeat, globs); + } + else + { + U16 finalBP = globs->bitpos + (USHORT) *repeat; + /* Store the limit. This comp may contain other comps as bitstring. */ + globs->maxBitpos = finalBP; + ActStructpos = globs->pstruct; + globs->pstruct += globs->pstructOffs; +#ifdef DEBUG_CCD +#ifdef CCD_SYMBOLS + TRACE_CCD (globs, "decoding composition %s as a bit array", + mcomp[melem[e_ref].elemRef].name); +#else + TRACE_CCD (globs, "decoding composition %d", melem[e_ref].elemRef); +#endif +#endif + + ccd_decodeComposition ((ULONG) (melem[e_ref].elemRef), globs); + if (finalBP < globs->bitpos) + { + ccd_recordFault (globs, ERR_BITSTR_COMP, CONTINUE, (USHORT) e_ref, + globs->pstruct + globs->pstructOffs); + } + bf_setBitpos (finalBP, globs); + /* Update maxBitpos to avoid an early end of decoding. */ + globs->maxBitpos = globs->buflen; + globs->pstruct = ActStructpos; + } + } + else + { + /* + * For pointer types, globs->pstruct is already set to point to + * the new memory area, and these types are not treated differently + * from non-pointer types. + */ + i=0; + switch (melem[e_ref].elemType) + { + case 'R': /* Pointer to (possible array of) basetype */ + case 'F': /* Code-transparent pointer to (possible array of) basetype */ + case 'V': + while (i < *repeat) + { + if (globs->bitpos < globs->maxBitpos) + { +#ifdef DEBUG_CCD +#ifdef CCD_SYMBOLS + TRACE_CCD (globs, "decoding var %s", + ccddata_get_alias((USHORT) e_ref, 1)); +#else + TRACE_CCD (globs, "decoding var %d", melem[e_ref].elemRef); +#endif +#endif + if (mvar[melem[e_ref].elemRef].cType EQ 'X') + bf_readBitChunk (mvar[melem[e_ref].elemRef].bSize, globs); + else + bf_readBits (mvar[melem[e_ref].elemRef].bSize, globs); + + globs->pstructOffs += mvar[melem[e_ref].elemRef].cSize; + + i++; + } + else + { + if (melem[e_ref].repType != 'i') + { + ccd_recordFault (globs, ERR_ELEM_LEN, CONTINUE, (USHORT) e_ref, + globs->pstruct + globs->pstructOffs); + } + break; + } + } + break; + + case 'D': /* Pointer to a composition */ + case 'P': /* Code transparent pointer to a comp */ + case 'C': /* Element is a composition. */ + case 'U': /* Element is a union. */ + /* + * Store the actual structure position. + */ + ActStructpos = globs->pstruct; + + globs->pstruct += globs->pstructOffs; + + while (i < *repeat) + { + if (globs->bitpos < globs->maxBitpos) + { +#ifdef DEBUG_CCD +#ifdef CCD_SYMBOLS + TRACE_CCD (globs, "decoding composition %s", + mcomp[melem[e_ref].elemRef].name); +#else + TRACE_CCD (globs, "decoding composition %d", melem[e_ref].elemRef); +#endif +#endif + /* + * recursiv call + */ + ccd_decodeComposition ((ULONG) (melem[e_ref].elemRef), globs); + globs->pstruct += mcomp[melem[e_ref].elemRef].cSize; + + i++; + } + else + { + if (melem[e_ref].repType != 'i') + { + ccd_recordFault (globs, ERR_ELEM_LEN, CONTINUE, (USHORT) e_ref, + globs->pstruct + globs->pstructOffs); + } + break; + } + } + /* + * restore the write pointer + */ + globs->pstruct = ActStructpos; + break; + + case 'S': /* Element is a spare. */ + { + spareLen = spare[melem[e_ref].elemRef].bSize; + /* + * Do not decode padding bits. They are not relevant. + * Just adjust the position pointer in the bit stream buffer. + */ + while (i < *repeat) + { + if (globs->bitpos < globs->maxBitpos) + { +#ifdef DEBUG_CCD + TRACE_CCD (globs, "decoding spare"); +#endif + bf_incBitpos (spareLen, globs); + i++; + } + else + break; + } + break; + } + } + *repeat = i; + } +} +#endif /* !RUN_FLASH */ + +#ifndef RUN_FLASH +/* ++--------------------------------------------------------------------+ +| PROJECT : CCD (6144) MODULE : CCD | +| STATE : code ROUTINE : cdc_encodeElemvalue | ++--------------------------------------------------------------------+ + + PURPOSE : Performs a standard encoding for a given elem table entry. + This means for non structured elements that 1-n bits are + read from the bitstream and write to a C-Variable + in a machine dependent format. For structured elements + an (indirect) recursive call to cc_decodeComposition() + is performed. If the element is a bitbuffer with variable + size, the repeat value gives the number of bits to write + from the buffer into the bitstream. +*/ + +void cdc_encodeElemvalue (ULONG e_ref, ULONG repeat, T_CCD_Globs *globs) +{ + UBYTE *ActStructpos = NULL; + ULONG i; + UBYTE spareLen; + + /* + * Element is a bit field of variable length. + */ + if (melem[e_ref].repType == 'b') + { + if (mvar[melem[e_ref].elemRef].cType EQ 'X') + bf_writeBitChunk (repeat, globs); + else + bf_writeBits (repeat, globs); + } + /* + * Element is a structured IE defined as bit string. + */ + else if (melem[e_ref].repType == 's') + { + U16 finalBP = (USHORT) (globs->bitpos + repeat); + ActStructpos = globs->pstruct; + globs->pstruct += globs->pstructOffs; +#ifdef DEBUG_CCD +#ifdef CCD_SYMBOLS + TRACE_CCD (globs, "encoding composition %s as a bit array", + mcomp[melem[e_ref].elemRef].name); +#else + TRACE_CCD (globs, "encoding composition %d", melem[e_ref].elemRef); +#endif +#endif + + ccd_encodeComposition ((ULONG) melem[e_ref].elemRef, globs); + if (finalBP < globs->bitpos) + { + ccd_recordFault (globs, ERR_BITSTR_COMP, CONTINUE, (USHORT) e_ref, + globs->pstruct + globs->pstructOffs); + } + bf_setBitpos (finalBP, globs); + globs->pstruct = ActStructpos; + } + else + { + /* + * For pointer types, globs->pstruct is already set to point to + * the new memory area, and these types are not treated differently + * from non-pointer types. + */ + switch(melem[e_ref].elemType) + { + case 'R': /* Pointer to (possible array of) basetype */ + case 'F': /* Code-transparent pointer to (possible array of) basetype */ + case 'V': + for (i=0; i<repeat; i++) + { + #ifdef DEBUG_CCD + #ifdef CCD_SYMBOLS + TRACE_CCD (globs, "encoding var %s", + ccddata_get_alias((USHORT) e_ref, 1)); + #else + TRACE_CCD (globs, "encoding var %s", melem[e_ref].elemRef); + #endif + #endif + if (mvar[melem[e_ref].elemRef].cType EQ 'X') + bf_writeBitChunk (mvar[melem[e_ref].elemRef].bSize, globs); + else + bf_writeBits (mvar[melem[e_ref].elemRef].bSize, globs); + + globs->pstructOffs += mvar[melem[e_ref].elemRef].cSize; + } + break; + + case 'D': /* Pointer to a composition (already dereferenced) */ + case 'P': /* Code transparent pointer to a comp (already dereferenced) */ + case 'C': /* Element is a composition. */ + case 'U': /* Element is a union. */ + /* + * store the actual structure position + */ + ActStructpos = globs->pstruct; + + globs->pstruct += globs->pstructOffs; + + for (i=0; i<repeat; i++) + { + #ifdef DEBUG_CCD + #ifdef CCD_SYMBOLS + TRACE_CCD (globs, "encoding composition %s", + mcomp[melem[e_ref].elemRef].name); + #else + TRACE_CCD (globs, "encoding composition %d", melem[e_ref].elemRef); + #endif + #endif + ccd_encodeComposition ((ULONG) melem[e_ref].elemRef, globs); /* recursiv call */ + globs->pstruct += mcomp[melem[e_ref].elemRef].cSize; + } + /* + * restore the write pointer + */ + globs->pstruct = ActStructpos; + break; + + case 'S': /* element is a spare */ + { + spareLen = spare[melem[e_ref].elemRef].bSize; + + /* + * encode the spare + */ + for (i=0; i < repeat; i++) + { + #ifdef DEBUG_CCD + TRACE_CCD (globs, "encoding spare"); + #endif + bf_codeLongNumber (spareLen, spare[melem[e_ref].elemRef].value, globs); + } + break; + } + } + } +} +#endif /* !RUN_FLASH */ + +#ifndef RUN_FLASH +/* ++--------------------------------------------------------------------+ +| PROJECT : CCD (6144) MODULE : CCD | +| STATE : code ROUTINE : cdc_STD_decode | ++--------------------------------------------------------------------+ + + PURPOSE : Performs a standard decoding for a given elem table entry. + This means for non structured elements that 1-n bits are + read from the bitstream and write to a C-Variable + in a machine dependent format. For structured elements + an (indirect) recursive call to cc_decodeComposition() + is performed. + The element may be conditional and/or repeatable. +*/ + +SHORT cdc_std_decode (const ULONG c_ref, const ULONG e_ref, T_CCD_Globs *globs) +{ + ULONG repeat, amount, act_offset; + BOOL is_variable; + U8 *old_pstruct = NULL; + ULONG cix_ref, num_prolog_steps, prolog_step_ref; + +#ifdef DEBUG_CCD + #ifndef CCD_SYMBOLS + TRACE_CCD (globs, "cdc_std_decode()"); + #else + TRACE_CCD (globs, "cdc_std_decode() %s", ccddata_get_alias((USHORT) e_ref, 1)); + #endif +#endif + + cix_ref = melem[e_ref].calcIdxRef; + num_prolog_steps = calcidx[cix_ref].numPrologSteps; + prolog_step_ref = calcidx[cix_ref].prologStepRef; + + /* + * if this element is conditional, check the condition + */ + if (calcidx[cix_ref].numCondCalcs NEQ 0 + AND ! ccd_conditionOK (e_ref, globs)) + return 1; + + /* + * if this element has a defined prologue + * we have to process it before decoding the bitstream + * If there are some epilogue expressions to be processed for this element + * (rare cases) the result here will be a reading of 0 to an internal + * register. The valid processing of expression takes place after the + * decoding of the element. + */ + if (num_prolog_steps) + { + ccd_performOperations (num_prolog_steps, prolog_step_ref, globs); + } + + /* + * if this element is repeatable, and the number of + * repeats depends on another element, calculate the repeater + */ + if (melem[e_ref].repType NEQ ' ') + { + is_variable = ccd_calculateRep (e_ref, &repeat, &act_offset, globs); + } + else + { + repeat = 1; + is_variable = FALSE; + } + + if (melem[e_ref].elemType NEQ 'S') + { + /* + * Element is not a SPARE. + * Setup the offset into the C-structure for this element + */ + globs->pstructOffs = melem[e_ref].structOffs; + + if (melem[e_ref].optional) + { + /* 20010621 MVJ: Dynamic array addition. + * Postpone optional flag setting for non-code transparent + * pointer types ('P', 'Q', 'R'). + * For these types, the optional flag is the pointer itself. + * These types cannot be set yet, as the pointer may be + * preceeded by a counter octet, a union tag id octet etc. + */ + if (melem[e_ref].elemType < 'P' OR melem[e_ref].elemType > 'R') + globs->pstruct[globs->pstructOffs++] = (UBYTE) TRUE; + } + + if (is_variable) + { + /* + * for variable sized elements store the min-value + * as counter into the C-Structure (c_xxx). + */ + if (act_offset > 65535) + *(ULONG *) (globs->pstruct + globs->pstructOffs++) = repeat; + else if (act_offset > 255) + *(USHORT *) (globs->pstruct + globs->pstructOffs++) = (USHORT) repeat; + else + globs->pstruct[globs->pstructOffs] = (UBYTE) repeat; + + globs->pstructOffs++; + } + } + #ifdef DYNAMIC_ARRAYS + /* + * MVJ: Dynamic array addition. + * Check for pointer types; allocate memory if necessary. + */ + if ( is_pointer_type(e_ref) ) { + U32 cSize; + U8 *addr; + + /* + * Find size to allocate; + * - Read from mcomp or mvar according to type + */ + cSize = (ULONG)((melem[e_ref].elemType EQ 'V' OR + melem[e_ref].elemType EQ 'R') + ? mvar[melem[e_ref].elemRef].cSize + : mcomp[melem[e_ref].elemRef].cSize + ) * repeat; + + /* + * Allocate additional memory + */ + addr = (U8 *)DP_ALLOC( cSize, globs->alloc_head, DP_NO_FRAME_GUESS); + + /* If no memory, log error and return immediately */ + if (addr EQ NULL) { + ccd_setError (globs, ERR_NO_MEM, + BREAK, + (USHORT) -1); + return 1; + } + else + memset (addr, 0, (size_t)cSize); + + /* + * Memory allocated; + * 1. Save old "globs->pstruct" variables + * 2. Store pointer to freshly allocated memory area in structure + * 3. Initialize pstruct to point to the freshly allocated memory area. + * 4. Initialize pstructOffs to 0 to start decoding at offset 0 + * in the new memory area. + */ + old_pstruct = globs->pstruct; + *(U8 **)(globs->pstruct + globs->pstructOffs) = addr; + globs->pstruct = addr; + globs->pstructOffs = 0; + } +#endif + + amount = repeat; + + cdc_decodeElemvalue (e_ref, &amount, globs); + + /* + * process the epilogue expression for this element if there is any + */ + if (num_prolog_steps) + { + if ( (calc[prolog_step_ref+1].operation EQ 'K') + || (calc[prolog_step_ref+1].operation EQ 'C') + || (calc[prolog_step_ref+1].operation EQ 's')) + { + ccd_performOperations (num_prolog_steps, prolog_step_ref, globs); + } + } + +#ifdef DYNAMIC_ARRAYS + /* + * Restore globs->pstruct for possible use below + */ + if (old_pstruct NEQ NULL) + { + globs->pstruct = old_pstruct; + } +#endif + if (amount NEQ repeat AND is_variable) + { + /* + * if the decoded elements are not equal the specified + * repeat value, because the bitstream or the IE ended, + * store the new c_xxx value. + */ + globs->pstructOffs = melem[e_ref].structOffs; + + if (melem[e_ref].optional) + globs->pstructOffs++; + + if (act_offset > 65535) + *(ULONG *) (globs->pstruct + globs->pstructOffs) = amount; + else if (act_offset > 255) + *(USHORT *) (globs->pstruct + globs->pstructOffs) = (USHORT) amount; + else + globs->pstruct[globs->pstructOffs] = (UBYTE) amount; + + if (melem[e_ref].repType NEQ 'i') + { + /* + * if this element is not of the repeat style 'interval' + * ([X..Y] where X and Y are constants) we have to generate + * an ccd error because some outstanding repeats are missing. + */ + ccd_setError (globs, ERR_MAND_ELEM_MISS, CONTINUE, (USHORT) -1); + } + } + + return 1; +} +#endif /* !RUN_FLASH */ + +#ifndef RUN_FLASH +/* ++--------------------------------------------------------------------+ +| PROJECT : CCD (6144) MODULE : CCD | +| STATE : code ROUTINE : cdc_std_encode | ++--------------------------------------------------------------------+ + + PURPOSE : Performs a standard encoding for a given elem table entry. + This means for non structured elements that m bytes read + from the C-Variable, converted to MSB-first format and + write 1-n bits at the end of the bitstream. + For structured elements an (indirect) recursive call + to ccd_encodeComposition() is performed. + The element may be conditional and/or repeatable. +*/ + +SHORT cdc_std_encode (const ULONG c_ref, const ULONG e_ref, T_CCD_Globs *globs) +{ + ULONG repeat, max_rep; + BOOL is_variable; + ULONG cix_ref, num_prolog_steps, prolog_step_ref; +#ifdef DYNAMIC_ARRAYS + U8 *old_pstruct = NULL; +#endif + +#ifdef DEBUG_CCD + #ifndef CCD_SYMBOLS + TRACE_CCD (globs, "cdc_std_encode()"); + #else + TRACE_CCD (globs, "cdc_std_encode() %s", ccddata_get_alias((USHORT) e_ref, 1)); + #endif +#endif + + cix_ref = melem[e_ref].calcIdxRef; + num_prolog_steps = calcidx[cix_ref].numPrologSteps; + prolog_step_ref = calcidx[cix_ref].prologStepRef; + /* + * if this element is conditional, check the condition + */ + if (calcidx[cix_ref].numCondCalcs NEQ 0 + AND ! ccd_conditionOK (e_ref, globs)) + return 1; + + /* + * if this element have a defined Prolog + * we have to process it before decoding the bitstream + */ + if (num_prolog_steps) + { + ccd_performOperations (num_prolog_steps, prolog_step_ref, globs); + } + + if (melem[e_ref].elemType NEQ 'S') + { + /* + * Element is not a SPARE. + * Setup the readpointer into the C-structure for this element + * MVJ: In case of pointer types, the pstructOffs must be + * the offset into the memory area pointed to. CCDGEN must + * ensure this holds true. + */ + globs->pstructOffs = melem[e_ref].structOffs; + + if (melem[e_ref].optional) + { + /* + * for optional elements check the valid-flag in the C-struct. + * Spare elements does not have a corresponding valid flag. For + * the spare elements we have to calculate and check the + * condition to decide if this elements have to be coded. + */ + /* 20010621 MVJ: Dynamic array addition. + * Postpone optional check for non-code transparent pointer + * types ('P', 'Q', 'R'). + * For these types, the optional flag is the pointer itself. + * These types cannot be checked yet, as the pointer may be + * preceeded by a counter octet, a union tag id octet etc. + */ + if (melem[e_ref].elemType < 'P' OR melem[e_ref].elemType > 'R') + { + if (globs->pstruct[globs->pstructOffs++] == FALSE) + { + return 1; + } +#ifdef DEBUG_CCD + else if (globs->pstruct [melem[e_ref].structOffs] != TRUE) + { + TRACE_CCD (globs, "Ambiguous value for valid flag!\n...assumed 1 for ccdID=%d", + e_ref); + } +#endif + } + } + + if ((melem[e_ref].repType EQ 'v' OR melem[e_ref].repType EQ 'i')) + { + /* + * for variable sized elements read the amount + * of repeats out of the C-Structure (c_xxx). + * If the number of repeats given by the C-Structure + * exceeds the allowed value (maxRepeat) CCD gives a warning! + */ + if (melem[e_ref].maxRepeat > 255) + { + ULONG count = (ULONG) (* (USHORT *)(globs->pstruct + globs->pstructOffs++)); + repeat = MINIMUM (count, (ULONG) melem[e_ref].maxRepeat); + if (repeat < count) + ccd_recordFault (globs, ERR_MAX_REPEAT, CONTINUE, + (USHORT) e_ref, globs->pstruct + globs->pstructOffs); + } + else + { + repeat = (ULONG)MINIMUM (globs->pstruct[globs->pstructOffs], + melem[e_ref].maxRepeat); + if ( repeat < (ULONG) (globs->pstruct[globs->pstructOffs]) ) + ccd_recordFault (globs, ERR_MAX_REPEAT, CONTINUE, + (USHORT) e_ref, globs->pstruct + globs->pstructOffs); + } + globs->pstructOffs++; + } + else + if (melem[e_ref].repType EQ 'c') + { + repeat = (ULONG) melem[e_ref].maxRepeat; + } + else + if (melem[e_ref].repType == 's' || melem[e_ref].repType == 'b') + { + switch (melem[e_ref].elemType) + { + case 'R': /* Pointer to (possible array of) basetype */ + case 'F': /* Code-transparent pointer to (possible array of) basetype */ + case 'V': + globs->maxBitpos = globs->bitpos + mvar[melem[e_ref].elemRef].bSize; + break; + case 'D': /* Pointer to a composition */ + case 'P': /* Code transparent pointer to a comp */ + case 'C': /* Element is a composition. */ + case 'E': /* Pointer to a union */ + case 'Q': /* Code transparent pointer to a union */ + case 'U': /* Element is a union. */ + globs->maxBitpos = globs->bitpos + mcomp[melem[e_ref].elemRef].bSize; + break; + } + + is_variable = ccd_calculateRep (e_ref, &repeat, &max_rep, globs); + } + else + repeat = 1; + + /* 20010621 MVJ: Dynamic array addition. + * Check for non-code transparent pointer types ('P', 'Q', 'R'). + * For these types, the optional flag is the pointer itself. + * ASSUMPTION: The pointer may be preceeded by a counter octet, + * a union tag id octet etc., but it is up to CCDGEN to ensure + * word alignment (by inserting alignment bytes). Therefore + * we just read from globs->pstruct[globs->pstructOffs]. + */ +#ifdef DEBUG_CCD + /* Check pointer alignment and re-align if necessary (should never happen) */ + if ( is_pointer_type(e_ref) AND ((globs->pstructOffs & 3) NEQ 0)) { + TRACE_CCD (globs, "cdc_STD_encode(): Pointer misaligned! pstruct=0x08x," + " pstructOffs=0x%08x", globs->pstruct, globs->pstructOffs); + globs->pstructOffs = (globs->pstructOffs + 3) & 3; + } +#endif +#ifdef DYNAMIC_ARRAYS + /* + * MVJ: Perform pointer dereference for pointer types. + * Also, check optionality for these types. + */ + if ( is_pointer_type(e_ref) ) { + U8 *deref_pstruct; + + /* Get pointer value */ + deref_pstruct = *(U8 **)&globs->pstruct[globs->pstructOffs]; + + /* + * Strictly speaking the 'D' to 'F' types should not need this + * check (should have returned after the optionality check above), + * but it will catch stray NULL pointers (or uninitialized + * valid flags) + */ + if (ccd_check_pointer(deref_pstruct) != ccdOK ) + { + ccd_recordFault (globs, ERR_INVALID_PTR, BREAK, (USHORT) e_ref, + &globs->pstruct[globs->pstructOffs]); + return 1; + } + /* + * Pointer not NULL; + * 1. Save old globs->pstruct and assign pointer to globs->pstruct + * as new base. + * 2. Set pstructOffs to 0 (zero) as the next offset will start + * in the new memory area. + */ + old_pstruct = globs->pstruct; + globs->pstruct = deref_pstruct; + globs->pstructOffs = 0; + } +#endif /* DYNAMIC_ARRAYS */ + } + else + { + /* + * for spare elements we have to calculate the conditions + * and the repetitions because there are no valid-flags and + * c_xxx variables in the C-structure to read. + */ + if (melem[e_ref].optional) + { + /* + * Spare elements does not have a corresponding valid flag. + * For the spare elements we have to calculate and check the + * condition to decide if this elements have to be coded. + */ + if (calcidx[cix_ref].numCondCalcs NEQ 0 + AND ! ccd_conditionOK (e_ref, globs)) + return 1; + } + /* + * if this spare is repeatable, calculate the amount of + * repeats because there is no corresponding c_xxx element + * in the C-Structure. + */ + if (melem[e_ref].repType NEQ ' ') + { + globs->maxBitpos = globs->bitpos + spare[melem[e_ref].elemRef].bSize; + is_variable = ccd_calculateRep (e_ref, &repeat, &max_rep, globs); + } + else + repeat = 1; + } + + cdc_encodeElemvalue (e_ref, repeat, globs); + +#ifdef DYNAMIC_ARRAYS + if ( old_pstruct NEQ NULL ) + globs->pstruct = old_pstruct; +#endif + + return 1; +} +#endif /* !RUN_FLASH */ + +/* + * some elementary bitfunctions + */ + +/* LSB,MSB Definitions for the CPUs */ + +#ifdef M_INTEL +#define MSB_POS 1 +#define LSB_POS 0 +#define MSW_POS 2 +#define LSW_POS 0 +#else /* M_INTEL */ +#ifdef M_MOTOROLA +#define MSB_POS 0 +#define LSB_POS 1 +#define MSW_POS 0 +#define LSW_POS 2 +#endif /* M_MOTOROLA */ +#endif /* M_INTEL */ + +extern const ULONG ccd_bitfun_mask[]; + +/* + * Table of one-bit values (2^X) + * This unused variable is commented now in order to avoid some compiler + * warnings like: variable "shift" (line xxx) is not used + +LOCAL const UBYTE shift[] = +{ + 128, 64, 32, 16, 8, 4, 2, 1 +}; + */ + +#ifndef RUN_FLASH +/* ++--------------------------------------------------------------------+ +| PROJECT : CCD (6144) MODULE : CCD | +| STATE : code ROUTINE : ccd_codeByte | ++--------------------------------------------------------------------+ + + PURPOSE: encodes the value of (value) into a CPU-dependent + bitstring. The length of the bitstring is specified + throu (bitlen). The function copies the bitstring into + the buffer (bitstream) at bit position (startbit). + The bitlen may be between 1 and 8. + +*/ + +BYTE CCDDATA_PREF(ccd_codeByte) (UBYTE * bitstream, + USHORT startbit, + USHORT bitlen, + UBYTE value) +{ + union + { /* Conversion structure 2 Byte <-> unsigned short */ + UBYTE c[2]; + USHORT s; + } + conv; + + UBYTE *p, lshift; + + USHORT m; + + p = bitstream + (startbit >> 3); + + lshift = (16 - ((startbit & 7) + bitlen)); + + conv.c[MSB_POS] = p[0]; + + conv.c[LSB_POS] = p[1]; + + m = ((USHORT) ccd_bitfun_mask[bitlen]) << lshift; + + conv.s &= ~m; + + conv.s |= ((value << lshift) & m); + + p[0] = conv.c[MSB_POS]; + + p[1] = conv.c[LSB_POS]; + + return ccdOK; +} +#endif /* !RUN_FLASH */ + +#ifndef RUN_FLASH +/* ++--------------------------------------------------------------------+ +| PROJECT : CCD (6144) MODULE : CCD | +| STATE : code ROUTINE : ccd_decodeByte | ++--------------------------------------------------------------------+ + + PURPOSE: decodes (bitlen) bits from the (bitstream) at position + (startbit) and converts them to a numeric value (value) + The bitlen may be between 1 and 8. + +*/ + +BYTE CCDDATA_PREF(ccd_decodeByte) (UBYTE *bitstream, + USHORT startbit, + USHORT bitlen, + UBYTE *value) +{ + union + { /* Conversion structure 2 Byte <-> unsigned short */ + UBYTE c[2]; + USHORT s; + } + conv; + + UBYTE *p; + + p = bitstream + (startbit >> 3); + + conv.c[MSB_POS] = *p++; + + conv.c[LSB_POS] = *p; + + conv.s >>= (16 - ((startbit & 7) + bitlen)); + + conv.s &= (USHORT) ccd_bitfun_mask[bitlen]; + + *value = (UBYTE) conv.s; + + return ccdOK; +} +#endif /* !RUN_FLASH */ + +#ifndef RUN_FLASH +/* ++--------------------------------------------------------------------+ +| PROJECT : CCD (6144) MODULE : CCD | +| STATE : code ROUTINE : ccd_codeLong | ++--------------------------------------------------------------------+ + + PURPOSE: encodes the value of (value) into a CPU-dependent + bitstring. The length of the bitstring is specified + throu (bitlen). The function copies the bitstring into + the buffer (bitstream) at bit position (startbit). + The bitlen may be between 1 and 32. + +*/ + +BYTE CCDDATA_PREF(ccd_codeLong) (UBYTE *bitstream, + USHORT startbit, + USHORT bitlen, + ULONG value) +{ + UBYTE *p; + + union + { /* Conversion structure 4 Byte <-> unsigned long */ + UBYTE c[4]; + ULONG l; + } + conv; + + p = bitstream + (startbit >> 3); + startbit &= 7; + + conv.l = value & ccd_bitfun_mask[bitlen]; + conv.l <<= (32-bitlen-startbit); + + p[0] &= (UCHAR)~ccd_bitfun_mask[8-startbit]; + + switch ((USHORT)(startbit+bitlen-1) >> 3) + { + case 0: + p[0] |= conv.c[MSW_POS+MSB_POS]; + break; + case 1: + p[0] |= conv.c[MSW_POS+MSB_POS]; + p[1] = conv.c[MSW_POS+LSB_POS]; + break; + case 2: + p[0] |= conv.c[MSW_POS+MSB_POS]; + p[1] = conv.c[MSW_POS+LSB_POS]; + p[2] = conv.c[LSW_POS+MSB_POS]; + break; + case 3: + p[0] |= conv.c[MSW_POS+MSB_POS]; + p[1] = conv.c[MSW_POS+LSB_POS]; + p[2] = conv.c[LSW_POS+MSB_POS]; + p[3] = conv.c[LSW_POS+LSB_POS]; + break; + default: + p[0] |= conv.c[MSW_POS+MSB_POS]; + p[1] = conv.c[MSW_POS+LSB_POS]; + p[2] = conv.c[LSW_POS+MSB_POS]; + p[3] = conv.c[LSW_POS+LSB_POS]; + p[4] = (UBYTE) ((value & 0xff) << (8-startbit)); + break; + } + return ccdOK; +} +#endif /* !RUN_FLASH */ + +#ifndef RUN_FLASH +/* ++--------------------------------------------------------------------+ +| PROJECT : CCD (6144) MODULE : CCD | +| STATE : code ROUTINE : ccd_decodeLong | ++--------------------------------------------------------------------+ + + PURPOSE: decodes (bitlen) bits from the (bitstream) at position + (startbit) and converts them to a numeric value (value) + The bitlen may be between 1 and 32. + +*/ + +BYTE CCDDATA_PREF(ccd_decodeLong) (UBYTE *bitstream, + USHORT startbit, + USHORT bitlen, + ULONG *value) +{ + UBYTE *p; + + union + { /* Conversion structure 4 Byte <-> unsigned long */ + UBYTE c[4]; + ULONG l; + } + conv; + + p = bitstream + (startbit >> 3); + startbit &= 7; + + conv.l = 0L; + switch ((USHORT)(startbit+bitlen-1) >> 3) + { + case 0: + conv.c[MSW_POS+MSB_POS] = p[0]; + conv.l <<= startbit; + conv.l >>= (32-bitlen); + break; + case 1: + conv.c[MSW_POS+MSB_POS] = p[0]; + conv.c[MSW_POS+LSB_POS] = p[1]; + conv.l <<= startbit; + conv.l >>= (32-bitlen); + break; + case 2: + conv.c[MSW_POS+MSB_POS] = p[0]; + conv.c[MSW_POS+LSB_POS] = p[1]; + conv.c[LSW_POS+MSB_POS] = p[2]; + conv.l <<= startbit; + conv.l >>= (32-bitlen); + break; + case 3: + conv.c[MSW_POS+MSB_POS] = p[0]; + conv.c[MSW_POS+LSB_POS] = p[1]; + conv.c[LSW_POS+MSB_POS] = p[2]; + conv.c[LSW_POS+LSB_POS] = p[3]; + conv.l <<= startbit; + conv.l >>= (32-bitlen); + break; + default: + conv.c[MSW_POS+MSB_POS] = p[0]; + conv.c[MSW_POS+LSB_POS] = p[1]; + conv.c[LSW_POS+MSB_POS] = p[2]; + conv.c[LSW_POS+LSB_POS] = p[3]; + conv.l <<= startbit; + conv.l >>= (32-bitlen); + conv.c[LSW_POS+LSB_POS] |= (p[4] >> (8-startbit)); + break; + } + *value = conv.l & ccd_bitfun_mask[bitlen]; + return ccdOK; +} +#endif /* !RUN_FLASH */ + +#ifndef RUN_FLASH +/* ++--------------------------------------------------------------------+ +| PROJECT : CCD (6144) MODULE : CCD | +| STATE : code ROUTINE : ccd_bitcopy | ++--------------------------------------------------------------------+ + + PURPOSE: copys bitlen bits from the source buffer to the destination + buffer. offset contains the position of the first bit in + the source bitfield. This function may perform a leftshift + to adjust the most significant bit on byte boundarys of the + first byte in dest. + +*/ + +void CCDDATA_PREF(ccd_bitcopy) (UBYTE *dest, + UBYTE *source, + USHORT bitlen, + USHORT offset) +{ + union + { /* Conversion structure 2 Byte <-> unsigned short */ + UBYTE c[2]; + USHORT s; + } + conv; + + USHORT l_shift = offset & 7; + USHORT bytes_to_copy = (bitlen >> 3); + + /* + * go to the byte that contains the first valid bit. + */ + source += (offset >> 3); + + + if (l_shift) + { + /* + * shift and copy each byte + */ + while (bytes_to_copy--) + { + conv.c[MSB_POS] = *source; + source++; + conv.c[LSB_POS] = *source; + conv.s <<= l_shift; + *dest = conv.c[MSB_POS]; + dest++; + } + } + else + { + /* + * no shift operation, do a memcopy; + */ + while (bytes_to_copy--) + { + *dest = *source; + dest++; + source++; + } + } + /* + * cutoff the garbage at the end of the bitstream + */ + + *(dest-1) &= (UCHAR)(~ccd_bitfun_mask[(8-(bitlen&7))&7]); +} +#endif /* !RUN_FLASH */