Checking in the TTS code from SVOX.
The wrapper for this will come in a later CL.
Note that this does not have any makefiles
so it should not affect the build.
diff --git a/lib/picoacph.c b/lib/picoacph.c
new file mode 100644
index 0000000..98feb6e
--- /dev/null
+++ b/lib/picoacph.c
@@ -0,0 +1,1430 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picoacph.c
+ *
+ * accentuation and phrasing
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#include "picoos.h"
+#include "picodbg.h"
+#include "picobase.h"
+#include "picodata.h"
+#include "picoacph.h"
+#include "picokdt.h"
+#include "picoklex.h"
+#include "picoktab.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+/* PU acphStep states */
+#define SA_STEPSTATE_COLLECT 0
+#define SA_STEPSTATE_PROCESS_PHR 12
+#define SA_STEPSTATE_PROCESS_ACC 13
+#define SA_STEPSTATE_FEED 2
+
+
+/* boundary strength state */
+#define SA_BOUNDSTRENGTH_SSEP 0 /* sentence separator */
+#define SA_BOUNDSTRENGTH_PPHR 1 /* primary phrase separator */
+
+
+/* subobject : AccPhrUnit
+ * shortcut : acph
+ * context size : one phrase, max. 30 non-PUNC items, for non-processed items
+ * one item if internal input empty
+ */
+
+/**
+ * @addtogroup picoacph
+ *
+ * <b> Pico Accentuation and Phrasing </b>\n
+ *
+ internal buffers:
+
+ - headx : array for extended item heads of fixed size (head plus
+ index for content, plus two fields for boundary strength/type)
+ - cbuf : buffer for item contents (referenced by index in
+ headx).
+
+ 0. bottom up filling of items in headx and cbuf
+
+ 1. phrasing (right-to-left):
+
+ e.g. from WP WP WP WP WP PUNC WP WP PUNC WP WP WP PUNC FLUSH \n
+ e.g. to BSBEG WP WP WP BPHR3 WP WP BPHR1 WP WP BSEND BSBEG WP WP WP BSEND BTERM \n
+ |1 |2 |3 |4 \n
+
+ 2-level bound state: The internal buffer contains one primary phrase (sometimes forced, if buffer
+ allmost full), with the trailing PUNCT item included (last item).\n
+ If the trailing PUNC is a a primary phrase separator, the
+ item is not output, but instead, the bound state is set to PPHR, so that the correct BOUND can
+ be output at the start of the next primary phrase.\n
+ Otherwise,
+ the item is converted to the corresponding BOUND and output. the bound state is set to SSEP,
+ so that a BOUND of type SBEG is output at the start of the next primary phrase.
+
+ trailing PUNC item bound states \n
+ SSEP PPHR \n
+ PUNC(SENTEND, X) B(B,X)>SSEP B(P1,X)>SSEP (X = T | Q | E) \n
+ PUNC(FLUSH, T) B(B,T)>SSEP* B(P1,T)>SSEP \n
+ PUNC(PHRASEEND, P) B(B,P)>PPHR B(P1,P)>PPHR \n
+ PUNC(PHRASEEND, FORC) B(B,P)>PPHR B(P1,P)>PPHR \n
+
+ If more than one sentence separators follow each other (e.g. SEND-FLUSH, SEND-SEND) then
+ all but the first will be treated as an (empty) phrase containing just this item.
+ If this (single) item is a flush, creation of SBEG is suppressed.
+
+
+ - dtphr phrasing tree ("subphrasing")
+ determines
+ - BOUND_PHR2
+ - BOUND_PHR3
+ - boundary strenghts are determined for every word (except the
+ first one) from right-to-left. The boundary types mark the phrase
+ type of the phrase following the boundary.
+ - number of items actually changed (new BOUND items added): because
+ of fixed size without content, two fields are contained in headx
+ to indicate if a BOUND needs to be added to the LEFT of the item.
+ -> headx further extended with boundary strength and type info to
+ indicate that to the left of the headx ele a BOUND needs to be
+ inserted when outputting.
+
+ 2. accentuation:
+ - number of items unchanged, content unchanged, only head info changes
+ -> changed in place in headx
+*/
+
+
+typedef struct {
+ picodata_itemhead_t head;
+ picoos_uint16 cind;
+ picoos_uint8 boundstrength; /* bstrength to the left, 0 if not set */
+ picoos_uint8 boundtype; /* btype for following phrase, 0 if not set */
+} picoacph_headx_t;
+
+
+typedef struct acph_subobj {
+ picoos_uint8 procState; /* for next processing step decision */
+ picoos_uint8 boundStrengthState; /* boundary strength state */
+
+ picoos_uint8 inspaceok; /* flag: headx/cbuf has space for an item */
+ picoos_uint8 needsmoreitems; /* flag: need more items */
+
+ picoos_uint8 tmpbuf[PICODATA_MAX_ITEMSIZE]; /* tmp. location for an item */
+
+ picoacph_headx_t headx[PICOACPH_MAXNR_HEADX];
+ picoos_uint16 headxBottom; /* bottom */
+ picoos_uint16 headxLen; /* length, 0 if empty */
+
+ picoos_uint8 cbuf[PICOACPH_MAXSIZE_CBUF];
+ picoos_uint16 cbufBufSize; /* actually allocated size */
+ picoos_uint16 cbufLen; /* length, 0 if empty */
+
+ /* tab knowledge base */
+ picoktab_Phones tabphones;
+
+ /* dtphr knowledge base */
+ picokdt_DtPHR dtphr;
+
+ /* dtacc knowledge base */
+ picokdt_DtACC dtacc;
+} acph_subobj_t;
+
+
+static pico_status_t acphInitialize(register picodata_ProcessingUnit this) {
+ acph_subobj_t * acph;
+ picoos_uint16 i;
+
+ PICODBG_DEBUG(("calling"));
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(this->common->em,
+ PICO_ERR_NULLPTR_ACCESS, NULL, NULL);
+ }
+ acph = (acph_subobj_t *) this->subObj;
+ acph->procState = SA_STEPSTATE_COLLECT;
+ acph->boundStrengthState = SA_BOUNDSTRENGTH_SSEP;
+
+ acph->inspaceok = TRUE;
+ acph->needsmoreitems = TRUE;
+
+ acph->headxBottom = 0;
+ acph->headxLen = 0;
+ acph->cbufBufSize = PICOACPH_MAXSIZE_CBUF;
+ acph->cbufLen = 0;
+
+ /* init headx, cbuf */
+ for (i = 0; i < PICOACPH_MAXNR_HEADX; i++){
+ acph->headx[i].head.type = 0;
+ acph->headx[i].head.info1 = 0;
+ acph->headx[i].head.info2 = 0;
+ acph->headx[i].head.len = 0;
+ acph->headx[i].cind = 0;
+ acph->headx[i].boundstrength = 0;
+ acph->headx[i].boundtype = 0;
+ }
+ for (i = 0; i < PICOACPH_MAXSIZE_CBUF; i++) {
+ acph->cbuf[i] = 0;
+ }
+
+ /* kb tabphones */
+ acph->tabphones =
+ picoktab_getPhones(this->voice->kbArray[PICOKNOW_KBID_TAB_PHONES]);
+ if (acph->tabphones == NULL) {
+ return picoos_emRaiseException(this->common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ PICODBG_DEBUG(("got tabphones"));
+
+#ifdef PICO_DEBUG_1
+ {
+ picoos_uint16 itmp;
+ for (itmp = 0; itmp < 256; itmp++) {
+ if (picoktab_hasVowelProp(acph->tabphones, itmp)) {
+ PICODBG_DEBUG(("tabphones hasVowel: %d", itmp));
+ }
+ if (picoktab_hasDiphthProp(acph->tabphones, itmp)) {
+ PICODBG_DEBUG(("tabphones hasDiphth: %d", itmp));
+ }
+ if (picoktab_hasGlottProp(acph->tabphones, itmp)) {
+ PICODBG_DEBUG(("tabphones hasGlott: %d", itmp));
+ }
+ if (picoktab_hasNonsyllvowelProp(acph->tabphones, itmp)) {
+ PICODBG_DEBUG(("tabphones hasNonsyllvowel: %d", itmp));
+ }
+ if (picoktab_hasSyllconsProp(acph->tabphones, itmp)) {
+ PICODBG_DEBUG(("tabphones hasSyllcons: %d", itmp));
+ }
+
+ if (picoktab_isPrimstress(acph->tabphones, itmp)) {
+ PICODBG_DEBUG(("tabphones isPrimstress: %d", itmp));
+ }
+ if (picoktab_isSecstress(acph->tabphones, itmp)) {
+ PICODBG_DEBUG(("tabphones isSecstress: %d", itmp));
+ }
+ if (picoktab_isSyllbound(acph->tabphones, itmp)) {
+ PICODBG_DEBUG(("tabphones isSyllbound: %d", itmp));
+ }
+ if (picoktab_isPause(acph->tabphones, itmp)) {
+ PICODBG_DEBUG(("tabphones isPause: %d", itmp));
+ }
+ }
+
+ PICODBG_DEBUG(("tabphones primstressID: %d",
+ picoktab_getPrimstressID(acph->tabphones)));
+ PICODBG_DEBUG(("tabphones secstressID: %d",
+ picoktab_getSecstressID(acph->tabphones)));
+ PICODBG_DEBUG(("tabphones syllboundID: %d",
+ picoktab_getSyllboundID(acph->tabphones)));
+ PICODBG_DEBUG(("tabphones pauseID: %d",
+ picoktab_getPauseID(acph->tabphones)));
+ }
+#endif
+
+
+ /* kb dtphr */
+ acph->dtphr = picokdt_getDtPHR(this->voice->kbArray[PICOKNOW_KBID_DT_PHR]);
+ if (acph->dtphr == NULL) {
+ return picoos_emRaiseException(this->common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ PICODBG_DEBUG(("got dtphr"));
+
+ /* kb dtacc */
+ acph->dtacc = picokdt_getDtACC(this->voice->kbArray[PICOKNOW_KBID_DT_ACC]);
+ if (acph->dtacc == NULL) {
+ return picoos_emRaiseException(this->common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ PICODBG_DEBUG(("got dtacc"));
+
+ return PICO_OK;
+}
+
+static picodata_step_result_t acphStep(register picodata_ProcessingUnit this,
+ picoos_int16 mode,
+ picoos_uint16 *numBytesOutput);
+
+static pico_status_t acphTerminate(register picodata_ProcessingUnit this)
+{
+ return PICO_OK;
+}
+
+static pico_status_t acphSubObjDeallocate(register picodata_ProcessingUnit this,
+ picoos_MemoryManager mm) {
+ mm = mm; /* avoid warning "var not used in this function"*/
+ if (NULL != this) {
+ picoos_deallocate(this->common->mm, (void *) &this->subObj);
+ }
+ return PICO_OK;
+}
+
+
+picodata_ProcessingUnit picoacph_newAccPhrUnit(picoos_MemoryManager mm,
+ picoos_Common common,
+ picodata_CharBuffer cbIn,
+ picodata_CharBuffer cbOut,
+ picorsrc_Voice voice) {
+ picodata_ProcessingUnit this;
+
+ this = picodata_newProcessingUnit(mm, common, cbIn, cbOut, voice);
+ if (this == NULL) {
+ return NULL;
+ }
+
+ this->initialize = acphInitialize;
+ PICODBG_DEBUG(("set this->step to acphStep"));
+ this->step = acphStep;
+ this->terminate = acphTerminate;
+ this->subDeallocate = acphSubObjDeallocate;
+ this->subObj = picoos_allocate(mm, sizeof(acph_subobj_t));
+ if (this->subObj == NULL) {
+ picoos_deallocate(mm, (void *)&this);
+ picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM, NULL, NULL);
+ return NULL;
+ }
+
+ acphInitialize(this);
+ return this;
+}
+
+
+/* ***********************************************************************/
+/* PROCESS_PHR/ACC support functions */
+/* ***********************************************************************/
+
+
+static picoos_uint8 acphGetNrSylls(register picodata_ProcessingUnit this,
+ register acph_subobj_t *acph,
+ const picoos_uint16 ind) {
+ picoos_uint8 i;
+ picoos_uint8 ch;
+ picoos_uint8 count;
+
+ count = 1;
+ for (i = 0; i < acph->headx[ind].head.len; i++) {
+ ch = acph->cbuf[acph->headx[ind].cind + i];
+ if (picoktab_isSyllbound(acph->tabphones, ch)) {
+ count++;
+ }
+ }
+ return count;
+}
+
+
+/* ***********************************************************************/
+/* PROCESS_PHR functions */
+/* ***********************************************************************/
+
+
+/* find next POS to the left of 'ind' and return its POS and index */
+static picoos_uint8 acphPhrItemSeqGetPosLeft(register picodata_ProcessingUnit this,
+ register acph_subobj_t *acph,
+ const picoos_uint16 ind,
+ picoos_uint16 *leftind) {
+ picoos_uint8 val;
+ picoos_int32 i;
+
+ val = PICOKDT_EPSILON;
+ for (i = ind - 1; ((val == PICOKDT_EPSILON) && (i >= 0)); i--) {
+ if ((acph->headx[i].head.type == PICODATA_ITEM_WORDPHON)) {
+ val = acph->headx[i].head.info1;
+ }
+ }
+ *leftind = i + 1;
+ return val;
+}
+
+
+/* right-to-left, for each WORDPHON do phr */
+static pico_status_t acphSubPhrasing(register picodata_ProcessingUnit this,
+ register acph_subobj_t *acph) {
+ picokdt_classify_result_t dtres;
+ picoos_uint8 valbuf[5];
+ picoos_uint16 nrwordspre;
+ picoos_uint16 nrwordsfol;
+ picoos_uint16 nrsyllsfol;
+ picoos_uint16 lastprev2; /* last index of POS(es) found to the left */
+ picoos_uint8 curpos; /* POS(es) of current word */
+ picoos_uint16 upbound; /* index of last WORDPHON item (with POS) */
+ picoos_uint8 okay;
+ picoos_uint8 nosubphrases;
+ picoos_int32 i;
+
+ /* set initial values */
+ okay = TRUE;
+ nosubphrases = TRUE;
+ curpos = PICOKDT_EPSILON; /* needs to be in 2^8 */
+
+ /* set upbound to last WORDPHON, don't worry about first one */
+ upbound = acph->headxLen - 1;
+ while ((upbound > 0) &&
+ (acph->headx[upbound].head.type != PICODATA_ITEM_WORDPHON)) {
+ upbound--;
+ }
+
+ /* zero or one WORDPHON, no subphrasing needed, but handling of
+ BOUND strength state is needed */
+ if (upbound <= 0) {
+ /* phrase not containing more than one WORDPHON */
+ PICODBG_DEBUG(("less than two WORDPHON in phrase -> no subphrasing"));
+ }
+
+ lastprev2 = upbound;
+
+ /* set initial nr pre/fol words/sylls, upbound is ind of last WORDPHON */
+ nrwordsfol = 0;
+ nrsyllsfol = 0;
+ nrwordspre = 0;
+ for (i = 0; i < upbound; i++) {
+ if (acph->headx[i].head.type == PICODATA_ITEM_WORDPHON) {
+ nrwordspre++;
+ }
+ }
+
+ nrwordspre++; /* because we later have a decrement before being used */
+
+
+ /* set POS of current word in valbuf[1], will be shifted right afterwards */
+ valbuf[1] = acph->headx[upbound].head.info1;
+ /* find first POS to the left and set valbuf[0] */
+ valbuf[0] = acphPhrItemSeqGetPosLeft(this, acph, lastprev2, &lastprev2);
+ for (i = 2; i < 5; i++) {
+ valbuf[i] = PICOKDT_EPSILON;
+ }
+
+ PICODBG_TRACE(("headxLen: %d", acph->headxLen));
+
+ /* at least two WORDPHON items */
+ /* process from right-to-left all items in headx, except for 1st WORDPHON */
+ for (i = upbound; (i > 0) && (nrwordspre > 1); i--) {
+ okay = TRUE;
+
+ PICODBG_TRACE(("iter: %d, type: %c", i, acph->headx[i].head.type));
+
+ /* if not (WORDPHON) */
+ if ((acph->headx[i].head.type != PICODATA_ITEM_WORDPHON)) {
+ continue;
+ }
+
+ PICODBG_TRACE(("iter: %d, curpos: %d", i, acph->headx[i].head.info1));
+
+ /* get and set POS of current item, must be WORDPHON */
+ curpos = acph->headx[i].head.info1;
+
+ /* no continue so far => at [i] we have a WORDPHON item */
+ /* shift all POS elements one position to the right */
+ valbuf[4] = valbuf[3];
+ valbuf[3] = valbuf[2];
+ valbuf[2] = valbuf[1];
+ valbuf[1] = valbuf[0];
+ /* find next POS to the left and set valbuf[0] */
+ valbuf[0] = acphPhrItemSeqGetPosLeft(this, acph, lastprev2, &lastprev2);
+
+ /* better check double than never */
+ if (curpos != valbuf[2]) {
+ PICODBG_WARN(("syncing POS"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_INVECTOR,
+ NULL, NULL);
+ valbuf[2] = curpos;
+ }
+
+ nrwordsfol++;
+ nrsyllsfol += acphGetNrSylls(this, acph, i);
+ nrwordspre--;
+
+ PICODBG_TRACE(("%d: [%d,%d|%d|%d,%d|%d,%d,%d]",
+ i, valbuf[0], valbuf[1], valbuf[2], valbuf[3],
+ valbuf[4], nrwordspre, nrwordsfol, nrsyllsfol));
+
+ /* no continue so far => subphrasing needed */
+ /* construct input vector, which is set in dtphr */
+ if (!picokdt_dtPHRconstructInVec(acph->dtphr, valbuf[0], valbuf[1],
+ valbuf[2], valbuf[3], valbuf[4],
+ nrwordspre, nrwordsfol, nrsyllsfol)) {
+ /* error constructing invec */
+ PICODBG_WARN(("problem with invec"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_INVECTOR,
+ NULL, NULL);
+ okay = FALSE;
+ }
+ /* classify */
+ if (okay && (!picokdt_dtPHRclassify(acph->dtphr))) {
+ /* error doing classification */
+ PICODBG_WARN(("problem classifying"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_CLASSIFICATION,
+ NULL, NULL);
+ okay = FALSE;
+ }
+ /* decompose */
+ if (okay && (!picokdt_dtPHRdecomposeOutClass(acph->dtphr, &dtres))) {
+ /* error decomposing */
+ PICODBG_WARN(("problem decomposing"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_OUTVECTOR,
+ NULL, NULL);
+ okay = FALSE;
+ }
+
+ if (okay && dtres.set) {
+ PICODBG_DEBUG(("%d - inpos: %d, out: %d", i,valbuf[2],dtres.class));
+ } else {
+ PICODBG_WARN(("problem determining subphrase boundary strength"));
+ dtres.class = PICODATA_ITEMINFO1_ERR;
+ }
+
+ if (dtres.class > 255) {
+ PICODBG_WARN(("dt class outside valid range, setting to PHR0"));
+ dtres.class = PICODATA_ITEMINFO1_BOUND_PHR0;
+ }
+ acph->headx[i].boundstrength = (picoos_uint8)dtres.class;
+ if ((dtres.class == PICODATA_ITEMINFO1_BOUND_PHR2) ||
+ (dtres.class == PICODATA_ITEMINFO1_BOUND_PHR3)) {
+ if (nosubphrases) {
+ /* it's the last secondary phrase in the primary phrase */
+ /* add type info */
+ switch (acph->headx[acph->headxLen - 1].head.info2) {
+ case PICODATA_ITEMINFO2_PUNC_SENT_T:
+ acph->headx[i].boundtype =
+ PICODATA_ITEMINFO2_BOUNDTYPE_T;
+ break;
+ case PICODATA_ITEMINFO2_PUNC_SENT_Q:
+ acph->headx[i].boundtype =
+ PICODATA_ITEMINFO2_BOUNDTYPE_Q;
+ break;
+ case PICODATA_ITEMINFO2_PUNC_SENT_E:
+ acph->headx[i].boundtype =
+ PICODATA_ITEMINFO2_BOUNDTYPE_E;
+ break;
+ case PICODATA_ITEMINFO2_PUNC_PHRASE:
+ case PICODATA_ITEMINFO2_PUNC_PHRASE_FORCED:
+ acph->headx[i].boundtype =
+ PICODATA_ITEMINFO2_BOUNDTYPE_P;
+ break;
+ default:
+ PICODBG_WARN(("invalid boundary type, not set"));
+ break;
+ }
+ nosubphrases = FALSE;
+
+ } else {
+ acph->headx[i].boundtype =
+ PICODATA_ITEMINFO2_BOUNDTYPE_P;
+ }
+ /* reset nr following words and sylls counters */
+ nrwordsfol = 0;
+ nrsyllsfol = 0;
+ }
+ }
+
+ /* process first item, add bound-info */
+ switch (acph->boundStrengthState) {
+ case SA_BOUNDSTRENGTH_SSEP:
+ acph->headx[0].boundstrength =
+ PICODATA_ITEMINFO1_BOUND_SBEG;
+ break;
+ case SA_BOUNDSTRENGTH_PPHR:
+ acph->headx[0].boundstrength =
+ PICODATA_ITEMINFO1_BOUND_PHR1;
+ break;
+ default:
+ PICODBG_WARN(("invalid boundary strength, not set"));
+ break;
+ }
+
+ /* set boundary strength state */
+ switch (acph->headx[acph->headxLen - 1].head.info1) {
+ case PICODATA_ITEMINFO1_PUNC_SENTEND:
+ case PICODATA_ITEMINFO1_PUNC_FLUSH:
+ acph->boundStrengthState = SA_BOUNDSTRENGTH_SSEP;
+ break;
+ case PICODATA_ITEMINFO1_PUNC_PHRASEEND:
+ acph->boundStrengthState = SA_BOUNDSTRENGTH_PPHR;
+ break;
+ default:
+ PICODBG_WARN(("invalid boundary strength state, not changed"));
+ break;
+ }
+
+ if (nosubphrases) {
+ /* process first item, add type info */
+ switch (acph->headx[acph->headxLen - 1].head.info2) {
+ case PICODATA_ITEMINFO2_PUNC_SENT_T:
+ acph->headx[0].boundtype =
+ PICODATA_ITEMINFO2_BOUNDTYPE_T;
+ break;
+ case PICODATA_ITEMINFO2_PUNC_SENT_Q:
+ acph->headx[0].boundtype =
+ PICODATA_ITEMINFO2_BOUNDTYPE_Q;
+ break;
+ case PICODATA_ITEMINFO2_PUNC_SENT_E:
+ acph->headx[0].boundtype =
+ PICODATA_ITEMINFO2_BOUNDTYPE_E;
+ break;
+ case PICODATA_ITEMINFO2_PUNC_PHRASE:
+ case PICODATA_ITEMINFO2_PUNC_PHRASE_FORCED:
+ acph->headx[0].boundtype =
+ PICODATA_ITEMINFO2_BOUNDTYPE_P;
+ break;
+ default:
+ PICODBG_WARN(("invalid boundary type, not set"));
+ break;
+ }
+ } else {
+ acph->headx[0].boundtype =
+ PICODATA_ITEMINFO2_BOUNDTYPE_P;
+ }
+
+ return PICO_OK;
+}
+
+
+/* ***********************************************************************/
+/* PROCESS_ACC functions */
+/* ***********************************************************************/
+
+/* find next POS to the left of 'ind' and return its POS and index */
+static picoos_uint8 acphAccItemSeqGetPosLeft(register picodata_ProcessingUnit this,
+ register acph_subobj_t *acph,
+ const picoos_uint16 ind,
+ picoos_uint16 *leftind) {
+ picoos_uint8 val;
+ picoos_int32 i;
+
+ val = PICOKDT_EPSILON;
+ for (i = ind - 1; ((val == PICOKDT_EPSILON) && (i >= 0)); i--) {
+ if ((acph->headx[i].head.type == PICODATA_ITEM_WORDPHON)) {
+ val = acph->headx[i].head.info1;
+ }
+ }
+ *leftind = i + 1;
+ return val;
+}
+
+
+/* s1: nr sylls in word before the first primary stressed syll,
+ s2: nr sylls in word after (but excluding) the first primary stressed syll */
+static picoos_uint8 acphAccNrSyllParts(register picodata_ProcessingUnit this,
+ register acph_subobj_t *acph,
+ const picoos_uint16 ind,
+ picoos_uint8 *s1,
+ picoos_uint8 *s2) {
+ picoos_uint16 pind;
+ picoos_uint16 pend; /* phone string start+len */
+ picoos_uint8 afterprim;
+
+ /* check ind is in valid range */
+ if (ind >= acph->headxLen) {
+ return FALSE;
+ }
+
+ *s1 = 0;
+ *s2 = 0;
+ afterprim = FALSE;
+ pend = acph->headx[ind].cind + acph->headx[ind].head.len;
+ for (pind = acph->headx[ind].cind; pind < pend; pind++) {
+ if (picoktab_isPrimstress(acph->tabphones, acph->cbuf[pind])) {
+ afterprim = TRUE;
+ } else if (picoktab_isSyllbound(acph->tabphones, acph->cbuf[pind])) {
+ if (afterprim) {
+ (*s2)++;
+ } else {
+ (*s1)++;
+ }
+ }
+ }
+ if (afterprim) {
+ (*s2)++;
+ } else {
+ (*s1)++;
+ }
+
+ /* exclude the stressed syllable */
+ if ((*s2) > 0) {
+ (*s2)--;
+ }
+ /* handle the case when there is no primstress */
+ if (!afterprim) {
+ (*s2) = (*s1);
+ }
+ return TRUE;
+}
+
+
+static picoos_uint8 acphAccGetNrsRight(register picodata_ProcessingUnit this,
+ register acph_subobj_t *acph,
+ const picoos_uint16 ind,
+ picoos_uint16 *nrwordsfol,
+ picoos_uint16 *nrsyllsfol,
+ picoos_uint16 *footwordsfol,
+ picoos_uint16 *footsyllsfol) {
+ picoos_uint16 i;
+ picoos_uint8 s1;
+ picoos_uint8 s2;
+
+ if (!acphAccNrSyllParts(this, acph, ind, &s1, &s2)) {
+ return FALSE;
+ }
+
+ *nrwordsfol = 0;
+ *nrsyllsfol = s2;
+ i = ind + 1;
+ while ((i < acph->headxLen) &&
+ (acph->headx[i].boundstrength == PICODATA_ITEMINFO1_BOUND_PHR0)) {
+ if (acph->headx[i].head.type == PICODATA_ITEM_WORDPHON) {
+ (*nrwordsfol)++;
+ *nrsyllsfol += acphGetNrSylls(this, acph, i);
+ }
+ i++;
+ }
+
+ *footwordsfol = 0;
+ *footsyllsfol = s2;
+ i = ind + 1;
+ while ((i < acph->headxLen) &&
+ (acph->headx[i].head.info2 != PICODATA_ACC1)) {
+ if (acph->headx[i].head.type == PICODATA_ITEM_WORDPHON) {
+ (*footwordsfol)++;
+ *footsyllsfol += acphGetNrSylls(this, acph, i);
+ }
+ i++;
+ }
+ if ((i < acph->headxLen) && (acph->headx[i].head.info2 == PICODATA_ACC1)) {
+ if (!acphAccNrSyllParts(this, acph, i, &s1, &s2)) {
+ return FALSE;
+ }
+ *footsyllsfol += s1;
+ }
+ return TRUE;
+}
+
+
+static picoos_uint8 acphAccGetNrsLeft(register picodata_ProcessingUnit this,
+ register acph_subobj_t *acph,
+ const picoos_uint16 ind,
+ picoos_uint16 *nrwordspre,
+ picoos_uint16 *nrsyllspre) {
+ picoos_int32 i;
+ picoos_uint8 s1;
+ picoos_uint8 s2;
+
+ if (!acphAccNrSyllParts(this, acph, ind, &s1, &s2)) {
+ return FALSE;
+ }
+
+ *nrwordspre = 0;
+ *nrsyllspre = s1;
+ i = ind - 1;
+ while ((i >= 0) &&
+ (acph->headx[i].boundstrength == PICODATA_ITEMINFO1_BOUND_PHR0)) {
+ if (acph->headx[i].head.type == PICODATA_ITEM_WORDPHON) {
+ (*nrwordspre)++;
+ *nrsyllspre += acphGetNrSylls(this, acph, i);
+ }
+ i--;
+ }
+
+ if ((acph->headx[i].boundstrength != PICODATA_ITEMINFO1_BOUND_PHR0) &&
+ (acph->headx[i].head.type == PICODATA_ITEM_WORDPHON)) {
+ (*nrwordspre)++;
+ *nrsyllspre += acphGetNrSylls(this, acph, i);
+ }
+ return TRUE;
+}
+
+
+/* return TRUE if wordphon contains no stress, FALSE otherwise */
+static picoos_uint8 acphIsWordWithoutStress(register picodata_ProcessingUnit this,
+ register acph_subobj_t *acph,
+ const picoos_uint16 ind) {
+ picoos_uint8 i;
+ picoos_uint16 pos;
+
+ pos = acph->headx[ind].cind;
+ for (i = 0; i < acph->headx[ind].head.len; i++) {
+ if (picoktab_isPrimstress(acph->tabphones, acph->cbuf[pos + i]) ||
+ picoktab_isSecstress(acph->tabphones, acph->cbuf[pos + i])) {
+ return FALSE;
+ }
+ }
+ return TRUE;
+}
+
+
+/* right-to-left, for each WORDPHON do acc */
+static pico_status_t acphAccentuation(register picodata_ProcessingUnit this,
+ register acph_subobj_t *acph) {
+ picokdt_classify_result_t dtres;
+ picoos_uint8 valbuf[5];
+ picoos_uint16 hist1;
+ picoos_uint16 hist2;
+ picoos_uint16 nrwordspre;
+ picoos_uint16 nrsyllspre;
+ picoos_uint16 nrwordsfol;
+ picoos_uint16 nrsyllsfol;
+ picoos_uint16 footwordsfol;
+ picoos_uint16 footsyllsfol;
+ picoos_uint16 lastprev2; /* last index of POS(es) found to the left */
+ picoos_uint8 curpos; /* POS(es) of current word */
+ picoos_uint16 prevout;
+ picoos_uint8 okay;
+ picoos_int32 upbound; /* index of last WORDPHON item (with POS) */
+ picoos_uint16 i;
+
+ /* set initial values */
+ okay = TRUE;
+ curpos = PICOKDT_EPSILON; /* needs to be < 2^8 */
+
+ /* set upbound to last WORDPHON */
+ upbound = acph->headxLen - 1;
+ while ((upbound >= 0) &&
+ (acph->headx[upbound].head.type != PICODATA_ITEM_WORDPHON)) {
+ upbound--;
+ }
+
+ if (upbound < 0) {
+ /* phrase containing zero WORDPHON */
+ PICODBG_DEBUG(("no WORDPHON in phrase -> no accentuation"));
+ return PICO_OK;
+ }
+
+ lastprev2 = upbound;
+
+ /* set initial history values */
+ prevout = PICOKDT_HISTORY_ZERO;
+ hist1 = PICOKDT_HISTORY_ZERO;
+ hist2 = PICOKDT_HISTORY_ZERO;
+
+ /* set initial nr pre/fol words/sylls, upbound is ind of last WORDPHON */
+ nrwordsfol = 0;
+ nrsyllsfol = 0;
+ footwordsfol = 0;
+ footsyllsfol = 0;
+ nrwordspre = 0;
+ nrsyllspre = 0;
+
+ /* set POS of current word in valbuf[1], will be shifted right afterwards */
+ valbuf[1] = acph->headx[upbound].head.info1;
+ /* find first POS to the left and set valbuf[0] */
+ valbuf[0] = acphAccItemSeqGetPosLeft(this, acph, lastprev2, &lastprev2);
+ for (i = 2; i < 5; i++) {
+ valbuf[i] = PICOKDT_EPSILON;
+ }
+
+ PICODBG_TRACE(("headxLen: %d", acph->headxLen));
+
+ /* process from right-to-left all items in headx */
+ for (i = upbound+1; i > 0; ) {
+ i--;
+
+ okay = TRUE;
+
+ PICODBG_TRACE(("iter: %d, type: %c", i, acph->headx[i].head.type));
+
+ /* if not (WORDPHON) */
+ if ((acph->headx[i].head.type != PICODATA_ITEM_WORDPHON)) {
+ continue;
+ }
+
+ PICODBG_TRACE(("iter: %d, curpos: %d", i, acph->headx[i].head.info1));
+
+ /* get and set POS of current item, must be WORDPHON */
+ curpos = acph->headx[i].head.info1;
+
+ /* no continue so far => at [i] we have a WORDPHON item */
+ /* shift all POS elements one position to the right */
+ valbuf[4] = valbuf[3];
+ valbuf[3] = valbuf[2];
+ valbuf[2] = valbuf[1];
+ valbuf[1] = valbuf[0];
+ /* find next POS to the left and set valbuf[0] */
+ valbuf[0] = acphAccItemSeqGetPosLeft(this, acph, lastprev2, &lastprev2);
+
+ /* better check double than never */
+ if (curpos != valbuf[2]) {
+ PICODBG_WARN(("syncing POS"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_INVECTOR,
+ NULL, NULL);
+ valbuf[2] = curpos;
+ }
+
+ /* set history values */
+ hist2 = hist1;
+ hist1 = prevout;
+
+ /* ************************************************************ */
+ /* many speedups possible by avoiding double calc of attribtues */
+ /* ************************************************************ */
+
+ /* get distances */
+ if ((!acphAccGetNrsRight(this, acph, i, &nrwordsfol, &nrsyllsfol,
+ &footwordsfol, &footsyllsfol)) ||
+ (!acphAccGetNrsLeft(this, acph, i, &nrwordspre, &nrsyllspre))) {
+ PICODBG_WARN(("problem setting distances in invec"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_INVECTOR,
+ NULL, NULL);
+ okay = FALSE;
+ }
+
+ PICODBG_TRACE(("%d: [%d,%d,%d,%d,%d|%d,%d|%d,%d,%d,%d|%d,%d]", i,
+ valbuf[0], valbuf[1], valbuf[2], valbuf[3], valbuf[4],
+ hist1, hist2, nrwordspre, nrsyllspre,
+ nrwordsfol, nrsyllsfol, footwordsfol, footsyllsfol));
+
+ /* no continue so far => accentuation needed */
+ /* construct input vector, which is set in dtacc */
+ if (!picokdt_dtACCconstructInVec(acph->dtacc, valbuf[0], valbuf[1],
+ valbuf[2], valbuf[3], valbuf[4],
+ hist1, hist2, nrwordspre, nrsyllspre,
+ nrwordsfol, nrsyllsfol, footwordsfol,
+ footsyllsfol)) {
+ /* error constructing invec */
+ PICODBG_WARN(("problem with invec"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_INVECTOR,
+ NULL, NULL);
+ okay = FALSE;
+ }
+ /* classify */
+ if (okay && (!picokdt_dtACCclassify(acph->dtacc, &prevout))) {
+ /* error doing classification */
+ PICODBG_WARN(("problem classifying"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_CLASSIFICATION,
+ NULL, NULL);
+ okay = FALSE;
+ }
+ /* decompose */
+ if (okay && (!picokdt_dtACCdecomposeOutClass(acph->dtacc, &dtres))) {
+ /* error decomposing */
+ PICODBG_WARN(("problem decomposing"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_OUTVECTOR,
+ NULL, NULL);
+ okay = FALSE;
+ }
+
+ if (dtres.class > 255) {
+ PICODBG_WARN(("dt class outside valid range, setting to ACC0"));
+ dtres.class = PICODATA_ACC0;
+ }
+
+ if (okay && dtres.set) {
+ PICODBG_DEBUG(("%d - inpos: %d, out: %d", i,valbuf[2],dtres.class));
+ if (acphIsWordWithoutStress(this, acph, i)) {
+ if (dtres.class != PICODATA_ACC0) {
+ acph->headx[i].head.info2 = PICODATA_ACC3;
+ } else {
+ acph->headx[i].head.info2 = (picoos_uint8)dtres.class;
+ }
+ } else {
+ acph->headx[i].head.info2 = (picoos_uint8)dtres.class;
+ }
+ PICODBG_DEBUG(("%d - after-nostress-corr: %d",
+ i, acph->headx[i].head.info2));
+ } else {
+ PICODBG_WARN(("problem determining accentuation level"));
+ dtres.class = PICODATA_ITEMINFO1_ERR;
+ }
+ }
+ return PICO_OK;
+}
+
+
+
+/* ***********************************************************************/
+/* acphStep support functions */
+/* ***********************************************************************/
+
+static picoos_uint8 acphPutBoundItem(register picodata_ProcessingUnit this,
+ register acph_subobj_t *acph,
+ const picoos_uint8 strength,
+ const picoos_uint8 type,
+ picoos_uint8 *dopuoutfull,
+ picoos_uint16 *numBytesOutput) {
+ pico_status_t rv = PICO_OK;
+ picoos_uint16 blen = 0;
+ picodata_itemhead_t tmphead;
+
+ *dopuoutfull = FALSE;
+
+ /* construct BOUND item in tmpbuf and put item */
+ tmphead.type = PICODATA_ITEM_BOUND;
+ tmphead.info1 = strength;
+ tmphead.info2 = type;
+ tmphead.len = 0;
+ rv = picodata_put_itemparts(&tmphead, NULL, 0, acph->tmpbuf,
+ PICODATA_MAX_ITEMSIZE, &blen);
+ if (rv != PICO_OK) {
+ PICODBG_ERROR(("problem creating BOUND item"));
+ picoos_emRaiseException(this->common->em, rv, NULL, NULL);
+ return FALSE;
+ }
+ /* put constructed item to ext. charbuf */
+ rv = picodata_cbPutItem(this->cbOut, acph->tmpbuf, blen, &blen);
+
+ *numBytesOutput += blen;
+ if (rv == PICO_EXC_BUF_OVERFLOW) {
+ PICODBG_DEBUG(("overflow in cb output buffer"));
+ *dopuoutfull = TRUE; /* ie. do PU_OUT_FULL later */
+ return FALSE;
+ } else if (rv != PICO_OK) {
+ PICODBG_ERROR(("problem putting BOUND item"));
+ picoos_emRaiseException(this->common->em, rv, NULL, NULL);
+ return FALSE;
+ }
+
+ PICODATA_INFO_ITEM(this->voice->kbArray[PICOKNOW_KBID_DBG],
+ (picoos_uint8 *)"acph: ", acph->tmpbuf, blen);
+
+ return TRUE;
+}
+
+
+
+/* ***********************************************************************/
+/* acphStep function */
+/* ***********************************************************************/
+
+/*
+complete phrase processed in one step, if not fast enough -> rework
+
+init, collect into internal buffer, process, and then feed to
+output buffer
+
+init state: INIT ext ext
+state trans: in hc1 hc2 out
+
+INIT | putItem = 0 0 +1 | BUSY -> COLL (put B-SBEG item,
+ set do-init to false)
+
+ inspace-ok-hc1
+ needs-more-items-(phrase-or-flush)
+COLL1 |getItems -n +n 0 1 | ATOMIC -> PPOSD (got items,
+ if flush set do-init)
+COLL2 |getItems -n +n 1 0 | ATOMIC -> PPOSD (got items, forced)
+COLL3 |getItems -n +n 1 1 | IDLE (got items, need more)
+COLL4 |getItems = = 1 1 | IDLE (got no items)
+
+PPOSD | posd = ~n~n | BUSY -> PWP (posd done)
+PWP | lex/g2p = ~n-n 0+n | BUSY -> PPHR (lex/g2p done)
+PPHR | phr = -n 0 +m=n | BUSY -> PACC (phr done, m>=n)
+PACC | acc = 0 0 ~m=n | BUSY -> FEED (acc done)
+
+ doinit-flag
+FEED | putItems 0 0 0 -m-n +m 0 | BUSY -> COLL (put items)
+FEED | putItems 0 0 0 -m-n +m 1 | BUSY -> INIT (put items)
+FEED | putItems 0 0 0 -d-d +d | OUT_FULL (put some items)
+*/
+
+static picodata_step_result_t acphStep(register picodata_ProcessingUnit this,
+ picoos_int16 mode,
+ picoos_uint16 *numBytesOutput) {
+ register acph_subobj_t *acph;
+ pico_status_t rv = PICO_OK;
+ pico_status_t rvP = PICO_OK;
+ picoos_uint16 blen = 0;
+ picoos_uint16 clen = 0;
+ picoos_uint16 i;
+
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ acph = (acph_subobj_t *) this->subObj;
+ mode = mode; /* avoid warning "var not used in this function"*/
+ *numBytesOutput = 0;
+ while (1) { /* exit via return */
+ PICODBG_DEBUG(("doing state %i, hLen|c1Len: %d|%d",
+ acph->procState, acph->headxLen, acph->cbufLen));
+
+ switch (acph->procState) {
+
+ /* *********************************************************/
+ /* collect state: get item(s) from charBuf and store in
+ * internal buffers, need a complete punctuation-phrase
+ */
+ case SA_STEPSTATE_COLLECT:
+
+ while (acph->inspaceok && acph->needsmoreitems && (PICO_OK ==
+ (rv = picodata_cbGetItem(this->cbIn, acph->tmpbuf,
+ PICODATA_MAX_ITEMSIZE, &blen)))) {
+ rvP = picodata_get_itemparts(acph->tmpbuf,
+ PICODATA_MAX_ITEMSIZE, &(acph->headx[acph->headxLen].head),
+ &(acph->cbuf[acph->cbufLen]), acph->cbufBufSize
+ - acph->cbufLen, &clen);
+ if (rvP != PICO_OK) {
+ PICODBG_ERROR(("problem getting item parts"));
+ picoos_emRaiseException(this->common->em, rvP,
+ NULL, NULL);
+ return PICODATA_PU_ERROR;
+ }
+
+ /* if CMD(...FLUSH...) -> PUNC(...FLUSH...),
+ construct PUNC-FLUSH item in headx */
+ if ((acph->headx[acph->headxLen].head.type
+ == PICODATA_ITEM_CMD)
+ && (acph->headx[acph->headxLen].head.info1
+ == PICODATA_ITEMINFO1_CMD_FLUSH)) {
+ acph->headx[acph->headxLen].head.type
+ = PICODATA_ITEM_PUNC;
+ acph->headx[acph->headxLen].head.info1
+ = PICODATA_ITEMINFO1_PUNC_FLUSH;
+ acph->headx[acph->headxLen].head.info2
+ = PICODATA_ITEMINFO2_PUNC_SENT_T;
+ acph->headx[acph->headxLen].head.len = 0;
+ }
+
+ /* check/set needsmoreitems */
+ if (acph->headx[acph->headxLen].head.type
+ == PICODATA_ITEM_PUNC) {
+ acph->needsmoreitems = FALSE;
+ }
+
+ /* check/set inspaceok, keep spare slot for forcing */
+ if ((acph->headxLen >= (PICOACPH_MAXNR_HEADX - 2))
+ || ((acph->cbufBufSize - acph->cbufLen)
+ < PICODATA_MAX_ITEMSIZE)) {
+ acph->inspaceok = FALSE;
+ }
+
+ if (clen > 0) {
+ acph->headx[acph->headxLen].cind = acph->cbufLen;
+ acph->cbufLen += clen;
+ } else {
+ acph->headx[acph->headxLen].cind = 0;
+ }
+ acph->headxLen++;
+ }
+
+ if (!acph->needsmoreitems) {
+ /* 1, phrase buffered */
+ acph->procState = SA_STEPSTATE_PROCESS_PHR;
+ return PICODATA_PU_ATOMIC;
+ } else if (!acph->inspaceok) {
+ /* 2, forced phrase end */
+ /* at least one slot is still free, use it to
+ force a trailing PUNC item */
+ acph->headx[acph->headxLen].head.type = PICODATA_ITEM_PUNC;
+ acph->headx[acph->headxLen].head.info1 =
+ PICODATA_ITEMINFO1_PUNC_PHRASEEND;
+ acph->headx[acph->headxLen].head.info2 =
+ PICODATA_ITEMINFO2_PUNC_PHRASE_FORCED;
+ acph->headx[acph->headxLen].head.len = 0;
+ acph->needsmoreitems = FALSE; /* not really needed for now */
+ acph->headxLen++;
+ PICODBG_WARN(("forcing phrase end, added PUNC_PHRASEEND"));
+ picoos_emRaiseWarning(this->common->em,
+ PICO_WARN_FALLBACK, NULL,
+ (picoos_char *)"forced phrase end");
+ acph->procState = SA_STEPSTATE_PROCESS_PHR;
+ return PICODATA_PU_ATOMIC;
+ } else if (rv == PICO_EOF) {
+ /* 3, 4 */
+ return PICODATA_PU_IDLE;
+ } else if ((rv == PICO_EXC_BUF_UNDERFLOW) ||
+ (rv == PICO_EXC_BUF_OVERFLOW)) {
+ /* error, no valid item in cb (UNDER) */
+ /* or tmpbuf not large enough, not possible (OVER) */
+ /* no exception raised, left for ctrl to handle */
+ PICODBG_ERROR(("buffer under/overflow, rv: %d", rv));
+ return PICODATA_PU_ERROR;
+ } else {
+ /* error, only possible if cbGetItem implementation
+ changes without this function being adapted*/
+ PICODBG_ERROR(("untreated return value, rv: %d", rv));
+ return PICODATA_PU_ERROR;
+ }
+ break;
+
+
+
+
+ /* *********************************************************/
+ /* process phr state: process items in headx and modify
+ * headx in place
+ */
+ case SA_STEPSTATE_PROCESS_PHR:
+ /* ensure there is an item in inBuf */
+ if (acph->headxLen > 0) {
+ /* we have a phrase in headx, cbuf1 (can be
+ single PUNC item), do phrasing and modify headx */
+
+ if (PICO_OK != acphSubPhrasing(this, acph)) {
+ picoos_emRaiseException(this->common->em,
+ PICO_ERR_OTHER, NULL, NULL);
+ return PICODATA_PU_ERROR;
+ }
+ acph->procState = SA_STEPSTATE_PROCESS_ACC;
+ } else if (acph->headxLen == 0) { /* no items in inBuf */
+ PICODBG_WARN(("no items in inBuf"));
+ acph->procState = SA_STEPSTATE_COLLECT;
+ return PICODATA_PU_BUSY;
+ }
+
+#if defined (PICO_DEBUG_NOTNEEDED)
+ if (1) {
+ picoos_uint8 i, j, ittype;
+ for (i = 0; i < acph->headxLen; i++) {
+ if ((acph->headx[i].boundstrength != 0) &&
+ (acph->headx[i].boundstrength !=
+ PICODATA_ITEMINFO1_BOUND_PHR0)) {
+ PICODBG_INFO(("acph-p: boundstrength '%c', "
+ "boundtype '%c'",
+ acph->headx[i].boundstrength,
+ acph->headx[i].boundtype));
+ }
+
+ ittype = acph->headx[i].head.type;
+ PICODBG_INFO_CTX();
+ PICODBG_INFO_MSG(("acph-p: ("));
+ PICODBG_INFO_MSG(("'%c',", ittype));
+ if ((32 <= acph->headx[i].head.info1) &&
+ (acph->headx[i].head.info1 < 127) &&
+ (ittype != PICODATA_ITEM_WORDPHON)) {
+ PICODBG_INFO_MSG(("'%c',",acph->headx[i].head.info1));
+ } else {
+ PICODBG_INFO_MSG(("%3d,", acph->headx[i].head.info1));
+ }
+ if ((32 <= acph->headx[i].head.info2) &&
+ (acph->headx[i].head.info2 < 127)) {
+ PICODBG_INFO_MSG(("'%c',",acph->headx[i].head.info2));
+ } else {
+ PICODBG_INFO_MSG(("%3d,", acph->headx[i].head.info2));
+ }
+ PICODBG_INFO_MSG(("%3d)", acph->headx[i].head.len));
+
+ for (j = 0; j < acph->headx[i].head.len; j++) {
+ if ((ittype == PICODATA_ITEM_CMD)) {
+ PICODBG_INFO_MSG(("%c",
+ acph->cbuf[acph->headx[i].cind+j]));
+ } else {
+ PICODBG_INFO_MSG(("%4d",
+ acph->cbuf[acph->headx[i].cind+j]));
+ }
+ }
+ PICODBG_INFO_MSG(("\n"));
+ }
+ }
+#endif
+
+ break;
+
+
+ /* *********************************************************/
+ /* process acc state: process items in headx and modify
+ * headx in place
+ */
+ case SA_STEPSTATE_PROCESS_ACC:
+ /* ensure there is an item in inBuf */
+ if (acph->headxLen > 0) {
+ /* we have a phrase in headx, cbuf (can be
+ single PUNC item), do accentuation and modify headx */
+ if (PICO_OK != acphAccentuation(this, acph)) {
+ picoos_emRaiseException(this->common->em,
+ PICO_ERR_OTHER, NULL, NULL);
+ return PICODATA_PU_ERROR;
+ }
+ acph->procState = SA_STEPSTATE_FEED;
+ } else if (acph->headxLen == 0) { /* no items in inBuf */
+ PICODBG_WARN(("no items in inBuf"));
+ acph->procState = SA_STEPSTATE_COLLECT;
+ return PICODATA_PU_BUSY;
+ }
+ break;
+
+
+ /* *********************************************************/
+ /* feed state: copy item in internal outBuf to output charBuf */
+ case SA_STEPSTATE_FEED: {
+ picoos_uint16 indupbound;
+ picoos_uint8 dopuoutfull;
+
+ PICODBG_DEBUG(("put out items (bot, len): (%d, %d)",
+ acph->headxBottom, acph->headxLen));
+
+ indupbound = acph->headxBottom + acph->headxLen;
+ dopuoutfull = FALSE;
+
+ if (acph->headxBottom == 0) {
+ /* construct first BOUND item in tmpbuf and put item */
+ /* produce BOUND unless it is followed by a term/flush) */
+ if (acph->headx[0].head.info1
+ != PICODATA_ITEMINFO1_PUNC_FLUSH) {
+ if (!acphPutBoundItem(this, acph,
+ acph->headx[0].boundstrength,
+ acph->headx[0].boundtype, &dopuoutfull,
+ numBytesOutput)) {
+ if (dopuoutfull) {
+ PICODBG_DEBUG(("feeding overflow"));
+ return PICODATA_PU_OUT_FULL;
+ } else {
+ /* ERR-msg and exception done in acphPutBoundItem */
+ return PICODATA_PU_ERROR;
+ }
+ }
+ }
+ }
+
+ /* for all items in headx, cbuf */
+ for (i = acph->headxBottom; i < indupbound; i++) {
+
+ switch (acph->headx[i].head.type) {
+ case PICODATA_ITEM_PUNC:
+ /* if sentence end, put SEND bound */
+ if ((acph->headx[i].head.info1 ==
+ PICODATA_ITEMINFO1_PUNC_SENTEND) &&
+ (i == (indupbound - 1))) {
+ /* construct and put BOUND item */
+ if (!acphPutBoundItem(this, acph,
+ PICODATA_ITEMINFO1_BOUND_SEND,
+ PICODATA_ITEMINFO2_NA,
+ &dopuoutfull, numBytesOutput)) {
+ if (dopuoutfull) {
+ PICODBG_DEBUG(("feeding overflow"));
+ return PICODATA_PU_OUT_FULL;
+ } else {
+ /* ERR-msg and exception done
+ in acphPutBoundItem */
+ return PICODATA_PU_ERROR;
+ }
+ }
+ } else if ((acph->headx[i].head.info1 ==
+ PICODATA_ITEMINFO1_PUNC_FLUSH) &&
+ (i == (indupbound - 1))) {
+ /* construct and put BOUND item */
+ if (!acphPutBoundItem(this, acph,
+ PICODATA_ITEMINFO1_BOUND_TERM,
+ PICODATA_ITEMINFO2_NA,
+ &dopuoutfull, numBytesOutput)) {
+ if (dopuoutfull) {
+ PICODBG_DEBUG(("feeding overflow"));
+ return PICODATA_PU_OUT_FULL;
+ } else {
+ /* ERR-msg and exception done
+ in acphPutBoundItem */
+ return PICODATA_PU_ERROR;
+ }
+ }
+ }
+ /* else, good-bye PUNC, not needed anymore */
+ break;
+ default:
+
+ /* PHR2/3 maybe existing, check and add
+ BOUND item now, if needed */
+ if ((acph->headx[i].boundstrength ==
+ PICODATA_ITEMINFO1_BOUND_PHR2) ||
+ (acph->headx[i].boundstrength ==
+ PICODATA_ITEMINFO1_BOUND_PHR3)) {
+ if (!acphPutBoundItem(this, acph,
+ acph->headx[i].boundstrength,
+ acph->headx[i].boundtype,
+ &dopuoutfull, numBytesOutput)) {
+ if (dopuoutfull) {
+ PICODBG_DEBUG(("feeding overflow"));
+ return PICODATA_PU_OUT_FULL;
+ } else {
+ /* ERR-msg and exception done
+ in acphPutBoundItem */
+ return PICODATA_PU_ERROR;
+ }
+ }
+ }
+
+ /* copy item unmodified */
+ rv = picodata_put_itemparts(&(acph->headx[i].head),
+ &(acph->cbuf[acph->headx[i].cind]),
+ acph->headx[i].head.len,
+ acph->tmpbuf, PICODATA_MAX_ITEMSIZE,
+ &blen);
+
+ rvP = picodata_cbPutItem(this->cbOut, acph->tmpbuf,
+ PICODATA_MAX_ITEMSIZE, &clen);
+
+ *numBytesOutput += clen;
+
+ PICODBG_DEBUG(("put item, status: %d", rvP));
+
+ if (rvP == PICO_OK) {
+ acph->headxBottom++;
+ acph->headxLen--;
+ } else if (rvP == PICO_EXC_BUF_OVERFLOW) {
+ /* try again next time, but PHR2/3
+ bound already added if existing,
+ ensure it's not output a 2nd
+ time */
+ PICODBG_DEBUG(("feeding overflow"));
+ acph->headx[i].boundstrength = 0;
+ return PICODATA_PU_OUT_FULL;
+ } else {
+ /* error, should never happen */
+ PICODBG_ERROR(("untreated return value, rvP: %d", rvP));
+ return PICODATA_PU_ERROR;
+ }
+
+ PICODATA_INFO_ITEM(this->voice->kbArray[PICOKNOW_KBID_DBG],
+ (picoos_uint8 *)"acph: ",
+ acph->tmpbuf, PICODATA_MAX_ITEMSIZE);
+
+ break;
+ } /*switch*/
+ } /*for*/
+
+ /* reset headx, cbuf */
+ acph->headxBottom = 0;
+ acph->headxLen = 0;
+ acph->cbufLen = 0;
+ for (i = 0; i < PICOACPH_MAXNR_HEADX; i++) {
+ acph->headx[i].boundstrength = 0;
+ }
+
+ /* reset collect state support variables */
+ acph->inspaceok = TRUE;
+ acph->needsmoreitems = TRUE;
+
+ acph->procState = SA_STEPSTATE_COLLECT;
+ return PICODATA_PU_BUSY;
+ break;
+ }
+
+ default:
+ break;
+ } /* switch */
+
+ } /* while */
+
+ /* should be never reached */
+ PICODBG_ERROR(("reached end of function"));
+ picoos_emRaiseException(this->common->em, PICO_ERR_OTHER, NULL, NULL);
+ return PICODATA_PU_ERROR;
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+
+/* end */
diff --git a/lib/picoacph.h b/lib/picoacph.h
new file mode 100644
index 0000000..266ab6e
--- /dev/null
+++ b/lib/picoacph.h
@@ -0,0 +1,205 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picoacph.h
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+
+/**
+ * @addtogroup picoacph
+ *
+itemtype, iteminfo1, iteminfo2, content -> TYPE(INFO1,INFO2)content
+in the following
+
+items input
+===========
+
+processed by sa (POS disambiguation):
+- WORDGRAPH(POSes,NA)graph
+- WORDINDEX(POSes,NA)POS|1ind1...POSN|indN
+- CMD(PICODATA_ITEMINFO1_CMD_FLUSH,PICODATA_ITEMINFO2_NA)
+
+processed by sa (Phrasing, Accentuation):
+- PUNC(PUNCtype,PUNCsubtype)
+
+unprocessed:
+- all other item types are forwarded through the PU without modification:
+ CMD
+
+
+minimal input size (before processing starts)
+==================
+
+processing (POS disambiguation, g2p, lexind, phrasing, accentuation)
+is possible with
+
+- one punctuation-phrase, consisting of a sequence (see below for
+ limits) of items terminated by a PUNC item.
+
+(possible but not implemented: as long as the internal buffer is
+empty, non-processed item types can be processed immediately)
+
+Ensuring terminal PUNC item:
+- when reading items from the external buffer a CMD(...FLUSH...) is
+ converted to a PUNC(...FLUSH...) item
+- If needed, a PUNC(PHRASE) is artificially added to ensure a phrase
+ fits in the PUs memory and processing can start.
+
+
+items processed and output
+==========================
+
+precondition:
+CMD(...FLUSH...) already converted to PUNC(...FLUSH...) and trailing
+PUNC item enforced if necessary.
+
+----
+-# PROCESS_POSD: processing input WORDGRAPH or WORDINDEX items, after
+POS disambiguation (POSes -> POS), results in a sequence of:
+ -
+ - WORDGRAPH(POS,NA)graph
+ - WORDINDEX(POS,NA)POS|ind
+ -
+ .
+-# PROCESS_WPHO: then, after lex-index lookup and G2P in a
+sequence of:
+ - WORDPHON(POS,NA)phon
+
+(phon containing primary and secondary word-level stress)
+
+----
+3. PROCESS_PHR: then, after processing these WORDPHON items,
+together with the trailing PUNC item results in:
+
+-> BOUND(BOUNDstrength,BOUNDtype)
+
+being added in the sequence of WORDPHON (respectively inserted instead
+of the PUNC). All PUNC, incl PUNC(...FLUSH...) now gone.
+
+----
+4. PROCESS_ACC: then, after processing the WORDPHON and BOUND items
+results in:
+
+-> WORDPHON(POS,ACC)phon
+
+A postprocessing step of accentuation is hard-coded in the
+accentuation module: In case the whole word does not have any stress
+at all (primary or secondary or both) then do the following mapping:
+
+ ACC0 nostress -> ACC0
+ ACC1 nostress -> ACC3
+ ACC2 nostress -> ACC3
+ ACC3 nostress -> ACC3
+
+----
+- POS
+ a single, unambiguous POS
+
+cf. picodata.h for
+- ACC (sentence-level accent (aka prominence)) %d
+ - PICODATA_ACC0
+ - PICODATA_ACC1
+ - PICODATA_ACC2 (<- maybe mapped to ACC1, ie. no ACC2 in output)
+ - PICODATA_ACC3
+
+- BOUNDstrength %d
+ - PICODATA_ITEMINFO1_BOUND_SBEG (at sentence start)
+ - PICODATA_ITEMINFO1_BOUND_SEND (at sentence end)
+ - PICODATA_ITEMINFO1_BOUND_TERM (replaces a flush)
+ - PICODATA_ITEMINFO1_BOUND_PHR1 (primary boundary)
+ - PICODATA_ITEMINFO1_BOUND_PHR2 (short break)
+ - PICODATA_ITEMINFO1_BOUND_PHR3 (secondary phrase boundary, no break)
+ - PICODATA_ITEMINFO1_BOUND_PHR0 (no break, not produced by sa, not existing
+ BOUND in item sequence equals PHR0 bound strength)
+
+- BOUNDtype (created in sa base on punctuation, indicates type of phrase
+ following the boundary) %d
+ - PICODATA_ITEMINFO2_BOUNDTYPE_P
+ - PICODATA_ITEMINFO2_BOUNDTYPE_T
+ - PICODATA_ITEMINFO2_BOUNDTYPE_Q
+ - PICODATA_ITEMINFO2_BOUNDTYPE_E
+
+
+output sequence (without CMDs):
+
+<output> = { BOUND(BOUND_SBEG,PHRASEtype) <sentence> BOUND(BOUND_SEND,..)} BOUND(BOUND_TERM,..)
+
+<sentence> = <phrase> { BOUND(BOUND_PHR1|2|3,BOUNDtype) <phrase> }
+
+<phrase> = WORDPHON(POS,ACC)phon { WORDPHON(POS,ACC)phon }
+
+Done in later PU: mapping ACC & word-level stress to syllable accent value
+ - ACC0 prim -> 0
+ - ACC1 prim -> 1
+ - ACC2 prim -> 2
+ - ACC3 prim -> 3
+ - ACC0 sec -> 0
+ - ACC1 sec -> 4
+ - ACC2 sec -> 4
+ - ACC3 sec -> 4
+
+other limitations
+=================
+
+- item size: header plus len=256 (valid for Pico in general)
+- see defines below for max nr of items. Item heads plus ref. to contents
+ buffer are stored in array with fixed size elements. Two restrictions:
+ - MAXNR_HEADX (max nr elements==items in headx array)
+ - CONTENTSSIZE (max size of all contents together
+ */
+
+
+#ifndef PICOACPH_H_
+#define PICOACPH_H_
+
+#include "picoos.h"
+#include "picodata.h"
+#include "picorsrc.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+/* nr item restriction: maximum number of extended item heads in headx */
+#define PICOACPH_MAXNR_HEADX 60
+
+/* nr item restriction: maximum size of all item contents together in cont */
+#define PICOACPH_MAXSIZE_CBUF 7680
+
+
+
+picodata_ProcessingUnit picoacph_newAccPhrUnit(
+ picoos_MemoryManager mm,
+ picoos_Common common,
+ picodata_CharBuffer cbIn,
+ picodata_CharBuffer cbOut,
+ picorsrc_Voice voice);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /*PICOACPH_H_*/
diff --git a/lib/picoapi.c b/lib/picoapi.c
new file mode 100644
index 0000000..c4ef8bf
--- /dev/null
+++ b/lib/picoapi.c
@@ -0,0 +1,801 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picoapi.c
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ */
+#include "picodefs.h"
+#include "picoos.h"
+#include "picodbg.h"
+#include "picorsrc.h"
+#include "picoctrl.h"
+#include "picoapi.h"
+#include "picoapid.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/* ****************************************************************************/
+/* System-level API functions */
+/* ****************************************************************************/
+
+#define MAGIC_MASK 0x5069636F /* Pico */
+
+#define SET_MAGIC_NUMBER(sys) \
+ (sys)->magic = ((picoos_uint32) (sys)) ^ MAGIC_MASK
+
+#define CHECK_MAGIC_NUMBER(sys) \
+ ((sys)->magic == (((picoos_uint32) (sys)) ^ MAGIC_MASK))
+
+
+
+/* *** Auxiliary routines (may also be called from picoextapi.c) **************/
+
+
+int is_valid_system_handle(pico_System system)
+{
+ return (system != NULL) && CHECK_MAGIC_NUMBER(system);
+}
+
+
+picoos_Common pico_sysGetCommon(pico_System this)
+{
+ if (this != NULL) {
+ return this->common;
+ } else {
+ return NULL;
+ }
+}
+
+
+
+/* *** System initialization and termination functions ************************/
+pico_Status pico_initialize_priv(
+ void *memory,
+ const pico_Uint32 size,
+ pico_Int16 enableMemProt,
+ pico_System *system
+ )
+{
+ pico_Status status = PICO_OK;
+
+ PICODBG_INITIALIZE(PICODBG_LOG_LEVEL_INFO);
+ PICODBG_ENABLE_COLORS(0);
+ /*PICODBG_SET_OUTPUT_FORMAT((PICODBG_SHOW_LEVEL | PICODBG_SHOW_SRCNAME));*/
+
+ if (memory == NULL) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else if (size == 0) {
+ status = PICO_ERR_INVALID_ARGUMENT;
+ } else if (system == NULL) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else {
+ byte_ptr_t rest_mem;
+ picoos_uint32 rest_mem_size;
+ pico_System sys;
+ picoos_MemoryManager sysMM;
+ picoos_ExceptionManager sysEM;
+
+ sys = (pico_System) picoos_raw_malloc(memory, size, sizeof(pico_system_t),
+ &rest_mem, &rest_mem_size);
+ if (sys != NULL) {
+ sysMM = picoos_newMemoryManager(rest_mem, rest_mem_size, enableMemProt ? TRUE : FALSE);
+ if (sysMM != NULL) {
+ sysEM = picoos_newExceptionManager(sysMM);
+ sys->common = picoos_newCommon(sysMM);
+ sys->rm = picorsrc_newResourceManager(sysMM, sys->common);
+ if ((sysEM != NULL) && (sys->common != NULL) && (sys->rm != NULL)) {
+ sys->common->em = sysEM;
+ sys->common->mm = sysMM;
+ sys->engine = NULL;
+
+ picorsrc_createDefaultResource(sys->rm /*,&defaultResource */);
+
+ SET_MAGIC_NUMBER(sys);
+ status = PICO_OK;
+ } else {
+ status = PICO_EXC_OUT_OF_MEM;
+ }
+ } else {
+ status = PICO_EXC_OUT_OF_MEM;
+ }
+ } else {
+ status = PICO_EXC_OUT_OF_MEM;
+ }
+ *system = sys;
+ }
+
+ if (status != PICO_OK) {
+ if (system != NULL) {
+ *system = NULL;
+ }
+ PICODBG_TERMINATE();
+ }
+
+ return status;
+}
+/**
+ * pico_initialize : initializes the pico system private memory
+ * @param memory : pointer to a free and already allocated memory area
+ * @param size : size of the memory area
+ * @param system : pointer to a pico_System struct
+ * @return PICO_OK : successful init, !PICO_OK : error on allocating private memory
+ * @callgraph
+ * @callergraph
+*/
+PICO_FUNC pico_initialize(
+ void *memory,
+ const pico_Uint32 size,
+ pico_System *system
+ )
+{
+ return pico_initialize_priv(memory, size, /*enableMemProt*/ FALSE, system);
+}
+
+/**
+ * pico_terminate : deallocates the pico system private memory
+ * @param system : pointer to a pico_System struct
+ * @return PICO_OK : successful de-init, !PICO_OK : error on de-allocating private memory
+ * @callgraph
+ * @callergraph
+*/
+PICO_FUNC pico_terminate(
+ pico_System *system
+ )
+{
+ pico_Status status = PICO_OK;
+
+ if ((system == NULL) || !is_valid_system_handle(*system)) {
+ status = PICO_ERR_INVALID_HANDLE;
+ } else {
+ pico_System sys = *system;
+
+ /* close engine(s) */
+ picoctrl_disposeEngine(sys->common->mm, sys->rm, &sys->engine);
+
+ /* close all resources */
+ picorsrc_disposeResourceManager(sys->common->mm, &sys->rm);
+
+ sys->magic ^= 0xFFFEFDFC;
+ *system = NULL;
+ }
+
+ PICODBG_TERMINATE();
+
+ return status;
+}
+
+
+
+/* *** System status and error/warning message retrieval function *************/
+
+/**
+ * pico_getSystemStatusMessage : Returns a description of the system status or errors
+ * @param system : pointer to a pico_System struct
+ * @param errCode : pico_System error code
+ * @param outMessage : memory area where to return a string
+ * @return PICO_OK : successful
+ * @return PICO_ERR_INVALID_HANDLE, PICO_ERR_NULLPTR_ACCESS : errors
+ * @callgraph
+ * @callergraph
+*/
+PICO_FUNC pico_getSystemStatusMessage(
+ pico_System system,
+ pico_Status errCode,
+ pico_Retstring outMessage
+ )
+{
+ pico_Status status = PICO_OK;
+
+ if (!is_valid_system_handle(system)) {
+ status = PICO_ERR_INVALID_HANDLE;
+ if (outMessage != NULL) {
+ picoos_strlcpy((picoos_char *) outMessage, (picoos_char *) "'system' not initialized", PICO_RETSTRINGSIZE);
+ }
+ } else if (outMessage == NULL) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else {
+ if (picoos_emGetExceptionCode(system->common->em) == PICO_OK) {
+ if (errCode == PICO_OK) {
+ picoos_strlcpy((picoos_char *) outMessage, (picoos_char *) "system ok", PICO_RETSTRINGSIZE);
+ } else {
+ /* exceptionManager was not informed yet; produce default message */
+ picoos_setErrorMsg((picoos_char *) outMessage, PICO_RETSTRINGSIZE, errCode, NULL, NULL, NULL);
+ }
+ } else {
+ picoos_emGetExceptionMessage(system->common->em, (picoos_char *) outMessage, PICO_RETSTRINGSIZE);
+ }
+ }
+
+ return status;
+}
+
+/**
+ * pico_getSystemStatusMessage : Returns the number of warnings
+ * @param system : pointer to a pico_System struct
+ * @param *outNrOfWarnings : pointer to location to receive number of warnings
+ * @return PICO_OK : successful
+ * @return PICO_ERR_INVALID_HANDLE, PICO_ERR_NULLPTR_ACCESS : errors
+ * @callgraph
+ * @callergraph
+*/
+PICO_FUNC pico_getNrSystemWarnings(
+ pico_System system,
+ pico_Int32 *outNrOfWarnings
+ )
+{
+ pico_Status status = PICO_OK;
+
+ if (!is_valid_system_handle(system)) {
+ status = PICO_ERR_INVALID_HANDLE;
+ if (outNrOfWarnings != NULL) {
+ *outNrOfWarnings = 0;
+ }
+ } else if (outNrOfWarnings == NULL) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else {
+ *outNrOfWarnings = picoos_emGetNumOfWarnings(system->common->em);
+ }
+
+ return status;
+}
+
+/**
+ * pico_getSystemWarning : Returns a description of a warning
+ * @param system : pointer to a pico_System struct
+ * @param warningIndex : warning index
+ * @param *outCode : pointer to receive the warning code
+ * @param outMessage : pointer to receive the output message
+ * @return PICO_OK : successful
+ * @return PICO_ERR_INVALID_HANDLE, PICO_ERR_NULLPTR_ACCESS : errors
+ * @callgraph
+ * @callergraph
+*/
+PICO_FUNC pico_getSystemWarning(
+ pico_System system,
+ const pico_Int32 warningIndex,
+ pico_Status *outCode,
+ pico_Retstring outMessage
+ )
+{
+ pico_Status status = PICO_OK;
+
+ if (!is_valid_system_handle(system)) {
+ status = PICO_ERR_INVALID_HANDLE;
+ if (outMessage != NULL) {
+ picoos_strlcpy((picoos_char *) outMessage, (picoos_char *) "'system' not initialized", PICO_RETSTRINGSIZE);
+ }
+ } else if (warningIndex < 0) {
+ status = PICO_ERR_INDEX_OUT_OF_RANGE;
+ } else if ((outCode == NULL) || (outMessage == NULL)) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else {
+ *outCode = picoos_emGetWarningCode(system->common->em, warningIndex);
+ picoos_emGetWarningMessage(system->common->em, warningIndex, (picoos_char *) outMessage, (picoos_uint16) PICO_RETSTRINGSIZE);
+ }
+
+ return status;
+}
+
+
+
+/* *** Resource loading and unloading functions *******************************/
+
+/**
+ * pico_loadResource : Loads a resource file into the Pico system
+ * @param system : pointer to a pico_System struct
+ * @param *lingwareFileName : lingware resource file name
+ * @param *outLingware : pointer to receive the loaded lingware resource memory area address
+ * @return PICO_OK : successful
+ * @return PICO_ERR_INVALID_HANDLE, PICO_ERR_NULLPTR_ACCESS : errors
+ * @callgraph
+ * @callergraph
+*/
+PICO_FUNC pico_loadResource(
+ pico_System system,
+ const pico_Char *lingwareFileName,
+ pico_Resource *outLingware
+ )
+{
+ pico_Status status = PICO_OK;
+
+ if (!is_valid_system_handle(system)) {
+ status = PICO_ERR_INVALID_HANDLE;
+ } else if ((lingwareFileName == NULL) || (outLingware == NULL)) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else {
+ PICODBG_DEBUG(("memory usage before resource loading"));
+ picoos_showMemUsage(system->common->mm, FALSE, TRUE);
+ picoos_emReset(system->common->em);
+ status = picorsrc_loadResource(system->rm, (picoos_char *) lingwareFileName, (picorsrc_Resource *) outLingware);
+ PICODBG_DEBUG(("memory used to load resource %s", lingwareFileName));
+ picoos_showMemUsage(system->common->mm, TRUE, FALSE);
+ }
+
+ return status;
+}
+
+/**
+ * pico_unloadResource : unLoads a resource file from the Pico system
+ * @param system : pointer to a pico_System struct
+ * @param *inoutLingware : pointer to the loaded lingware resource memory area address
+ * @return PICO_OK : successful
+ * @return PICO_ERR_INVALID_HANDLE, PICO_ERR_NULLPTR_ACCESS : errors
+ * @callgraph
+ * @callergraph
+*/
+PICO_FUNC pico_unloadResource(
+ pico_System system,
+ pico_Resource *inoutLingware
+ )
+{
+ pico_Status status = PICO_OK;
+
+ if (!is_valid_system_handle(system)) {
+ status = PICO_ERR_INVALID_HANDLE;
+ } else if (inoutLingware == NULL) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else if (!picoctrl_isValidResourceHandle(*((picorsrc_Resource *) inoutLingware))) {
+ status = PICO_ERR_INVALID_HANDLE;
+ } else {
+ PICODBG_DEBUG(("memory usage before resource unloading"));
+ picoos_showMemUsage(system->common->mm, FALSE, TRUE);
+ picoos_emReset(system->common->em);
+ status = picorsrc_unloadResource(system->rm, (picorsrc_Resource *) inoutLingware);
+ PICODBG_DEBUG(("memory released by resource unloading"));
+ picoos_showMemUsage(system->common->mm, TRUE, FALSE);
+ }
+
+ return status;
+}
+
+/* *** Resource inspection functions *******************************/
+/**
+ * pico_getResourceName : Gets a resource name
+ * @param system : pointer to a pico_System struct
+ * @param resource : pointer to the loaded resource memory area address
+ * @param outName : pointer to the area to receuive the resource name
+ * @return PICO_OK : successful
+ * @return PICO_ERR_INVALID_HANDLE, PICO_ERR_NULLPTR_ACCESS : errors
+ * @callgraph
+ * @callergraph
+*/
+PICO_FUNC pico_getResourceName(
+ pico_System system,
+ pico_Resource resource,
+ pico_Retstring outName) {
+
+ if (!is_valid_system_handle(system)) {
+ return PICO_ERR_INVALID_HANDLE;
+ } else if (NULL == outName) {
+ return PICO_ERR_NULLPTR_ACCESS;
+ }
+ return picorsrc_rsrcGetName((picorsrc_Resource)resource, (picoos_char *) outName, PICO_RETSTRINGSIZE);
+}
+
+
+/* *** Voice definition functions *********************************************/
+
+/**
+ * pico_createVoiceDefinition : Creates a voice definition
+ * @param system : pointer to a pico_System struct
+ * @param *voiceName : pointer to the area to receive the voice definition
+ * @return PICO_OK : successful
+ * @return PICO_ERR_INVALID_HANDLE, PICO_ERR_NULLPTR_ACCESS : errors
+ * @callgraph
+ * @callergraph
+*/
+PICO_FUNC pico_createVoiceDefinition(
+ pico_System system,
+ const pico_Char *voiceName
+ )
+{
+ pico_Status status = PICO_OK;
+
+ if (!is_valid_system_handle(system)) {
+ status = PICO_ERR_INVALID_HANDLE;
+ } else if (voiceName == NULL) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else if (picoos_strlen((picoos_char *) voiceName) == 0) {
+ status = PICO_ERR_INVALID_ARGUMENT;
+ } else {
+ picoos_emReset(system->common->em);
+ status = picorsrc_createVoiceDefinition(system->rm, (picoos_char *) voiceName);
+ }
+
+ return status;
+}
+
+/**
+ * pico_addResourceToVoiceDefinition : Adds a mapping pair to a voice definition
+ * @param system : pointer to a pico_System struct
+ * @param *voiceName : pointer to the area containing the voice definition
+ * @param *resourceName : pointer to the area containing the resource name
+ * @return PICO_OK : successful
+ * @return PICO_ERR_INVALID_HANDLE, PICO_ERR_NULLPTR_ACCESS : errors
+ * @callgraph
+ * @callergraph
+*/
+PICO_FUNC pico_addResourceToVoiceDefinition(
+ pico_System system,
+ const pico_Char *voiceName,
+ const pico_Char *resourceName
+ )
+{
+ pico_Status status = PICO_OK;
+
+ if (!is_valid_system_handle(system)) {
+ status = PICO_ERR_INVALID_HANDLE;
+ } else if (voiceName == NULL) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else if (picoos_strlen((picoos_char *) voiceName) == 0) {
+ status = PICO_ERR_INVALID_ARGUMENT;
+ } else if (resourceName == NULL) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else if (picoos_strlen((picoos_char *) resourceName) == 0) {
+ status = PICO_ERR_INVALID_ARGUMENT;
+ } else {
+ picoos_emReset(system->common->em);
+ status = picorsrc_addResourceToVoiceDefinition(system->rm, (picoos_char *) voiceName, (picoos_char *) resourceName);
+ }
+
+ return status;
+}
+
+/**
+ * pico_releaseVoiceDefinition : Releases a voice definition
+ * @param system : pointer to a pico_System struct
+ * @param *voiceName : pointer to the area containing the voice definition
+ * @return PICO_OK : successful
+ * @return PICO_ERR_INVALID_HANDLE, PICO_ERR_NULLPTR_ACCESS : errors
+ * @callgraph
+ * @callergraph
+*/
+PICO_FUNC pico_releaseVoiceDefinition(
+ pico_System system,
+ const pico_Char *voiceName
+ )
+{
+ pico_Status status = PICO_OK;
+
+ if (!is_valid_system_handle(system)) {
+ status = PICO_ERR_INVALID_HANDLE;
+ } else if (voiceName == NULL) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else if (picoos_strlen((picoos_char *) voiceName) == 0) {
+ status = PICO_ERR_INVALID_ARGUMENT;
+ } else {
+ picoos_emReset(system->common->em);
+ status = picorsrc_releaseVoiceDefinition(system->rm, (picoos_char *) voiceName);
+ }
+
+ return status;
+}
+
+
+
+/* *** Engine creation and deletion functions *********************************/
+
+/**
+ * pico_newEngine : Creates and initializes a new Pico engine
+ * @param system : pointer to a pico_System struct
+ * @param *voiceName : pointer to the area containing the voice definition
+ * @param *outEngine : pointer to the Pico engine handle
+ * @return PICO_OK : successful
+ * @return PICO_ERR_INVALID_HANDLE, PICO_ERR_NULLPTR_ACCESS : errors
+ * @callgraph
+ * @callergraph
+*/
+PICO_FUNC pico_newEngine(
+ pico_System system,
+ const pico_Char *voiceName,
+ pico_Engine *outEngine
+ )
+{
+ pico_Status status = PICO_OK;
+
+ PICODBG_DEBUG(("creating engine for voice '%s'", (picoos_char *) voiceName));
+
+ if (!is_valid_system_handle(system)) {
+ status = PICO_ERR_INVALID_HANDLE;
+ } else if (voiceName == NULL) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else if (picoos_strlen((picoos_char *) voiceName) == 0) {
+ status = PICO_ERR_INVALID_ARGUMENT;
+ } else if (outEngine == NULL) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else {
+ picoos_emReset(system->common->em);
+ if (system->engine == NULL) {
+ *outEngine = (pico_Engine) picoctrl_newEngine(system->common->mm, system->rm, voiceName);
+ if (*outEngine != NULL) {
+ system->engine = (picoctrl_Engine) *outEngine;
+ } else {
+ status = picoos_emRaiseException(system->common->em, PICO_EXC_OUT_OF_MEM,
+ (picoos_char *) "out of memory creating new engine", NULL);
+ }
+ } else {
+ status = picoos_emRaiseException(system->common->em, PICO_EXC_MAX_NUM_EXCEED,
+ NULL, (picoos_char *) "no more than %i engines", 1);
+ }
+ }
+
+ return status;
+}
+
+/**
+ * pico_disposeEngine : Disposes a Pico engine
+ * @param system : pointer to a pico_System struct
+ * @param *inoutEngine : pointer to the Pico engine handle
+ * @return PICO_OK : successful
+ * @return PICO_ERR_INVALID_HANDLE, PICO_ERR_NULLPTR_ACCESS : errors
+ * @callgraph
+ * @callergraph
+*/
+PICO_FUNC pico_disposeEngine(
+ pico_System system,
+ pico_Engine *inoutEngine
+ )
+{
+ pico_Status status = PICO_OK;
+
+ if (!is_valid_system_handle(system)) {
+ status = PICO_ERR_INVALID_HANDLE;
+ } else if (inoutEngine == NULL) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else if (!picoctrl_isValidEngineHandle(*((picoctrl_Engine *) inoutEngine))) {
+ status = PICO_ERR_INVALID_HANDLE;
+ } else {
+ picoos_emReset(system->common->em);
+ picoctrl_disposeEngine(system->common->mm, system->rm, (picoctrl_Engine *) inoutEngine);
+ system->engine = NULL;
+ status = picoos_emGetExceptionCode(system->common->em);
+ }
+
+ return status;
+}
+
+
+
+/* ****************************************************************************/
+/* Engine-level API functions */
+/* ****************************************************************************/
+
+/**
+ * pico_putTextUtf8 : Puts UTF8 text into Pico text input buffer
+ * @param engine : pointer to a Pico engine handle
+ * @param *text : pointer to the text buffer
+ * @param textSize : text buffer size
+ * @param *bytesPut : pointer to variable to receive the number of bytes put
+ * @return PICO_OK : successful
+ * @return PICO_ERR_INVALID_HANDLE, PICO_ERR_NULLPTR_ACCESS : errors
+ * @callgraph
+ * @callergraph
+ */
+PICO_FUNC pico_putTextUtf8(
+ pico_Engine engine,
+ const pico_Char *text,
+ const pico_Int16 textSize,
+ pico_Int16 *bytesPut)
+{
+ pico_Status status = PICO_OK;
+
+ if (!picoctrl_isValidEngineHandle((picoctrl_Engine) engine)) {
+ status = PICO_ERR_INVALID_HANDLE;
+ } else if (text == NULL) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else if (textSize < 0) {
+ status = PICO_ERR_INVALID_ARGUMENT;
+ } else if (bytesPut == NULL) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else {
+ picoctrl_engResetExceptionManager((picoctrl_Engine) engine);
+ status = picoctrl_engFeedText((picoctrl_Engine) engine, (picoos_char *)text, textSize, bytesPut);
+ }
+
+ return status;
+}
+
+/**
+ * pico_getData : Gets speech data from the engine.
+ * @param engine : pointer to a Pico engine handle
+ * @param *buffer : pointer to output buffer
+ * @param bufferSize : out buffer size
+ * @param *bytesReceived : pointer to a variable to receive the number of bytes received
+ * @param *outDataType : pointer to a variable to receive the type of buffer received
+ * @return PICO_OK : successful
+ * @return PICO_ERR_INVALID_HANDLE, PICO_ERR_NULLPTR_ACCESS : errors
+ * @callgraph
+ * @callergraph
+*/
+PICO_FUNC pico_getData(
+ pico_Engine engine,
+ void *buffer,
+ const pico_Int16 bufferSize,
+ pico_Int16 *bytesReceived,
+ pico_Int16 *outDataType
+ )
+{
+ pico_Status status = PICO_OK;
+
+ if (!picoctrl_isValidEngineHandle((picoctrl_Engine) engine)) {
+ status = PICO_STEP_ERROR;
+ } else if (buffer == NULL) {
+ status = PICO_STEP_ERROR;
+ } else if (bufferSize < 0) {
+ status = PICO_STEP_ERROR;
+ } else if (bytesReceived == NULL) {
+ status = PICO_STEP_ERROR;
+ } else {
+ picoctrl_engResetExceptionManager((picoctrl_Engine) engine);
+ status = picoctrl_engFetchOutputItemBytes((picoctrl_Engine) engine, (picoos_char *)buffer, bufferSize, bytesReceived);
+ if ((status != PICO_STEP_IDLE) && (status != PICO_STEP_BUSY)) {
+ status = PICO_STEP_ERROR;
+ }
+ }
+
+ *outDataType = PICO_DATA_PCM_16BIT;
+ return status;
+}
+
+/**
+ * pico_resetEngine : Resets the engine
+ * @param engine : pointer to a Pico engine handle
+ * @return PICO_OK : successful
+ * @return PICO_ERR_INVALID_HANDLE, PICO_ERR_NULLPTR_ACCESS : errors
+ * @callgraph
+ * @callergraph
+*/
+PICO_FUNC pico_resetEngine(
+ pico_Engine engine)
+{
+ pico_Status status = PICO_OK;
+
+ if (!picoctrl_isValidEngineHandle((picoctrl_Engine) engine)) {
+ status = PICO_ERR_INVALID_HANDLE;
+ } else {
+ picoctrl_engResetExceptionManager((picoctrl_Engine) engine);
+ status = picoctrl_engReset((picoctrl_Engine) engine);
+ }
+
+ return status;
+}
+
+/**
+ * pico_getEngineStatusMessage : Returns the engine status or error description
+ * @param engine : pointer to a Pico engine handle
+ * @param errCode : error code
+ * @param outMessage : pointer to a memory area to receive the output message
+ * @return PICO_OK : successful
+ * @return PICO_ERR_INVALID_HANDLE, PICO_ERR_NULLPTR_ACCESS : errors
+ * @callgraph
+ * @callergraph
+*/
+PICO_FUNC pico_getEngineStatusMessage(
+ pico_Engine engine,
+ pico_Status errCode,
+ pico_Retstring outMessage
+ )
+{
+ pico_Status status = PICO_OK;
+
+ PICODBG_DEBUG(("got error code %i", errCode));
+
+ if (!picoctrl_isValidEngineHandle((picoctrl_Engine) engine)) {
+ status = PICO_ERR_INVALID_HANDLE;
+ if (outMessage != NULL) {
+ picoos_strlcpy((picoos_char *) outMessage, (picoos_char *) "'engine' not initialized", PICO_RETSTRINGSIZE);
+ }
+ } else if (outMessage == NULL) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else {
+ picoos_Common common = picoctrl_engGetCommon((picoctrl_Engine) engine);
+ if (picoos_emGetExceptionCode(common->em) == PICO_OK) {
+ if (errCode == PICO_OK) {
+ picoos_strlcpy((picoos_char *) outMessage, (picoos_char *) "engine ok", PICO_RETSTRINGSIZE);
+ } else {
+ /* exceptionManager was not informed yet; produce default message */
+ picoos_setErrorMsg((picoos_char *) outMessage, PICO_RETSTRINGSIZE, errCode, NULL, NULL, NULL);
+ }
+ } else {
+ picoos_emGetExceptionMessage(common->em, (picoos_char *) outMessage, PICO_RETSTRINGSIZE);
+ }
+ }
+
+ return status;
+}
+
+/**
+ * pico_getNrEngineWarnings : Returns the number of warnings
+ * @param engine : pointer to a Pico engine handle
+ * @param *outNrOfWarnings: pointer to a variable to receive the number of warnings
+ * @return PICO_OK : successful
+ * @return PICO_ERR_INVALID_HANDLE, PICO_ERR_NULLPTR_ACCESS : errors
+ * @callgraph
+ * @callergraph
+*/
+PICO_FUNC pico_getNrEngineWarnings(
+ pico_Engine engine,
+ pico_Int32 *outNrOfWarnings
+ )
+{
+ pico_Status status = PICO_OK;
+
+ if (!picoctrl_isValidEngineHandle((picoctrl_Engine) engine)) {
+ status = PICO_ERR_INVALID_HANDLE;
+ if (outNrOfWarnings != NULL) {
+ *outNrOfWarnings = 0;
+ }
+ } else if (outNrOfWarnings == NULL) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else {
+ picoos_Common common = picoctrl_engGetCommon((picoctrl_Engine) engine);
+ *outNrOfWarnings = picoos_emGetNumOfWarnings(common->em);
+ }
+
+ return status;
+}
+
+/**
+ * pico_getEngineWarning : Returns a description of a warning
+ * @param engine : pointer to a Pico engine handle
+ * @param warningIndex : warning index
+ * @param *outCode: pointer to a variable to receive the warning code
+ * @param outMessage: pointer to a memory area to receive the warning description
+ * @return PICO_OK : successful
+ * @return PICO_ERR_INVALID_HANDLE, PICO_ERR_NULLPTR_ACCESS : errors
+ * @callgraph
+ * @callergraph
+*/
+PICO_FUNC pico_getEngineWarning(
+ pico_Engine engine,
+ const pico_Int32 warningIndex,
+ pico_Status *outCode,
+ pico_Retstring outMessage
+ )
+{
+ pico_Status status = PICO_OK;
+
+ if (!picoctrl_isValidEngineHandle((picoctrl_Engine) engine)) {
+ status = PICO_ERR_INVALID_HANDLE;
+ if (outMessage != NULL) {
+ picoos_strlcpy((picoos_char *) outMessage, (picoos_char *) "'engine' not initialized", PICO_RETSTRINGSIZE);
+ }
+ } else if (warningIndex < 0) {
+ status = PICO_ERR_INDEX_OUT_OF_RANGE;
+ } else if ((outCode == NULL) || (outMessage == NULL)) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else {
+ picoos_Common common = picoctrl_engGetCommon((picoctrl_Engine) engine);
+ *outCode = picoos_emGetWarningCode(common->em, warningIndex);
+ picoos_emGetWarningMessage(common->em, warningIndex, (picoos_char *) outMessage, (picoos_uint16) PICO_RETSTRINGSIZE);
+ }
+
+ return status;
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+
+/* end */
diff --git a/lib/picoapi.h b/lib/picoapi.h
new file mode 100644
index 0000000..0dad702
--- /dev/null
+++ b/lib/picoapi.h
@@ -0,0 +1,469 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picoapi.h
+ *
+ * SVOX Pico application programming interface
+ * (SVOX Pico version 1.0 and later)
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ */
+
+
+/**
+ * @addtogroup picoapi
+ *
+@b Basic_Concepts
+
+@e SVOX_Pico_System
+
+The SVOX Pico 'system' is the entity that manages data common to all
+SVOX Pico engines, e.g. linguistic data needed to do text-to-speech
+(TTS) synthesis, license key, etc. All API functions on the Pico
+system level take a 'pico_System' handle as the first parameter.
+
+@e SVOX_Pico_Engine
+
+A SVOX Pico 'engine' provides the functions needed to perform actual
+synthesis. Currently there can be only one engine instance at a time
+(concurrent engines will be possible in the future). All API functions
+at the engine level take a 'pico_Engine' handle as the first
+parameter.
+
+@e SVOX_Pico_Resource
+
+A SVOX Pico 'resource' denotes all the language- and speaker-dependent
+data needed to do TTS synthesis. In the following, the term 'resource'
+may be used interchangeably with the term 'lingware'. A resource file
+contains a set of knowledge bases for an entire TTS voice or parts of
+it.
+
+
+@b Basic_Usage
+
+In its most basic form, an application must call the following
+functions in order to perform TTS synthesis:
+
+ - pico_initialize
+ - pico_loadResource
+ - pico_createVoiceDefinition
+ - pico_addResourceToVoiceDefinition
+ - pico_newEngine
+ - pico_putTextUtf8
+ - pico_getData (several times)
+ - pico_disposeEngine
+ - pico_releaseVoiceDefinition
+ - pico_unloadResource
+ - pico_terminate
+
+It is possible to repeatedly run the above sequence, i.e., the SVOX
+Pico system may be initialized and terminated multiple times. This may
+be useful in applications that need TTS functionality only from time
+to time.
+
+
+@b Conventions
+
+@e Function_arguments
+
+All arguments that only return values are marked by a leading 'out...'
+in their name. All arguments that are used as input and output values
+are marked by a leading 'inout...'. All other arguments are read-only
+(input) arguments.
+
+@e Error_handling
+
+All API functions return a status code which is one of the status
+constants defined in picodefs.h. In case of an error, a more detailed
+description of the status can be retrieved by calling function
+'pico_getSystemStatusMessage' (or 'pico_getEngineStatusMessage'
+if the error happened on the SVOX Pico engine level).
+
+Unlike errors, warnings do not prevent an API function from performing
+its function, but output might not be as intended. Functions
+'pico_getNrSystemWarnings' and 'pico_getNrEngineWarnings' respectively
+can be used to determine whether an API function caused any
+warnings. Details about warnings can be retrieved by calling
+'pico_getSystemWarning' and 'pico_getEngineWarning' respectively.
+
+*/
+
+
+
+#ifndef PICOAPI_H_
+#define PICOAPI_H_
+
+
+
+#include "picodefs.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+#ifdef _WIN32
+# define PICO_EXPORT __declspec( dllexport )
+#else
+# define PICO_EXPORT extern
+#endif
+
+#define PICO_FUNC PICO_EXPORT pico_Status
+
+
+
+/* ********************************************************************/
+/* PICO data types */
+/* ********************************************************************/
+
+/* Handle types (opaque) for Pico system, resource, engine ************/
+
+typedef struct pico_system *pico_System;
+typedef struct pico_resource *pico_Resource;
+typedef struct pico_engine *pico_Engine;
+
+
+/* Signed/unsigned integer data types *********************************/
+
+#define PICO_INT16_MAX 32767
+#define PICO_UINT16_MAX 0xffff
+#define PICO_INT32_MAX 2147483647
+#define PICO_UINT32_MAX 0xffffffff
+
+#include <limits.h>
+
+#if (SHRT_MAX == PICO_INT16_MAX)
+typedef short pico_Int16;
+#else
+#error "platform not supported"
+#endif
+
+#if (USHRT_MAX == PICO_UINT16_MAX)
+typedef unsigned short pico_Uint16;
+#else
+#error "platform not supported"
+#endif
+
+#if (INT_MAX == PICO_INT32_MAX)
+typedef int pico_Int32;
+#else
+#error "platform not supported"
+#endif
+
+#if (UINT_MAX == PICO_UINT32_MAX)
+typedef unsigned int pico_Uint32;
+#else
+#error "platform not supported"
+#endif
+
+
+/* Char data type *****************************************************/
+
+typedef unsigned char pico_Char;
+
+
+/* String type to be used when ASCII string values are returned *******/
+
+#define PICO_RETSTRINGSIZE 200 /* maximum length of returned strings */
+
+typedef char pico_Retstring[PICO_RETSTRINGSIZE];
+
+
+
+/* ********************************************************************/
+/* System-level API functions */
+/* ********************************************************************/
+
+/* System initialization and termination functions ********************/
+
+/**
+ Initializes the Pico system and returns its handle in 'outSystem'.
+ 'memory' and 'size' define the location and maximum size of memory
+ in number of bytes that the Pico system will use. The minimum size
+ required depends on the number of engines and configurations of
+ lingware to be used. No additional memory will be allocated by the
+ Pico system. This function must be called before any other API
+ function is called. It may only be called once (e.g. at application
+ startup), unless a call to 'pico_terminate'.
+*/
+PICO_FUNC pico_initialize(
+ void *memory,
+ const pico_Uint32 size,
+ pico_System *outSystem
+ );
+
+/**
+ Terminates the Pico system. Lingware resources still being loaded
+ are unloaded automatically. The memory area provided to Pico in
+ 'pico_initialize' is released. The system handle becomes
+ invalid. It is not allowed to call this function as long as Pico
+ engine instances are existing. No API function may be called after
+ this function, except for 'pico_initialize', which reinitializes
+ the system.
+*/
+PICO_FUNC pico_terminate(
+ pico_System *system
+ );
+
+
+/* System status and error/warning message retrieval ******************/
+
+/**
+ Returns in 'outMessage' a description of the system status or of an
+ error that occurred with the most recently called system-level API
+ function.
+*/
+PICO_FUNC pico_getSystemStatusMessage(
+ pico_System system,
+ pico_Status errCode,
+ pico_Retstring outMessage
+ );
+
+/**
+ Returns in 'outNrOfWarnings' the number of warnings that occurred
+ with the most recently called system-level API function.
+*/
+PICO_FUNC pico_getNrSystemWarnings(
+ pico_System system,
+ pico_Int32 *outNrOfWarnings
+ );
+
+/**
+ Returns in 'outMessage' a description of a warning that occurred
+ with the most recently called system-level API function.
+ 'warningIndex' must be in the range 0..N-1 where N is the number of
+ warnings returned by 'pico_getNrSystemWarnings'. 'outCode' returns
+ the warning as an integer code (cf. PICO_WARN_*).
+*/
+PICO_FUNC pico_getSystemWarning(
+ pico_System system,
+ const pico_Int32 warningIndex,
+ pico_Status *outCode,
+ pico_Retstring outMessage
+ );
+
+
+/* Resource loading and unloading functions ***************************/
+
+/**
+ Loads a resource file into the Pico system. The number of resource
+ files loaded in parallel is limited by PICO_MAX_NUM_RESOURCES.
+ Loading of a resource file may be done at any time (even in
+ parallel to a running engine doing TTS synthesis), but with the
+ general restriction that functions taking a system handle as their
+ first argument must be called in a mutually exclusive fashion. The
+ loaded resource will be available only to engines started after the
+ resource is fully loaded, i.e., not to engines currently
+ running.
+*/
+PICO_FUNC pico_loadResource(
+ pico_System system,
+ const pico_Char *resourceFileName,
+ pico_Resource *outResource
+ );
+
+/**
+ Unloads a resource file from the Pico system. If no engine uses the
+ resource file, the resource is removed immediately and its
+ associated internal memory is released, otherwise
+ PICO_EXC_RESOURCE_BUSY is returned.
+*/
+PICO_FUNC pico_unloadResource(
+ pico_System system,
+ pico_Resource *inoutResource
+ );
+
+/* *** Resource inspection functions *******************************/
+
+/**
+ Gets the unique resource name of a loaded resource
+*/
+PICO_FUNC pico_getResourceName(
+ pico_System system,
+ pico_Resource resource,
+ pico_Retstring outName);
+
+
+/* Voice definition ***************************************************/
+
+/**
+ Creates a voice definition. Resources must be added to the created
+ voice with 'pico_addResourceToVoiceDefinition' before using the
+ voice in 'pico_newEngine'. It is an error to create a voice
+ definition with a previously defined voice name. In that case use
+ 'pico_releaseVoiceName' first.
+*/
+PICO_FUNC pico_createVoiceDefinition(
+ pico_System system,
+ const pico_Char *voiceName
+ );
+
+/**
+ Adds a mapping pair ('voiceName', 'resourceName') to the voice
+ definition. Multiple mapping pairs can added to a voice defintion.
+ When calling 'pico_newEngine' with 'voiceName', the corresponding
+ resources from the mappings will be used with that engine. */
+
+PICO_FUNC pico_addResourceToVoiceDefinition(
+ pico_System system,
+ const pico_Char *voiceName,
+ const pico_Char *resourceName
+ );
+
+
+/**
+ Releases the voice definition 'voiceName'.
+
+*/
+PICO_FUNC pico_releaseVoiceDefinition(
+ pico_System system,
+ const pico_Char *voiceName
+ );
+
+
+/* Engine creation and deletion functions *****************************/
+
+/**
+ Creates and initializes a new Pico engine instance and returns its
+ handle in 'outEngine'. Only one instance per system is currently
+ possible.
+*/
+PICO_FUNC pico_newEngine(
+ pico_System system,
+ const pico_Char *voiceName,
+ pico_Engine *outEngine
+ );
+
+
+/**
+ Disposes a Pico engine and releases all memory it occupied. The
+ engine handle becomes invalid.
+*/
+PICO_FUNC pico_disposeEngine(
+ pico_System system,
+ pico_Engine *inoutEngine
+ );
+
+
+
+/* ********************************************************************/
+/* Engine-level API functions */
+/* ********************************************************************/
+
+/**
+ Puts text 'text' encoded in UTF8 into the Pico text input buffer.
+ 'textSize' is the maximum size in number of bytes accessible in
+ 'text'. The input text may also contain text-input commands to
+ change, for example, speed or pitch of the resulting speech
+ output. The number of bytes actually copied to the Pico text input
+ buffer is returned in 'outBytesPut'. Sentence ends are
+ automatically detected. '\0' characters may be embedded in 'text'
+ to finish text input or separate independently to be synthesized
+ text parts from each other. Repeatedly calling 'pico_getData' will
+ result in the content of the text input buffer to be synthesized
+ (up to the last sentence end or '\0' character detected). To empty
+ the internal buffers without finishing synthesis, use the function
+ 'pico_resetEngine'.
+*/
+PICO_FUNC pico_putTextUtf8(
+ pico_Engine engine,
+ const pico_Char *text,
+ const pico_Int16 textSize,
+ pico_Int16 *outBytesPut
+ );
+
+/**
+ Gets speech data from the engine. Every time this function is
+ called, the engine performs, within a short time slot, a small
+ amount of processing its input text, and then gives control back to
+ the calling application. Ie. after calling 'pico_putTextUtf8'
+ (incl. a final embedded '\0'), this function needs to be called
+ repeatedly till 'outBytesReceived' bytes are returned in
+ 'outBuffer'. The type of data returned in 'outBuffer' (e.g. 8 or 16
+ bit PCM samples) is returned in 'outDataType' and depends on the
+ lingware resources. Possible 'outDataType' values are listed in
+ picodefs.h (PICO_DATA_*).
+ This function returns PICO_STEP_BUSY while processing input and
+ producing speech output. Once all data is returned and there is no
+ more input text available in the Pico text input buffer,
+ PICO_STEP_IDLE is returned. All other function return values
+ indicate a system error.
+*/
+PICO_FUNC pico_getData(
+ pico_Engine engine,
+ void *outBuffer,
+ const pico_Int16 bufferSize,
+ pico_Int16 *outBytesReceived,
+ pico_Int16 *outDataType
+ );
+
+/**
+ Resets the engine and clears all engine-internal buffers, in
+ particular text input and signal data output buffers.
+*/
+PICO_FUNC pico_resetEngine(
+ pico_Engine engine
+ );
+
+
+/* Engine status and error/warning message retrieval ******************/
+
+/**
+ Returns in 'outMessage' a description of the engine status or of an
+ error that occurred with the most recently called engine-level API
+ function.
+*/
+PICO_FUNC pico_getEngineStatusMessage(
+ pico_Engine engine,
+ pico_Status errCode,
+ pico_Retstring outMessage
+ );
+
+/**
+ Returns in 'outNrOfWarnings' the number of warnings that occurred
+ with the most recently called engine-level API function.
+*/
+PICO_FUNC pico_getNrEngineWarnings(
+ pico_Engine engine,
+ pico_Int32 *outNrOfWarnings
+ );
+
+/**
+ Returns in 'outMessage' a description of a warning that occurred
+ with the most recently called engine-level API function.
+ 'warningIndex' must be in the range 0..N-1 where N is the number of
+ warnings returned by 'pico_getNrEngineWarnings'. 'outCode' returns
+ the warning as an integer code (cf. PICO_WARN_*).
+*/
+PICO_FUNC pico_getEngineWarning(
+ pico_Engine engine,
+ const pico_Int32 warningIndex,
+ pico_Status *outCode,
+ pico_Retstring outMessage
+ );
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /*PICOAPI_H_*/
diff --git a/lib/picoapid.h b/lib/picoapid.h
new file mode 100644
index 0000000..ef62701
--- /dev/null
+++ b/lib/picoapid.h
@@ -0,0 +1,70 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picoapid.h
+ *
+ * Pico api definitions commonly used by picoapi and picoapiext
+ *
+ * This header file must be part of the runtime-only pico system and therefore
+ * must not include picoapiext.h!
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#ifndef PICOAPID_H_
+#define PICOAPID_H_
+
+#include "picodefs.h"
+#include "picoapi.h"
+#include "picoos.h"
+#include "picorsrc.h"
+#include "picoctrl.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* Pico system descriptor */
+typedef struct pico_system {
+ picoos_uint32 magic; /* magic number used to validate handles */
+ picoos_Common common;
+ picorsrc_ResourceManager rm;
+ picoctrl_Engine engine;
+} pico_system_t;
+
+
+/* declared in picoapi.c */
+extern int is_valid_system_handle(pico_System system);
+extern picoos_Common pico_sysGetCommon(pico_System this);
+
+
+#if 0
+{
+#endif
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* PICOAPID_H_ */
diff --git a/lib/picobase.c b/lib/picobase.c
new file mode 100644
index 0000000..9246551
--- /dev/null
+++ b/lib/picobase.c
@@ -0,0 +1,1243 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picobase.c
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#include "picoos.h"
+#include "picodbg.h"
+#include "picodefs.h"
+#include "picobase.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+/**
+ * @addtogroup picobase
+ *
+ * @b Unicode_UTF8_functions
+ *
+ * UTF8
+ * scalar value 1st Byte 2nd Byte 3rd Byte 4th Byte
+ * 00000000 0xxxxxxx 0xxxxxxx
+ * 00000yyy yyxxxxxx 110yyyyy 10xxxxxx
+ * zzzzyyyy yyxxxxxx 1110zzzz 10yyyyyy 10xxxxxx
+ * 000uuuuu zzzzyyyy yyxxxxx 11110uuu 10uuzzzz 10yyyyyy 10xxxxxx
+ *
+*/
+picoos_int32 picobase_utf8_length(const picoos_uint8 *utf8str,
+ const picoos_uint16 maxlen) {
+
+ picoos_uint16 i;
+ picoos_uint16 len;
+ picoos_uint8 follow;
+ picoos_uint8 ok;
+
+ ok = TRUE;
+ i = 0;
+ len = 0;
+ follow = 0;
+ while (ok && (i < maxlen) && (utf8str[i] != '\000')) {
+ if (follow > 0) {
+ if ((utf8str[i] >= (picoos_uint8)'\200') &&
+ (utf8str[i] < (picoos_uint8)'\300')) {
+ follow--;
+ } else {
+ ok = FALSE;
+ }
+ } else if (utf8str[i] < (picoos_uint8)'\200') {
+ len++;
+ } else if (utf8str[i] >= (picoos_uint8)'\370') {
+ ok = FALSE;
+ } else if (utf8str[i] >= (picoos_uint8)'\360') {
+ follow = 3;
+ len++;
+ } else if (utf8str[i] >= (picoos_uint8)'\340') {
+ follow = 2;
+ len++;
+ } else if (utf8str[i] >= (picoos_uint8)'\300') {
+ follow = 1;
+ len++;
+ } else {
+ ok = FALSE;
+ }
+ i++;
+ }
+ if (ok) {
+ return len;
+ } else {
+ return -1;
+ }
+}
+
+
+static picoos_uint32 base_utf32_lowercase (picoos_uint32 utf32)
+{
+
+ picoos_uint32 lc;
+
+ lc = utf32;
+ if (((utf32 >= 65313) && (utf32 <= 65338))) {
+ lc = (utf32 + 32);
+ } else if (((utf32 >= 66560) && (utf32 <= 66599))) {
+ lc = (utf32 + 40);
+ } else if (((utf32 >= 7680) && (utf32 <= 9423))) {
+ switch (utf32) {
+ case 7680: case 7681: case 7682: case 7683: case 7684: case 7685: case 7686: case 7687: case 7688: case 7689:
+ case 7690: case 7691: case 7692: case 7693: case 7694: case 7695: case 7696: case 7697: case 7698: case 7699: case 7700: case 7701:
+ case 7702: case 7703: case 7704: case 7705: case 7706: case 7707: case 7708: case 7709: case 7710: case 7711: case 7712: case 7713:
+ case 7714: case 7715: case 7716: case 7717: case 7718: case 7719: case 7720: case 7721: case 7722: case 7723: case 7724: case 7725:
+ case 7726: case 7727: case 7728: case 7729: case 7730: case 7731: case 7732: case 7733: case 7734: case 7735: case 7736: case 7737:
+ case 7738: case 7739: case 7740: case 7741: case 7742: case 7743: case 7744: case 7745: case 7746: case 7747: case 7748: case 7749:
+ case 7750: case 7751: case 7752: case 7753: case 7754: case 7755: case 7756: case 7757: case 7758: case 7759: case 7760: case 7761:
+ case 7762: case 7763: case 7764: case 7765: case 7766: case 7767: case 7768: case 7769: case 7770: case 7771: case 7772: case 7773:
+ case 7774: case 7775: case 7776: case 7777: case 7778: case 7779: case 7780: case 7781: case 7782: case 7783: case 7784: case 7785:
+ case 7786: case 7787: case 7788: case 7789: case 7790: case 7791: case 7792: case 7793: case 7794: case 7795: case 7796: case 7797:
+ case 7798: case 7799: case 7800: case 7801: case 7802: case 7803: case 7804: case 7805: case 7806: case 7807: case 7808: case 7809:
+ case 7810: case 7811: case 7812: case 7813: case 7814: case 7815: case 7816: case 7817: case 7818: case 7819: case 7820: case 7821:
+ case 7822: case 7823: case 7824: case 7825: case 7826: case 7827: case 7828: case 7840: case 7841: case 7842: case 7843:
+ case 7844: case 7845: case 7846: case 7847: case 7848: case 7849: case 7850: case 7851: case 7852: case 7853: case 7854: case 7855:
+ case 7856: case 7857: case 7858: case 7859: case 7860: case 7861: case 7862: case 7863: case 7864: case 7865: case 7866: case 7867:
+ case 7868: case 7869: case 7870: case 7871: case 7872: case 7873: case 7874: case 7875: case 7876: case 7877: case 7878: case 7879:
+ case 7880: case 7881: case 7882: case 7883: case 7884: case 7885: case 7886: case 7887: case 7888: case 7889: case 7890: case 7891:
+ case 7892: case 7893: case 7894: case 7895: case 7896: case 7897: case 7898: case 7899: case 7900: case 7901: case 7902: case 7903:
+ case 7904: case 7905: case 7906: case 7907: case 7908: case 7909: case 7910: case 7911: case 7912: case 7913: case 7914: case 7915:
+ case 7916: case 7917: case 7918: case 7919: case 7920: case 7921: case 7922: case 7923: case 7924: case 7925: case 7926: case 7927:
+ case 7928:
+ if ( !(((utf32) % 2 == 1))) {
+ lc = (utf32 + 1);
+ }
+ break;
+ case 7944: case 7945: case 7946: case 7947: case 7948: case 7949: case 7950: case 7951: case 7960:
+ case 7961: case 7962: case 7963: case 7964: case 7965: case 7976: case 7977: case 7978: case 7979: case 7980: case 7981:
+ case 7982: case 7983: case 7992: case 7993: case 7994: case 7995: case 7996: case 7997: case 7998: case 7999:
+ case 8008: case 8009: case 8010: case 8011: case 8012: case 8013: case 8040: case 8041: case 8042: case 8043: case 8044:
+ case 8045: case 8046: case 8047: case 8072: case 8073: case 8074: case 8075: case 8076: case 8077: case 8078: case 8079:
+ case 8088: case 8089: case 8090: case 8091: case 8092: case 8093: case 8094: case 8095: case 8104: case 8105:
+ case 8106: case 8107: case 8108: case 8109: case 8110: case 8111:
+ lc = (utf32 - 8);
+ break;
+ case 8025: case 8026: case 8027: case 8028: case 8029: case 8030: case 8031:
+ if (((utf32) % 2 == 1)) {
+ lc = (utf32 - 8);
+ }
+ break;
+ case 8544: case 8545: case 8546: case 8547: case 8548: case 8549: case 8550: case 8551: case 8552: case 8553:
+ case 8554: case 8555: case 8556: case 8557: case 8558: case 8559:
+ lc = (utf32 + 16);
+ break;
+ case 9398: case 9399: case 9400: case 9401: case 9402: case 9403: case 9404: case 9405: case 9406: case 9407:
+ case 9408: case 9409: case 9410: case 9411: case 9412: case 9413: case 9414: case 9415: case 9416: case 9417: case 9418: case 9419:
+ case 9420: case 9421: case 9422: case 9423:
+ lc = (utf32 + 26);
+ break;
+ case 8120:
+ lc = 8112;
+ break;
+ case 8121:
+ lc = 8113;
+ break;
+ case 8122:
+ lc = 8048;
+ break;
+ case 8123:
+ lc = 8049;
+ break;
+ case 8124:
+ lc = 8115;
+ break;
+ case 8136:
+ lc = 8050;
+ break;
+ case 8137:
+ lc = 8051;
+ break;
+ case 8138:
+ lc = 8052;
+ break;
+ case 8139:
+ lc = 8053;
+ break;
+ case 8140:
+ lc = 8131;
+ break;
+ case 8152:
+ lc = 8144;
+ break;
+ case 8153:
+ lc = 8145;
+ break;
+ case 8154:
+ lc = 8054;
+ break;
+ case 8155:
+ lc = 8055;
+ break;
+ case 8168:
+ lc = 8160;
+ break;
+ case 8169:
+ lc = 8161;
+ break;
+ case 8170:
+ lc = 8058;
+ break;
+ case 8171:
+ lc = 8059;
+ break;
+ case 8172:
+ lc = 8165;
+ break;
+ case 8184:
+ lc = 8056;
+ break;
+ case 8185:
+ lc = 8057;
+ break;
+ case 8186:
+ lc = 8060;
+ break;
+ case 8187:
+ lc = 8061;
+ break;
+ case 8188:
+ lc = 8179;
+ break;
+ case 8486:
+ lc = 969;
+ break;
+ case 8490:
+ lc = 107;
+ break;
+ case 8491:
+ lc = 229;
+ break;
+ default:
+ break;
+ }
+ } else {
+ switch (utf32) {
+ case 65: case 66: case 67: case 68: case 69: case 70: case 71: case 72: case 73: case 74:
+ case 75: case 76: case 77: case 78: case 79: case 80: case 81: case 82: case 83: case 84: case 85: case 86:
+ case 87: case 88: case 89: case 90: case 192: case 193: case 194: case 195: case 196: case 197: case 198:
+ case 199: case 200: case 201: case 202: case 203: case 204: case 205: case 206: case 207: case 208: case 209: case 210:
+ case 211: case 212: case 213: case 214: case 216: case 217: case 218: case 219: case 220: case 221: case 222:
+ case 913: case 914: case 915: case 916: case 917: case 918: case 919: case 920: case 921: case 922: case 923:
+ case 924: case 925: case 926: case 927: case 928: case 929: case 931: case 932: case 933: case 934: case 935:
+ case 936: case 937: case 938: case 939: case 1040: case 1041: case 1042: case 1043: case 1044: case 1045: case 1046:
+ case 1047: case 1048: case 1049: case 1050: case 1051: case 1052: case 1053: case 1054: case 1055: case 1056: case 1057: case 1058:
+ case 1059: case 1060: case 1061: case 1062: case 1063: case 1064: case 1065: case 1066: case 1067: case 1068: case 1069: case 1070:
+ case 1071:
+ lc = (utf32 + 32);
+ break;
+ case 256: case 257: case 258: case 259: case 260: case 261: case 262: case 263: case 264: case 265:
+ case 266: case 267: case 268: case 269: case 270: case 271: case 272: case 273: case 274: case 275: case 276: case 277:
+ case 278: case 279: case 280: case 281: case 282: case 283: case 284: case 285: case 286: case 287: case 288: case 289:
+ case 290: case 291: case 292: case 293: case 294: case 295: case 296: case 297: case 298: case 299: case 300: case 301:
+ case 302: case 303: case 305: case 306: case 307: case 308: case 309: case 310: case 330: case 331:
+ case 332: case 333: case 334: case 335: case 336: case 337: case 338: case 339: case 340: case 341: case 342: case 343:
+ case 344: case 345: case 346: case 347: case 348: case 349: case 350: case 351: case 352: case 353: case 354: case 355:
+ case 356: case 357: case 358: case 359: case 360: case 361: case 362: case 363: case 364: case 365: case 366: case 367:
+ case 368: case 369: case 370: case 371: case 372: case 373: case 374: case 416: case 417: case 418: case 419:
+ case 420: case 478: case 479: case 480: case 481: case 482: case 483: case 484: case 485: case 486: case 487:
+ case 488: case 489: case 490: case 491: case 492: case 493: case 494: case 504: case 505: case 506: case 507:
+ case 508: case 509: case 510: case 511: case 512: case 513: case 514: case 515: case 516: case 517: case 518: case 519:
+ case 520: case 521: case 522: case 523: case 524: case 525: case 526: case 527: case 528: case 529: case 530: case 531:
+ case 532: case 533: case 534: case 535: case 536: case 537: case 538: case 539: case 540: case 541: case 542:
+ case 546: case 547: case 548: case 549: case 550: case 551: case 552: case 553: case 554: case 555: case 556: case 557:
+ case 558: case 559: case 560: case 561: case 562: case 984: case 985: case 986: case 987: case 988: case 989:
+ case 990: case 991: case 992: case 993: case 994: case 995: case 996: case 997: case 998: case 999: case 1000: case 1001:
+ case 1002: case 1003: case 1004: case 1005: case 1006: case 1120: case 1121: case 1122: case 1123: case 1124: case 1125:
+ case 1126: case 1127: case 1128: case 1129: case 1130: case 1131: case 1132: case 1133: case 1134: case 1135: case 1136: case 1137:
+ case 1138: case 1139: case 1140: case 1141: case 1142: case 1143: case 1144: case 1145: case 1146: case 1147: case 1148: case 1149:
+ case 1150: case 1151: case 1152: case 1162: case 1163: case 1164: case 1165: case 1166: case 1167: case 1168: case 1169:
+ case 1170: case 1171: case 1172: case 1173: case 1174: case 1175: case 1176: case 1177: case 1178: case 1179: case 1180: case 1181:
+ case 1182: case 1183: case 1184: case 1185: case 1186: case 1187: case 1188: case 1189: case 1190: case 1191: case 1192: case 1193:
+ case 1194: case 1195: case 1196: case 1197: case 1198: case 1199: case 1200: case 1201: case 1202: case 1203: case 1204: case 1205:
+ case 1206: case 1207: case 1208: case 1209: case 1210: case 1211: case 1212: case 1213: case 1214: case 1232: case 1233:
+ case 1234: case 1235: case 1236: case 1237: case 1238: case 1239: case 1240: case 1241: case 1242: case 1243: case 1244: case 1245:
+ case 1246: case 1247: case 1248: case 1249: case 1250: case 1251: case 1252: case 1253: case 1254: case 1255: case 1256: case 1257:
+ case 1258: case 1259: case 1260: case 1261: case 1262: case 1263: case 1264: case 1265: case 1266: case 1267: case 1268:
+ case 1280: case 1281: case 1282: case 1283: case 1284: case 1285: case 1286: case 1287: case 1288: case 1289: case 1290: case 1291:
+ case 1292: case 1293: case 1294:
+ if ( !(((utf32) % 2 == 1))) {
+ lc = (utf32 + 1);
+ }
+ break;
+ case 313: case 314: case 315: case 316: case 317: case 318: case 319: case 320: case 321: case 322:
+ case 323: case 324: case 325: case 326: case 327: case 377: case 378: case 379: case 380: case 381:
+ case 459: case 460: case 461: case 462: case 463: case 464: case 465: case 466: case 467: case 468: case 469: case 470:
+ case 471: case 472: case 473: case 474: case 475: case 1217: case 1218: case 1219: case 1220: case 1221: case 1222:
+ case 1223: case 1224: case 1225: case 1226: case 1227: case 1228: case 1229:
+ if (((utf32) % 2 == 1)) {
+ lc = (utf32 + 1);
+ }
+ break;
+ case 1024: case 1025: case 1026: case 1027: case 1028: case 1029: case 1030: case 1031: case 1032: case 1033:
+ case 1034: case 1035: case 1036: case 1037: case 1038: case 1039:
+ lc = (utf32 + 80);
+ break;
+ case 1329: case 1330: case 1331: case 1332: case 1333: case 1334: case 1335: case 1336: case 1337: case 1338:
+ case 1339: case 1340: case 1341: case 1342: case 1343: case 1344: case 1345: case 1346: case 1347: case 1348: case 1349: case 1350:
+ case 1351: case 1352: case 1353: case 1354: case 1355: case 1356: case 1357: case 1358: case 1359: case 1360: case 1361: case 1362:
+ case 1363: case 1364: case 1365: case 1366:
+ lc = (utf32 + 48);
+ break;
+ case 304:
+ lc = 105;
+ break;
+ case 376:
+ lc = 255;
+ break;
+ case 385:
+ lc = 595;
+ break;
+ case 386:
+ lc = 387;
+ break;
+ case 388:
+ lc = 389;
+ break;
+ case 390:
+ lc = 596;
+ break;
+ case 391:
+ lc = 392;
+ break;
+ case 393:
+ lc = 598;
+ break;
+ case 394:
+ lc = 599;
+ break;
+ case 395:
+ lc = 396;
+ break;
+ case 398:
+ lc = 477;
+ break;
+ case 399:
+ lc = 601;
+ break;
+ case 400:
+ lc = 603;
+ break;
+ case 401:
+ lc = 402;
+ break;
+ case 403:
+ lc = 608;
+ break;
+ case 404:
+ lc = 611;
+ break;
+ case 406:
+ lc = 617;
+ break;
+ case 407:
+ lc = 616;
+ break;
+ case 408:
+ lc = 409;
+ break;
+ case 412:
+ lc = 623;
+ break;
+ case 413:
+ lc = 626;
+ break;
+ case 415:
+ lc = 629;
+ break;
+ case 422:
+ lc = 640;
+ break;
+ case 423:
+ lc = 424;
+ break;
+ case 425:
+ lc = 643;
+ break;
+ case 428:
+ lc = 429;
+ break;
+ case 430:
+ lc = 648;
+ break;
+ case 431:
+ lc = 432;
+ break;
+ case 433:
+ lc = 650;
+ break;
+ case 434:
+ lc = 651;
+ break;
+ case 435:
+ lc = 436;
+ break;
+ case 437:
+ lc = 438;
+ break;
+ case 439:
+ lc = 658;
+ break;
+ case 440:
+ lc = 441;
+ break;
+ case 444:
+ lc = 445;
+ break;
+ case 452:
+ lc = 454;
+ break;
+ case 453:
+ lc = 454;
+ break;
+ case 455:
+ lc = 457;
+ break;
+ case 456:
+ lc = 457;
+ break;
+ case 458:
+ lc = 460;
+ break;
+ case 497:
+ lc = 499;
+ break;
+ case 498:
+ lc = 499;
+ break;
+ case 500:
+ lc = 501;
+ break;
+ case 502:
+ lc = 405;
+ break;
+ case 503:
+ lc = 447;
+ break;
+ case 544:
+ lc = 414;
+ break;
+ case 902:
+ lc = 940;
+ break;
+ case 904:
+ lc = 941;
+ break;
+ case 905:
+ lc = 942;
+ break;
+ case 906:
+ lc = 943;
+ break;
+ case 908:
+ lc = 972;
+ break;
+ case 910:
+ lc = 973;
+ break;
+ case 911:
+ lc = 974;
+ break;
+ case 1012:
+ lc = 952;
+ break;
+ case 1015:
+ lc = 1016;
+ break;
+ case 1017:
+ lc = 1010;
+ break;
+ case 1018:
+ lc = 1019;
+ break;
+ case 1272:
+ lc = 1273;
+ break;
+ default:
+ break;
+ }
+ }
+ return lc;
+}
+
+/**
+ * Converts utf32 input to uppercase
+ * @param utf32 : a single character encoded in UTF32
+ * @return a single uppercase character encoded in UTF32
+*/
+static picoos_uint32 base_utf32_uppercase (picoos_uint32 utf32)
+{
+ picoos_uint32 lc;
+
+ lc = utf32;
+ if (((utf32 >= 65345) && (utf32 <= 65370))) {
+ lc = (utf32 - 32);
+ } else if (((utf32 >= 66600) && (utf32 <= 66639))) {
+ lc = (utf32 - 40);
+ } else if (((utf32 >= 7681) && (utf32 <= 9449))) {
+ switch (utf32) {
+ case 7681: case 7682: case 7683: case 7684: case 7685: case 7686: case 7687: case 7688: case 7689: case 7690:
+ case 7691: case 7692: case 7693: case 7694: case 7695: case 7696: case 7697: case 7698: case 7699: case 7700: case 7701: case 7702:
+ case 7703: case 7704: case 7705: case 7706: case 7707: case 7708: case 7709: case 7710: case 7711: case 7712: case 7713: case 7714:
+ case 7715: case 7716: case 7717: case 7718: case 7719: case 7720: case 7721: case 7722: case 7723: case 7724: case 7725: case 7726:
+ case 7727: case 7728: case 7729: case 7730: case 7731: case 7732: case 7733: case 7734: case 7735: case 7736: case 7737: case 7738:
+ case 7739: case 7740: case 7741: case 7742: case 7743: case 7744: case 7745: case 7746: case 7747: case 7748: case 7749: case 7750:
+ case 7751: case 7752: case 7753: case 7754: case 7755: case 7756: case 7757: case 7758: case 7759: case 7760: case 7761: case 7762:
+ case 7763: case 7764: case 7765: case 7766: case 7767: case 7768: case 7769: case 7770: case 7771: case 7772: case 7773: case 7774:
+ case 7775: case 7776: case 7777: case 7778: case 7779: case 7780: case 7781: case 7782: case 7783: case 7784: case 7785: case 7786:
+ case 7787: case 7788: case 7789: case 7790: case 7791: case 7792: case 7793: case 7794: case 7795: case 7796: case 7797: case 7798:
+ case 7799: case 7800: case 7801: case 7802: case 7803: case 7804: case 7805: case 7806: case 7807: case 7808: case 7809: case 7810:
+ case 7811: case 7812: case 7813: case 7814: case 7815: case 7816: case 7817: case 7818: case 7819: case 7820: case 7821: case 7822:
+ case 7823: case 7824: case 7825: case 7826: case 7827: case 7828: case 7829: case 7841: case 7842: case 7843: case 7844:
+ case 7845: case 7846: case 7847: case 7848: case 7849: case 7850: case 7851: case 7852: case 7853: case 7854: case 7855: case 7856:
+ case 7857: case 7858: case 7859: case 7860: case 7861: case 7862: case 7863: case 7864: case 7865: case 7866: case 7867: case 7868:
+ case 7869: case 7870: case 7871: case 7872: case 7873: case 7874: case 7875: case 7876: case 7877: case 7878: case 7879: case 7880:
+ case 7881: case 7882: case 7883: case 7884: case 7885: case 7886: case 7887: case 7888: case 7889: case 7890: case 7891: case 7892:
+ case 7893: case 7894: case 7895: case 7896: case 7897: case 7898: case 7899: case 7900: case 7901: case 7902: case 7903: case 7904:
+ case 7905: case 7906: case 7907: case 7908: case 7909: case 7910: case 7911: case 7912: case 7913: case 7914: case 7915: case 7916:
+ case 7917: case 7918: case 7919: case 7920: case 7921: case 7922: case 7923: case 7924: case 7925: case 7926: case 7927: case 7928:
+ case 7929:
+ if (((utf32) % 2 == 1)) {
+ lc = (utf32 - 1);
+ }
+ break;
+ case 7936: case 7937: case 7938: case 7939: case 7940: case 7941: case 7942: case 7943: case 7952:
+ case 7953: case 7954: case 7955: case 7956: case 7957: case 7968: case 7969: case 7970: case 7971: case 7972: case 7973:
+ case 7974: case 7975: case 7984: case 7985: case 7986: case 7987: case 7988: case 7989: case 7990: case 7991:
+ case 8000: case 8001: case 8002: case 8003: case 8004: case 8005: case 8032: case 8033: case 8034: case 8035: case 8036:
+ case 8037: case 8038: case 8039: case 8064: case 8065: case 8066: case 8067: case 8068: case 8069: case 8070: case 8071:
+ case 8080: case 8081: case 8082: case 8083: case 8084: case 8085: case 8086: case 8087: case 8096: case 8097:
+ case 8098: case 8099: case 8100: case 8101: case 8102: case 8103:
+ lc = (utf32 + 8);
+ break;
+ case 8017: case 8018: case 8019: case 8020: case 8021: case 8022: case 8023:
+ if (((utf32) % 2 == 1)) {
+ lc = (utf32 + 8);
+ }
+ break;
+ case 8560: case 8561: case 8562: case 8563: case 8564: case 8565: case 8566: case 8567: case 8568: case 8569:
+ case 8570: case 8571: case 8572: case 8573: case 8574: case 8575:
+ lc = (utf32 - 16);
+ break;
+ case 9424: case 9425: case 9426: case 9427: case 9428: case 9429: case 9430: case 9431: case 9432: case 9433:
+ case 9434: case 9435: case 9436: case 9437: case 9438: case 9439: case 9440: case 9441: case 9442: case 9443: case 9444: case 9445:
+ case 9446: case 9447: case 9448: case 9449:
+ lc = (utf32 - 26);
+ break;
+ case 7835:
+ lc = 7776;
+ break;
+ case 8048:
+ lc = 8122;
+ break;
+ case 8049:
+ lc = 8123;
+ break;
+ case 8050:
+ lc = 8136;
+ break;
+ case 8051:
+ lc = 8137;
+ break;
+ case 8052:
+ lc = 8138;
+ break;
+ case 8053:
+ lc = 8139;
+ break;
+ case 8054:
+ lc = 8154;
+ break;
+ case 8055:
+ lc = 8155;
+ break;
+ case 8056:
+ lc = 8184;
+ break;
+ case 8057:
+ lc = 8185;
+ break;
+ case 8058:
+ lc = 8170;
+ break;
+ case 8059:
+ lc = 8171;
+ break;
+ case 8060:
+ lc = 8186;
+ break;
+ case 8061:
+ lc = 8187;
+ break;
+ case 8112:
+ lc = 8120;
+ break;
+ case 8113:
+ lc = 8121;
+ break;
+ case 8115:
+ lc = 8124;
+ break;
+ case 8126:
+ lc = 921;
+ break;
+ case 8131:
+ lc = 8140;
+ break;
+ case 8144:
+ lc = 8152;
+ break;
+ case 8145:
+ lc = 8153;
+ break;
+ case 8160:
+ lc = 8168;
+ break;
+ case 8161:
+ lc = 8169;
+ break;
+ case 8165:
+ lc = 8172;
+ break;
+ case 8179:
+ lc = 8188;
+ break;
+ default:
+ break;
+ }
+ } else {
+ switch (utf32) {
+ case 97: case 98: case 99: case 100: case 101: case 102: case 103: case 104: case 105: case 106:
+ case 107: case 108: case 109: case 110: case 111: case 112: case 113: case 114: case 115: case 116: case 117: case 118:
+ case 119: case 120: case 121: case 122: case 224: case 225: case 226: case 227: case 228: case 229: case 230:
+ case 231: case 232: case 233: case 234: case 235: case 236: case 237: case 238: case 239: case 240: case 241: case 242:
+ case 243: case 244: case 245: case 246: case 247: case 248: case 249: case 250: case 251: case 252: case 253: case 254:
+ case 945: case 946: case 947: case 948: case 949: case 950: case 951: case 952: case 953: case 954: case 955:
+ case 956: case 957: case 958: case 959: case 960: case 961: case 963: case 964: case 965: case 966: case 967:
+ case 968: case 969: case 970: case 971: case 1072: case 1073: case 1074: case 1075: case 1076: case 1077: case 1078:
+ case 1079: case 1080: case 1081: case 1082: case 1083: case 1084: case 1085: case 1086: case 1087: case 1088: case 1089: case 1090:
+ case 1091: case 1092: case 1093: case 1094: case 1095: case 1096: case 1097: case 1098: case 1099: case 1100: case 1101: case 1102:
+ case 1103:
+ if ((utf32 != 247)) {
+ lc = (utf32 - 32);
+ }
+ break;
+ case 257: case 258: case 259: case 260: case 261: case 262: case 263: case 264: case 265: case 266:
+ case 267: case 268: case 269: case 270: case 271: case 272: case 273: case 274: case 275: case 276: case 277: case 278:
+ case 279: case 280: case 281: case 282: case 283: case 284: case 285: case 286: case 287: case 288: case 289: case 290:
+ case 291: case 292: case 293: case 294: case 295: case 296: case 297: case 298: case 299: case 300: case 301: case 302:
+ case 303: case 304: case 306: case 307: case 308: case 309: case 310: case 311: case 331: case 332:
+ case 333: case 334: case 335: case 336: case 337: case 338: case 339: case 340: case 341: case 342: case 343: case 344:
+ case 345: case 346: case 347: case 348: case 349: case 350: case 351: case 352: case 353: case 354: case 355: case 356:
+ case 357: case 358: case 359: case 360: case 361: case 362: case 363: case 364: case 365: case 366: case 367: case 368:
+ case 369: case 370: case 371: case 372: case 373: case 374: case 375: case 417: case 418: case 419: case 420:
+ case 421: case 481: case 482: case 483: case 484: case 485: case 486: case 487: case 488: case 489: case 490:
+ case 491: case 492: case 493: case 494: case 495: case 507: case 508: case 509: case 510: case 511:
+ case 513: case 514: case 515: case 516: case 517: case 518: case 519: case 520: case 521: case 522: case 523: case 524:
+ case 525: case 526: case 527: case 528: case 529: case 530: case 531: case 532: case 533: case 534: case 535: case 536:
+ case 537: case 538: case 539: case 540: case 541: case 542: case 543: case 544: case 546: case 547: case 548:
+ case 549: case 550: case 551: case 552: case 553: case 554: case 555: case 556: case 557: case 558: case 559: case 560:
+ case 561: case 562: case 563: case 985: case 986: case 987: case 988: case 989: case 990: case 991: case 992:
+ case 993: case 994: case 995: case 996: case 997: case 998: case 999: case 1000: case 1001: case 1002: case 1003: case 1004:
+ case 1005: case 1006: case 1007: case 1121: case 1122: case 1123: case 1124: case 1125: case 1126: case 1127: case 1128:
+ case 1129: case 1130: case 1131: case 1132: case 1133: case 1134: case 1135: case 1136: case 1137: case 1138: case 1139: case 1140:
+ case 1141: case 1142: case 1143: case 1144: case 1145: case 1146: case 1147: case 1148: case 1149: case 1150: case 1151: case 1152:
+ case 1153: case 1163: case 1164: case 1165: case 1166: case 1167: case 1168: case 1169: case 1170: case 1171: case 1172:
+ case 1173: case 1174: case 1175: case 1176: case 1177: case 1178: case 1179: case 1180: case 1181: case 1182: case 1183: case 1184:
+ case 1185: case 1186: case 1187: case 1188: case 1189: case 1190: case 1191: case 1192: case 1193: case 1194: case 1195: case 1196:
+ case 1197: case 1198: case 1199: case 1200: case 1201: case 1202: case 1203: case 1204: case 1205: case 1206: case 1207: case 1208:
+ case 1209: case 1210: case 1211: case 1212: case 1213: case 1214: case 1215: case 1233: case 1234: case 1235: case 1236:
+ case 1237: case 1238: case 1239: case 1240: case 1241: case 1242: case 1243: case 1244: case 1245: case 1246: case 1247: case 1248:
+ case 1249: case 1250: case 1251: case 1252: case 1253: case 1254: case 1255: case 1256: case 1257: case 1258: case 1259: case 1260:
+ case 1261: case 1262: case 1263: case 1264: case 1265: case 1266: case 1267: case 1268: case 1269: case 1281: case 1282:
+ case 1283: case 1284: case 1285: case 1286: case 1287: case 1288: case 1289: case 1290: case 1291: case 1292: case 1293: case 1294:
+ case 1295:
+ if (((utf32) % 2 == 1)) {
+ lc = (utf32 - 1);
+ }
+ break;
+ case 314: case 315: case 316: case 317: case 318: case 319: case 320: case 321: case 322: case 323:
+ case 324: case 325: case 326: case 327: case 328: case 378: case 379: case 380: case 381: case 382:
+ case 464: case 465: case 466: case 467: case 468: case 469: case 470: case 471: case 472: case 473: case 474: case 475:
+ case 476: case 1218: case 1219: case 1220: case 1221: case 1222: case 1223: case 1224: case 1225: case 1226: case 1227:
+ case 1228: case 1229: case 1230:
+ if ( !(((utf32) % 2 == 1))) {
+ lc = (utf32 - 1);
+ }
+ break;
+ case 1104: case 1105: case 1106: case 1107: case 1108: case 1109: case 1110: case 1111: case 1112: case 1113:
+ case 1114: case 1115: case 1116: case 1117: case 1118: case 1119:
+ lc = (utf32 - 80);
+ break;
+ case 1377: case 1378: case 1379: case 1380: case 1381: case 1382: case 1383: case 1384: case 1385: case 1386:
+ case 1387: case 1388: case 1389: case 1390: case 1391: case 1392: case 1393: case 1394: case 1395: case 1396: case 1397: case 1398:
+ case 1399: case 1400: case 1401: case 1402: case 1403: case 1404: case 1405: case 1406: case 1407: case 1408: case 1409: case 1410:
+ case 1411: case 1412: case 1413: case 1414:
+ lc = (utf32 - 48);
+ break;
+ case 181:
+ lc = 924;
+ break;
+ case 255:
+ lc = 376;
+ break;
+ case 305:
+ lc = 73;
+ break;
+ case 383:
+ lc = 83;
+ break;
+ case 387:
+ lc = 386;
+ break;
+ case 389:
+ lc = 388;
+ break;
+ case 392:
+ lc = 391;
+ break;
+ case 396:
+ lc = 395;
+ break;
+ case 402:
+ lc = 401;
+ break;
+ case 405:
+ lc = 502;
+ break;
+ case 409:
+ lc = 408;
+ break;
+ case 414:
+ lc = 544;
+ break;
+ case 424:
+ lc = 423;
+ break;
+ case 429:
+ lc = 428;
+ break;
+ case 432:
+ lc = 431;
+ break;
+ case 436:
+ lc = 435;
+ break;
+ case 438:
+ lc = 437;
+ break;
+ case 441:
+ lc = 440;
+ break;
+ case 445:
+ lc = 444;
+ break;
+ case 447:
+ lc = 503;
+ break;
+ case 453:
+ lc = 452;
+ break;
+ case 454:
+ lc = 452;
+ break;
+ case 456:
+ lc = 455;
+ break;
+ case 457:
+ lc = 455;
+ break;
+ case 459:
+ lc = 458;
+ break;
+ case 460:
+ lc = 458;
+ break;
+ case 462:
+ lc = 461;
+ break;
+ case 477:
+ lc = 398;
+ break;
+ case 479:
+ lc = 478;
+ break;
+ case 498:
+ lc = 497;
+ break;
+ case 499:
+ lc = 497;
+ break;
+ case 501:
+ lc = 500;
+ break;
+ case 505:
+ lc = 504;
+ break;
+ case 595:
+ lc = 385;
+ break;
+ case 596:
+ lc = 390;
+ break;
+ case 598:
+ lc = 393;
+ break;
+ case 599:
+ lc = 394;
+ break;
+ case 601:
+ lc = 399;
+ break;
+ case 603:
+ lc = 400;
+ break;
+ case 608:
+ lc = 403;
+ break;
+ case 611:
+ lc = 404;
+ break;
+ case 616:
+ lc = 407;
+ break;
+ case 617:
+ lc = 406;
+ break;
+ case 623:
+ lc = 412;
+ break;
+ case 626:
+ lc = 413;
+ break;
+ case 629:
+ lc = 415;
+ break;
+ case 640:
+ lc = 422;
+ break;
+ case 643:
+ lc = 425;
+ break;
+ case 648:
+ lc = 430;
+ break;
+ case 650:
+ lc = 433;
+ break;
+ case 651:
+ lc = 434;
+ break;
+ case 658:
+ lc = 439;
+ break;
+ case 837:
+ lc = 921;
+ break;
+ case 940:
+ lc = 902;
+ break;
+ case 941:
+ lc = 904;
+ break;
+ case 942:
+ lc = 905;
+ break;
+ case 943:
+ lc = 906;
+ break;
+ case 962:
+ lc = 931;
+ break;
+ case 972:
+ lc = 908;
+ break;
+ case 973:
+ lc = 910;
+ break;
+ case 974:
+ lc = 911;
+ break;
+ case 976:
+ lc = 914;
+ break;
+ case 977:
+ lc = 920;
+ break;
+ case 981:
+ lc = 934;
+ break;
+ case 982:
+ lc = 928;
+ break;
+ case 1008:
+ lc = 922;
+ break;
+ case 1009:
+ lc = 929;
+ break;
+ case 1010:
+ lc = 1017;
+ break;
+ case 1013:
+ lc = 917;
+ break;
+ case 1016:
+ lc = 1015;
+ break;
+ case 1019:
+ lc = 1018;
+ break;
+ case 1273:
+ lc = 1272;
+ break;
+ default:
+ break;
+ }
+ }
+ return lc;
+}
+
+/**
+ * Gets the UTF8 character 'utf8char' from the UTF8 string 'utf8str' from
+ * position 'pos'
+ * @param utf8str: utf8 string
+ * @param pos: position from where the utf8 character is copied
+ * (also output set as position of the next utf8 character in the utf8 string)
+ * @param utf8char: zero terminated utf8 character containing 1 to 4 bytes (output)
+*/
+static void picobase_get_utf8char (picoos_uint8 utf8[], picoos_int32 * pos, picobase_utf8char utf8char)
+{
+
+ int i;
+ int l;
+
+ utf8char[0] = 0;
+ l = picobase_det_utf8_length(utf8[*pos]);
+ i = 0;
+ while ((((i < l) && (i < PICOBASE_UTF8_MAXLEN)) && (utf8[*pos] != 0))) {
+ utf8char[i] = utf8[*pos];
+ (*pos)++;
+ i++;
+ }
+ utf8char[i] = 0;
+}
+
+
+picoos_uint8 picobase_get_next_utf8char(const picoos_uint8 *utf8s,
+ const picoos_uint32 utf8slenmax,
+ picoos_uint32 *pos,
+ picobase_utf8char utf8char) {
+ picoos_uint8 i;
+ picoos_uint8 len;
+ picoos_uint32 poscnt;
+
+ utf8char[0] = 0;
+ len = picobase_det_utf8_length(utf8s[*pos]);
+ if ((((*pos) + len) > utf8slenmax) ||
+ (len > PICOBASE_UTF8_MAXLEN)) {
+ return FALSE;
+ }
+
+ poscnt = *pos;
+ i = 0;
+ while ((i < len) && (utf8s[poscnt] != 0)) {
+ utf8char[i] = utf8s[poscnt];
+ poscnt++;
+ i++;
+ }
+ utf8char[i] = 0;
+ if ((i < len) && (utf8s[poscnt] == 0)) {
+ return FALSE;
+ }
+ *pos = poscnt;
+ return TRUE;
+}
+
+picoos_uint8 picobase_get_next_utf8charpos(const picoos_uint8 *utf8s,
+ const picoos_uint32 utf8slenmax,
+ picoos_uint32 *pos) {
+ picoos_uint8 i;
+ picoos_uint8 len;
+ picoos_uint32 poscnt;
+
+ len = picobase_det_utf8_length(utf8s[*pos]);
+ if ((((*pos) + len) > utf8slenmax) ||
+ (len > PICOBASE_UTF8_MAXLEN)){
+ return FALSE;
+ }
+
+ poscnt = *pos;
+ i = 0;
+ while ((i < len) && (utf8s[poscnt] != 0)) {
+ poscnt++;
+ i++;
+ }
+ if ((i < len) && (utf8s[poscnt] == 0)) {
+ return FALSE;
+ }
+ *pos = poscnt;
+ return TRUE;
+}
+
+picoos_uint8 picobase_get_prev_utf8char(const picoos_uint8 *utf8s,
+ const picoos_uint32 utf8slenmin,
+ picoos_uint32 *pos,
+ picobase_utf8char utf8char) {
+ picoos_uint8 i, j;
+ picoos_uint8 len;
+ picoos_uint32 poscnt;
+
+ utf8char[0] = 0;
+ if ((*pos) == 0) {
+ return FALSE;
+ }
+ poscnt = (*pos) - 1;
+ i = 1;
+ while ((i <= PICOBASE_UTF8_MAXLEN) && (poscnt >= utf8slenmin) &&
+ (utf8s[poscnt] != 0)) {
+ len = picobase_det_utf8_length(utf8s[poscnt]);
+ if (len == i) {
+ for (j = 0; j < len; j++) {
+ utf8char[j] = utf8s[poscnt + j];
+ }
+ utf8char[j] = 0;
+ *pos = poscnt;
+ return TRUE;
+ }
+ i++;
+ poscnt--;
+ }
+ return FALSE;
+}
+
+picoos_uint8 picobase_get_prev_utf8charpos(const picoos_uint8 *utf8s,
+ const picoos_uint32 utf8slenmin,
+ picoos_uint32 *pos) {
+ picoos_uint8 i;
+ picoos_uint8 len;
+ picoos_uint32 poscnt;
+
+ if ((*pos) == 0) {
+ return FALSE;
+ }
+ poscnt = (*pos) - 1;
+ i = 1;
+ while ((i <= PICOBASE_UTF8_MAXLEN) && (poscnt >= utf8slenmin) &&
+ (utf8s[poscnt] != 0)) {
+ len = picobase_det_utf8_length(utf8s[poscnt]);
+ if (len == i) {
+ *pos = poscnt;
+ return TRUE;
+ }
+ i++;
+ poscnt--;
+ }
+ return FALSE;
+}
+
+/**
+ * Converts utf8 input to utf32
+ * @param utf8[] : character encoded in utf8
+ * @param done : boolean indicating the completion of the operation (FALSE: conversion not done)
+ * @return a single character encoded in UTF32
+*/
+static picobase_utf32 picobase_utf8_to_utf32 (picoos_uint8 utf8[], picoos_uint8 * done)
+{
+ (*done) = TRUE;
+ if ((utf8[0] < (picoos_uint8)'\200')) {
+ return utf8[0];
+ } else if ((utf8[0] >= (picoos_uint8)'\370')) {
+ return 0;
+ } else if ((utf8[0] >= (picoos_uint8)'\360')) {
+ return ((((262144 * (utf8[0] % 8)) + (4096 * (utf8[1] % 64))) + (64 * (utf8[2] % 64))) + (utf8[3] % 64));
+ } else if ((utf8[0] >= (picoos_uint8)'\340')) {
+ return (((4096 * (utf8[0] % 16)) + (64 * (utf8[1] % 64))) + (utf8[2] % 64));
+ } else if ((utf8[(0)] >= (picoos_uint8)'\300')) {
+ return ((64 * (utf8[0] % 32)) + (utf8[1] % 64));
+ } else {
+ (*done) = FALSE;
+ return 0;
+ }
+}
+
+static picoos_int32 picobase_utf32_to_utf8 (picobase_utf32 utf32, picobase_utf8 utf8[], picoos_int32 utf8MaxLen, picoos_uint8 * done)
+{
+ picoos_int32 len;
+
+ (*done) = TRUE;
+ if (utf8MaxLen >= 4) {
+ if (utf32 < 128) {
+ len = 1;
+ utf8[0] = utf32;
+ } else if (utf32 < 2048) {
+ len = 2;
+ utf8[1] = (128 + (utf32 % 64));
+ utf32 = (utf32 / 64);
+ utf8[0] = (192 + (utf32 % 32));
+ } else if (utf32 < 65536) {
+ len = 3;
+ utf8[2] = (128 + (utf32 % 64));
+ utf32 = (utf32 / 64);
+ utf8[1] = (128 + (utf32 % 64));
+ utf32 = (utf32 / 64);
+ utf8[0] = (224 + utf32);
+ } else if (utf32 < 1048576) {
+ len = 4;
+ utf8[3] = (128 + (utf32 % 64));
+ utf32 = (utf32 / 64);
+ utf8[2] = (128 + (utf32 % 64));
+ utf32 = (utf32 / 64);
+ utf8[1] = (128 + (utf32 % 64));
+ utf32 = (utf32 / 64);
+ utf8[0] = (240 + utf32);
+ } else {
+ (*done) = FALSE;
+ return 0;
+ }
+ if (len <= (utf8MaxLen-1)) {
+ utf8[len] = 0;
+ }
+ return len;
+ } else {
+ (*done) = FALSE;
+ return 0;
+ }
+}
+
+
+extern picoos_int32 picobase_lowercase_utf8_str (picoos_uchar utf8str[], picoos_char lowercase[], int lowercaseMaxLen, picoos_uint8 * done)
+{
+ picobase_utf8char utf8char;
+ picoos_int32 i;
+ picoos_int32 j;
+ picoos_int32 k;
+ picoos_int32 l;
+ picobase_utf32 utf32;
+ picoos_uint8 done1;
+
+ k = 0;
+ i = 0;
+ (*done) = TRUE;
+ while (utf8str[i] != 0) {
+ picobase_get_utf8char(utf8str,& i,utf8char);
+ utf32 = picobase_utf8_to_utf32(utf8char, & done1);
+ utf32 = base_utf32_lowercase(utf32);
+ l = picobase_utf32_to_utf8(utf32, utf8char, PICOBASE_UTF8_MAXLEN, & done1);
+ j = 0;
+ while ((j < l) && (k < (lowercaseMaxLen-1))) {
+ lowercase[k] = utf8char[j];
+ k++;
+ j++;
+ }
+ *done = *done && (j == l);
+ }
+ lowercase[k] = 0;
+ return k;
+}
+
+
+extern picoos_int32 picobase_uppercase_utf8_str (picoos_uchar utf8str[], picoos_char uppercase[], int uppercaseMaxLen, picoos_uint8 * done)
+{
+ picobase_utf8char utf8char;
+ picoos_int32 i;
+ picoos_int32 j;
+ picoos_int32 k;
+ picoos_int32 l;
+ picobase_utf32 utf32;
+ picoos_uint8 done1;
+
+ k = 0;
+ i = 0;
+ (*done) = TRUE;
+ while (utf8str[i] != 0) {
+ picobase_get_utf8char(utf8str,& i,utf8char);
+ utf32 = picobase_utf8_to_utf32(utf8char, & done1);
+ utf32 = base_utf32_uppercase(utf32);
+ l = picobase_utf32_to_utf8(utf32, utf8char, PICOBASE_UTF8_MAXLEN, & done1);
+ j = 0;
+ while ((j < l) && (k < (uppercaseMaxLen-1))) {
+ uppercase[k] = utf8char[j];
+ k++;
+ j++;
+ }
+ *done = *done && (j == l);
+ }
+ uppercase[k] = 0;
+ return k;
+}
+
+
+extern picoos_bool picobase_is_utf8_uppercase (picoos_uchar utf8str[], picoos_int32 utf8strmaxlen)
+{
+ picobase_utf8char utf8char;
+ picoos_int32 i;
+ picoos_uint32 utf32;
+ picoos_bool done;
+ picoos_bool isUpperCase;
+
+ isUpperCase = TRUE;
+ i = 0;
+ while (isUpperCase && (i <= utf8strmaxlen-1) && (utf8str[i] != 0)) {
+ picobase_get_utf8char(utf8str,& i,utf8char);
+ utf32 = picobase_utf8_to_utf32(utf8char,& done);
+ isUpperCase = isUpperCase && (utf32 == base_utf32_uppercase(utf32));
+ }
+ return isUpperCase;
+}
+
+
+extern picoos_bool picobase_is_utf8_lowercase (picoos_uchar utf8str[], picoos_int32 utf8strmaxlen)
+{
+ picobase_utf8char utf8char;
+ picoos_int32 i;
+ picoos_uint32 utf32;
+ picoos_bool done;
+ picoos_bool isLowerCase;
+
+ isLowerCase = TRUE;
+ i = 0;
+ while (isLowerCase && (i <= utf8strmaxlen-1) && (utf8str[i] != 0)) {
+ picobase_get_utf8char(utf8str,& i,utf8char);
+ utf32 = picobase_utf8_to_utf32(utf8char,& done);
+ isLowerCase = isLowerCase && (utf32 == base_utf32_lowercase(utf32));
+ }
+ return isLowerCase;
+}
+
+
+#ifdef __cplusplus
+}
+#endif
+
+
+
+/* end */
diff --git a/lib/picobase.h b/lib/picobase.h
new file mode 100644
index 0000000..1c384e1
--- /dev/null
+++ b/lib/picobase.h
@@ -0,0 +1,171 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picobase.h
+ *
+ * base functionality
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#ifndef PICOBASE_H_
+#define PICOBASE_H_
+
+#include "picoos.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+/* maximum number of bytes of an UTF8 character */
+#define PICOBASE_UTF8_MAXLEN 4
+
+typedef picoos_uint8 picobase_utf8char[PICOBASE_UTF8_MAXLEN+1]; /* always zero terminated */
+typedef picoos_uint8 picobase_utf8;
+typedef picoos_uint16 picobase_utf16;
+typedef picoos_uint32 picobase_utf32;
+
+/* ***************************************************************/
+/* Unicode UTF8 functions */
+/* ***************************************************************/
+
+/**
+ * Determines the number of UTF8 characters contained in
+ * the UTF8 string 'utf8str' of maximum length maxlen (in bytes)
+ * @param utf8str : a string encoded in UTF8
+ * @param maxlen : max length (in bytes) accessible in utf8str
+ * @return >=0 : length of the UTF8 string in number of UTF8 characters
+ * up to the first '\0' or maxlen
+ * @return <0 : not starting with a valid UTF8 character
+ * @remarks strict implementation, not allowing invalid utf8
+*/
+picoos_int32 picobase_utf8_length(const picoos_uint8 *utf8str,
+ const picoos_uint16 maxlen);
+
+
+/**
+ * Determines the number of bytes an UTF8 character used based
+ * on the first byte of the UTF8 character
+ * @param firstchar: the first (and maybe only) byte of an UTF8 character
+ * @return positive value in {1,4} : number of bytes of the UTF8 character
+ * @return 0 :if not a valid UTF8 character start
+ * @remarks strict implementation, not allowing invalid utf8
+*/
+/* picoos_uint8 picobase_det_utf8_length(const picoos_uint8 firstchar); */
+
+#define picobase_det_utf8_length(x) ( ((x)<(picoos_uint8)'\200')?1:(((x)>=(picoos_uint8)'\370')?0:(((x)>=(picoos_uint8)'\360')?4:(((x)>=(picoos_uint8)'\340')?3:(((x)>=(picoos_uint8)'\300')?2:0)))) )
+
+/**
+ * Converts the content of 'utf8str' to lowercase and stores it on 'lowercase'
+ * on the first byte of the UTF8 character
+ * @param utf8str : utf8 string
+ * @param lowercaseMaxLen : maximal number of bytes available in 'lowercase'
+ * @param lowercase : string converted to lowercase (output)
+ * @param done : flag to report success/failure of the operation (output)
+ * @return TRUE if successful, FALSE otherwise
+*/
+picoos_int32 picobase_lowercase_utf8_str (picoos_uchar utf8str[], picoos_char lowercase[], picoos_int32 lowercaseMaxLen, picoos_uint8 * done);
+
+/**
+ * Converts the content of 'utf8str' to upperrcase and stores it on 'uppercase'
+ * @param utf8str : utf8 string
+ * @param uppercase : string converted to uppercase (output)
+ * @param uppercaseMaxLen : maximal number of bytes available in 'uppercase'
+ * @param done : flag to report success/failure of the operation (output)
+ * @return TRUE if successful, FALSE otherwise
+*/
+picoos_int32 picobase_uppercase_utf8_str (picoos_uchar utf8str[], picoos_char uppercase[], int uppercaseMaxLen, picoos_uint8 * done);
+
+/**
+ * Gets next UTF8 character 'utf8char' from the UTF8 string
+ * 'utf8s' starting at position 'pos'
+ * @param utf8s : UTF8 string
+ * @param utf8slenmax : max length accessible in utf8s
+ * @param pos : position from where the UTF8 character is checked and copied
+ * (set also as output to the position directly following the UTF8 char)
+ * @param utf8char : zero terminated UTF8 character containing 1 to 4 bytes (output)
+ * @return TRUE if okay
+ * @return FALSE if there is no valid UTF8 char or no more UTF8 char available within utf8len
+*/
+picoos_uint8 picobase_get_next_utf8char(const picoos_uint8 *utf8s,
+ const picoos_uint32 utf8slenmax,
+ picoos_uint32 *pos,
+ picobase_utf8char utf8char);
+
+/**
+ * Same as picobase_get_next_utf8char
+ * without copying the char to utf8char
+*/
+picoos_uint8 picobase_get_next_utf8charpos(const picoos_uint8 *utf8s,
+ const picoos_uint32 utf8slenmax,
+ picoos_uint32 *pos);
+
+/**
+ * Gets previous UTF8 character 'utf8char' from the UTF8 string
+ * 'utf8s' starting the backward search at position 'pos-1'
+ * @param utf8s : UTF8 string
+ * @param utf8slenmin : min length accessible in utf8s
+ * @param pos : the search for the prev UTF8 char starts at [pos-1]
+ * (set also as output to the start position of the prev UTF8 character)
+ * @param utf8char : zero terminated UTF8 character containing 1 to 4 bytes (output)
+ * @return TRUE if okay
+ * @return FALSE if there is no valid UTF8 char preceeding pos or no more UTF8 char available within utf8len
+*/
+picoos_uint8 picobase_get_prev_utf8char(const picoos_uint8 *utf8s,
+ const picoos_uint32 utf8slenmin,
+ picoos_uint32 *pos,
+ picobase_utf8char utf8char);
+
+/**
+ * Same as picobase_get_prev_utf8char
+ * without copying the char to utf8char
+*/
+picoos_uint8 picobase_get_prev_utf8charpos(const picoos_uint8 *utf8s,
+ const picoos_uint32 utf8slenmin,
+ picoos_uint32 *pos);
+
+
+/**
+ * returns TRUE if the input string is UTF8 and uppercase
+ * @param str : UTF8 string
+ * @param strmaxlen : max length for the input string
+ * @return TRUE if string is UTF8 and uppercase
+ * @return FALSE otherwise
+*/
+extern picoos_bool picobase_is_utf8_uppercase (picoos_uchar str[], picoos_int32 strmaxlen);
+
+/**
+ * returns TRUE if the input string is UTF8 and lowercase
+ * @param str : UTF8 string
+ * @param strmaxlen : max length for the input string
+ * @return TRUE if string is UTF8 and lowercase
+ * @return FALSE otherwise
+*/
+extern picoos_bool picobase_is_utf8_lowercase (picoos_uchar str[], picoos_int32 strmaxlen);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /*PICOBASE_H_*/
diff --git a/lib/picocep.c b/lib/picocep.c
new file mode 100644
index 0000000..2510c13
--- /dev/null
+++ b/lib/picocep.c
@@ -0,0 +1,1965 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picocep.c
+ *
+ * Phonetic to Acoustic Mapping PU - Implementation
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+/**
+ * @addtogroup picocep
+ * <b> Pico Cepstral Smoothing </b>\n
+ *
+ itemtype, iteminfo1, iteminfo2, content -> TYPE(INFO1,INFO2)content
+ in the following
+
+ items input
+
+ processed:
+
+ - PHONE(PHONID,StatesPerPhone)[FramesPerState|f0Index|mgcIndex]{StatesPerPhone}
+
+ (StatesPerPhone is a constant (5) for the time being. The content size is therfore allways 34)
+
+ unprocessed:
+ - all other item types are forwarded through the PU without modification
+
+ items output
+
+ FRAME(PHONID,F0 [?])ceps{25}
+
+ each PHONE produces at least StatesPerPhone FRAME (if each state has FramesPerState == 1), but usually more.
+
+ minimal input size (before processing starts)
+
+ other limitations
+
+ */
+#include "picodefs.h"
+#include "picoos.h"
+#include "picodbg.h"
+#include "picodata.h"
+#include "picokpdf.h"
+#include "picodsp.h"
+#include "picocep.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+#define PICOCEP_MAXWINLEN 10000 /* maximum number of frames that can be smoothed, i.e. maximum sentence length */
+#define PICOCEP_MSGSTR_SIZE 32
+#define PICOCEP_IN_BUFF_SIZE PICODATA_BUFSIZE_DEFAULT
+
+#define PICOCEP_OUT_DATA_FORMAT PICODATA_ITEMINFO1_FRAME_PAR_DATA_FORMAT_FIXED /* we output coefficients as fixed point values */
+
+#define PICOCEP_STEPSTATE_COLLECT 0
+#define PICOCEP_STEPSTATE_PROCESS_PARSE 1
+#define PICOCEP_STEPSTATE_PROCESS_SMOOTH 2
+#define PICOCEP_STEPSTATE_PROCESS_FRAME 3
+#define PICOCEP_STEPSTATE_FEED 4
+
+#define PICOCEP_LFZINVPOW 31 /* cannot be higher than 31 because 1<<invpow must fit in uint32 */
+#define PICOCEP_MGCINVPOW 24
+#define PICOCEP_LFZDOUBLEDEC 1
+#define PICOCEP_MGCDOUBLEDEC 0
+
+typedef enum picocep_WantMeanOrIvar
+{
+ PICOCEP_WANTMEAN, PICOCEP_WANTIVAR
+} picocep_WantMeanOrIvar_t;
+
+typedef enum picocep_WantStaticOrDeltax
+{
+ PICOCEP_WANTSTATIC, PICOCEP_WANTDELTA, PICOCEP_WANTDELTA2
+} picocep_WantStaticOrDelta_t;
+
+/*
+ * Fixedpoint arithmetic (might go into a separate module if general enough and needed by other modules)
+ */
+
+#if defined(PICO_DEBUG) || defined(PICO_DEVEL_MODE)
+int numlongmult = 0, numshortmult = 0;
+#endif
+
+#define POW1 (0x1)
+#define POW2 (0x2)
+#define POW3 (0x4)
+#define POW4 (0x8)
+#define POW5 (0x10)
+#define POW6 (0x20)
+#define POW7 (0x40)
+#define POW8 (0x80)
+#define POW9 (0x100)
+#define POW10 (0x200)
+#define POW11 (0x400)
+#define POW12 (0x800)
+#define POW13 (0x1000)
+#define POW14 (0x2000)
+#define POW15 (0x4000)
+#define POW16 (0x8000)
+#define POW17 (0x10000)
+#define POW18 (0x20000)
+#define POW19 (0x40000)
+#define POW20 (0x80000)
+#define POW21 (0x100000)
+#define POW22 (0x200000)
+#define POW23 (0x400000)
+#define POW24 (0x800000)
+#define POW25 (0x1000000)
+#define POW26 (0x2000000)
+#define POW27 (0x4000000)
+#define POW28 (0x8000000)
+#define POW29 (0x10000000)
+#define POW30 (0x20000000)
+#define POW31 (0x40000000)
+
+/* item num restriction: maximum number of extended item heads in headx */
+#define PICOCEP_MAXNR_HEADX 60
+/* item num restriction: maximum size of all item contents together in cont */
+#define PICOCEP_MAXSIZE_CBUF 7680 /* (128 * PICOCEP_MAXNR_HEADX) */
+
+typedef struct
+{
+ picodata_itemhead_t head;
+ picoos_uint16 cind;
+ picoos_uint16 frame; /* sync position */
+} picoacph_headx_t;
+
+/*----------------------------------------------------------
+ // Name : cep_subobj
+ // Function: subobject definition for the cep processing
+ // Shortcut: cep
+ //---------------------------------------------------------*/
+typedef struct cep_subobj
+{
+ /*----------------------PU voice management------------------------------*/
+ /* picorsrc_Voice voice; */
+ /*----------------------PU state management------------------------------*/
+ picoos_uint8 procState; /* where to take up work at next processing step */
+ picoos_bool needMoreInput; /* more data necessary to start processing */
+ /* picoos_uint8 force; *//* forced processing (needMoreData but buffers full */
+ picoos_uint8 sentenceEnd;
+ picoos_uint8 feedFollowState;
+ picoos_bool inIgnoreState;
+ /*----------------------PU input management------------------------------*/
+ picoos_uint8 inBuf[PICODATA_MAX_ITEMSIZE]; /* internal input buffer */
+ picoos_uint16 inBufSize; /* actually allocated size */
+ picoos_uint16 inReadPos, inWritePos; /* next pos to read/write from/to inBuf*/
+ picoos_uint16 nextInPos;
+
+ picoacph_headx_t headx[PICOCEP_MAXNR_HEADX];
+ picoos_uint16 headxBottom; /* bottom */
+ picoos_uint16 headxWritePos; /* next free position; headx is empty if headxBottom == headxWritePos */
+
+ picoos_uint8 cbuf[PICOCEP_MAXSIZE_CBUF];
+ picoos_uint16 cbufBufSize; /* actually allocated size */
+ picoos_uint16 cbufWritePos; /* length, 0 if empty */
+
+ /*----------------------PU output management-----------------------------*/
+ picodata_itemhead_t framehead;
+ picoos_uint8 outBuf[PICODATA_MAX_ITEMSIZE]; /* internal output buffer (one item) */
+ picoos_uint16 outBufSize; /* allocated outBuf size */
+ picoos_uint16 outReadPos, outWritePos; /* next pos to read/write from/to outBuf*/
+
+ picoos_uint32 nNumFrames;
+ /*---------------------- other working variables ---------------------------*/
+
+ picoos_int32 diag0[PICOCEP_MAXWINLEN], diag1[PICOCEP_MAXWINLEN],
+ diag2[PICOCEP_MAXWINLEN], WUm[PICOCEP_MAXWINLEN],
+ invdiag0[PICOCEP_MAXWINLEN];
+
+ /*---------------------- constants --------------------------------------*/
+ picoos_int32 xi[5], x1[2], x2[3], xm[3], xn[2];
+ picoos_int32 xsqi[5], xsq1[2], xsq2[3], xsqm[3], xsqn[2];
+
+ picoos_uint32 scmeanpowLFZ, scmeanpowMGC;
+ picoos_uint32 scmeanLFZ, scmeanMGC;
+
+ /*---------------------- indices --------------------------------------*/
+ /* index buffer to hold indices as input for smoothing */
+ picoos_uint16 indicesLFZ[PICOCEP_MAXWINLEN];
+ picoos_uint16 indicesMGC[PICOCEP_MAXWINLEN];
+ picoos_uint16 indexReadPos, indexWritePos;
+ picoos_uint16 activeEndPos; /* end position of indices to be considered */
+
+ /* this is used for input and output */
+ picoos_uint8 phoneId[PICOCEP_MAXWINLEN]; /* synchronised with indexReadPos */
+
+ /*---------------------- coefficients --------------------------------------*/
+ /* output coefficients buffer */
+ picoos_int16 * outF0;
+ picoos_uint16 outF0ReadPos, outF0WritePos;
+ picoos_int16 * outXCep;
+ picoos_uint32 outXCepReadPos, outXCepWritePos; /* uint32 needed for MAXWINLEN*ceporder > 2^16 */
+ picoos_uint8 * outVoiced;
+ picoos_uint16 outVoicedReadPos, outVoicedWritePos;
+
+ /*---------------------- LINGWARE related data -------------------*/
+ /* pdflfz knowledge base */
+ picokpdf_PdfMUL pdflfz, pdfmgc;
+
+} cep_subobj_t;
+
+/**
+ * picocep_highestBit
+ * @brief find the highest non-zero bit in input x
+ * @remarks this may be implemented by comparing x to powers of 2
+ * or instead of calling this function perform multiplication
+ * and consult overflow register if available on target
+ * @note implemented as a series of macros
+ */
+
+#define picocep_highestBitNZ(x) (x>=POW17?(x>=POW25?(x>=POW29?(x>=POW31?31:(x>=POW30?30:29)):(x>=POW27?(x>=POW28?28:27):(x>=POW26?26:25))):(x>=POW21?(x>=POW23?(x>=POW24?24:23):(x>=POW22?22:21)):(x>=POW19?(x>=POW20?20:19):(x>=POW18?18:17)))):(x>=POW9?(x>=POW13?(x>=POW15?(x>=POW16?16:15):(x>=POW14?14:13)):(x>=POW11?(x>=POW12?12:11):(x>=POW10?10:9))):(x>=POW5?(x>=POW7?(x>=POW8?8:7):(x>=POW6?6:5)):(x>=POW3?(x>=POW4?4:3):(x>=POW2?2:1)))))
+#define picocep_highestBitU(x) (x==0?0:picocep_highestBitNZ(x))
+#define picocep_highestBitS(x,zz) (x==0?0:(x<0?((zz)=(-x),picocep_highestBitNZ(zz)):picocep_highestBitNZ(x)))
+
+/* ------------------------------------------------------------------------------
+ Internal function definitions
+ ---------------------------------------------------------------------------------*/
+
+static void initSmoothing(cep_subobj_t * cep);
+
+static picoos_int32 getFromPdf(picokpdf_PdfMUL pdf, picoos_uint32 vecstart,
+ picoos_uint8 cepnum, picocep_WantMeanOrIvar_t wantMeanOrIvar,
+ picocep_WantStaticOrDelta_t wantStaticOrDeltax);
+
+static void invMatrix(cep_subobj_t * cep, picoos_uint16 N,
+ picoos_int16 *smoothcep, picoos_uint8 cepnum,
+ picokpdf_PdfMUL pdf, picoos_uint8 invpow, picoos_uint8 invDoubleDec);
+
+static picoos_uint8 makeWUWandWUm(cep_subobj_t * cep, picokpdf_PdfMUL pdf,
+ picoos_uint16 *indices, picoos_uint16 b, picoos_uint16 N,
+ picoos_uint8 cepnum);
+
+static void getDirect(picokpdf_PdfMUL pdf, picoos_uint16 *indices,
+ picoos_uint16 activeEndPos,
+ picoos_uint8 cepnum, picoos_int16 *smoothcep);
+
+static void getVoiced(picokpdf_PdfMUL pdf, picoos_uint16 *indices,
+ picoos_uint16 activeEndPos,
+ picoos_uint8 *smoothcep);
+
+static picoos_uint16 get_pi_uint16(picoos_uint8 * buf, picoos_uint16 *pos);
+
+static void treat_phone(cep_subobj_t * cep, picodata_itemhead_t * ihead);
+
+static picoos_uint8 forwardingItem(picodata_itemhead_t * ihead);
+
+static picodata_step_result_t cepStep(register picodata_ProcessingUnit this,
+ picoos_int16 mode, picoos_uint16 * numBytesOutput);
+
+/* --------------------------------------------
+ * generic PU management
+ * --------------------------------------------
+ */
+
+/**
+ * initialization of a cep PU (processing unit)
+ * @param this : handle to a cep PU struct
+ * @return PICO_OK : init succeded
+ * @return PICO_ERR_OTHER : init failed
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t cepInitialize(register picodata_ProcessingUnit this)
+{
+ /*pico_status_t nRes;*/
+ cep_subobj_t * cep;
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ cep = (cep_subobj_t *) this->subObj;
+ /* inBuf */
+ cep->inBufSize = PICODATA_BUFSIZE_CEP;
+ cep->inReadPos = 0;
+ cep->inWritePos = 0;
+ /* headx and cbuf */
+ cep->headxBottom = cep->headxWritePos = 0;
+ cep->cbufBufSize = PICOCEP_MAXSIZE_CBUF;
+ cep->cbufWritePos = 0;
+ /* outBuf */
+ cep->outBufSize = PICODATA_MAX_ITEMSIZE;
+ cep->outReadPos = 0;
+ cep->outWritePos = 0;
+ /* indices* */
+ cep->indexReadPos = 0;
+ cep->indexWritePos = 0;
+ /* outCep, outF0, outVoiced */
+ cep->outXCepReadPos = 0;
+ cep->outXCepWritePos = 0;
+ cep->outVoicedReadPos = 0;
+ cep->outVoicedWritePos = 0;
+ cep->outF0ReadPos = 0;
+ cep->outF0WritePos = 0;
+
+ cep->needMoreInput = 0;
+ cep->inIgnoreState = 0;
+ cep->sentenceEnd = FALSE;
+ cep->procState = PICOCEP_STEPSTATE_COLLECT;
+
+ cep->nNumFrames = 0;
+
+ /*-----------------------------------------------------------------
+ * MANAGE Item I/O control management
+ ------------------------------------------------------------------*/
+ cep->activeEndPos = PICOCEP_MAXWINLEN;
+
+ /* kb pdflfz */
+ cep->pdflfz = picokpdf_getPdfMUL(
+ this->voice->kbArray[PICOKNOW_KBID_PDF_LFZ]);
+
+ /* kb pdfmgc */
+ cep->pdfmgc = picokpdf_getPdfMUL(
+ this->voice->kbArray[PICOKNOW_KBID_PDF_MGC]);
+
+ /* kb tab phones */
+ /* cep->phones =
+ picoktab_getPhones(this->voice->kbArray[PICOKNOW_KBID_TAB_PHONES]); */
+
+ /*---------------------- other working variables ---------------------------*/
+ /* define the (constant) FRAME_PAR item header */
+ cep->framehead.type = PICODATA_ITEM_FRAME_PAR;
+ cep->framehead.info1 = PICOCEP_OUT_DATA_FORMAT;
+ cep->framehead.info2 = cep->pdfmgc->ceporder;
+ cep->framehead.len = sizeof(picoos_uint16) + (cep->framehead.info2 + 4)
+ * sizeof(picoos_uint16);
+
+ /* constants used in makeWUWandWUm */
+ initSmoothing(cep);
+
+ cep->scmeanpowLFZ = cep->pdflfz->bigpow - cep->pdflfz->meanpow;
+ cep->scmeanpowMGC = cep->pdfmgc->bigpow - cep->pdfmgc->meanpow;
+
+ cep->scmeanLFZ = (1 << (picoos_uint32) cep->scmeanpowLFZ);
+
+ cep->scmeanMGC = (1 << (picoos_uint32) cep->scmeanpowMGC);
+
+ return PICO_OK;
+}/*cepInitialize*/
+
+/**
+ * termination of a cep PU (processing unit)
+ * @param this : handle to a cep PU struct
+ * @return PICO_OK : termination succeded
+ * @return PICO_ERR_OTHER : termination failed
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t cepTerminate(register picodata_ProcessingUnit this)
+{
+ return PICO_OK;
+}
+
+/**
+ * deallocation of a cep PU internal sub object
+ * @param this : handle to a cep PU struct
+ * @param mm : handle of the engine memory manager
+ * @return PICO_OK : deallocation succeded
+ * @return PICO_ERR_OTHER : deallocation failed
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t cepSubObjDeallocate(register picodata_ProcessingUnit this,
+ picoos_MemoryManager mm)
+{
+
+ mm = mm; /* avoid warning "var not used in this function"*/
+#if defined(PICO_DEVEL_MODE)
+ printf("number of long mult is %d, number of short mult is %i\n",numlongmult,numshortmult);
+#else
+ PICODBG_INFO_MSG(("number of long mult is %d, number of short mult is %i\n",numlongmult,numshortmult));
+#endif
+ if (NULL != this) {
+ cep_subobj_t * cep = (cep_subobj_t *) this->subObj;
+ picoos_deallocate(this->common->mm, (void *) &cep->outXCep);
+ picoos_deallocate(this->common->mm, (void *) &cep->outVoiced);
+ picoos_deallocate(this->common->mm, (void *) &cep->outF0);
+ picoos_deallocate(this->common->mm, (void *) &this->subObj);
+ }
+ return PICO_OK;
+}
+
+/**
+ * creates a new cep PU (processing unit)
+ * @param mm : engine memory manager object pointer
+ * @param common : engine common object pointer
+ * @param cbIn : PU input buffer
+ * @param cbOut : PU output buffer
+ * @param voice : the voice descriptor object
+ * @return a valid PU handle if creation succeded
+ * @return NULL : creation failed
+ * @callgraph
+ * @callergraph
+ */
+picodata_ProcessingUnit picocep_newCepUnit(picoos_MemoryManager mm,
+ picoos_Common common, picodata_CharBuffer cbIn,
+ picodata_CharBuffer cbOut, picorsrc_Voice voice)
+{
+ picodata_ProcessingUnit this = picodata_newProcessingUnit(mm, common, cbIn,
+ cbOut, voice);
+ cep_subobj_t * cep;
+
+ if (this == NULL) {
+ return NULL;
+ }
+ this->initialize = cepInitialize;
+
+ PICODBG_DEBUG(("set this->step to cepStep"));
+
+ this->step = cepStep;
+ this->terminate = cepTerminate;
+ this->subDeallocate = cepSubObjDeallocate;
+ this->subObj = picoos_allocate(mm, sizeof(cep_subobj_t));
+
+ cep = (cep_subobj_t *) this->subObj;
+
+ if (this->subObj == NULL) {
+ picoos_deallocate(mm, (void*) &this);
+ return NULL;
+ };
+
+ /* allocate output coeeficient buffers */
+ cep->outF0 = (picoos_int16 *) picoos_allocate(this->common->mm,
+ PICOCEP_MAXWINLEN * PICOKPDF_MAX_MUL_LFZ_CEPORDER
+ * sizeof(picoos_int16));
+ cep->outXCep = (picoos_int16 *) picoos_allocate(this->common->mm,
+ PICOCEP_MAXWINLEN * PICOKPDF_MAX_MUL_MGC_CEPORDER
+ * sizeof(picoos_int16));
+ cep->outVoiced = (picoos_uint8 *) picoos_allocate(this->common->mm,
+ PICOCEP_MAXWINLEN * sizeof(picoos_uint8));
+
+ if ((NULL == cep->outF0) || (NULL == cep->outXCep) || (NULL
+ == cep->outVoiced)) {
+ picoos_deallocate(this->common->mm, (void *) &(cep->outF0));
+ picoos_deallocate(this->common->mm, (void *) &(cep->outXCep));
+ picoos_deallocate(this->common->mm, (void *) &(cep->outVoiced));
+ picoos_deallocate(mm, (void*) &cep);
+ picoos_deallocate(mm, (void*) &this);
+ return NULL;
+ }
+ cepInitialize(this);
+
+ return this;
+}/*picocep_newCepUnit*/
+
+/* --------------------------------------------
+ * processing and internal functions
+ * --------------------------------------------
+ */
+
+/**
+ * multiply by 1<<pow and check overflow
+ * @param a : input value
+ * @param pow : shift value
+ * @return multiplied value
+ * @callgraph
+ * @callergraph
+ */
+static picoos_int32 picocep_fixptmultpow(picoos_int32 a, picoos_uint8 pow)
+{
+ picoos_int32 b;
+ picoos_int32 zzz;
+
+ if (picocep_highestBitS(a,zzz) + pow < 32) {
+ b = a << pow;
+ } else {
+ /* clip to maximum positive or negative value */
+ b = 1 << 31; /* maximum negative value */
+ if (a > 0) {
+ b -= 1; /* maximum positive value */
+ }PICODBG_WARN(("picocep_fixptmultpow warning: overflow in fixed point multiplication %i*1<<%i. Clipping to %i\n", a, pow, b));
+ }
+ return b;
+}
+
+/**
+ * divide by 1<<pow with rounding
+ * @param a : input value
+ * @param pow : shift value
+ * @return divided value
+ * @callgraph
+ * @callergraph
+ */
+static picoos_int32 picocep_fixptdivpow(picoos_int32 a, picoos_uint8 pow)
+{
+ picoos_int32 big;
+
+ if (a == 0) {
+ return a;
+ }
+ big = 1 << (pow - 1);
+ if (a > 0) {
+ a = (a + big) >> pow;
+ } else {
+ a = -1 * ((-1 * a + big) >> pow);
+ }
+
+ return a;
+}
+
+/**
+ * fixed point multiplication of x and y for large values of x or y or both
+ * @param x,y : operands 1 & 2, in fixed point S:M:N representation
+ * @param bigpow (int) : normalization factor=2**N, where N=number of binary decimal digits
+ * @param invDoubleDec : boolean indicating that x has double decimal size.
+ * do extra division by 1<<bigpow so that result has again single decimal size
+ * @return z(int) : result, in fixed point S:M:N representation
+ * @callgraph
+ * @callergraph
+ */
+static picoos_int32 picocep_fixptmultdouble(picoos_int32 x, picoos_int32 y,
+ picoos_uint8 bigpow, picoos_uint8 invDoubleDec)
+{
+ picoos_int32 a, b, c, d, e, z;
+ picoos_int32 big;
+
+ big = 1 << bigpow;
+
+ /* a = floor(x/big); */
+ if (x >= 0) {
+ a = x >> bigpow;
+ b = x - (a << bigpow);
+ } else {
+ a = -1 * ((x * -1) >> bigpow); /* most significant 2 bytes of x */
+ b = x - (a << bigpow);
+ }
+
+ /* least significant 2 bytes of x i.e. x modulo big */
+ /* c = floor(y/big); */
+ if (y >= 0) {
+ c = y >> bigpow;
+ d = y - (c << bigpow);
+ } else {
+ c = -1 * ((y * -1) >> bigpow);
+ d = y - (c << bigpow);
+ }
+
+ if (invDoubleDec == 1) {
+ e = a * d + b * c + picocep_fixptdivpow(b * d, bigpow);
+ z = a * c + picocep_fixptdivpow(e, bigpow);
+ } else {
+ z = ((a * c) << bigpow) + (a * d + b * c) + picocep_fixptdivpow(b * d,
+ bigpow); /* 4 mult and 3 add instead of 1 mult. */
+ }
+
+ return z;
+}
+
+/**
+ * fixed point multiplication of x and y
+ * @param x,y : operands 1 & 2, in fixed point S:M:N representation
+ * @param bigpow (int) : normalization factor=2**N, where N=number of binary decimal digits
+ * @param invDoubleDec : boolean indicating that x has double decimal size.
+ * do extra division by 1<<bigpow so that result has again single decimal size
+ * @return z(int) : result, in fixed point S:M:N representation
+ * Notes
+ * - input and output values are 32 bit signed integers
+ * meant to represent a S.M.N encoding of a floating point value where
+ * - S : 1 sign bit
+ * - M : number of binary integer digits (M=32-1-N)
+ * - N : number of binary decimal digits (N=log2(big))
+ * the routine supports 2 methods
+ * -# standard multiplication of x and y
+ * -# long multiplication of x and y
+ * under PICO_DEBUG the number of double and single precision multiplications is monitored for accuracy/performance tuning
+ * Calls
+ * - picocep_highestBit
+ * - picocep_fixptmultdouble
+ * @callgraph
+ * @callergraph
+ */
+static picoos_int32 picocep_fixptmult(picoos_int32 x, picoos_int32 y,
+ picoos_uint8 bigpow, picoos_uint8 invDoubleDec)
+{
+ picoos_int32 z;
+ picoos_uint8 multsz, pow;
+ picoos_int32 zz1, zz2;
+
+ /* in C, the evaluation order of f() + g() is not defined, so
+ * if both have a side effect on e.g. zz, the outcome of zz is not defined.
+ * For that reason, picocep_highestBitS(x,zz) + picocep_highestBitS(y,zz)
+ * would generate a warning "operation on zz may be undefined" which we
+ * avoid by using two different variables zz1 and zz2 */
+ multsz = picocep_highestBitS(x,zz1) + picocep_highestBitS(y,zz2);
+ pow = bigpow;
+ if (invDoubleDec == 1) {
+ pow += bigpow;
+ }
+
+ if (multsz <= 30) { /* x*y < 1<<30 is safe including rounding in picocep_fixptdivpow, x*y < 1<<31 is safe but not with rounding */
+ /* alternatively perform multiplication and consult overflow register */
+ z = picocep_fixptdivpow(x * y, pow);
+#if defined(PICO_DEBUG)
+ numshortmult++; /* keep track of number of short multiplications */
+#endif
+ } else {
+#if defined(PICO_DEBUG)
+ if (multsz> 31 + pow) {
+ PICODBG_WARN(("picocep_fixptmult warning: overflow in fixed point multiplication %i*%i, multsz = %i, pow = %i, decrease bigpow\n", x, y, multsz, pow));
+ }
+#endif
+ z = picocep_fixptmultdouble(x, y, bigpow, invDoubleDec); /* perform long multiplication for large x and y */
+#if defined(PICO_DEBUG)
+ numlongmult++; /* keep track of number of long multiplications */
+#endif
+ }
+ return z;
+}/* picocep_fixptmult */
+
+/**
+ * fixed point ^division of a vs b
+ * @param a,b : operands 1 & 2, in fixed point S:M:N representation
+ * @param bigpow (int) : normalization factor=2**N, where N=number of binary decimal digits
+ * @return z(int) : result, in fixed point S:M:N representation
+ * Notes
+ * - input and output values are 32 bit signed integers
+ * meant to represent a S.M.N encoding of a floating point value where
+ * - S : 1 sign bit
+ * - M : number of binary integer digits (M=32-1-N)
+ * - N : number of binary decimal digits (N=log2(big))
+ * - standard implementation of division by iteratively calculating
+ * the remainder and setting corresponding bit
+ * calculate integer part by integer division (DIV),
+ * then add X bits of precision in decimal part
+ * @callgraph
+ * @callergraph
+ */
+static picoos_int32 picocep_fixptdiv(picoos_int32 a, picoos_int32 b,
+ picoos_uint8 bigpow)
+{
+ picoos_int32 r, c, f, h, stop;
+ r = (a < 0) ? -a : a; /* take absolute value; b is already guaranteed to be positive in smoothing operation */
+ if (r == 0) {
+ return 0;
+ }
+ c = 0;
+ stop = 0;
+
+ /* can speed up slightly by setting stop = 2 => slightly less precision */
+ h = r / b; /* in first loop h can be multiple bits, after first loop h can only be 0 or 1 */
+ /* faster implementation on target by using comparison instead of DIV? */
+ /* For our LDL even in first loop h <= 1, but not in backward step */
+ c = c + (h << bigpow); /* after first loop simply set bit */
+ r = r - h * b; /* corresponds to modulo operation */
+ bigpow--;
+ r <<= 1;
+
+ while ((bigpow > stop) && (r != 0)) { /* calculate bigpow bits after fixed point */
+ /* can speed up slightly by setting stop = 2 => slightly less precision */
+ if (r >= b) {
+ c += (1 << bigpow); /* after first loop simply set bit */
+ r -= b; /* corresponds to modulo operation */
+ }
+ bigpow--;
+ r <<= 1;
+ }
+
+ if (r != 0) {
+ f = r + (b >> 1);
+ if (f >= b) {
+ if (f >= b + b) {
+ c += 2;
+ } else {
+ c++;
+ }
+ }
+ }
+ /* final step: do rounding (experimentally improves accuracy for our problem) */
+ c = (a >= 0) ? c : -c; /* b is guaranteed to be positive because corresponds to diag0 */
+ return c;
+}/* picocep_fixptdiv */
+
+/**
+ * perform inversion of diagonal element of WUW matrix
+ * @param d : diagonal element to be inverted
+ * @param rowscpow (int) : fixed point base for each dimension of the vectors stored in the database
+ * @param bigpow (int) : fixed point base used during cepstral smoothing
+ * @param invpow : fixed point base of inverted pivot elements
+ * @return inverted pivot element
+ * @note
+ * - d is guaranteed positive
+ * @callgraph
+ * @callergraph
+ */
+static picoos_int32 picocep_fixptInvDiagEle(picoos_uint32 d,
+ picoos_uint8* rowscpow, picoos_uint8 bigpow, picoos_uint8 invpow)
+{
+ picoos_uint32 r, b, c, h, f, stop;
+ picoos_uint8 dlen;
+ /* picoos_int32 zz; */
+ c = 0;
+ stop = 0;
+
+ dlen = picocep_highestBitU(d);
+ if (invpow + bigpow > 30 + dlen) { /* c must be < 2^32, hence d which is >= 2^(dlen-1) must be > 2^(invpow+bigpow-32), or invpow+bigpow must be <= dlen+30*/
+ *rowscpow = invpow + bigpow - 30 - dlen;PICODBG_DEBUG(("input to picocep_fixptInvDiagEle is %i <= 1<<%i = 1<<invpow+bigpow-32. Choose lower invpow. For now scaling row by 1<<%i\n", d, invpow+bigpow-32, *rowscpow));
+ } else {
+ *rowscpow = 0;
+ }
+ r = 1 << invpow;
+ b = d << (*rowscpow);
+
+ /* first */
+ h = r / b;
+ if (h > 0) {
+ c += (h << bigpow);
+ r -= h * b;
+ }
+ bigpow--;
+ r <<= 1;
+
+ /* loop */
+ while ((bigpow > stop) && (r != 0)) {
+ if (r >= b) {
+ c += (1 << bigpow);
+ r -= b;
+ }
+ bigpow--;
+ r <<= 1;
+ }
+
+ if (r != 0) {
+ f = r + (b >> 1);
+ if (f >= b) {
+ if (f >= b + b) {
+ c += 2;
+ } else {
+ c++;
+ }
+ }
+ }
+
+ return c;
+}/* picocep_fixptInvDiagEle */
+
+/**
+ * perform division of two operands a and b by multiplication by inverse of b
+ * @param a (int32) : operand 1 in fixed point S:M:N representation
+ * @param invb(uint32) : inverse of operand b, in fixed point P:Q representation (sign is positive)
+ * @param bigpow(uint8) : N = bigpow when invDoubleDec==0, else N = 2*bigpow
+ * @param invpow(uint8) : Q = invpow = number of binary decimal digits for invb
+ * @param invDoubleDec : boolean to indicate that a and the return value c have 2*N binary decimal digits instead of N
+ * @return c(int32) : result in fixed point S:v:w where w = 2*N when invDoubleDec == 1
+ * @note Calls
+ * - picocep_fixptmult
+ * @callgraph
+ * @callergraph
+ */
+static picoos_int32 picocep_fixptinv(picoos_int32 a, picoos_uint32 invb,
+ picoos_uint8 bigpow, picoos_uint8 invpow, picoos_uint8 invDoubleDec)
+{
+ picoos_int32 c;
+ picoos_int8 normpow;
+
+ c = picocep_fixptmult(a, invb, bigpow, invDoubleDec);
+
+ /* if invDoubleDec==0, picocep_fixptmult assumes a and invb are in base 1<<bigpow and returns c = (a*b)/1<<bigpow
+ Since invb is in base 1<<invpow instead of 1<<bigpow, normalize c by 1<<(bigpow-invpow)
+ if invDoubleDec==1:
+ multiply additionally by 1<<bigpow*2 (for invb and c) so that base of c is again 2*bigpow
+ this can be seen by setting a=A*big, b=B*big, invb=big2/B, mult(a,invb) = a*invb/(big*big) = A/B*big*big2/(big*big) = A/B*big2/big
+ and we want c = A/B*big*big => normfactor = big^3/big2
+ */
+ if (invDoubleDec == 1) {
+ normpow = 3 * bigpow;
+ } else {
+ normpow = bigpow;
+ }
+ if (normpow < invpow) {
+ /* divide with rounding */
+ c = picocep_fixptdivpow(c, invpow - normpow);
+ } else {
+ c = picocep_fixptmultpow(c, normpow - invpow);
+ }
+ return c;
+}
+
+/**
+ * initializes the coefficients to calculate delta and delta-delta values and the squares of the coefficients
+ * @param cep : the CEP PU sub-object handle
+ * @callgraph
+ * @callergraph
+ */
+static void initSmoothing(cep_subobj_t * cep)
+{
+ cep->xi[0] = 1;
+ cep->xi[1] = -1;
+ cep->xi[2] = 2;
+ cep->xi[3] = -4;
+ cep->xi[4] = 2;
+ cep->xsqi[0] = 1;
+ cep->xsqi[1] = 1;
+ cep->xsqi[2] = 4;
+ cep->xsqi[3] = 16;
+ cep->xsqi[4] = 4;
+
+ cep->x1[0] = -1;
+ cep->x1[1] = 2;
+ cep->xsq1[0] = 1;
+ cep->xsq1[1] = 4;
+
+ cep->x2[0] = -1;
+ cep->x2[1] = -4;
+ cep->x2[2] = 2;
+ cep->xsq2[0] = 1;
+ cep->xsq2[1] = 16;
+ cep->xsq2[2] = 4;
+
+ cep->xm[0] = 1;
+ cep->xm[1] = 2;
+ cep->xm[2] = -4;
+ cep->xsqm[0] = 1;
+ cep->xsqm[1] = 4;
+ cep->xsqm[2] = 16;
+
+ cep->xn[0] = 1;
+ cep->xn[1] = 2;
+ cep->xsqn[0] = 1;
+ cep->xsqn[1] = 4;
+}
+
+/**
+ * matrix inversion
+ * @param cep : PU sub object pointer
+ * @param N
+ * @param smoothcep : pointer to picoos_int16, sequence of smoothed cepstral vectors
+ * @param cepnum : cepstral dimension to be treated
+ * @param pdf : pdf resource
+ * @param invpow : fixed point base for inverse
+ * @param invDoubleDec : boolean indicating that result of picocep_fixptinv has fixed point base 2*bigpow
+ * picocep_fixptmult absorbs double decimal size by dividing its result by extra factor big
+ * @return void
+ * @remarks diag0, diag1, diag2, WUm, invdiag0 globals needed in this function (object members in pico)
+ * @callgraph
+ * @callergraph
+ */
+static void invMatrix(cep_subobj_t * cep, picoos_uint16 N,
+ picoos_int16 *smoothcep, picoos_uint8 cepnum,
+ picokpdf_PdfMUL pdf, picoos_uint8 invpow, picoos_uint8 invDoubleDec)
+{
+ picoos_int32 j, v1, v2, h;
+ picoos_uint32 k;
+ picoos_uint8 rowscpow, prevrowscpow;
+ picoos_uint8 ceporder = pdf->ceporder;
+ picoos_uint8 bigpow = pdf->bigpow;
+ picoos_uint8 meanpow = pdf->meanpow;
+
+ /* LDL factorization */
+ prevrowscpow = 0;
+ cep->invdiag0[0] = picocep_fixptInvDiagEle(cep->diag0[0], &rowscpow,
+ bigpow, invpow); /* inverse has fixed point basis 1<<invpow */
+ cep->diag1[0] = picocep_fixptinv((cep->diag1[0]) << rowscpow,
+ cep->invdiag0[0], bigpow, invpow, invDoubleDec); /* perform division via inverse */
+ cep->diag2[0] = picocep_fixptinv((cep->diag2[0]) << rowscpow,
+ cep->invdiag0[0], bigpow, invpow, invDoubleDec);
+ cep->WUm[0] = (cep->WUm[0]) << rowscpow; /* if diag0 too low, multiply LHS and RHS of row in matrix equation by 1<<rowscpow */
+ for (j = 1; j < N; j++) {
+ /* do forward substitution */
+ cep->WUm[j] = cep->WUm[j] - picocep_fixptmult(cep->diag1[j - 1],
+ cep->WUm[j - 1], bigpow, invDoubleDec);
+ if (j > 1) {
+ cep->WUm[j] = cep->WUm[j] - picocep_fixptmult(cep->diag2[j - 2],
+ cep->WUm[j - 2], bigpow, invDoubleDec);
+ }
+
+ /* update row j */
+ v1 = picocep_fixptmult((cep->diag1[j - 1]) / (1 << rowscpow),
+ cep->diag0[j - 1], bigpow, invDoubleDec); /* undo scaling by 1<<rowscpow because diag1(j-1) refers to symm ele in column j-1 not in row j-1 */
+ cep->diag0[j] = cep->diag0[j] - picocep_fixptmult(cep->diag1[j - 1],
+ v1, bigpow, invDoubleDec);
+ if (j > 1) {
+ v2 = picocep_fixptmult((cep->diag2[j - 2]) / (1 << prevrowscpow),
+ cep->diag0[j - 2], bigpow, invDoubleDec); /* undo scaling by 1<<prevrowscpow because diag1(j-2) refers to symm ele in column j-2 not in row j-2 */
+ cep->diag0[j] = cep->diag0[j] - picocep_fixptmult(
+ cep->diag2[j - 2], v2, bigpow, invDoubleDec);
+ }
+ prevrowscpow = rowscpow;
+ cep->invdiag0[j] = picocep_fixptInvDiagEle(cep->diag0[j], &rowscpow,
+ bigpow, invpow); /* inverse has fixed point basis 1<<invpow */
+ cep->WUm[j] = (cep->WUm[j]) << rowscpow;
+ if (j < N - 1) {
+ h = picocep_fixptmult(cep->diag2[j - 1], v1, bigpow, invDoubleDec);
+ cep->diag1[j] = picocep_fixptinv((cep->diag1[j] - h) << rowscpow,
+ cep->invdiag0[j], bigpow, invpow, invDoubleDec); /* eliminate column j below pivot */
+ }
+ if (j < N - 2) {
+ cep->diag2[j] = picocep_fixptinv((cep->diag2[j]) << rowscpow,
+ cep->invdiag0[j], bigpow, invpow, invDoubleDec); /* eliminate column j below pivot */
+ }
+ }
+
+ /* divide all entries of WUm by diag0 */
+ for (j = 0; j < N; j++) {
+ cep->WUm[j] = picocep_fixptinv(cep->WUm[j], cep->invdiag0[j], bigpow,
+ invpow, invDoubleDec);
+ if (invDoubleDec == 1) {
+ cep->WUm[j] = picocep_fixptdivpow(cep->WUm[j], bigpow);
+ }
+ }
+
+ /* backward substitution */
+ for (j = N - 2; j >= 0; j--) {
+ cep->WUm[j] = cep->WUm[j] - picocep_fixptmult(cep->diag1[j], cep->WUm[j
+ + 1], bigpow, invDoubleDec);
+ if (j < N - 2) {
+ cep->WUm[j] = cep->WUm[j] - picocep_fixptmult(cep->diag2[j],
+ cep->WUm[j + 2], bigpow, invDoubleDec);
+ }
+ }
+ /* copy N frames into smoothcep (only for coeff # "cepnum") */
+ /* coefficients normalized to occupy short; for correct waveform energy, divide by (1<<(bigpow-meanpow)) then convert e.g. to picoos_single */
+ k = cepnum;
+ for (j = 0; j < N; j++) {
+ smoothcep[k] = (picoos_int16)(cep->WUm[j]/(1<<meanpow));
+ k += ceporder;
+ }
+
+}/* invMatrix*/
+
+/**
+ * Calculate matrix products needed to implement the solution
+ * @param cep : PU sub object pointer
+ * @param pdf : pointer to picoos_uint8, sequence of pdf vectors, each vector of length 1+ceporder*2+numdeltas*3+ceporder*3
+ * @param indices : indices of pdf vectors for all frames in current sentence
+ * @param b, N : to be smoothed frames indices (range will be from b to b+N-1)
+ * @param cepnum : cepstral dimension to be treated
+ * @return void
+ * @remarks diag0, diag1, diag2, WUm, invdiag0 globals needed in this function (object members in pico)
+ * @remarks WUW --> At x W x A
+ * @remarks WUm --> At x W x b
+ * @callgraph
+ * @callergraph
+ */
+static picoos_uint8 makeWUWandWUm(cep_subobj_t * cep, picokpdf_PdfMUL pdf,
+ picoos_uint16 *indices, picoos_uint16 b, picoos_uint16 N,
+ picoos_uint8 cepnum)
+{
+ picoos_uint16 Id[2], Idd[3];
+ /*picoos_uint32 vecstart, k;*/
+ picoos_uint32 vecstart;
+ picoos_int32 *x = NULL, *xsq = NULL;
+ picoos_int32 mean, ivar;
+ picoos_uint16 i, j, numd = 0, numdd = 0;
+ picoos_uint8 vecsize = pdf->vecsize;
+ picoos_int32 prev_WUm, prev_diag0, prev_diag1, prev_diag1_1, prev_diag2;
+
+ prev_WUm = prev_diag0 = prev_diag1 = prev_diag1_1 = prev_diag2 = 0;
+ for (i = 0; i < N; i++) {
+
+ if ((1 < i) && (i < N - 2)) {
+ x = cep->xi;
+ xsq = cep->xsqi;
+ numd = 2;
+ numdd = 3;
+ Id[0] = Idd[0] = i - 1;
+ Id[1] = Idd[2] = i + 1;
+ Idd[1] = i;
+ } else if (i == 0) {
+ x = cep->x1;
+ xsq = cep->xsq1;
+ numd = numdd = 1;
+ Id[0] = Idd[0] = 1;
+ } else if (i == 1) {
+ x = cep->x2;
+ xsq = cep->xsq2;
+ numd = 1;
+ numdd = 2;
+ Id[0] = Idd[1] = 2;
+ Idd[0] = 1;
+ } else if (i == N - 2) {
+ x = cep->xm;
+ xsq = cep->xsqm;
+ numd = 1;
+ numdd = 2;
+ Id[0] = Idd[0] = N - 3;
+ Idd[1] = N - 2;
+ } else if (i == N - 1) {
+ x = cep->xn;
+ xsq = cep->xsqn;
+ numd = numdd = 1;
+ Id[0] = Idd[0] = N - 2;
+ }
+
+ /* process static means and static inverse variances */
+ if (i > 0 && indices[b + i] == indices[b + i - 1]) {
+ cep->diag0[i] = prev_diag0;
+ cep->WUm[i] = prev_WUm;
+ } else {
+ vecstart = indices[b + i] * vecsize;
+ ivar = getFromPdf(pdf, vecstart, cepnum, PICOCEP_WANTIVAR,
+ PICOCEP_WANTSTATIC);
+ prev_diag0 = cep->diag0[i] = ivar << 2; /* multiply ivar by 4 (4 used to be first entry of xsq) */
+ mean = getFromPdf(pdf, vecstart, cepnum, PICOCEP_WANTMEAN,
+ PICOCEP_WANTSTATIC);
+ prev_WUm = cep->WUm[i] = mean << 1; /* multiply mean by 2 (2 used to be first entry of x) */
+ }
+
+ /* process delta means and delta inverse variances */
+ for (j = 0; j < numd; j++) {
+ vecstart = indices[b + Id[j]] * vecsize;
+ ivar = getFromPdf(pdf, vecstart, cepnum, PICOCEP_WANTIVAR,
+ PICOCEP_WANTDELTA);
+ cep->diag0[i] += xsq[j] * ivar;
+
+ mean = getFromPdf(pdf, vecstart, cepnum, PICOCEP_WANTMEAN,
+ PICOCEP_WANTDELTA);
+ if (mean != 0) {
+ cep->WUm[i] += x[j] * mean;
+ }
+ }
+
+ /* process delta delta means and delta delta inverse variances */
+ for (j = 0; j < numdd; j++) {
+ vecstart = indices[b + Idd[j]] * vecsize;
+ ivar = getFromPdf(pdf, vecstart, cepnum, PICOCEP_WANTIVAR,
+ PICOCEP_WANTDELTA2);
+ cep->diag0[i] += xsq[numd + j] * ivar;
+
+ mean = getFromPdf(pdf, vecstart, cepnum, PICOCEP_WANTMEAN,
+ PICOCEP_WANTDELTA2);
+ if (mean != 0) {
+ cep->WUm[i] += x[numd + j] * mean;
+ }
+ }
+
+ cep->diag0[i] = (cep->diag0[i] + 2) / 4; /* long DIV with rounding */
+ cep->WUm[i] = (cep->WUm[i] + 1) / 2; /* long DIV with rounding */
+
+ /* calculate diag(A,-1) */
+ if (i < N - 1) {
+ if (i < N - 2) {
+ if (i > 0 && indices[b + i + 1] == indices[b + i]) {
+ cep->diag1[i] = prev_diag1;
+ } else {
+ vecstart = indices[b + i + 1] * vecsize;
+ /*
+ diag1[i] = getFromPdf(pdf, vecstart, numvuv, ceporder, numdeltas, cepnum,
+ bigpow, meanpowUm, ivarpow, PICOCEP_WANTIVAR, PICOCEP_WANTDELTA2);
+ */
+ prev_diag1 = cep->diag1[i] = getFromPdf(pdf, vecstart,
+ cepnum, PICOCEP_WANTIVAR, PICOCEP_WANTDELTA2);
+ }
+ /*
+ k = vecstart +pdf->numvuv+pdf->ceporder*2 + pdf->numdeltas*3 +
+ pdf->ceporder*2 +cepnum;
+ cep->diag1[i] = (picoos_int32)(pdf->content[k]) << pdf->bigpow;
+ */
+ } else {
+ cep->diag1[i] = 0;
+ }
+ if (i > 0) {
+ if (i > 1 && indices[b + i] == indices[b + i - 1]) {
+ cep->diag1[i] += prev_diag1_1;
+ } else {
+ vecstart = indices[b + i] * vecsize;
+ /*
+ k = vecstart + pdf->numvuv + pdf->ceporder * 2 + pdf->numdeltas * 3 + pdf->ceporder * 2 + cepnum;
+ cep->diag1[i] += (picoos_int32)(pdf->content[k]) << pdf->bigpow; */
+ /* cepnum'th delta delta ivar */
+
+ prev_diag1_1 = getFromPdf(pdf, vecstart, cepnum,
+ PICOCEP_WANTIVAR, PICOCEP_WANTDELTA2);
+ cep->diag1[i] += prev_diag1_1;
+ }
+
+ } /*i < N-1 */
+ cep->diag1[i] *= -2;
+ }
+ }
+
+ /* calculate diag(A,-2) */
+ for (i = 0; i < N - 2; i++) {
+ if (i > 0 && indices[b + i + 1] == indices[b + i]) {
+ cep->diag2[i] = prev_diag2;
+ } else {
+ vecstart = indices[b + i + 1] * vecsize;
+ /*
+ k = vecstart + pdf->numvuv + pdf->ceporder * 2 + pdf->numdeltas * 3 + pdf->ceporder * 2 + cepnum;
+ cep->diag2[i] = (picoos_int32)(pdf->content[k]) << pdf->bigpow;
+ k -= pdf->ceporder;
+ ivar = (picoos_int32)(pdf->content[k]) << pdf->bigpow;
+ */
+ cep->diag2[i] = getFromPdf(pdf, vecstart, cepnum, PICOCEP_WANTIVAR,
+ PICOCEP_WANTDELTA2);
+ ivar = getFromPdf(pdf, vecstart, cepnum, PICOCEP_WANTIVAR,
+ PICOCEP_WANTDELTA);
+ cep->diag2[i] -= (ivar + 2) / 4;
+ prev_diag2 = cep->diag2[i];
+ }
+ }
+
+ return 0;
+}/* makeWUWandWUm */
+
+/**
+ * Retrieve actual values for MGC from PDF resource
+ * @param pdf : pointer to picoos_uint8, sequence of pdf vectors, each vector of length 1+ceporder*2+numdeltas*3+ceporder*3
+ * @param vecstart : indices of pdf vectors for all frames in current sentence
+ * @param cepnum : cepstral dimension to be treated
+ * @param wantMeanOrIvar : flag to select mean or variance values
+ * @param wantStaticOrDeltax : flag to select static or delta values
+ * @return the actual value retrieved
+ * @callgraph
+ * @callergraph
+ */
+static picoos_int32 getFromPdf(picokpdf_PdfMUL pdf, picoos_uint32 vecstart,
+ picoos_uint8 cepnum, picocep_WantMeanOrIvar_t wantMeanOrIvar,
+ picocep_WantStaticOrDelta_t wantStaticOrDeltax)
+{
+ picoos_uint8 s, ind;
+ picoos_uint8 *p;
+ picoos_uint8 ceporder, ceporder2, cc;
+ picoos_uint32 k;
+ picoos_int32 mean = 0, ivar = 0;
+
+ if (pdf->numdeltas == 0xFF) {
+ switch (wantMeanOrIvar) {
+ case PICOCEP_WANTMEAN:
+ switch (wantStaticOrDeltax) {
+ case PICOCEP_WANTSTATIC:
+ p = pdf->content
+ + (vecstart + pdf->numvuv + cepnum * 2); /* cepnum'th static mean */
+ mean = ((picoos_int32) ((picoos_int16) (*(p + 1) << 8))
+ | *p) << (pdf->meanpowUm[cepnum]);
+ break;
+ case PICOCEP_WANTDELTA:
+ cc = pdf->ceporder + cepnum;
+ p = pdf->content + (vecstart + pdf->numvuv + cc * 2); /* cepnum'th delta mean */
+ mean = ((picoos_int32) ((picoos_int16) (*(p + 1) << 8))
+ | *p) << (pdf->meanpowUm[cc]);
+ break;
+ case PICOCEP_WANTDELTA2:
+ cc = pdf->ceporder * 2 + cepnum;
+ p = pdf->content + (vecstart + pdf->numvuv + cc * 2); /* cepnum'th delta delta mean */
+ mean = ((picoos_int32) ((picoos_int16) (*(p + 1) << 8))
+ | *p) << (pdf->meanpowUm[cc]);
+ break;
+ default:
+ /* should never come here */
+ PICODBG_ERROR(("unknown type wantStaticOrDeltax = %i", wantStaticOrDeltax));
+ }
+ return mean;
+ break;
+ case PICOCEP_WANTIVAR:
+ switch (wantStaticOrDeltax) {
+ case PICOCEP_WANTSTATIC:
+ k = vecstart + pdf->numvuv + pdf->ceporder * 6 + cepnum; /* cepnum'th static ivar */
+ ivar = (picoos_int32) (pdf->content[k])
+ << (pdf->ivarpow[cepnum]);
+ break;
+ case PICOCEP_WANTDELTA:
+ ceporder = pdf->ceporder;
+ k = vecstart + pdf->numvuv + ceporder * 7 + cepnum; /* cepnum'th delta ivar */
+ ivar = (picoos_int32) (pdf->content[k])
+ << (pdf->ivarpow[ceporder + cepnum]);
+ break;
+ case PICOCEP_WANTDELTA2:
+ ceporder = pdf->ceporder;
+ k = vecstart + pdf->numvuv + ceporder * 8 + cepnum; /* cepnum'th delta delta ivar */
+ ivar = (picoos_int32) (pdf->content[k])
+ << (pdf->ivarpow[2 * ceporder + cepnum]);
+ break;
+ default:
+ /* should never get here */
+ PICODBG_ERROR(("unknown type wantStaticOrDeltax = %i", wantStaticOrDeltax));
+ }
+ return ivar;
+ break;
+ default:
+ /* should never come here */
+ PICODBG_ERROR(("unknown type wantMeanOrIvar = %n", wantMeanOrIvar));
+ return 0;
+ }
+ } else {
+ switch (wantMeanOrIvar) {
+ case PICOCEP_WANTMEAN:
+ switch (wantStaticOrDeltax) {
+ case PICOCEP_WANTSTATIC:
+ p = pdf->content
+ + (vecstart + pdf->numvuv + cepnum * 2); /* cepnum'th static mean */
+ mean = ((picoos_int32) ((picoos_int16) (*(p + 1) << 8))
+ | *p) << (pdf->meanpowUm[cepnum]);
+ return mean;
+ break;
+ case PICOCEP_WANTDELTA:
+ ceporder = pdf->ceporder;
+ s = 0;
+ ind = 0;
+ while ((s < pdf->numdeltas) && (ind < cepnum || (ind
+ == 0 && cepnum == 0))) { /* rawmean deltas are sparse so investigate indices in column */
+ k = vecstart + pdf->numvuv + ceporder * 2 + s; /* s'th delta index */
+ ind = (picoos_uint8) (pdf->content[k]); /* is already picoos_uint8 but just to make explicit */
+ if (ind == cepnum) {
+ k = vecstart + pdf->numvuv + ceporder * 2
+ + pdf->numdeltas + s * 2; /* s'th sparse delta mean, corresponds to cepnum'th delta mean */
+ mean
+ = ((picoos_int32) ((picoos_int16) ((pdf->content[k
+ + 1]) << 8)) | pdf->content[k])
+ << (pdf->meanpowUm[ceporder
+ + cepnum]);
+ return mean;
+ }
+ s++;
+ }
+ return 0;
+ break;
+ case PICOCEP_WANTDELTA2:
+ ceporder = pdf->ceporder;
+ ceporder2 = ceporder * 2;
+ s = pdf->numdeltas;
+ ind = 2 * ceporder;
+ while ((s-- > 0) && (ind > ceporder + cepnum)) { /* rawmean deltas are sparse so investigate indices in column */
+ k = vecstart + pdf->numvuv + ceporder2 + s; /* s'th delta index */
+ ind = (picoos_uint8) (pdf->content[k]); /* is already picoos_uint8 but just to make explicit */
+ if (ind == ceporder + cepnum) {
+ k = vecstart + pdf->numvuv + ceporder2
+ + pdf->numdeltas + s * 2; /* s'th sparse delta mean, corresponds to cepnum'th delta delta mean */
+ mean
+ = ((picoos_int32) ((picoos_int16) ((pdf->content[k
+ + 1]) << 8)) | pdf->content[k])
+ << (pdf->meanpowUm[ceporder2
+ + cepnum]);
+ return mean;
+ }
+ }
+ return 0;
+ break;
+ default:
+ PICODBG_ERROR(("getFromPdf: unknown type wantStaticOrDeltax = %i\n", wantStaticOrDeltax));
+ return 0;
+ }
+ break;
+ case PICOCEP_WANTIVAR:
+ switch (wantStaticOrDeltax) {
+ case PICOCEP_WANTSTATIC:
+ k = vecstart + pdf->numvuv + pdf->ceporder * 2
+ + pdf->numdeltas * 3 + cepnum; /* cepnum'th static ivar */
+ ivar = (picoos_int32) (pdf->content[k])
+ << (pdf->ivarpow[cepnum]);
+ break;
+ case PICOCEP_WANTDELTA:
+ ceporder = pdf->ceporder;
+ k = vecstart + pdf->numvuv + ceporder * 3
+ + pdf->numdeltas * 3 + cepnum; /* cepnum'th delta ivar */
+ ivar = (picoos_int32) (pdf->content[k])
+ << (pdf->ivarpow[ceporder + cepnum]);
+ break;
+ case PICOCEP_WANTDELTA2:
+ ceporder2 = 2 * pdf->ceporder;
+ k = vecstart + pdf->numvuv + ceporder2 + pdf->numdeltas
+ * 3 + ceporder2 + cepnum; /* cepnum'th delta delta ivar */
+ ivar = (picoos_int32) (pdf->content[k])
+ << (pdf->ivarpow[ceporder2 + cepnum]);
+ break;
+ default:
+ PICODBG_ERROR(("unknown type wantStaticOrDeltax = %i", wantStaticOrDeltax));
+ }
+ return ivar;
+ break;
+ default:
+ PICODBG_ERROR(("unknown type wantMeanOrIvar = %i", wantMeanOrIvar));
+ return 0;
+ }
+ }
+ return 0;
+}
+
+/**
+ * Retrieve actual values for MGC from PDF resource - Variant "Direct"
+ * @param pdf : pointer to picoos_uint8, sequence of pdf vectors, each vector of length 1+ceporder*2+numdeltas*3+ceporder*3
+ * @param indices : indices of pdf vectors for all frames in current sentence
+ * @param activeEndPos : ??
+ * @param cepnum : cepstral dimension to be treated
+ * @param smoothcep : ??
+ * @return the actual value retrieved
+ * @callgraph
+ * @callergraph
+ */
+static void getDirect(picokpdf_PdfMUL pdf, picoos_uint16 *indices,
+ picoos_uint16 activeEndPos,
+ picoos_uint8 cepnum, picoos_int16 *smoothcep)
+{
+ picoos_uint16 i;
+ picoos_uint32 j;
+ picoos_uint32 vecstart;
+ picoos_int32 mean, ivar;
+ picoos_int32 prev_mean;
+ picoos_uint8 vecsize = pdf->vecsize;
+ picoos_uint8 order = pdf->ceporder;
+
+ j = cepnum;
+ prev_mean = 0;
+ for (i = 0; i < activeEndPos; i++) {
+ if (i > 0 && indices[i] == indices[i - 1]) {
+ mean = prev_mean;
+ } else {
+ vecstart = indices[i] * vecsize;
+ mean = getFromPdf(pdf, vecstart, cepnum, PICOCEP_WANTMEAN,
+ PICOCEP_WANTSTATIC);
+ ivar = getFromPdf(pdf, vecstart, cepnum, PICOCEP_WANTIVAR,
+ PICOCEP_WANTSTATIC);
+ prev_mean = mean = picocep_fixptdiv(mean, ivar, pdf->bigpow);
+ }
+ smoothcep[j] = (picoos_int16)(mean/(1<<pdf->meanpow));
+ j += order;
+ }
+}
+
+/**
+ * Retrieve actual values for voicing from PDF resource
+ * @param pdf : pointer to picoos_uint8, sequence of pdf vectors, each vector of length 1+ceporder*2+numdeltas*3+ceporder*3
+ * @param indices : indices of pdf vectors for all frames in current sentence
+ * @param activeEndPos : end position of indices to be considered
+ * @return smoothcep : the values retrieved
+ * @callgraph
+ * @callergraph
+ */
+static void getVoiced(picokpdf_PdfMUL pdf, picoos_uint16 *indices,
+ picoos_uint16 activeEndPos,
+ picoos_uint8 *smoothcep)
+{
+ picoos_uint16 i, j;
+ picoos_uint32 vecstart;
+ picoos_uint8 vecsize = pdf->vecsize;
+
+ if (pdf->numvuv == 0) {
+ /* do nothing */
+ } else {
+ for (i = 0, j = 0; i < activeEndPos; i++, j++) {
+ vecstart = indices[i] * vecsize;
+ smoothcep[j] = pdf->content[vecstart]; /* odd value is voiced, even if unvoiced */
+ }
+ }
+}
+
+/** reads platform-independent uint16 from buffer at *pos and advances *pos
+ * @param buf : buffer picoos_uint8
+ * @param *pos : start position inside buffer of pi uint16
+ * @return the uint16
+ * @callgraph
+ * @callergraph
+ */
+static picoos_uint16 get_pi_uint16(picoos_uint8 * buf, picoos_uint16 *pos)
+{
+ picoos_uint16 res;
+ res = buf[(*pos)] | ((picoos_uint16) buf[(*pos) + 1] << 8);
+ *pos += 2;
+ return res;
+}
+/**
+ * Looks up indices of one phone item and fills index buffers. Consumes Item
+ * @param cep : the CEP PU sub object pointer
+ * @param ihead : pointer to the start of the phone item
+ * @callgraph
+ * @callergraph
+ */
+static void treat_phone(cep_subobj_t * cep, picodata_itemhead_t * ihead)
+{
+ picoos_uint16 state, frame, frames;
+ picoos_uint16 indlfz, indmgc;
+ picoos_uint16 pos;
+ picoos_uint8 bufferFull;
+
+ /* treat all states
+ * for each state, repeat putting the index into the index buffer framesperstate times.
+ */
+ /* set state and frame to the first state and frame in the phone to be considered */
+ state = 0; /* the first state to be considered */
+ frame = 0; /* the first frame to be considered */
+ /* numFramesPerState: 2 byte, lf0Index: 2 byte, mgcIndex: 2 byte -> 6 bytes per state */
+ PICODBG_DEBUG(("skipping to phone state %i ",state));
+ pos = cep->inReadPos + PICODATA_ITEM_HEADSIZE + state * 6;
+ /* */
+ PICODBG_DEBUG(("state info starts at inBuf pos %i ",pos));
+ /* get the current frames per state */
+ frames = get_pi_uint16(cep->inBuf, &pos);
+ /* */
+ PICODBG_DEBUG(("number of frames for this phone state: %i",frames));
+ /* */
+ PICODBG_DEBUG(("PARSE starting with frame %i",frame));
+
+ bufferFull = cep->indexWritePos >= PICOCEP_MAXWINLEN;
+ while ((state < ihead->info2) && (bufferFull == FALSE)) {
+
+ /* get the current state's lf0 and mgc indices and adjust according to state */
+ /* the indices have to be calculated as follows:
+ * new index = (index-1) + stateoffset(state) */
+
+ indlfz = get_pi_uint16(cep->inBuf, &pos); /* lfz index */
+ indlfz += -1 + cep->pdflfz->stateoffset[state]; /* transform index */
+ indmgc = get_pi_uint16(cep->inBuf, &pos); /* mgc index */
+ indmgc += -1 + cep->pdfmgc->stateoffset[state]; /* transform index */
+
+ /* are we reaching the end of the index buffers? */
+ if ((cep->indexWritePos - frame) + frames > PICOCEP_MAXWINLEN) {
+ /* number of frames that will still fit */
+ frames = PICOCEP_MAXWINLEN - (cep->indexWritePos - frame);
+ bufferFull = TRUE;
+ PICODBG_DEBUG(("smoothing buffer full at state=%i frame=%i",state, frame));
+ }
+ while (frame < frames) {
+ cep->indicesMGC[cep->indexWritePos] = indmgc;
+ cep->indicesLFZ[cep->indexWritePos] = indlfz;
+ cep->phoneId[cep->indexWritePos] = ihead->info1;
+ cep->indexWritePos++;
+ frame++;
+ }
+ /* proceed to next state */
+ PICODBG_DEBUG(("finished state %i with %i frames, now at index write pos %i",
+ state, frames,cep->indexWritePos));
+ state++;
+ if (state < ihead->info2) {
+ frame = 0;
+ frames = get_pi_uint16(cep->inBuf, &pos);
+ }
+ }
+ /* consume the phone item */
+ cep->inReadPos = cep->nextInPos;
+ /* */
+ PICODBG_DEBUG(("finished phone, advancing inReadPos to %i",cep->inReadPos));
+}
+
+/**
+ * Returns true if an Item has to be forwarded to next PU
+ * @param ihead : pointer to item head structure
+ * @return TRUE : the item should be forwarded
+ * @return FALSE : the item should be consumed
+ * @callgraph
+ * @callergraph
+ */
+static picoos_uint8 forwardingItem(picodata_itemhead_t * ihead)
+{
+ if ((PICODATA_ITEM_CMD == ihead->type) && (PICODATA_ITEMINFO1_CMD_IGNSIG
+ == ihead->info1)) {
+ return FALSE;
+ } else {
+ return TRUE;
+ }
+}
+
+/**
+ * performs a step of the cep processing
+ * @param this : pointer to current PU (Control Unit)
+ * @param mode : mode for the PU (not used)
+ * @param numBytesOutput : pointer to output number fo bytes produced
+ * @return one of the "picodata_step_result_t" values
+ * @callgraph
+ * @callergraph
+ */
+static picodata_step_result_t cepStep(register picodata_ProcessingUnit this,
+ picoos_int16 mode, picoos_uint16 * numBytesOutput)
+{
+ register cep_subobj_t * cep;
+ picodata_itemhead_t ihead /* , ohead */;
+ picoos_uint8 * icontents;
+ pico_status_t sResult = PICO_OK;
+ picoos_uint16 blen, clen;
+ picoos_uint16 numinb, numoutb;
+
+#if defined (PICO_DEBUG)
+ picoos_char msgstr[PICOCEP_MSGSTR_SIZE];
+#endif
+ numinb = 0;
+ numoutb = 0;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ cep = (cep_subobj_t *) this->subObj;
+ mode = mode; /* avoid warning "var not used in this function"*/
+
+ /*Init number of output bytes*/
+ *numBytesOutput = 0;
+
+ while (1) { /* exit via return */
+
+ PICODBG_DEBUG(("doing pu state %i", cep->procState));
+
+ switch (cep->procState) {
+
+ case PICOCEP_STEPSTATE_COLLECT:
+ /* *************** item collector ***********************************/
+
+ PICODBG_TRACE(("COLLECT"));
+
+ /*collecting items from the PU input buffer*/
+ sResult = picodata_cbGetItem(this->cbIn,
+ &(cep->inBuf[cep->inWritePos]), cep->inBufSize
+ - cep->inWritePos, &blen);
+ if (PICO_EOF == sResult) { /* there are no more items available and we always need more data here */
+ PICODBG_DEBUG(("COLLECT need more data, returning IDLE"));
+ return PICODATA_PU_IDLE;
+ }
+
+ PICODBG_DEBUG(("got item, status: %d", sResult));
+
+ if ((PICO_OK == sResult) && (blen > 0)) {
+ /* we now have one item */
+ cep->inWritePos += blen;
+ cep->procState = PICOCEP_STEPSTATE_PROCESS_PARSE;
+ } else {
+ /* ignore item and stay in collect */
+ PICODBG_ERROR(("COLLECT got bad result %i", sResult));
+ cep->inReadPos = cep->inWritePos = 0;
+ }
+ /* return PICODATA_PU_ATOMIC; */
+ break;
+
+ case PICOCEP_STEPSTATE_PROCESS_PARSE:
+ /* **************** put item indices into index buffers (with repetition) ******************/
+
+ PICODBG_TRACE(("PARSE"));
+
+ PICODBG_DEBUG(("getting info from inBuf in range: [%i,%i[", cep->inReadPos, cep->inWritePos));
+ if (cep->inWritePos <= cep->inReadPos) {
+ /* no more items in inBuf */
+ /* we try to get more data */
+ PICODBG_DEBUG(("no more items in inBuf, try to collect more"));
+ /* cep->needMoreInput = TRUE; */
+ cep->inReadPos = cep->inWritePos = 0;
+ cep->procState = PICOCEP_STEPSTATE_COLLECT;
+ break;
+ }
+ /* look at the current item */
+ /*verify that current item is valid */
+ if (!picodata_is_valid_item(cep->inBuf + cep->inReadPos,
+ cep->inWritePos - cep->inReadPos)) {
+ PICODBG_ERROR(("found invalid item"));
+ sResult = picodata_get_iteminfo(
+ cep->inBuf + cep->inReadPos, cep->inWritePos
+ - cep->inReadPos, &ihead, &icontents);PICODBG_DEBUG(("PARSE bad item %s",picodata_head_to_string(&ihead,msgstr,PICOCEP_MSGSTR_SIZE)));
+
+ return PICODATA_PU_ERROR;
+ }
+
+ sResult = picodata_get_iteminfo(cep->inBuf + cep->inReadPos,
+ cep->inWritePos - cep->inReadPos, &ihead, &icontents);
+
+ if (PICO_EXC_BUF_UNDERFLOW == sResult) {
+ /* no more items in inBuf */
+ /* we try to get more data */
+ PICODBG_DEBUG(("no more items in inBuf, try to collect more"));
+ /* cep->needMoreInput = TRUE; */
+ cep->inReadPos = cep->inWritePos = 0;
+ cep->procState = PICOCEP_STEPSTATE_COLLECT;
+ break;
+ } else if (PICO_OK != sResult) {
+ PICODBG_ERROR(("unknown exception (sResult == %i)",sResult));
+ return (picodata_step_result_t) picoos_emRaiseException(
+ this->common->em, sResult, NULL, NULL);
+ break;
+ }
+
+ PICODBG_DEBUG(("PARSE looking at item %s",picodata_head_to_string(&ihead,msgstr,PICOCEP_MSGSTR_SIZE)));
+
+ cep->nextInPos = PICODATA_ITEM_HEADSIZE + ihead.len;
+
+ /* we decide what to do next depending on the item and the state of the index buffer:
+ * - process buffer if buffer not empty and sentence end or flush or ignsig/start (don't consume item yet)
+ * - fill buffer with (part of) phone contents if item is a phone
+ * - consume or copy for later output otherwise
+ */
+
+ if (cep->inIgnoreState) {
+ if ((PICODATA_ITEM_CMD == ihead.type)
+ && (PICODATA_ITEMINFO1_CMD_IGNSIG == ihead.info1)
+ && (PICODATA_ITEMINFO2_CMD_END == ihead.info2)) {
+ cep->inIgnoreState = 0;
+ }PICODBG_DEBUG(("cep: PARSE consuming item of inBuf"));
+ cep->inReadPos = cep->nextInPos;
+ break;
+ }
+
+ /* see if it is a sentence end boundary or termination boundary (flush) and there are indices to smooth -> smooth */
+ if ((PICODATA_ITEM_BOUND == ihead.type)
+ && ((PICODATA_ITEMINFO1_BOUND_SEND == ihead.info1)
+ || (PICODATA_ITEMINFO1_BOUND_TERM == ihead.info1))
+ && (cep->indexWritePos > 0)) {
+ /* we smooth the buffer */
+ cep->activeEndPos = cep->indexWritePos;
+ cep->sentenceEnd = TRUE;
+ /* output whatever we got */
+ PICODBG_DEBUG(("cep: PARSE found sentence terminator; setting activeEndPos to %i",cep->activeEndPos));
+ cep->procState = PICOCEP_STEPSTATE_PROCESS_SMOOTH;
+ break;
+ } else if (PICODATA_ITEM_PHONE == ihead.type) {
+ /* it is a phone */
+ PICODBG_DEBUG(("cep: PARSE treating PHONE"));
+ treat_phone(cep, &ihead);
+
+ } else {
+ if ((PICODATA_ITEM_CMD == ihead.type)
+ && (PICODATA_ITEMINFO1_CMD_IGNSIG == ihead.info1)
+ && (PICODATA_ITEMINFO2_CMD_START == ihead.info2)) {
+ cep->inIgnoreState = 1;
+ }
+ /* sentence end or flush remaining after frame or other non-processable item, e.g. command */
+ /* do we have to forward? */
+ if (forwardingItem(&ihead)) {
+ /* if no active frames, output immediately */
+ if (cep->indexWritePos <= 0) {
+ /* copy item to outBuf */
+ PICODBG_DEBUG(("PARSE copy item in inBuf to outBuf"));
+ picodata_copy_item(cep->inBuf + cep->inReadPos,
+ cep->inWritePos - cep->inReadPos,
+ cep->outBuf, cep->outBufSize, &blen);
+ cep->outWritePos += blen;
+ PICODATA_INFO_ITEM(this->voice->kbArray[PICOKNOW_KBID_DBG],
+ (picoos_uint8 *)"cep: do forward item ",
+ cep->outBuf, PICODATA_MAX_ITEMSIZE);
+ /* output item and then go to parse to treat a new item. */
+ cep->feedFollowState
+ = PICOCEP_STEPSTATE_PROCESS_PARSE;
+ cep->procState = PICOCEP_STEPSTATE_FEED;
+ } else if ((cep->headxWritePos < PICOCEP_MAXNR_HEADX)
+ && (cep->cbufWritePos + ihead.len
+ < cep->cbufBufSize)) {
+ /* there is enough space to store item */
+ PICODBG_DEBUG(("unhandled item (type %c, length %i). Storing associated with index %i",ihead.type, ihead.len, cep->indexWritePos));
+ sResult
+ = picodata_get_itemparts(
+ cep->inBuf + cep->inReadPos,
+ cep->inWritePos - cep->inReadPos,
+ &(cep->headx[cep->headxWritePos].head),
+ &(cep->cbuf[cep->cbufWritePos]),
+ cep->cbufBufSize
+ - cep->cbufWritePos, &clen);
+
+ if (sResult != PICO_OK) {
+ PICODBG_ERROR(("problem getting item parts"));
+ picoos_emRaiseException(this->common->em,
+ sResult, NULL, NULL);
+ return PICODATA_PU_ERROR;
+ }
+ /* remember sync position */
+ cep->headx[cep->headxWritePos].frame
+ = cep->indexWritePos;
+
+ if (clen > 0) {
+ cep->headx[cep->headxWritePos].cind
+ = cep->cbufWritePos;
+ cep->cbufWritePos += clen;
+ } else {
+ cep->headx[cep->headxWritePos].cind = 0;
+ }
+ cep->headxWritePos++;
+ } else {
+ /* buffer full, smooth and output whatever we got */
+ PICODBG_DEBUG(("PARSE is forced to smooth prematurely; setting activeEndPos to %i", cep->activeEndPos));
+ cep->procState = PICOCEP_STEPSTATE_PROCESS_SMOOTH;
+ /* don't consume item yet */
+ break;
+ }
+ } else {
+ }PICODBG_DEBUG(("cep: PARSE consuming item of inBuf"));
+ cep->inReadPos = cep->nextInPos;
+ }
+ break;
+
+ case PICOCEP_STEPSTATE_PROCESS_SMOOTH:
+ /* **************** smooth (indexed) coefficients and store smoothed in outBuffers ****************/
+
+ PICODBG_TRACE(("SMOOTH"));
+ {
+ picokpdf_PdfMUL pdf;
+
+ /* picoos_uint16 framesTreated = 0; */
+ picoos_uint8 cepnum;
+ picoos_uint16 N;
+
+ N = cep->activeEndPos; /* numframes in current step */
+
+ /* the range to be smoothed starts at 0 and is N long */
+
+ /* smooth each cepstral dimension separately */
+ /* still to be experimented if higher order coeff can remain unsmoothed, i.e. simple copy from pdf */
+
+ /* reset the f0, ceps and voiced outfuffers */
+ cep->outXCepReadPos = cep->outXCepWritePos = 0;
+ cep->outVoicedReadPos = cep->outVoicedWritePos = 0;
+ cep->outF0ReadPos = cep->outF0WritePos = 0;
+
+ PICODBG_DEBUG(("smoothing %d frames\n", N));
+
+ /* smooth f0 */
+ pdf = cep->pdflfz;
+ for (cepnum = 0; cepnum < pdf->ceporder; cepnum++) {
+ if (cep->activeEndPos <= 0) {
+ /* do nothing */
+ } else if (3 < N) {
+ makeWUWandWUm(cep, pdf, cep->indicesLFZ, 0, N,
+ cepnum); /* update diag0, diag1, diag2, WUm */
+ invMatrix(cep, N, cep->outF0 + cep->outF0WritePos, cepnum, pdf,
+ PICOCEP_LFZINVPOW, PICOCEP_LFZDOUBLEDEC);
+ } else {
+ getDirect(pdf, cep->indicesLFZ, cep->activeEndPos,
+ cepnum, cep->outF0 + cep->outF0WritePos);
+ }
+ }/* end for cepnum */
+ cep->outF0WritePos += cep->activeEndPos * pdf->ceporder;
+
+ /* smooth mgc */
+ pdf = cep->pdfmgc;
+ for (cepnum = 0; cepnum < pdf->ceporder; cepnum++) {
+ if (cep->activeEndPos <= 0) {
+ /* do nothing */
+ } else if (3 < N) {
+ makeWUWandWUm(cep, pdf, cep->indicesMGC, 0, N,
+ cepnum); /* update diag0, diag1, diag2, WUm */
+ invMatrix(cep, N, cep->outXCep
+ + cep->outXCepWritePos, cepnum,
+ pdf, PICOCEP_MGCINVPOW,
+ PICOCEP_MGCDOUBLEDEC);
+ } else {
+ getDirect(pdf, cep->indicesMGC, cep->activeEndPos,
+ cepnum, cep->outXCep + cep->outXCepWritePos);
+ }
+ }/* end for cepnum */
+ cep->outXCepWritePos += cep->activeEndPos * pdf->ceporder;
+
+ getVoiced(pdf, cep->indicesMGC, cep->activeEndPos, cep->outVoiced
+ + cep->outVoicedWritePos);
+ cep->outVoicedWritePos += cep->activeEndPos;
+
+ }
+ /* setting indexReadPos to the next active index to be used. (will be advanced by FRAME when
+ * reading the phoneId */
+ cep->indexReadPos = 0;
+ cep->procState = PICOCEP_STEPSTATE_PROCESS_FRAME;
+ return PICODATA_PU_BUSY; /*data to feed*/
+
+ break;
+
+ case PICOCEP_STEPSTATE_PROCESS_FRAME:
+
+ /* *************** creating output items (type FRAME) ***********************************/
+
+ PICODBG_TRACE(("FRAME"));
+
+ if ((cep->headxBottom < cep->headxWritePos)
+ && (cep->headx[cep->headxBottom].frame
+ <= cep->indexReadPos)) {
+
+ /* output item in headx/cbuf */
+ /* copy item to outBuf */
+ PICODBG_DEBUG(("FRAME copy item in inBuf to outBuf"));
+ picodata_put_itemparts(
+ &(cep->headx[cep->headxBottom].head),
+ &(cep->cbuf[cep->headx[cep->headxBottom].cind]),
+ cep->headx[cep->headxBottom].head.len, cep->outBuf,
+ cep->outBufSize, &blen);
+ cep->outWritePos += blen;
+ /* consume item in headx/cbuf */
+ PICODBG_DEBUG(("PARSE consuming item of headx/cbuf"));
+ cep->headxBottom++;
+
+ /* output item and then go to parse to treat a new item. */
+ cep->feedFollowState = PICOCEP_STEPSTATE_PROCESS_FRAME;
+ cep->procState = PICOCEP_STEPSTATE_FEED;
+ break;
+ }
+
+ if (cep->indexReadPos < cep->activeEndPos) {
+ /*------------ there are frames to output ----------------------------------------*/
+ /* still frames to output, create new FRAME_PAR item */
+
+ cep->nNumFrames++;
+
+ PICODBG_DEBUG(("FRAME creating FRAME_PAR: active: [0,%i[, read=%i, write=%i",
+ cep->activeEndPos, cep->indexReadPos, cep->indexWritePos
+ ));
+
+ /* create FRAME_PAR items from cep->outXX one by one */
+
+ /* converting the ceps shorts into floats:
+ * scmeanpow = pdf->bigpow - pdf->meanpow;
+ * for all sval:
+ * fval = (picoos_single) sval / scmeanpow;
+ */
+
+ cep->outWritePos = cep->outReadPos = 0;
+ cep->outBuf[cep->outWritePos++] = cep->framehead.type;
+ cep->outBuf[cep->outWritePos++] = cep->framehead.info1;
+ cep->outBuf[cep->outWritePos++] = cep->framehead.info2;
+ cep->outBuf[cep->outWritePos++] = cep->framehead.len;
+
+ PICODBG_DEBUG(("FRAME writing position after header: %i",cep->outWritePos));
+
+ {
+ picoos_uint16 tmpUint16;
+ picoos_int16 tmpInt16;
+ picoos_uint16 i;
+
+ /* */
+ PICODBG_DEBUG(("FRAME reading phoneId[%i] = %c:",cep->indexReadPos, cep->phoneId[cep->indexReadPos]));
+ /* */
+
+ tmpUint16
+ = (picoos_uint16) cep->phoneId[cep->indexReadPos];
+
+ picoos_mem_copy((void *) &tmpUint16,
+ (void *) &cep->outBuf[cep->outWritePos],
+ sizeof(tmpUint16));
+
+ cep->outWritePos += sizeof(tmpUint16);
+
+ PICODBG_DEBUG(("FRAME writing position after phone id: %i",cep->outWritePos));
+
+ for (i = 0; i < cep->pdflfz->ceporder; i++) {
+
+ tmpUint16 = (cep->outVoiced[cep->outVoicedReadPos]
+ & 0x01) ? cep->outF0[cep->outF0ReadPos]
+ : (picoos_uint16) 0;
+
+ picoos_mem_copy((void *) &tmpUint16,
+ (void *) &cep->outBuf[cep->outWritePos],
+ sizeof(tmpUint16));
+ cep->outWritePos += sizeof(tmpUint16);
+
+ tmpUint16
+ = (picoos_uint16) (cep->outVoiced[cep->outVoicedReadPos]);
+ picoos_mem_copy((void *) &tmpUint16,
+ (void *) &cep->outBuf[cep->outWritePos],
+ sizeof(tmpUint16));
+ cep->outWritePos += sizeof(tmpUint16);
+ tmpUint16
+ = (picoos_uint16) (cep->outF0[cep->outF0ReadPos]);
+ picoos_mem_copy((void *) &tmpUint16,
+ (void *) &cep->outBuf[cep->outWritePos],
+ sizeof(tmpUint16));
+ cep->outWritePos += sizeof(tmpUint16);
+
+ cep->outVoicedReadPos++;
+ cep->outF0ReadPos++;
+ }
+
+ PICODBG_DEBUG(("FRAME writing position after f0: %i",cep->outWritePos));
+
+ for (i = 0; i < cep->pdfmgc->ceporder; i++) {
+ tmpInt16 = cep->outXCep[cep->outXCepReadPos++];
+ picoos_mem_copy((void *) &tmpInt16,
+ (void *) &cep->outBuf[cep->outWritePos],
+ sizeof(tmpInt16));
+ cep->outWritePos += sizeof(tmpInt16);
+ }
+
+ PICODBG_DEBUG(("FRAME writing position after cepstrals: %i",cep->outWritePos));
+
+ tmpUint16
+ = (picoos_uint16) cep->indicesMGC[cep->indexReadPos++];
+
+ picoos_mem_copy((void *) &tmpUint16,
+ (void *) &cep->outBuf[cep->outWritePos],
+ sizeof(tmpUint16));
+
+ PICODBG_DEBUG(("FRAME writing position after mgc index: %i",cep->outWritePos));
+
+ cep->outWritePos += sizeof(tmpUint16);
+
+ }
+ /* finished to create FRAME_PAR, now output and then come back*/
+ cep->feedFollowState = PICOCEP_STEPSTATE_PROCESS_FRAME;
+ cep->procState = PICOCEP_STEPSTATE_FEED;
+
+ } else if (cep->sentenceEnd) {
+ /*------------ no more frames to output at end of sentence ----------------------------------------*/
+ PICODBG_INFO(("End of sentence - Processed frames : %d",
+ cep->nNumFrames));
+ cep->nNumFrames = 0;PICODBG_DEBUG(("FRAME no more active frames for this sentence"));
+ /* no frames left in this sentence*/
+ /* reset for new sentence */
+ initSmoothing(cep);
+ cep->sentenceEnd = FALSE;
+ cep->indexReadPos = cep->indexWritePos = 0;
+ cep->activeEndPos = PICOCEP_MAXWINLEN;
+ cep->headxBottom = cep->headxWritePos = 0;
+ cep->cbufWritePos = 0;
+ cep->procState = PICOCEP_STEPSTATE_PROCESS_PARSE;
+ } else {
+ /*------------ no more frames can be output but sentence end not reached ----------------------------------------*/
+ PICODBG_DEBUG(("Maximum number of frames per sentence reached"));
+ cep->procState = PICOCEP_STEPSTATE_PROCESS_PARSE;
+ }
+ /*----------------------------------------------------*/
+ break;
+
+ case PICOCEP_STEPSTATE_FEED:
+ /* ***********************************************************************/
+ /* FEED: combine input item with pos/phon pairs to output item */
+ /* ***********************************************************************/
+
+ PICODBG_DEBUG(("FEED"));
+
+ PICODBG_DEBUG(("FEED putting outBuf item into cb"));
+
+ /*feeding items to PU output buffer*/
+ sResult = picodata_cbPutItem(this->cbOut, cep->outBuf,
+ cep->outBufSize, &blen);
+
+ if (PICO_EXC_BUF_OVERFLOW == sResult) {
+ /* we have to redo this item */
+ PICODBG_DEBUG(("FEED got overflow, returning PICODATA_PU_OUT_FULL"));
+ return PICODATA_PU_OUT_FULL;
+ } else if (PICO_OK == sResult) {
+
+ if (cep->outBuf[0] != 'k') {
+ PICODATA_INFO_ITEM(this->voice->kbArray[PICOKNOW_KBID_DBG],
+ (picoos_uint8 *)"cep: ",
+ cep->outBuf, PICODATA_MAX_ITEMSIZE);
+ }
+
+ *numBytesOutput += blen;
+ /*-------------------------*/
+ /*reset the output pointers*/
+ /*-------------------------*/
+ if (cep->outReadPos >= cep->outWritePos) {
+ cep->outReadPos = 0;
+ cep->outWritePos = 0;
+ }
+ cep->procState = cep->feedFollowState;
+ PICODBG_DEBUG(("FEED ok, going back to procState %i", cep->procState));
+ return PICODATA_PU_BUSY;
+ } else {
+ PICODBG_DEBUG(("FEED got exception %i when trying to output item",sResult));
+ cep->procState = cep->feedFollowState;
+ return (picodata_step_result_t) sResult;
+ }
+ break;
+
+ default:
+ /*NOT feeding items*/
+ sResult = PICO_EXC_BUF_IGNORE;
+ break;
+ }/*end switch (cep->procState) */
+ return PICODATA_PU_BUSY; /*check if there is more data to process after feeding*/
+
+ }/*end while*/
+ /* we should never come here */
+ return PICODATA_PU_ERROR;
+}/*cepStep*/
+
+/* Picocep.c end */
diff --git a/lib/picocep.h b/lib/picocep.h
new file mode 100644
index 0000000..79a5d7c
--- /dev/null
+++ b/lib/picocep.h
@@ -0,0 +1,54 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picocep.h
+ *
+ * Cepstral Smoothing PU - Header file
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#ifndef PICOCEP_H_
+#define PICOCEP_H_
+
+#include "picoos.h"
+#include "picodata.h"
+#include "picorsrc.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+/* ******************************************************************************
+ * items related to the generic interface
+ ********************************************************************************/
+picodata_ProcessingUnit picocep_newCepUnit(picoos_MemoryManager mm,
+ picoos_Common common, picodata_CharBuffer cbIn,
+ picodata_CharBuffer cbOut, picorsrc_Voice voice);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /*PICOCEP_H_*/
diff --git a/lib/picoctrl.c b/lib/picoctrl.c
new file mode 100644
index 0000000..5af70e3
--- /dev/null
+++ b/lib/picoctrl.c
@@ -0,0 +1,840 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picoctrl.c
+ *
+ * Control PU -- Implementation
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#include "picodefs.h"
+#include "picoos.h"
+#include "picodbg.h"
+#include "picodata.h"
+#include "picorsrc.h"
+
+/* processing unit definitions */
+#include "picotok.h"
+#include "picopr.h"
+#include "picowa.h"
+#include "picosa.h"
+#include "picoacph.h"
+#include "picospho.h"
+#include "picopam.h"
+#include "picocep.h"
+#include "picosig.h"
+#if defined(PICO_DEVEL_MODE)
+#include "../history/picosink.h"
+#endif
+
+#include "picoctrl.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+/**
+ * @addtogroup picoctrl
+ * @b Control
+ * The "control" is a processing unit (PU) that contains and governs a sequence of sub-PUs
+ * (TTS processing chain).
+ * At each step (ctrlStep) it passes control to one of the sub-PUs (currrent PU). It may re-assign
+ * the role of "current PU" to another sub-PU, according to the status information returned from each PU.
+ */
+
+/*----------------------------------------------------------
+ * object : Control
+ * shortcut : ctrl
+ * derived from : picodata_ProcessingUnit
+ * implements a ProcessingUnit by creating and controlling
+ * a sequence of Processing Units (of possibly different
+ * implementations) exchanging data via CharBuffers
+ * ---------------------------------------------------------*/
+/* control sub-object */
+typedef struct ctrl_subobj {
+ picoos_uint8 numProcUnits;
+ picoos_uint8 curPU;
+ picoos_uint8 lastItemTypeProduced;
+ picodata_ProcessingUnit procUnit [PICOCTRL_MAX_PROC_UNITS];
+ picodata_step_result_t procStatus [PICOCTRL_MAX_PROC_UNITS];
+ picodata_CharBuffer procCbOut [PICOCTRL_MAX_PROC_UNITS];
+} ctrl_subobj_t;
+
+/**
+ * performs Control PU initialization
+ * @param this : pointer to Control PU
+ * @return PICO_OK : processing done
+ * @return PICO_ERR_OTHER : init error
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t ctrlInitialize(register picodata_ProcessingUnit this) {
+ register ctrl_subobj_t * ctrl;
+ pico_status_t status= PICO_OK;
+ picoos_int8 i;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ ctrl = (ctrl_subobj_t *) this->subObj;
+ ctrl->curPU = 0;
+ ctrl->lastItemTypeProduced=0; /*no item produced by default*/
+ status = PICO_OK;
+ for (i = 0; i < ctrl->numProcUnits; i++) {
+ if (PICO_OK == status) {
+ status = ctrl->procUnit[i]->initialize(ctrl->procUnit[i]);
+ PICODBG_DEBUG(("(re-)initializing procUnit[%i] returned status %i",i, status));
+ }
+ if (PICO_OK == status) {
+ status = picodata_cbReset(ctrl->procCbOut[i]);
+ PICODBG_DEBUG(("(re-)initializing procCbOut[%i] returned status %i",i, status));
+ }
+ }
+ if (PICO_OK != status) {
+ picoos_emRaiseException(this->common->em,status,NULL,(picoos_char*)"problem (re-)initializing the engine");
+ }
+ return status;
+}/*ctrlInitialize*/
+
+
+/**
+ * performs one processing step
+ * @param this : pointer to Control PU
+ * @param mode : activation mode (unused)
+ * @param bytesOutput : number of bytes produced during this step (output)
+ * @return PICO_OK : processing done
+ * @return PICO_EXC_OUT_OF_MEM : no more memory available
+ * @return PICO_ERR_OTHER : other error
+ * @callgraph
+ * @callergraph
+ */
+static picodata_step_result_t ctrlStep(register picodata_ProcessingUnit this,
+ picoos_int16 mode, picoos_uint16 * bytesOutput) {
+ /* rules/invariants:
+ * - all pu's above current have status idle except possibly pu+1, which may be busy.
+ * (The latter is set if any pu->step produced output)
+ * - a pu returns idle iff its cbIn is empty and it has no more data ready for output */
+
+ register ctrl_subobj_t * ctrl = (ctrl_subobj_t *) this->subObj;
+ picodata_step_result_t status;
+ picoos_uint16 puBytesOutput;
+#if defined(PICO_DEVEL_MODE)
+ picoos_uint8 btype;
+#endif
+
+ *bytesOutput = 0;
+ ctrl->lastItemTypeProduced=0; /*no item produced by default*/
+
+ /* --------------------- */
+ /* do step of current pu */
+ /* --------------------- */
+ status = ctrl->procStatus[ctrl->curPU] = ctrl->procUnit[ctrl->curPU]->step(
+ ctrl->procUnit[ctrl->curPU], mode, &puBytesOutput);
+
+ if (puBytesOutput) {
+
+#if defined(PICO_DEVEL_MODE)
+ /*store the type of item produced*/
+ btype = picodata_cbGetFrontItemType(ctrl->procUnit[ctrl->curPU]->cbOut);
+ ctrl->lastItemTypeProduced=(picoos_uint8)btype;
+#endif
+
+ if (ctrl->curPU < ctrl->numProcUnits-1) {
+ /* data was output to internal PU buffers : set following pu to busy */
+ ctrl->procStatus[ctrl->curPU + 1] = PICODATA_PU_BUSY;
+ } else {
+ /* data was output to caller output buffer */
+ *bytesOutput = puBytesOutput;
+ }
+ }
+ /* recalculate state depending on pu status returned from curPU */
+ switch (status) {
+ case PICODATA_PU_ATOMIC:
+ PICODBG_DEBUG(("got PICODATA_PU_ATOMIC"));
+ return status;
+ break;
+
+ case PICODATA_PU_BUSY:
+ PICODBG_DEBUG(("got PICODATA_PU_BUSY"));
+ if ( (ctrl->curPU+1 < ctrl->numProcUnits) && (PICODATA_PU_BUSY
+ == ctrl->procStatus[ctrl->curPU+1])) {
+ ctrl->curPU++;
+ }
+ return status;
+ break;
+
+ case PICODATA_PU_IDLE:
+ PICODBG_DEBUG(("got PICODATA_PU_IDLE"));
+ if ( (ctrl->curPU+1 < ctrl->numProcUnits) && (PICODATA_PU_BUSY
+ == ctrl->procStatus[ctrl->curPU+1])) {
+ /* still data to process below */
+ ctrl->curPU++;
+ } else if (0 == ctrl->curPU) { /* all pu's are idle */
+ /* nothing to do */
+ } else { /* find non-idle pu above */
+ PICODBG_DEBUG((
+ "find non-idle pu above from pu %d with status %d",
+ ctrl->curPU, ctrl->procStatus[ctrl->curPU]));
+ while ((ctrl->curPU > 0) && (PICODATA_PU_IDLE
+ == ctrl->procStatus[ctrl->curPU])) {
+ ctrl->curPU--;
+ }
+ ctrl->procStatus[ctrl->curPU] = PICODATA_PU_BUSY;
+ }
+ PICODBG_DEBUG(("going to pu %d with status %d",
+ ctrl->curPU, ctrl->procStatus[ctrl->curPU]));
+ /*update last scheduled PU*/
+ return ctrl->procStatus[ctrl->curPU];
+ break;
+
+ case PICODATA_PU_OUT_FULL:
+ PICODBG_DEBUG(("got PICODATA_PU_OUT_FULL"));
+ if (ctrl->curPU+1 < ctrl->numProcUnits) { /* let pu below empty buffer */
+ ctrl->curPU++;
+ ctrl->procStatus[ctrl->curPU] = PICODATA_PU_BUSY;
+ } else {
+ /* nothing more to do, out_full will be returned to caller */
+ }
+ return ctrl->procStatus[ctrl->curPU];
+ break;
+ default:
+ return PICODATA_PU_ERROR;
+ break;
+ }
+}/*ctrlStep*/
+
+/**
+ * terminates Control PU
+ * @param this : pointer to Control PU
+ * @return PICO_OK : processing done
+ * @return PICO_ERR_OTHER : other error
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t ctrlTerminate(register picodata_ProcessingUnit this) {
+ pico_status_t status = PICO_OK;
+ picoos_int16 i;
+ register ctrl_subobj_t * ctrl;
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ ctrl = (ctrl_subobj_t *) this->subObj;
+ for (i = 0; i < ctrl->numProcUnits; i++) {
+ status = ctrl->procUnit[i]->terminate(ctrl->procUnit[i]);
+ PICODBG_DEBUG(("terminating procUnit[%i] returned status %i",i, status));
+ if (PICO_OK != status) {
+ return status;
+ }
+ }
+ return status;
+}/*ctrlTerminate*/
+
+/**
+ * deallocates Control PU's subobject
+ * @param this : pointer to Control PU
+ * @return PICO_OK : processing done
+ * @return PICO_ERR_OTHER : other error
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t ctrlSubObjDeallocate(register picodata_ProcessingUnit this,
+ picoos_MemoryManager mm) {
+ register ctrl_subobj_t * ctrl;
+ picoos_int16 i;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ ctrl = (ctrl_subobj_t *) this->subObj;
+ mm = mm; /* fix warning "var not used in this function"*/
+ /* deallocate members (procCbOut and procUnit) */
+ for (i = ctrl->numProcUnits-1; i >= 0; i--) {
+ picodata_disposeProcessingUnit(this->common->mm,&ctrl->procUnit[i]);
+ picodata_disposeCharBuffer(this->common->mm, &ctrl->procCbOut[i]);
+ }
+ /* deallocate object itself */
+ picoos_deallocate(this->common->mm, (void *) &this->subObj);
+
+ return PICO_OK;
+}/*ctrlSubObjDeallocate*/
+
+/**
+ * inserts a new PU in the TTS processing chain
+ * @param this : pointer to Control PU
+ * @param puType : type of the PU to be inserted
+ * @param last : if true, inserted PU is the last in the TTS processing chain
+ * @return PICO_OK : processing done
+ * @return PICO_EXC_OUT_OF_MEM : no more memory available
+ * @return PICO_ERR_OTHER : other error
+ * @remarks Calls the PU object creation method
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t ctrlAddPU(register picodata_ProcessingUnit this,
+ picodata_putype_t puType,
+ picoos_bool levelAwareCbOut,
+ picoos_bool last)
+{
+ picoos_uint16 bufSize;
+ register ctrl_subobj_t * ctrl;
+ picodata_CharBuffer cbIn;
+ picoos_uint8 newPU;
+ if (this == NULL) {
+ return PICO_ERR_OTHER;
+ }
+ ctrl = (ctrl_subobj_t *) this->subObj;
+ if (ctrl == NULL) {
+ return PICO_ERR_OTHER;
+ }
+ newPU = ctrl->numProcUnits;
+ if (0 == newPU) {
+ PICODBG_DEBUG(("taking cbIn of this because adding first pu"));
+ cbIn = this->cbIn;
+ } else {
+ PICODBG_DEBUG(("taking cbIn of previous pu"));
+ cbIn = ctrl->procCbOut[newPU-1];
+ }
+ if (last) {
+ PICODBG_DEBUG(("taking cbOut of this because adding last pu"));
+ ctrl->procCbOut[newPU] = this->cbOut;
+ } else {
+ PICODBG_DEBUG(("creating intermediate cbOut of pu[%i]", newPU));
+ bufSize = picodata_get_default_buf_size(puType);
+ ctrl->procCbOut[newPU] = picodata_newCharBuffer(this->common->mm,
+ this->common,bufSize);
+
+ PICODBG_DEBUG(("intermediate cbOut of pu[%i] (address %i)", newPU,
+ (picoos_uint32) ctrl->procCbOut[newPU]));
+ if (NULL == ctrl->procCbOut[newPU]) {
+ return PICO_EXC_OUT_OF_MEM;
+ }
+ }
+ ctrl->procStatus[newPU] = PICODATA_PU_IDLE;
+ /*...............*/
+ switch (puType) {
+ case PICODATA_PUTYPE_TOK:
+ PICODBG_DEBUG(("creating TokenizeUnit for pu %i", newPU));
+ ctrl->procUnit[newPU] = picotok_newTokenizeUnit(this->common->mm,
+ this->common, cbIn, ctrl->procCbOut[newPU], this->voice);
+ break;
+ case PICODATA_PUTYPE_PR:
+ PICODBG_DEBUG(("creating PreprocUnit for pu %i", newPU));
+ ctrl->procUnit[newPU] = picopr_newPreprocUnit(this->common->mm,
+ this->common, cbIn, ctrl->procCbOut[newPU], this->voice);
+ break;
+ case PICODATA_PUTYPE_WA:
+ PICODBG_DEBUG(("creating WordAnaUnit for pu %i", newPU));
+ ctrl->procUnit[newPU] = picowa_newWordAnaUnit(this->common->mm,
+ this->common, cbIn, ctrl->procCbOut[newPU], this->voice);
+ break;
+ case PICODATA_PUTYPE_SA:
+ PICODBG_DEBUG(("creating SentAnaUnit for pu %i", newPU));
+ ctrl->procUnit[newPU] = picosa_newSentAnaUnit(this->common->mm,
+ this->common, cbIn, ctrl->procCbOut[newPU], this->voice);
+ break;
+ case PICODATA_PUTYPE_ACPH:
+ PICODBG_DEBUG(("creating AccPhrUnit for pu %i", newPU));
+ ctrl->procUnit[newPU] = picoacph_newAccPhrUnit(this->common->mm,
+ this->common, cbIn, ctrl->procCbOut[newPU], this->voice);
+ break;
+ case PICODATA_PUTYPE_SPHO:
+ PICODBG_DEBUG(("creating SentPhoUnit for pu %i", newPU));
+ ctrl->procUnit[newPU] = picospho_newSentPhoUnit(this->common->mm,
+ this->common, cbIn, ctrl->procCbOut[newPU], this->voice);
+ break;
+ case PICODATA_PUTYPE_PAM:
+ PICODBG_DEBUG(("creating PAMUnit for pu %i", newPU));
+ ctrl->procUnit[newPU] = picopam_newPamUnit(this->common->mm,
+ this->common, cbIn, ctrl->procCbOut[newPU], this->voice);
+ break;
+ case PICODATA_PUTYPE_CEP:
+ PICODBG_DEBUG(("creating CepUnit for pu %i", newPU));
+ ctrl->procUnit[newPU] = picocep_newCepUnit(this->common->mm,
+ this->common, cbIn, ctrl->procCbOut[newPU], this->voice);
+ break;
+#if defined(PICO_DEVEL_MODE)
+ case PICODATA_PUTYPE_SINK:
+ PICODBG_DEBUG(("creating SigUnit for pu %i", newPU));
+ ctrl->procUnit[newPU] = picosink_newSinkUnit(this->common->mm,
+ this->common, cbIn, ctrl->procCbOut[newPU], this->voice);
+ break;
+#endif
+ case PICODATA_PUTYPE_SIG:
+ PICODBG_DEBUG(("creating SigUnit for pu %i", newPU));
+ ctrl->procUnit[newPU] = picosig_newSigUnit(this->common->mm,
+ this->common, cbIn, ctrl->procCbOut[newPU], this->voice);
+ break;
+ default:
+ ctrl->procUnit[newPU] = picodata_newProcessingUnit(
+ this->common->mm, this->common, cbIn,
+ ctrl->procCbOut[newPU], this->voice);
+ break;
+ }
+ if (NULL == ctrl->procUnit[newPU]) {
+ picodata_disposeCharBuffer(this->common->mm,&ctrl->procCbOut[newPU]);
+ return PICO_EXC_OUT_OF_MEM;
+ }
+ ctrl->numProcUnits++;
+ return PICO_OK;
+}/*ctrlAddPU*/
+
+/*forward declaration : see below for full function body*/
+void picoctrl_disposeControl(picoos_MemoryManager mm,
+ picodata_ProcessingUnit * this);
+
+/**
+ * initializes a control PU object
+ * @param mm : memory manager
+ * @param common : the common object
+ * @param cbIn : the input char buffer
+ * @param cbOut : the output char buffer
+ * @param voice : the voice object
+ * @return the pointer to the PU object created if OK
+ * @return PICO_EXC_OUT_OF_MEM : no more memory available
+ * @return NULL otherwise
+ * @callgraph
+ * @callergraph
+ */
+picodata_ProcessingUnit picoctrl_newControl(picoos_MemoryManager mm,
+ picoos_Common common, picodata_CharBuffer cbIn,
+ picodata_CharBuffer cbOut, picorsrc_Voice voice) {
+ picoos_int16 i;
+ register ctrl_subobj_t * ctrl;
+ picodata_ProcessingUnit this = picodata_newProcessingUnit(mm, common, cbIn,
+ cbOut,voice);
+ if (this == NULL) {
+ return NULL;
+ }
+
+ this->initialize = ctrlInitialize;
+ this->step = ctrlStep;
+ this->terminate = ctrlTerminate;
+ this->subDeallocate = ctrlSubObjDeallocate;
+
+ this->subObj = picoos_allocate(mm, sizeof(ctrl_subobj_t));
+ if (this->subObj == NULL) {
+ picoos_deallocate(mm, (void **)(void*)&this);
+ return NULL;
+ }
+
+ ctrl = (ctrl_subobj_t *) this->subObj;
+
+ for (i=0; i < PICOCTRL_MAX_PROC_UNITS; i++) {
+ ctrl->procUnit[i] = NULL;
+ ctrl->procStatus[i] = PICODATA_PU_IDLE;
+ ctrl->procCbOut[i] = NULL;
+ }
+ ctrl->numProcUnits = 0;
+
+ if (
+ (PICO_OK == ctrlAddPU(this,PICODATA_PUTYPE_TOK, FALSE, /*last*/FALSE)) &&
+ (PICO_OK == ctrlAddPU(this,PICODATA_PUTYPE_PR, FALSE, FALSE)) &&
+ (PICO_OK == ctrlAddPU(this,PICODATA_PUTYPE_WA, FALSE, FALSE)) &&
+ (PICO_OK == ctrlAddPU(this,PICODATA_PUTYPE_SA, FALSE, FALSE)) &&
+ (PICO_OK == ctrlAddPU(this,PICODATA_PUTYPE_ACPH, FALSE, FALSE)) &&
+ (PICO_OK == ctrlAddPU(this,PICODATA_PUTYPE_SPHO, FALSE, FALSE)) &&
+ (PICO_OK == ctrlAddPU(this,PICODATA_PUTYPE_PAM, FALSE, FALSE)) &&
+ (PICO_OK == ctrlAddPU(this,PICODATA_PUTYPE_CEP, FALSE, FALSE)) &&
+ (PICO_OK == ctrlAddPU(this,PICODATA_PUTYPE_SIG, FALSE, TRUE))
+ ) {
+
+ /* we don't call ctrlInitialize here because ctrlAddPU does initialize the PUs allready and the only thing
+ * remaining to initialize is:
+ */
+ ctrl->curPU = 0;
+ return this;
+ } else {
+ picoctrl_disposeControl(this->common->mm,&this);
+ return NULL;
+ }
+
+}/*picoctrl_newControl*/
+
+/**
+ * disposes a Control PU
+ * @param mm : memory manager
+ * @param this : pointer to Control PU
+ *
+ * @return void
+ * @callgraph
+ * @callergraph
+ */
+void picoctrl_disposeControl(picoos_MemoryManager mm,
+ picodata_ProcessingUnit * this)
+{
+ picodata_disposeProcessingUnit(mm, this);
+}/*picoctrl_disposeControl*/
+
+/* **************************************************************************
+ *
+ * Engine
+ *
+ ****************************************************************************/
+/** object : Engine
+ * shortcut : eng
+ */
+typedef struct picoctrl_engine {
+ picoos_uint32 magic; /* magic number used to validate handles */
+ void *raw_mem;
+ picoos_Common common;
+ picorsrc_Voice voice;
+ picodata_ProcessingUnit control;
+ picodata_CharBuffer cbIn, cbOut;
+} picoctrl_engine_t;
+
+
+#define MAGIC_MASK 0x5069436F /* PiCo */
+
+#define SET_MAGIC_NUMBER(eng) \
+ (eng)->magic = ((picoos_uint32) (eng)) ^ MAGIC_MASK
+
+#define CHECK_MAGIC_NUMBER(eng) \
+ ((eng)->magic == (((picoos_uint32) (eng)) ^ MAGIC_MASK))
+
+/**
+ * performs an engine reset
+ * @param this : the engine object
+ * @return PICO_OK : reset performed
+ * @return otherwise error code
+ * @callgraph
+ * @callergraph
+ */
+pico_status_t picoctrl_engReset(picoctrl_Engine this)
+{
+ pico_status_t status;
+
+ if (NULL == this) {
+ return PICO_ERR_NULLPTR_ACCESS;
+ }
+ picoos_emReset(this->common->em);
+
+ status = this->control->terminate(this->control);
+ if (PICO_OK == status) {
+ status = this->control->initialize(this->control);
+ }
+ if (PICO_OK == status) {
+ status = picodata_cbReset(this->cbIn);
+ }
+ if (PICO_OK == status) {
+ status = picodata_cbReset(this->cbOut);
+ }
+ if (PICO_OK != status) {
+ picoos_emRaiseException(this->common->em,status,NULL,(picoos_char*) "problem resetting engine");
+ }
+ return status;
+}
+
+/**
+ * checks an engine handle
+ * @param this : the engine object
+ * @return PICO_OK : reset performed
+ * @return non-zero if 'this' is a valid engine handle
+ * @return zero otherwise
+ * @callgraph
+ * @callergraph
+ */
+picoos_int16 picoctrl_isValidEngineHandle(picoctrl_Engine this)
+{
+ return (this != NULL) && CHECK_MAGIC_NUMBER(this);
+}/*picoctrl_isValidEngineHandle*/
+
+/**
+ * creates a new engine object
+ * @param mm : memory manager to be used for this engine
+ * @param rm : resource manager to be used for this engine
+ * @param voiceName : voice definition to be used for this engine
+ * @return PICO_OK : reset performed
+ * @return new engine handle
+ * @return NULL otherwise
+ * @callgraph
+ * @callergraph
+ */
+picoctrl_Engine picoctrl_newEngine(picoos_MemoryManager mm,
+ picorsrc_ResourceManager rm, const picoos_char * voiceName) {
+ picoos_uint8 done= TRUE;
+
+ picoos_uint16 bSize;
+
+ picoos_MemoryManager engMM;
+ picoos_ExceptionManager engEM;
+
+ picoctrl_Engine this = (picoctrl_Engine) picoos_allocate(mm, sizeof(*this));
+
+ PICODBG_DEBUG(("creating engine for voice '%s'",voiceName));
+
+ done = (NULL != this);
+
+ if (done) {
+ this->magic = 0;
+ this->common = NULL;
+ this->voice = NULL;
+ this->control = NULL;
+ this->cbIn = NULL;
+ this->cbOut = NULL;
+
+ this->raw_mem = picoos_allocate(mm, PICOCTRL_DEFAULT_ENGINE_SIZE);
+ if (NULL == this->raw_mem) {
+ done = FALSE;
+ }
+ }
+
+ if (done) {
+ engMM = picoos_newMemoryManager(this->raw_mem, PICOCTRL_DEFAULT_ENGINE_SIZE,
+ /*enableMemProt*/ FALSE);
+ done = (NULL != engMM);
+ }
+ if (done) {
+ this->common = picoos_newCommon(engMM);
+ engEM = picoos_newExceptionManager(engMM);
+ done = (NULL != this->common) && (NULL != engEM);
+ }
+ if (done) {
+ this->common->mm = engMM;
+ this->common->em = engEM;
+
+ done = (PICO_OK == picorsrc_createVoice(rm,voiceName,&(this->voice)));
+ }
+ if (done) {
+ bSize = picodata_get_default_buf_size(PICODATA_PUTYPE_TEXT);
+
+ this->cbIn = picodata_newCharBuffer(this->common->mm,
+ this->common, bSize);
+ bSize = picodata_get_default_buf_size(PICODATA_PUTYPE_SIG);
+
+ this->cbOut = picodata_newCharBuffer(this->common->mm,
+ this->common, bSize);
+
+ PICODBG_DEBUG(("cbOut has address %i", (picoos_uint32) this->cbOut));
+
+
+ this->control = picoctrl_newControl(this->common->mm, this->common,
+ this->cbIn, this->cbOut, this->voice);
+ done = (NULL != this->cbIn) && (NULL != this->cbOut)
+ && (NULL != this->control);
+ }
+ if (done) {
+ SET_MAGIC_NUMBER(this);
+ } else {
+ if (NULL != this) {
+ if (NULL != this->voice) {
+ picorsrc_releaseVoice(rm,&(this->voice));
+ }
+ if(NULL != this->raw_mem) {
+ picoos_deallocate(mm,&(this->raw_mem));
+ }
+ picoos_deallocate(mm,(void *)&this);
+ }
+ }
+ return this;
+}/*picoctrl_newEngine*/
+
+/**
+ * disposes an engine object
+ * @param mm : memory manager associated to the engine
+ * @param rm : resource manager associated to the engine
+ * @param this : handle of the engine to dispose
+ * @return PICO_OK : reset performed
+ * @return void
+ * @callgraph
+ * @callergraph
+ */
+void picoctrl_disposeEngine(picoos_MemoryManager mm, picorsrc_ResourceManager rm,
+ picoctrl_Engine * this)
+{
+ if (NULL != (*this)) {
+ if (NULL != (*this)->voice) {
+ picorsrc_releaseVoice(rm,&((*this)->voice));
+ }
+ if(NULL != (*this)->control) {
+ picoctrl_disposeControl((*this)->common->mm,&((*this)->control));
+ }
+ if(NULL != (*this)->raw_mem) {
+ picoos_deallocate(mm,&((*this)->raw_mem));
+ }
+ (*this)->magic ^= 0xFFFEFDFC;
+ picoos_deallocate(mm,(void **)this);
+ }
+}/*picoctrl_disposeEngine*/
+
+/**
+ * resets the exception manager of an engine
+ * @param this : handle of the engine
+ * @return void
+ * @callgraph
+ * @callergraph
+ */
+void picoctrl_engResetExceptionManager(
+ picoctrl_Engine this
+ )
+{
+ picoos_emReset(this->common->em);
+}/*picoctrl_engResetExceptionManager*/
+
+/**
+ * returns the engine common pointer
+ * @param this : handle of the engine
+ * @return PICO_OK : reset performed
+ * @return the engine common pointer
+ * @return NULL if error
+ * @callgraph
+ * @callergraph
+ */
+picoos_Common picoctrl_engGetCommon(picoctrl_Engine this) {
+ if (NULL == this) {
+ return NULL;
+ } else {
+ return this->common;
+ }
+}/*picoctrl_engGetCommon*/
+
+/**
+ * feed raw 'text' into 'engine'. text may contain '\\0'.
+ * @param this : handle of the engine
+ * @param text : the input text
+ * @param textSize : size of the input text
+ * @param *bytesPut : the number of bytes effectively consumed from 'text'.
+ * @return PICO_OK : feeding succeded
+ * @return PICO_ERR_OTHER : if error
+ * @callgraph
+ * @callergraph
+ */
+pico_status_t picoctrl_engFeedText(picoctrl_Engine this,
+ picoos_char * text,
+ picoos_int16 textSize, picoos_int16 * bytesPut) {
+ if (NULL == this) {
+ return PICO_ERR_OTHER;
+ }
+ PICODBG_DEBUG(("get \"%.100s\"", text));
+ *bytesPut = 0;
+ while ((*bytesPut < textSize) && (PICO_OK == picodata_cbPutCh(this->cbIn, text[*bytesPut]))) {
+ (*bytesPut)++;
+ }
+
+ return PICO_OK;
+}/*picoctrl_engFeedText*/
+
+/**
+ * gets engine output bytes
+ * @param this : handle of the engine
+ * @param buffer : the destination buffer
+ * @param bufferSize : max size of the destinatioon buffer
+ * @param *bytesReceived : the number of bytes effectively returned
+ * @return PICO_OK : feeding succeded
+ * @return PICO_ERR_OTHER : if error
+ * @callgraph
+ * @callergraph
+ */
+picodata_step_result_t picoctrl_engFetchOutputItemBytes(
+ picoctrl_Engine this,
+ picoos_char *buffer,
+ picoos_int16 bufferSize,
+ picoos_int16 *bytesReceived) {
+ picoos_uint16 ui;
+ picodata_step_result_t stepResult;
+ pico_status_t rv;
+
+ if (NULL == this) {
+ return (picodata_step_result_t)PICO_STEP_ERROR;
+ }
+ PICODBG_DEBUG(("doing one step"));
+ stepResult = this->control->step(this->control,/* mode */0,&ui);
+ if (PICODATA_PU_ERROR != stepResult) {
+ PICODBG_TRACE(("filling output buffer"));
+ rv = picodata_cbGetSpeechData(this->cbOut, (picoos_uint8 *)buffer,
+ bufferSize, &ui);
+
+ if (ui > 255) { /* because picoapi uses signed int16 */
+ return (picodata_step_result_t)PICO_STEP_ERROR;
+ } else {
+ *bytesReceived = ui;
+ }
+ if ((rv == PICO_EXC_BUF_UNDERFLOW) || (rv == PICO_EXC_BUF_OVERFLOW)) {
+ PICODBG_ERROR(("problem getting speech data"));
+ return (picodata_step_result_t)PICO_STEP_ERROR;
+ }
+ /* rv must now be PICO_OK or PICO_EOF */
+ PICODBG_ASSERT(((PICO_EOF == rv) || (PICO_OK == rv)));
+ if ((PICODATA_PU_IDLE == stepResult) && (PICO_EOF == rv)) {
+ PICODBG_DEBUG(("IDLE"));
+ return (picodata_step_result_t)PICO_STEP_IDLE;
+ } else if (PICODATA_PU_ERROR == stepResult) {
+ PICODBG_DEBUG(("ERROR"));
+ return (picodata_step_result_t)PICO_STEP_ERROR;
+ } else {
+ PICODBG_DEBUG(("BUSY"));
+ return (picodata_step_result_t)PICO_STEP_BUSY;
+ }
+ } else {
+ return (picodata_step_result_t)PICO_STEP_ERROR;
+ }
+}/*picoctrl_engFetchOutputItemBytes*/
+
+/**
+ * returns the last scheduled PU
+ * @param this : handle of the engine
+ * @return a value >= 0 : last scheduled PU index
+ * @remarks designed to be used for performance evaluation
+ * @callgraph
+ * @callergraph
+ */
+picodata_step_result_t picoctrl_getLastScheduledPU(
+ picoctrl_Engine this
+ )
+{
+ ctrl_subobj_t * ctrl;
+ if (NULL == this || NULL == this->control->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ ctrl = (ctrl_subobj_t *) ((*this).control->subObj);
+ return (picodata_step_result_t) ctrl->curPU;
+}/*picoctrl_getLastScheduledPU*/
+
+/**
+ * returns the last item type produced by the last scheduled PU
+ * @param this : handle of the engine
+ * @return a value >= 0 : item type (see picodata.h for item types)
+ * @return a value = 0 : no item produced
+ * @remarks designed to be used for performance evaluation
+ * @callgraph
+ * @callergraph
+ */
+picodata_step_result_t picoctrl_getLastProducedItemType(
+ picoctrl_Engine this
+ )
+{
+ ctrl_subobj_t * ctrl;
+ if (NULL == this || NULL == this->control->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ ctrl = (ctrl_subobj_t *) ((*this).control->subObj);
+ return (picodata_step_result_t) ctrl->lastItemTypeProduced;
+}/*picoctrl_getLastProducedItemType*/
+
+
+#ifdef __cplusplus
+}
+#endif
+
+/* Picoctrl.c end */
diff --git a/lib/picoctrl.h b/lib/picoctrl.h
new file mode 100644
index 0000000..1dfea54
--- /dev/null
+++ b/lib/picoctrl.h
@@ -0,0 +1,101 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picoctrl.h
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#ifndef PICOCTRL_H_
+#define PICOCTRL_H_
+
+#include "picodefs.h"
+#include "picoos.h"
+#include "picorsrc.h"
+#include "picodata.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+#define PICOCTRL_MAX_PROC_UNITS 25
+
+/* temporarily increased for preprocessing
+#define PICOCTRL_DEFAULT_ENGINE_SIZE 200000
+*/
+#define PICOCTRL_DEFAULT_ENGINE_SIZE 1000000
+
+typedef struct picoctrl_engine * picoctrl_Engine;
+
+picoos_int16 picoctrl_isValidEngineHandle(picoctrl_Engine this);
+
+picoctrl_Engine picoctrl_newEngine (
+ picoos_MemoryManager mm,
+ picorsrc_ResourceManager rm,
+ const picoos_char * voiceName
+ );
+
+void picoctrl_disposeEngine(
+ picoos_MemoryManager mm,
+ picorsrc_ResourceManager rm,
+ picoctrl_Engine * this
+ );
+
+pico_status_t picoctrl_engFeedText(
+ picoctrl_Engine engine,
+ picoos_char * text,
+ picoos_int16 textSize,
+ picoos_int16 * bytesPut);
+
+pico_status_t picoctrl_engReset(
+ picoctrl_Engine engine
+ );
+
+picoos_Common picoctrl_engGetCommon(picoctrl_Engine this);
+
+picodata_step_result_t picoctrl_engFetchOutputItemBytes(
+ picoctrl_Engine engine,
+ picoos_char * buffer,
+ picoos_int16 bufferSize,
+ picoos_int16 * bytesReceived
+);
+
+void picoctrl_engResetExceptionManager(
+ picoctrl_Engine this
+ );
+
+
+picodata_step_result_t picoctrl_getLastScheduledPU(
+ picoctrl_Engine engine
+ );
+
+picodata_step_result_t picoctrl_getLastProducedItemType(
+ picoctrl_Engine engine
+ );
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /*PICOCTRL_H_*/
diff --git a/lib/picodata.c b/lib/picodata.c
new file mode 100644
index 0000000..244e4b6
--- /dev/null
+++ b/lib/picodata.c
@@ -0,0 +1,1152 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picodata.c
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#include "picodefs.h"
+#include "picoos.h"
+#include "picodbg.h"
+#include "picorsrc.h"
+#include "picotrns.h"
+#include "picodata.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+/* we output coefficients as fixed point values */
+#define PICOCEP_OUT_DATA_FORMAT PICODATA_ITEMINFO1_FRAME_PAR_DATA_FORMAT_FIXED
+
+/* ***************************************************************
+ * CharBuffer *
+ *****************************************************************/
+
+/*
+ * method signatures
+ */
+typedef pico_status_t (* picodata_cbPutItemMethod) (register picodata_CharBuffer this,
+ const picoos_uint8 *buf, const picoos_uint16 blenmax,
+ picoos_uint16 *blen);
+
+typedef pico_status_t (* picodata_cbGetItemMethod) (register picodata_CharBuffer this,
+ picoos_uint8 *buf, const picoos_uint16 blenmax,
+ picoos_uint16 *blen, const picoos_uint8 issd);
+
+typedef pico_status_t (* picodata_cbSubResetMethod) (register picodata_CharBuffer this);
+typedef pico_status_t (* picodata_cbSubDeallocateMethod) (register picodata_CharBuffer this, picoos_MemoryManager mm);
+
+typedef struct picodata_char_buffer
+{
+ picoos_char *buf;
+ picoos_uint16 rear; /* next free position to write */
+ picoos_uint16 front; /* next position to read */
+ picoos_uint16 len; /* empty: len = 0, full: len = size */
+ picoos_uint16 size;
+
+ picoos_Common common;
+
+ picodata_cbGetItemMethod getItem;
+ picodata_cbPutItemMethod putItem;
+
+ picodata_cbSubResetMethod subReset;
+ picodata_cbSubDeallocateMethod subDeallocate;
+ void * subObj;
+} char_buffer_t;
+
+
+static pico_status_t data_cbPutItem(register picodata_CharBuffer this,
+ const picoos_uint8 *buf, const picoos_uint16 blenmax,
+ picoos_uint16 *blen);
+
+static pico_status_t data_cbGetItem(register picodata_CharBuffer this,
+ picoos_uint8 *buf, const picoos_uint16 blenmax,
+ picoos_uint16 *blen, const picoos_uint8 issd);
+
+pico_status_t picodata_cbReset(register picodata_CharBuffer this)
+{
+ this->rear = 0;
+ this->front = 0;
+ this->len = 0;
+ if (NULL != this->subObj) {
+ return this->subReset(this);
+ } else {
+ return PICO_OK;
+ }
+}
+
+/* CharBuffer constructor */
+picodata_CharBuffer picodata_newCharBuffer(picoos_MemoryManager mm,
+ picoos_Common common,
+ picoos_objsize_t size)
+{
+ picodata_CharBuffer this;
+
+ this = (picodata_CharBuffer) picoos_allocate(mm, sizeof(*this));
+ PICODBG_DEBUG(("new character buffer, size=%i", size));
+ if (NULL == this) {
+ return NULL;
+ }
+ this->buf = picoos_allocate(mm, size);
+ if (NULL == this->buf) {
+ picoos_deallocate(mm, (void*) &this);
+ return NULL;
+ }
+ this->size = size;
+ this->common = common;
+
+ this->getItem = data_cbGetItem;
+ this->putItem = data_cbPutItem;
+
+ this->subReset = NULL;
+ this->subDeallocate = NULL;
+ this->subObj = NULL;
+
+ picodata_cbReset(this);
+ return this;
+}
+
+void picodata_disposeCharBuffer(picoos_MemoryManager mm,
+ picodata_CharBuffer *this)
+{
+ if (NULL != (*this)) {
+ /* terminate */
+ if (NULL != (*this)->subObj) {
+ (*this)->subDeallocate(*this,mm);
+ }
+ picoos_deallocate(mm,(void*)&(*this)->buf);
+ picoos_deallocate(mm,(void*)this);
+ }
+}
+
+pico_status_t picodata_cbPutCh(register picodata_CharBuffer this,
+ picoos_char ch)
+{
+ if (this->len < this->size) {
+ this->buf[this->rear++] = ch;
+ this->rear %= this->size;
+ this->len++;
+ return PICO_OK;
+ } else {
+ return PICO_EXC_BUF_OVERFLOW;
+ }
+}
+
+
+picoos_int16 picodata_cbGetCh(register picodata_CharBuffer this)
+{
+ picoos_char ch;
+ if (this->len > 0) {
+ ch = this->buf[this->front++];
+ this->front %= this->size;
+ this->len--;
+ return ch;
+ } else {
+ return PICO_EOF;
+ }
+}
+
+/* ***************************************************************
+ * items: CharBuffer functions *
+ *****************************************************************/
+
+static pico_status_t data_cbGetItem(register picodata_CharBuffer this,
+ picoos_uint8 *buf, const picoos_uint16 blenmax,
+ picoos_uint16 *blen, const picoos_uint8 issd)
+{
+ picoos_uint16 i;
+
+ if (this->len < PICODATA_ITEM_HEADSIZE) { /* item not in cb? */
+ *blen = 0;
+ if (this->len == 0) { /* is cb empty? */
+ PICODBG_DEBUG(("no item to get"));
+ return PICO_EOF;
+ } else { /* cb not empty, but not a valid item */
+ PICODBG_WARN(("problem getting item, incomplete head, underflow"));
+ }
+ return PICO_EXC_BUF_UNDERFLOW;
+ }
+ *blen = PICODATA_ITEM_HEADSIZE + (picoos_uint8)(this->buf[((this->front) +
+ PICODATA_ITEMIND_LEN) % this->size]);
+
+ /* if getting speech data in item */
+ if (issd) {
+ /* check item type */
+ if (this->buf[this->front] != PICODATA_ITEM_FRAME) {
+ PICODBG_WARN(("item type mismatch for speech data: %c",
+ this->buf[this->front]));
+ for (i=0; i<*blen; i++) {
+ this->front++;
+ this->front %= this->size;
+ this->len--;
+ }
+ *blen = 0;
+ return PICO_OK;
+ }
+ }
+
+ if (*blen > this->len) { /* item in cb not complete? */
+ PICODBG_WARN(("problem getting item, incomplete content, underflow; "
+ "blen=%d, len=%d", *blen, this->len));
+ *blen = 0;
+ return PICO_EXC_BUF_UNDERFLOW;
+ }
+ if (blenmax < *blen) { /* buf not large enough? */
+ PICODBG_WARN(("problem getting item, overflow"));
+ *blen = 0;
+ return PICO_EXC_BUF_OVERFLOW;
+ }
+
+ /* if getting speech data in item */
+ if (issd) {
+ /* skip item header */
+ for (i = 0; i < PICODATA_ITEM_HEADSIZE; i++) {
+ this->front++;
+ this->front %= this->size;
+ this->len--;
+ }
+ *blen -= PICODATA_ITEM_HEADSIZE;
+ }
+
+ /* all ok, now get item (or speech data only) */
+ for (i = 0; i < *blen; i++) {
+ buf[i] = (picoos_uint8)(this->buf[this->front++]);
+ this->front %= this->size;
+ this->len--;
+ }
+
+#if defined(PICO_DEBUG)
+ if (issd) {
+ PICODBG_DEBUG(("got speech data: %d samples", *blen));
+ } else {
+ PICODBG_DEBUG(("got item: %c|%d|%d|%d||", buf[PICODATA_ITEMIND_TYPE],
+ buf[PICODATA_ITEMIND_INFO1], buf[PICODATA_ITEMIND_INFO2],
+ buf[PICODATA_ITEMIND_LEN]));
+ for (i=PICODATA_ITEM_HEADSIZE; i<*blen; i++) {
+ if (buf[PICODATA_ITEMIND_TYPE] == PICODATA_ITEM_WORDGRAPH) {
+ PICODBG_DEBUG(("%c", buf[i]));
+ } else {
+ PICODBG_DEBUG((" %d", buf[i]));
+ }
+ }
+ }
+#endif
+
+ return PICO_OK;
+}
+
+static pico_status_t data_cbPutItem(register picodata_CharBuffer this,
+ const picoos_uint8 *buf, const picoos_uint16 blenmax,
+ picoos_uint16 *blen)
+{
+ picoos_uint16 i;
+
+ if (blenmax < PICODATA_ITEM_HEADSIZE) { /* itemlen not accessible? */
+ PICODBG_WARN(("problem putting item, underflow"));
+ *blen = 0;
+ return PICO_EXC_BUF_UNDERFLOW;
+ }
+ *blen = buf[PICODATA_ITEMIND_LEN] + PICODATA_ITEM_HEADSIZE;
+ if (*blen > (this->size - this->len)) { /* cb not enough space? */
+ PICODBG_WARN(("problem putting item, overflow"));
+ *blen = 0;
+ return PICO_EXC_BUF_OVERFLOW;
+ }
+ if (*blen > blenmax) { /* item in buf not completely accessible? */
+ PICODBG_WARN(("problem putting item, underflow"));
+ *blen = 0;
+ return PICO_EXC_BUF_UNDERFLOW;
+ }
+ /* all ok, now put complete item */
+
+#if defined(PICO_DEBUG)
+ PICODBG_DEBUG(("putting item: %c|%d|%d|%d||",
+ buf[PICODATA_ITEMIND_TYPE],
+ buf[PICODATA_ITEMIND_INFO1],
+ buf[PICODATA_ITEMIND_INFO2],
+ buf[PICODATA_ITEMIND_LEN]));
+ for (i=PICODATA_ITEM_HEADSIZE; i<*blen; i++) {
+ if (buf[PICODATA_ITEMIND_TYPE] == PICODATA_ITEM_WORDGRAPH) {
+ PICODBG_DEBUG(("%c", buf[i]));
+ } else {
+ PICODBG_DEBUG((" %d", buf[i]));
+ }
+ }
+#endif
+
+ for (i = 0; i < *blen; i++) {
+ /* put single byte */
+ this->buf[this->rear++] = (picoos_char)buf[i];
+ this->rear %= this->size;
+ this->len++;
+ }
+ return PICO_OK;
+}
+
+/*----------------------------------------------------------
+ * Names : picodata_cbGetItem
+ * picodata_cbGetSpeechData
+ * Function: gets one item from 'this' and writes it on 'blenmax' sized 'buf'.
+ * gets one item from 'this' and writes the speech data to
+ * 'blenmax' sized 'buf'.
+ * Returns : PICO_OK : one item was copied
+ * PICO_EOF : 'this' is empty; no new items available
+ * PICO_BUF_UNDERFLOW : 'this' doesn't contain a full/valid item
+ * PICO_BUF_OVERFLOW : 'buf[blenmax]' too small to hold item
+ * on return, '*blen' contains the number of bytes written to 'buf'
+ * ---------------------------------------------------------*/
+pico_status_t picodata_cbGetItem(register picodata_CharBuffer this,
+ picoos_uint8 *buf, const picoos_uint16 blenmax,
+ picoos_uint16 *blen)
+{
+ return this->getItem(this, buf, blenmax, blen, FALSE);
+}
+
+pico_status_t picodata_cbGetSpeechData(register picodata_CharBuffer this,
+ picoos_uint8 *buf, const picoos_uint16 blenmax,
+ picoos_uint16 *blen)
+{
+
+ return this->getItem(this, buf, blenmax, blen, TRUE);
+}
+
+
+pico_status_t picodata_cbPutItem(register picodata_CharBuffer this,
+ const picoos_uint8 *buf, const picoos_uint16 blenmax,
+ picoos_uint16 *blen)
+{
+
+ return this->putItem(this,buf,blenmax,blen);
+}
+
+/* unsafe, just for measuring purposes */
+picoos_uint8 picodata_cbGetFrontItemType(register picodata_CharBuffer this)
+{
+ return this->buf[this->front];
+}
+/* ***************************************************************
+ * items: support function *
+ *****************************************************************/
+
+picoos_uint8 is_valid_itemtype(const picoos_uint8 ch) {
+ switch (ch) {
+ case PICODATA_ITEM_WSEQ_GRAPH:
+ case PICODATA_ITEM_TOKEN:
+ case PICODATA_ITEM_WORDGRAPH:
+ case PICODATA_ITEM_WORDINDEX:
+ case PICODATA_ITEM_WORDPHON:
+ case PICODATA_ITEM_SYLLPHON:
+ case PICODATA_ITEM_BOUND:
+ case PICODATA_ITEM_PUNC:
+ case PICODATA_ITEM_CMD:
+ case PICODATA_ITEM_PHONE:
+ case PICODATA_ITEM_FRAME:
+ case PICODATA_ITEM_FRAME_PAR:
+ return TRUE;
+ break;
+ case PICODATA_ITEM_OTHER:
+ default:
+ break;
+ }
+ PICODBG_WARN(("item type problem: %c", ch));
+ return FALSE;
+}
+
+picoos_uint8 picodata_is_valid_itemhead(const picodata_itemhead_t *head) {
+ if ((NULL != head) && is_valid_itemtype(head->type)) {
+ return TRUE;
+ } else {
+ PICODBG_WARN(("item header problem"));
+ return FALSE;
+ }
+}
+
+/* ***************************************************/
+
+
+pico_status_t picodata_get_itemparts_nowarn(
+ const picoos_uint8 *buf, const picoos_uint16 blenmax,
+ picodata_itemhead_t *head, picoos_uint8 *content,
+ const picoos_uint16 clenmax, picoos_uint16 *clen)
+{
+ if (blenmax >= PICODATA_ITEM_HEADSIZE) {
+ head->type = buf[PICODATA_ITEMIND_TYPE];
+ head->info1 = buf[PICODATA_ITEMIND_INFO1];
+ head->info2 = buf[PICODATA_ITEMIND_INFO2];
+ head->len = buf[PICODATA_ITEMIND_LEN];
+ *clen = head->len;
+ if (blenmax >= (*clen + PICODATA_ITEM_HEADSIZE)) {
+ if (clenmax >= head->len) {
+ picoos_uint16 i;
+ for (i=0; i<head->len; i++) {
+ content[i] = buf[PICODATA_ITEM_HEADSIZE+i];
+ }
+ return PICO_OK;
+ }
+ *clen = 0;
+ return PICO_EXC_BUF_OVERFLOW;
+ }
+ }
+ *clen = 0;
+ return PICO_EXC_BUF_UNDERFLOW;
+}
+
+pico_status_t picodata_get_itemparts(
+ const picoos_uint8 *buf, const picoos_uint16 blenmax,
+ picodata_itemhead_t *head, picoos_uint8 *content,
+ const picoos_uint16 clenmax, picoos_uint16 *clen)
+{
+ pico_status_t status = picodata_get_itemparts_nowarn(buf,blenmax,head,content,clenmax,clen);
+ if (PICO_EXC_BUF_OVERFLOW == status) {
+ PICODBG_WARN(("problem getting item, overflow"));
+ } else if (PICO_EXC_BUF_UNDERFLOW == status) {
+ PICODBG_WARN(("problem getting item, overflow"));
+ }
+ return status;
+}
+pico_status_t picodata_put_itemparts(const picodata_itemhead_t *head,
+ const picoos_uint8 *content, const picoos_uint16 clenmax,
+ picoos_uint8 *buf, const picoos_uint16 blenmax, picoos_uint16 *blen)
+{
+ *blen = head->len + PICODATA_ITEM_HEADSIZE;
+ if (blenmax >= *blen) {
+ if (clenmax >= head->len) {
+ picoos_uint16 i;
+ buf[PICODATA_ITEMIND_TYPE] = head->type;
+ buf[PICODATA_ITEMIND_INFO1] = head->info1;
+ buf[PICODATA_ITEMIND_INFO2] = head->info2;
+ buf[PICODATA_ITEMIND_LEN] = head->len;
+ for (i=0; i<head->len; i++) {
+ buf[PICODATA_ITEM_HEADSIZE+i] = content[i];
+ }
+ return PICO_OK;
+ }
+ PICODBG_WARN(("problem putting item, underflow"));
+ *blen = 0;
+ return PICO_EXC_BUF_UNDERFLOW;
+ }
+ PICODBG_WARN(("problem putting item, overflow"));
+ *blen = 0;
+ return PICO_EXC_BUF_OVERFLOW;
+}
+
+pico_status_t picodata_get_iteminfo(
+ picoos_uint8 *buf, const picoos_uint16 blenmax,
+ picodata_itemhead_t *head, picoos_uint8 **content) {
+ if (blenmax >= PICODATA_ITEM_HEADSIZE) {
+ head->type = buf[PICODATA_ITEMIND_TYPE];
+ head->info1 = buf[PICODATA_ITEMIND_INFO1];
+ head->info2 = buf[PICODATA_ITEMIND_INFO2];
+ head->len = buf[PICODATA_ITEMIND_LEN];
+ if (head->len == 0) {
+ *content = NULL;
+ } else {
+ *content = &buf[PICODATA_ITEM_HEADSIZE];
+ }
+ return PICO_OK;
+ }
+ return PICO_EXC_BUF_UNDERFLOW;
+}
+
+pico_status_t picodata_copy_item(const picoos_uint8 *inbuf,
+ const picoos_uint16 inlenmax, picoos_uint8 *outbuf,
+ const picoos_uint16 outlenmax, picoos_uint16 *numb)
+{
+ if (picodata_is_valid_item(inbuf, inlenmax)) {
+ *numb = PICODATA_ITEM_HEADSIZE + inbuf[PICODATA_ITEMIND_LEN];
+ } else {
+ *numb = 0;
+ }
+ if (*numb > 0) {
+ if (outlenmax >= inlenmax) {
+ picoos_uint16 i;
+ for (i=0; i<*numb; i++) {
+ outbuf[i] = inbuf[i];
+ }
+ return PICO_OK;
+ } else {
+ PICODBG_WARN(("buffer problem, out: %d > in: %d", outlenmax, inlenmax));
+ *numb = 0;
+ return PICO_EXC_BUF_OVERFLOW;
+ }
+ } else {
+ PICODBG_WARN(("item problem in inbuf"));
+ return PICO_ERR_OTHER;
+ }
+}
+
+pico_status_t picodata_set_iteminfo1(picoos_uint8 *buf,
+ const picoos_uint16 blenmax, const picoos_uint8 info) {
+ if (PICODATA_ITEMIND_INFO1 < blenmax) {
+ buf[PICODATA_ITEMIND_INFO1] = info;
+ return PICO_OK;
+ } else
+ return PICO_EXC_BUF_UNDERFLOW;
+}
+
+pico_status_t picodata_set_iteminfo2(picoos_uint8 *buf,
+ const picoos_uint16 blenmax, const picoos_uint8 info) {
+ if (PICODATA_ITEMIND_INFO2 < blenmax) {
+ buf[PICODATA_ITEMIND_INFO2] = info;
+ return PICO_OK;
+ } else
+ return PICO_EXC_BUF_UNDERFLOW;
+}
+
+/* sets the len field in the header contained in the item in buf;
+ return values:
+ PICO_OK <- all ok
+ PICO_EXC_BUF_UNDERFLOW <- underflow in buf
+*/
+pico_status_t picodata_set_itemlen(picoos_uint8 *buf,
+ const picoos_uint16 blenmax, const picoos_uint8 len) {
+ if (PICODATA_ITEMIND_LEN < blenmax) {
+ buf[PICODATA_ITEMIND_LEN] = len;
+ return PICO_OK;
+ } else
+ return PICO_EXC_BUF_UNDERFLOW;
+}
+
+picoos_uint8 picodata_is_valid_item(const picoos_uint8 *item,
+ const picoos_uint16 ilenmax)
+{
+ if (ilenmax >= PICODATA_ITEM_HEADSIZE) {
+ picodata_itemhead_t head;
+ head.type = item[0];
+ head.info1 = item[1];
+ head.info2 = item[2];
+ head.len = item[3];
+ if ((ilenmax >= (head.len + PICODATA_ITEM_HEADSIZE)) &&
+ picodata_is_valid_itemhead(&head))
+ {
+ return TRUE;
+ }
+ }
+ return FALSE;
+}
+
+/* ***************************************************************
+ * ProcessingUnit *
+ *****************************************************************/
+picoos_uint16 picodata_get_default_buf_size (picodata_putype_t puType)
+{
+ return (PICODATA_PUTYPE_TEXT == puType) ? PICODATA_BUFSIZE_TEXT
+ : (PICODATA_PUTYPE_TOK == puType) ? PICODATA_BUFSIZE_TOK
+ : (PICODATA_PUTYPE_PR == puType) ? PICODATA_BUFSIZE_PR
+ : (PICODATA_PUTYPE_PR == puType) ? PICODATA_BUFSIZE_PR
+ : (PICODATA_PUTYPE_WA == puType) ? PICODATA_BUFSIZE_WA
+ : (PICODATA_PUTYPE_SA == puType) ? PICODATA_BUFSIZE_SA
+ : (PICODATA_PUTYPE_ACPH == puType) ? PICODATA_BUFSIZE_ACPH
+ : (PICODATA_PUTYPE_SPHO == puType) ? PICODATA_BUFSIZE_SPHO
+ : (PICODATA_PUTYPE_PAM == puType) ? PICODATA_BUFSIZE_PAM
+ : (PICODATA_PUTYPE_CEP == puType) ? PICODATA_BUFSIZE_CEP
+ : (PICODATA_PUTYPE_SIG == puType) ? PICODATA_BUFSIZE_SIG
+ : (PICODATA_PUTYPE_SINK == puType) ? PICODATA_BUFSIZE_SINK
+ : PICODATA_BUFSIZE_DEFAULT;
+}
+
+
+typedef struct simple_pu_data
+{
+ picorsrc_Voice voice;
+} simple_pu_data_t;
+
+static pico_status_t puSimpleInitialize (register picodata_ProcessingUnit this) {
+ return PICO_OK;
+}
+
+static pico_status_t puSimpleTerminate (register picodata_ProcessingUnit this) {
+ return PICO_OK;
+}
+
+static picodata_step_result_t puSimpleStep (register picodata_ProcessingUnit this,
+ picoos_int16 mode,
+ picoos_uint16 * numBytesOutput) {
+ picoos_int16 ch;
+ picoos_int16 result = PICO_OK;
+ mode = mode; /*PP 13.10.08 : fix warning "var not used in this function"*/
+ *numBytesOutput = 0;
+ while ((result == PICO_OK) && (ch = picodata_cbGetCh(this->cbIn)) != PICO_EOF) {
+ result = picodata_cbPutCh(this->cbOut,(picoos_char) ch);
+ (*numBytesOutput)++;
+ }
+ if (PICO_OK != result) {
+ (*numBytesOutput)--;
+ }
+ if (PICO_OK == result) {
+ return PICODATA_PU_IDLE;
+ } else {
+ return PICODATA_PU_ERROR;
+ }
+}
+
+
+picodata_ProcessingUnit picodata_newProcessingUnit(
+ picoos_MemoryManager mm,
+ picoos_Common common,
+ picodata_CharBuffer cbIn,
+ picodata_CharBuffer cbOut,
+ picorsrc_Voice voice)
+{
+ picodata_ProcessingUnit this = (picodata_ProcessingUnit) picoos_allocate(mm, sizeof(*this));
+ if (this == NULL) {
+ picoos_emRaiseException(common->em,PICO_EXC_OUT_OF_MEM,NULL,NULL);
+ return NULL;
+ }
+ this->common = common;
+ this->cbIn = cbIn;
+ this->cbOut = cbOut;
+ this->voice = voice;
+ this->initialize = puSimpleInitialize;
+ this->terminate = puSimpleTerminate;
+ this->step = puSimpleStep;
+ this->subDeallocate = NULL;
+ this->subObj = NULL;
+ return this;
+}
+
+void picodata_disposeProcessingUnit(
+ picoos_MemoryManager mm,
+ picodata_ProcessingUnit * this)
+{
+ if (NULL != (*this)) {
+ /* terminate */
+ if (NULL != (*this)->subObj) {
+ (*this)->subDeallocate(*this,mm);
+ }
+ picoos_deallocate(mm,(void *)this);
+ }
+
+}
+
+
+picodata_CharBuffer picodata_getCbIn(picodata_ProcessingUnit this)
+{
+ if (NULL == this) {
+ picoos_emRaiseException(this->common->em,PICO_ERR_NULLPTR_ACCESS,NULL,NULL);
+ return NULL;
+ } else {
+ return this->cbIn;
+ }
+}
+
+picodata_CharBuffer picodata_getCbOut(picodata_ProcessingUnit this)
+{
+ if (NULL == this) {
+ picoos_emRaiseException(this->common->em,PICO_ERR_NULLPTR_ACCESS,NULL,NULL);
+ return NULL;
+ } else {
+ return this->cbOut;
+ }
+}
+
+pico_status_t picodata_setCbIn(picodata_ProcessingUnit this, picodata_CharBuffer cbIn)
+{
+ if (NULL == this) {
+ picoos_emRaiseException(this->common->em,PICO_ERR_NULLPTR_ACCESS,NULL,NULL);
+ return PICO_ERR_OTHER;
+ } else {
+ this->cbIn = cbIn;
+ return PICO_OK;
+ }
+
+}
+
+pico_status_t picodata_setCbOut(picodata_ProcessingUnit this, picodata_CharBuffer cbOut)
+{
+ if (NULL == this) {
+ picoos_emRaiseException(this->common->em,PICO_ERR_NULLPTR_ACCESS,NULL,NULL);
+ return PICO_ERR_OTHER;
+ } else {
+ this->cbOut = cbOut;
+ return PICO_OK;
+ }
+}
+
+
+/* ***************************************************************
+ * auxiliary functions *
+ *****************************************************************/
+
+static void transDurUniform(
+ picoos_uint8 frame_duration_exp, /* 2's exponent of frame duration in ms, e.g. 2 for 4ms, 3 for 8ms */
+ picoos_int8 array_length,
+ picoos_uint8 * inout,
+ picoos_int16 inputdur, /* input duration in ms */
+ picoos_int16 targetdur, /* target duration in ms */
+ picoos_int16 * restdur /* in/out, rest in ms */
+ )
+{
+ picoos_int8 i;
+ picoos_int32 fact, rest;
+
+ /* calculate rest and factor in number of frames (in PICODATA_PICODATA_PRECISION) */
+ rest = (*restdur) << (PICODATA_PRECISION - frame_duration_exp);
+ fact = (targetdur << (PICODATA_PRECISION - frame_duration_exp)) / inputdur;
+
+ for (i = 0; i < array_length; i++) {
+ rest += fact * inout[i];
+ /* instead of rounding, we carry the rest to the next state */
+ inout[i] = rest >> PICODATA_PRECISION;
+ rest -= inout[i] << PICODATA_PRECISION;
+ }
+ (*restdur) = rest >> (PICODATA_PRECISION - frame_duration_exp);
+}
+
+static void transDurWeighted(
+ picoos_uint8 frame_duration_exp, /* 2's exponent of frame duration in ms, e.g. 2 for 4ms, 3 for 8ms */
+ picoos_int8 array_length,
+ picoos_uint8 * inout,
+ const picoos_uint16 * weight, /* integer weights */
+ picoos_int16 inputdur, /* input duration in ms */
+ picoos_int16 targetdur, /* target duration in ms */
+ picoos_int16 * restdur /* in/out, rest in ms */
+ )
+{
+ picoos_int8 i;
+ picoos_int32 fact, rest, out, weighted_sum;
+
+ /* calculate rest and factor in number of frames (in PICODATA_PICODATA_PRECISION) */
+ rest = (*restdur) << (PICODATA_PRECISION - frame_duration_exp);
+ weighted_sum = 0;
+ for (i=0; i < array_length; i++) {
+ weighted_sum += inout[i] * weight[i];
+ }
+ if (0 == weighted_sum) {
+ transDurUniform(frame_duration_exp,array_length,inout,inputdur,targetdur,restdur);
+ return;
+ }
+ /* get the additive change factor in PICODATA_PRECISION: */
+ if (targetdur > inputdur) {
+ fact = ((targetdur - inputdur) << (PICODATA_PRECISION-frame_duration_exp))/ weighted_sum;
+ } else {
+ fact = -((inputdur - targetdur) << (PICODATA_PRECISION-frame_duration_exp))/ weighted_sum;
+ }
+
+ /* input[i] * fact * weight[i] is the additive modification in PICODATA_PRECISION */
+ for (i=0; i < array_length; i++) {
+ rest += fact * inout[i] * weight[i];
+ /* instead of rounding, we carry the rest to the next state */
+ out = inout[i] + (rest >> PICODATA_PRECISION);
+ if (out < 0) {
+ out = 0;
+ }
+ rest -= ((out-inout[i]) << PICODATA_PRECISION);
+ inout[i] = out;
+ }
+ (*restdur) = rest >> (PICODATA_PRECISION - frame_duration_exp);
+}
+
+
+
+void picodata_transformDurations(
+ picoos_uint8 frame_duration_exp,
+ picoos_int8 array_length,
+ picoos_uint8 * inout,
+ const picoos_uint16 * weight, /* integer weights */
+ picoos_int16 mintarget, /* minimum target duration in ms */
+ picoos_int16 maxtarget, /* maximum target duration in ms */
+ picoos_int16 facttarget, /* factor to be multiplied with original length to get the target
+ the factor is fixed-point with PICODATA_PRECISION PICODATA_PRECISION, i.e.
+ the factor as float would be facttarget / PICODATA_PRECISION_FACT
+ if factor is 0, only min/max are considered */
+ picoos_int16 * dur_rest /* in/out, rest in ms */
+ )
+{
+ picoos_int32 inputdur, targetdur;
+ picoos_int8 i;
+
+ /* find the original duration in ms */
+ inputdur = 0;
+ for (i=0; i < array_length; i++) {
+ inputdur += inout[i];
+ }
+
+ PICODBG_TRACE(("######## transforming duration fact=%i, limits = [%i,%i] (input frames: %i)",facttarget,mintarget,maxtarget, inputdur));
+
+ inputdur <<= frame_duration_exp;
+
+ /* find the target duration */
+ if (facttarget) {
+ targetdur = (facttarget * inputdur + PICODATA_PREC_HALF) >> PICODATA_PRECISION;
+ } else {
+ targetdur = inputdur;
+ }
+
+ /* we need to modify input if there is an explicit factor or input is not in the target range */
+ if (facttarget || (targetdur < mintarget) || (maxtarget < targetdur)) {
+ /* make sure we are in the limits */
+ if (targetdur < mintarget) {
+ targetdur = mintarget;
+ } else if (maxtarget < targetdur) {
+ targetdur = maxtarget;
+ }
+ /* perform modification */
+ if (NULL == weight) {
+ transDurUniform(frame_duration_exp,array_length,inout,inputdur,targetdur,dur_rest);
+ } else {
+ transDurWeighted(frame_duration_exp,array_length,inout,weight,inputdur,targetdur,dur_rest);
+ }
+ }
+}
+
+
+
+extern picoos_uint8 picodata_getPuTypeFromExtension(picoos_uchar * filename, picoos_bool input)
+{
+ if (input) {
+ if (picoos_has_extension(filename, PICODATA_PUTYPE_TOK_INPUT_EXTENSION)) {
+ return PICODATA_ITEMINFO2_CMD_TO_TOK;
+ }
+ else if (picoos_has_extension(filename, PICODATA_PUTYPE_PR_INPUT_EXTENSION)) {
+ return PICODATA_ITEMINFO2_CMD_TO_PR;
+ }
+ else if (picoos_has_extension(filename, PICODATA_PUTYPE_WA_INPUT_EXTENSION)) {
+ return PICODATA_ITEMINFO2_CMD_TO_WA;
+ }
+ else if (picoos_has_extension(filename, PICODATA_PUTYPE_SA_INPUT_EXTENSION)) {
+ return PICODATA_ITEMINFO2_CMD_TO_SA;
+ }
+ else if (picoos_has_extension(filename, PICODATA_PUTYPE_ACPH_INPUT_EXTENSION)) {
+ return PICODATA_ITEMINFO2_CMD_TO_ACPH;
+ }
+ else if (picoos_has_extension(filename, PICODATA_PUTYPE_SPHO_INPUT_EXTENSION)) {
+ return PICODATA_ITEMINFO2_CMD_TO_SPHO;
+ }
+ else if (picoos_has_extension(filename, PICODATA_PUTYPE_PAM_INPUT_EXTENSION)) {
+ return PICODATA_ITEMINFO2_CMD_TO_PAM;
+ }
+ else if (picoos_has_extension(filename, PICODATA_PUTYPE_CEP_INPUT_EXTENSION)) {
+ return PICODATA_ITEMINFO2_CMD_TO_CEP;
+ }
+ else if (picoos_has_extension(filename, PICODATA_PUTYPE_SIG_INPUT_EXTENSION) ||
+ picoos_has_extension(filename, PICODATA_PUTYPE_WAV_INPUT_EXTENSION)) {
+ return PICODATA_ITEMINFO2_CMD_TO_SIG;
+ }
+ else {
+ return PICODATA_ITEMINFO2_CMD_TO_UNKNOWN;
+ }
+ }
+ else {
+ if (picoos_has_extension(filename, PICODATA_PUTYPE_TOK_OUTPUT_EXTENSION)) {
+ return PICODATA_ITEMINFO2_CMD_TO_TOK;
+ }
+ else if (picoos_has_extension(filename, PICODATA_PUTYPE_PR_OUTPUT_EXTENSION)) {
+ return PICODATA_ITEMINFO2_CMD_TO_PR;
+ }
+ else if (picoos_has_extension(filename, PICODATA_PUTYPE_WA_OUTPUT_EXTENSION)) {
+ return PICODATA_ITEMINFO2_CMD_TO_WA;
+ }
+ else if (picoos_has_extension(filename, PICODATA_PUTYPE_SA_OUTPUT_EXTENSION)) {
+ return PICODATA_ITEMINFO2_CMD_TO_SA;
+ }
+ else if (picoos_has_extension(filename, PICODATA_PUTYPE_ACPH_OUTPUT_EXTENSION)) {
+ return PICODATA_ITEMINFO2_CMD_TO_ACPH;
+ }
+ else if (picoos_has_extension(filename, PICODATA_PUTYPE_SPHO_OUTPUT_EXTENSION)) {
+ return PICODATA_ITEMINFO2_CMD_TO_SPHO;
+ }
+ else if (picoos_has_extension(filename, PICODATA_PUTYPE_PAM_OUTPUT_EXTENSION)) {
+ return PICODATA_ITEMINFO2_CMD_TO_PAM;
+ }
+ else if (picoos_has_extension(filename, PICODATA_PUTYPE_CEP_OUTPUT_EXTENSION)) {
+ return PICODATA_ITEMINFO2_CMD_TO_CEP;
+ }
+ else if (picoos_has_extension(filename, PICODATA_PUTYPE_SIG_OUTPUT_EXTENSION) ||
+ picoos_has_extension(filename, PICODATA_PUTYPE_WAV_OUTPUT_EXTENSION)) {
+ return PICODATA_ITEMINFO2_CMD_TO_SIG;
+ }
+ else {
+ return PICODATA_ITEMINFO2_CMD_TO_UNKNOWN;
+ }
+ }
+ return PICODATA_ITEMINFO2_CMD_TO_UNKNOWN;
+}
+
+
+
+
+/**
+ *
+ * @param transducer
+ * @param common
+ * @param xsampa_parser
+ * @param svoxpa_parser
+ * @param xsampa2svoxpa_mapper
+ * @param inputPhones
+ * @param alphabet
+ * @param outputPhoneIds
+ * @param maxOutputPhoneIds
+ * @return
+ */
+pico_status_t picodata_mapPAStrToPAIds(picotrns_SimpleTransducer transducer,
+ picoos_Common common, picokfst_FST xsampa_parser,
+ picokfst_FST svoxpa_parser, picokfst_FST xsampa2svoxpa_mapper,
+ picoos_uchar * inputPhones, picoos_uchar * alphabet,
+ picoos_uint8 * outputPhoneIds, picoos_int32 maxOutputPhoneIds)
+{
+ pico_status_t status = PICO_OK;
+ if (picoos_strcmp(alphabet, PICODATA_XSAMPA) == 0) {
+ if ((NULL != xsampa_parser) && (NULL != xsampa2svoxpa_mapper)) {
+ picotrns_stInitialize(transducer);
+ status = picotrns_stAddWithPlane(transducer, inputPhones,
+ PICOKFST_PLANE_ASCII);
+ if (PICO_OK != status) {
+ picoos_emRaiseWarning(common->em, status, NULL,
+ (picoos_char *) "phoneme sequence too long (%s)",
+ inputPhones);
+ } else {
+ status = picotrns_stTransduce(transducer, xsampa_parser);
+ if (PICO_OK != status) {
+ picoos_emRaiseWarning(common->em, status, NULL,
+ (picoos_char *) "not recognized as xsampa (%s)",
+ inputPhones);
+ } else {
+ status = picotrns_stTransduce(transducer, xsampa2svoxpa_mapper);
+ if (PICO_OK != status) {
+ picoos_emRaiseWarning(common->em, status, NULL,
+ (picoos_char *) "illeagal phoneme sequence (%s)",
+ inputPhones);
+ } else {
+ status = picotrns_stGetSymSequence(transducer, outputPhoneIds,
+ maxOutputPhoneIds);
+ }
+ }
+ }
+ return status;
+ }
+ } else if (picoos_strcmp(alphabet, PICODATA_SVOXPA) == 0) {
+ if ((NULL != svoxpa_parser)) {
+ picotrns_stInitialize(transducer);
+ status = picotrns_stAddWithPlane(transducer, inputPhones,
+ PICOKFST_PLANE_ASCII);
+ if (PICO_OK == status) {
+ status = picotrns_stTransduce(transducer, svoxpa_parser);
+ }
+ if (PICO_OK == status) {
+ status = picotrns_stGetSymSequence(transducer, outputPhoneIds,
+ maxOutputPhoneIds);
+ }
+ return status;
+ }
+ }
+ picoos_strlcpy(outputPhoneIds, (picoos_char *) "", maxOutputPhoneIds);
+ picoos_emRaiseWarning(common->em, PICO_EXC_NAME_ILLEGAL, NULL,
+ (picoos_char *) "alphabet not supported (%s)", alphabet);
+ return PICO_EXC_NAME_ILLEGAL;
+
+}
+
+#if defined (PICO_DEBUG)
+/* ***************************************************************
+ * For Debugging only *
+ *****************************************************************/
+
+
+/* put string representation of 'itemtype' into 'str' (allocated size 'strsize')
+ * return 'str' */
+static picoos_char * data_itemtype_to_string(const picoos_uint8 itemtype,
+ picoos_char * str, picoos_uint16 strsize)
+{
+ picoos_char * tmpstr;
+ switch (itemtype) {
+ case PICODATA_ITEM_BOUND:
+ tmpstr = (picoos_char *)"BOUND";
+ break;
+ case PICODATA_ITEM_FRAME_PAR:
+ tmpstr = (picoos_char *)"FRAME_PAR";
+ break;
+ case PICODATA_ITEM_PHONE:
+ tmpstr = (picoos_char *)"PHONE";
+ break;
+ case PICODATA_ITEM_CMD:
+ tmpstr = (picoos_char *)"CMD";
+ break;
+ case PICODATA_ITEM_ERR:
+ tmpstr = (picoos_char *)"ERR";
+ break;
+ case PICODATA_ITEM_FRAME:
+ tmpstr = (picoos_char *)"FRAME";
+ break;
+ case PICODATA_ITEM_OTHER:
+ tmpstr = (picoos_char *)"OTHER";
+ break;
+ case PICODATA_ITEM_PUNC:
+ tmpstr = (picoos_char *)"PUNC";
+ break;
+ case PICODATA_ITEM_SYLLPHON:
+ tmpstr = (picoos_char *)"SYLLPHON";
+ break;
+ case PICODATA_ITEM_WORDGRAPH:
+ tmpstr = (picoos_char *)"WORDGRAPH";
+ break;
+ case PICODATA_ITEM_WORDINDEX:
+ tmpstr = (picoos_char *)"WORDINDEX";
+ break;
+ case PICODATA_ITEM_WORDPHON:
+ tmpstr = (picoos_char *)"WORDPHON";
+ break;
+ case PICODATA_ITEM_WSEQ_GRAPH:
+ tmpstr = (picoos_char *)"WSEQ_GRAPH";
+ break;
+ default:
+ tmpstr = (picoos_char *)"UNKNOWN";
+ break;
+ }
+ picopal_slprintf((picopal_char *) str, strsize, (picopal_char *)"%s",
+ tmpstr);
+ return str;
+}
+
+
+picoos_char * picodata_head_to_string(const picodata_itemhead_t *head,
+ picoos_char * str, picoos_uint16 strsize)
+{
+ picoos_uint16 typelen;
+
+ if (NULL == head) {
+ picoos_strlcpy(str,(picoos_char *)"[head is NULL]",strsize);
+ } else {
+ data_itemtype_to_string(head->type, str, strsize);
+ typelen = picoos_strlen(str);
+ picopal_slprintf((picopal_char *) str+typelen, strsize-typelen,
+ (picopal_char *)"|%c|%c|%i", head->info1, head->info2,
+ head->len);
+ }
+
+ return str;
+}
+
+void picodata_info_item(const picoknow_KnowledgeBase kb,
+ const picoos_uint8 *pref6ch,
+ const picoos_uint8 *item,
+ const picoos_uint16 itemlenmax,
+ const picoos_char *filterfn)
+{
+#define SA_USE_PHST 1
+ picoos_uint16 i;
+ picoos_uint8 ch;
+
+ if ((itemlenmax < 4) || (item == NULL)) {
+ PICODBG_INFO_MSG(("invalid item\n"));
+ }
+
+ /* first 6 char used for prefix */
+ PICODBG_INFO_MSG_F(filterfn, ("%6s(", pref6ch));
+
+ /* type */
+ ch = item[0];
+ if ((32 <= ch) && (ch < 127)) {
+ PICODBG_INFO_MSG_F(filterfn, ("'%c',", ch));
+ } else {
+ PICODBG_INFO_MSG_F(filterfn, ("%3d,", ch));
+ }
+ /* info1 */
+ ch = item[1];
+ if ((32 <= ch) && (ch < 127))
+ switch (item[0]) {
+ case PICODATA_ITEM_PUNC:
+ case PICODATA_ITEM_BOUND:
+ case PICODATA_ITEM_CMD:
+ case PICODATA_ITEM_TOKEN:
+ case PICODATA_ITEM_FRAME_PAR:
+ PICODBG_INFO_MSG_F(filterfn, ("'%c',", ch));
+ break;
+ default:
+ PICODBG_INFO_MSG_F(filterfn, ("%3d,", ch));
+ }
+ else
+ PICODBG_INFO_MSG_F(filterfn, ("%3d,", ch));
+
+ /* info2 */
+ ch = item[2];
+ if ((32 <= ch) && (ch < 127))
+ switch (item[0]) {
+ case PICODATA_ITEM_PUNC:
+ case PICODATA_ITEM_BOUND:
+ case PICODATA_ITEM_CMD:
+ case PICODATA_ITEM_WORDPHON:
+ case PICODATA_ITEM_SYLLPHON:
+ PICODBG_INFO_MSG_F(filterfn, ("'%c',", ch));
+ break;
+ default:
+ PICODBG_INFO_MSG_F(filterfn, ("%3d,", ch));
+ }
+ else
+ PICODBG_INFO_MSG_F(filterfn, ("%3d,", ch));
+
+ /* len */
+ ch = item[3];
+ PICODBG_INFO_MSG_F(filterfn, ("%3d)", ch));
+
+ for (i = 0; i < ch; i++) {
+ if ((item[0] == PICODATA_ITEM_WSEQ_GRAPH) ||
+ (item[0] == PICODATA_ITEM_TOKEN) ||
+ (item[0] == PICODATA_ITEM_WORDGRAPH) ||
+ ((item[0] == PICODATA_ITEM_CMD) && !((item[1] == PICODATA_ITEMINFO1_CMD_SPEED) ||
+ (item[1] == PICODATA_ITEMINFO1_CMD_PITCH) ||
+ (item[1] == PICODATA_ITEMINFO1_CMD_VOLUME) ||
+ (item[1] == PICODATA_ITEMINFO1_CMD_SPELL) ||
+ (item[1] == PICODATA_ITEMINFO1_CMD_SIL)))) {
+ PICODBG_INFO_MSG_F(filterfn, ("%c", item[4 + i]));
+ } else {
+ PICODBG_INFO_MSG_F(filterfn, ("%4d", item[4 + i]));
+ }
+ }
+
+#if defined (SA_USE_PHST)
+ {
+#include "picokdbg.h"
+ picoos_uint8 j;
+ picokdbg_Dbg kdbg;
+ kdbg = picokdbg_getDbg(kb);
+
+ if ((item[0] == PICODATA_ITEM_WORDPHON) ||
+ (item[0] == PICODATA_ITEM_SYLLPHON) ||
+ ((item[0] == PICODATA_ITEM_CMD) && (item[1] == PICODATA_ITEMINFO1_CMD_PHONEME))) {
+ if (picokdbg_getPhoneSym(kdbg, item[4])) {
+ PICODBG_INFO_MSG_F(filterfn, (" "));
+ for (j = 0; j < item[3]; j++) {
+ PICODBG_INFO_MSG_F(filterfn, ("%s",
+ picokdbg_getPhoneSym(kdbg, item[4 + j])));
+ }
+ }
+ }
+ }
+#endif
+
+ PICODBG_INFO_MSG_F(filterfn, ("\n"));
+}
+
+
+#endif /* PICO_DEBUG */
+
+#ifdef __cplusplus
+}
+#endif
+
+
+/* picodata.c end */
diff --git a/lib/picodata.h b/lib/picodata.h
new file mode 100644
index 0000000..637a3ff
--- /dev/null
+++ b/lib/picodata.h
@@ -0,0 +1,643 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picodata.h
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+#ifndef PICODATA_H_
+#define PICODATA_H_
+
+#include "picodefs.h"
+#include "picoos.h"
+#include "picotrns.h"
+#include "picokfst.h"
+#include "picorsrc.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* ***************************************************************
+ * Constants *
+ *****************************************************************/
+
+#define PICODATA_MAX_ITEMS_PER_PHRASE 30
+
+/**
+ * @addtogroup picodata
+ * <b> Pico Data : Item Format </b>\n
+ *
+ The item header is identical for all item types and PUs. Item types
+ that are not handled by a PU are copied.
+
+ Item Header structure\n
+ ---------------------
+ - Byte Content
+ - 0x00 item type
+ - 0x01 item info 1
+ - 0x02 item info 2
+ - 0x03 item length in bytes (not including the header)
+
+ depending on the item type/info, a specific subheader may follow
+ (included in length)
+*/
+
+/* item header fields (tmp.: use item functions below to acces header fields */
+#define PICODATA_ITEMIND_TYPE 0
+#define PICODATA_ITEMIND_INFO1 1
+#define PICODATA_ITEMIND_INFO2 2
+#define PICODATA_ITEMIND_LEN 3
+
+/* ***************************************************************
+ * CharBuffer *
+ *****************************************************************/
+typedef struct picodata_char_buffer * picodata_CharBuffer;
+
+picodata_CharBuffer picodata_newCharBuffer(picoos_MemoryManager mm,
+ picoos_Common common, picoos_objsize_t size);
+
+void picodata_disposeCharBuffer(picoos_MemoryManager mm,
+ picodata_CharBuffer * this);
+
+/* should not be used for PUs but only for feeding the initial cb */
+pico_status_t picodata_cbPutCh(register picodata_CharBuffer this, picoos_char ch);
+
+/* should not be used for PUs other than first PU in the chain (picotok) */
+picoos_int16 picodata_cbGetCh(register picodata_CharBuffer this);
+
+/* reset cb (as if after newCharBuffer) */
+pico_status_t picodata_cbReset (register picodata_CharBuffer this);
+
+/* ** CharBuffer item functions, cf. below in items section ****/
+
+/* ***************************************************************
+ * items *
+ *****************************************************************/
+
+/* item header size */
+#define PICODATA_ITEM_HEADSIZE 4
+
+typedef struct picodata_itemhead
+{
+ picoos_uint8 type;
+ picoos_uint8 info1;
+ picoos_uint8 info2;
+ picoos_uint8 len;
+} picodata_itemhead_t;
+
+
+/* -------------- System wide defines referred to by items -------- */
+/* ---- These maybe better stored in a knowledge module/resoruce*/
+#define PICODATA_ACC0 '\x30' /* 48 '0' */
+#define PICODATA_ACC1 '\x31' /* 49 '1' */
+#define PICODATA_ACC2 '\x32' /* 50 '2' */
+#define PICODATA_ACC3 '\x33' /* 51 '3' */
+#define PICODATA_ACC4 '\x34' /* 52 '4' */
+
+/* reserved for future use:
+ * user-imposed Part-Of-Speech ids for user lexica and phoneme tags
+ * These values should be applied BEFORE POS-disambiguation. The POS lingware either assigns the same
+ * ids to corresponding internal unique or composed POS or else the POS-D will consider these values
+ * "default" */
+#define PICODATA_POS_XNPR 20
+#define PICODATA_POS_XN 21
+#define PICODATA_POS_XV 22
+#define PICODATA_POS_XA 23
+#define PICODATA_POS_XADV 24
+#define PICODATA_POS_XX 25
+
+/* ------------------------- item types ---------------------------- */
+/* new item types, info1, info2 to be defined during PU development */
+/* make sure this stays in sync with "is_valid_itemtype" function */
+#define PICODATA_ITEM_WSEQ_GRAPH '\x73' /* 115, 's' */
+#define PICODATA_ITEM_TOKEN '\x74' /* 116 't' */
+#define PICODATA_ITEM_WORDGRAPH '\x67' /* 103 'g' */
+#define PICODATA_ITEM_WORDINDEX '\x69' /* 105 'i' */
+#define PICODATA_ITEM_WORDPHON '\x77' /* 119 'w' */
+#define PICODATA_ITEM_SYLLPHON '\x79' /* 121 'y' */
+#define PICODATA_ITEM_BOUND '\x62' /* 98 'b' */
+/* #define PICODATA_ITEM_BOUND_DUR '\x64' */ /* 100 'd' */ /* duration-constrained bound */
+#define PICODATA_ITEM_PUNC '\x70' /* 112 'p' */
+#define PICODATA_ITEM_CMD '\x63' /* 99 'c' */
+#define PICODATA_ITEM_PHONE '\x68' /* 104 'h' */ /*reserved for PAM*/
+#define PICODATA_ITEM_FRAME_PAR '\x6b' /* 107 'k' */ /*reserved for CEP*/
+#define PICODATA_ITEM_FRAME '\x66' /* 102 'f' */ /*reserved for SIG*/
+#define PICODATA_ITEM_OTHER '\x6f' /* 111 'o' */
+#define PICODATA_ITEM_ERR '\x00' /* 0 '^@' */
+
+/* generic iteminfo1 */
+#define PICODATA_ITEMINFO1_ERR '\x00' /* 0 '^@' */ /* error state */
+#define PICODATA_ITEMINFO1_NA '\x01' /* 1 '^A' */ /* not applicable */
+
+/* generic iteminfo2 */
+#define PICODATA_ITEMINFO2_ERR '\x00' /* 0 '^@' */ /* error state */
+#define PICODATA_ITEMINFO2_NA '\x01' /* 1 '^A' */ /* not applicable */
+
+/* ------------------------- PUNC item type ---------------------------- */
+/* iteminfo1 */
+#define PICODATA_ITEMINFO1_PUNC_SENTEND '\x73' /* 115 's' */
+#define PICODATA_ITEMINFO1_PUNC_PHRASEEND '\x70' /* 112 'p' */
+#define PICODATA_ITEMINFO1_PUNC_FLUSH '\x66' /* 102 'f' */
+/* iteminfo2 */
+#define PICODATA_ITEMINFO2_PUNC_SENT_T '\x74' /* 116 't' */
+#define PICODATA_ITEMINFO2_PUNC_SENT_Q '\x71' /* 113 'q' */
+#define PICODATA_ITEMINFO2_PUNC_SENT_E '\x65' /* 101 'e' */
+#define PICODATA_ITEMINFO2_PUNC_PHRASE '\x70' /* 112 'p' */
+#define PICODATA_ITEMINFO2_PUNC_PHRASE_FORCED '\x66' /* 102 'f' */
+/* len for PUNC item is ALWAYS = 0 */
+/* ------------------------- BOUND item type ---------------------------- */
+/* iteminfo1 : phrase strength*/
+#define PICODATA_ITEMINFO1_BOUND_SBEG '\x62' /* 98 'b', at sentence begin */
+#define PICODATA_ITEMINFO1_BOUND_SEND '\x73' /* 115 's', at sentence end */
+#define PICODATA_ITEMINFO1_BOUND_TERM '\x74' /* 116 't', replaces a flush */
+#define PICODATA_ITEMINFO1_BOUND_PHR0 '\x30' /* 48 '0', no break, no item */
+#define PICODATA_ITEMINFO1_BOUND_PHR1 '\x31' /* 49 '1', pri. phrase bound. */
+#define PICODATA_ITEMINFO1_BOUND_PHR2 '\x32' /* 50 '2', short break */
+#define PICODATA_ITEMINFO1_BOUND_PHR3 '\x33' /* 51 '3', sec. phr. bound., no break*/
+/* iteminfo2 : phrase type*/
+#define PICODATA_ITEMINFO2_BOUNDTYPE_P '\x50' /* 80 'P' */
+#define PICODATA_ITEMINFO2_BOUNDTYPE_T '\x54' /* 84 'T' */
+#define PICODATA_ITEMINFO2_BOUNDTYPE_Q '\x51' /* 81 'Q' */
+#define PICODATA_ITEMINFO2_BOUNDTYPE_E '\x45' /* 69 'E' */
+/* len for BOUND item is ALWAYS = 0 */
+/* ------------------------- CMD item type ---------------------------- */
+/* iteminfo1 */
+#define PICODATA_ITEMINFO1_CMD_FLUSH 'f' /* 102 flush command (all PUs)*/
+#define PICODATA_ITEMINFO1_CMD_PLAY 'p' /* 112 play command : PU in info 2 will read items from file-->Filename in item content.*/
+#define PICODATA_ITEMINFO1_CMD_SAVE 's' /* 115 save command : PU in info 2 will save items to file-->Filename in item content.*/
+#define PICODATA_ITEMINFO1_CMD_UNSAVE 'u' /* 117 save command : PU in info 2 will stop saving items to file*/
+#define PICODATA_ITEMINFO1_CMD_PROSDOMAIN 'd' /* 100 prosody domain : domain type in info 2, domain name in item content */
+#define PICODATA_ITEMINFO1_CMD_SPELL 'e' /* 101 spell command : info 2 contains start/stop info,
+ spell type/pause len as little endian uint16 in item content */
+#define PICODATA_ITEMINFO1_CMD_IGNSIG 'i' /* ignore signal command : info 2 contains start/stop info */
+#define PICODATA_ITEMINFO1_CMD_PHONEME 'o' /* phoneme command : info 2 contains start/stop info, phonemes in item content */
+#define PICODATA_ITEMINFO1_CMD_IGNORE 'I' /* ignore text command : info 2 contains start/stop info */
+#define PICODATA_ITEMINFO1_CMD_SIL 'z' /* silence command : info 2 contains type of silence;
+ silence duration as little endian uint16 in item content */
+#define PICODATA_ITEMINFO1_CMD_CONTEXT 'c' /* context command : context name in item content */
+#define PICODATA_ITEMINFO1_CMD_VOICE 'v' /* context command : voice name in item content */
+#define PICODATA_ITEMINFO1_CMD_MARKER 'm' /* marker command : marker name in item content */
+#define PICODATA_ITEMINFO1_CMD_PITCH 'P' /* 80 pitch command : abs/rel info in info 2; pitch level as little endian
+ uint16 in item content; relative value is in promille */
+#define PICODATA_ITEMINFO1_CMD_SPEED 'R' /* 82 speed command : abs/rel info in info 2, speed level as little endian
+ uint16 in item content; elative value is in promille */
+#define PICODATA_ITEMINFO1_CMD_VOLUME 'V' /* 86 volume command : abs/rel info in info 2, volume level as little endian
+ uint16 in item content; relative value is in promille */
+#define PICODATA_ITEMINFO1_CMD_SPEAKER 'S' /* 83 speaker command : abs/rel info in info 2, speaker level as little endian
+ uint16 in item content; relative value is in promille */
+
+/* iteminfo2 for PLAY/SAVE */
+#define PICODATA_ITEMINFO2_CMD_TO_TOK 't' /* CMD+PLAY/SAVE+TOKENISATION*/
+#define PICODATA_ITEMINFO2_CMD_TO_PR 'g' /* CMD+PLAY/SAVE+PREPROC*/
+#define PICODATA_ITEMINFO2_CMD_TO_WA 'w' /* CMD+PLAY/SAVE+WORDANA*/
+#define PICODATA_ITEMINFO2_CMD_TO_SA 'a' /* CMD+PLAY/SAVE+SENTANA*/
+#define PICODATA_ITEMINFO2_CMD_TO_ACPH 'h' /* CMD+PLAY/SAVE+ACCENTUATION&PHRASING*/
+#define PICODATA_ITEMINFO2_CMD_TO_SPHO 'p' /* CMD+PLAY/SAVE+ACCENTUATION&PHRASING*/
+#define PICODATA_ITEMINFO2_CMD_TO_PAM 'q' /* CMD+PLAY/SAVE+PHONETIC-ACOUSTIC MAPPING*/
+#define PICODATA_ITEMINFO2_CMD_TO_CEP 'c' /* CMD+PLAY/SAVE+CEP_SMOOTHER*/
+#define PICODATA_ITEMINFO2_CMD_TO_SIG 's' /* CMD+PLAY/SAVE+SIG_GEN */
+
+#if 0
+#define PICODATA_ITEMINFO2_CMD_TO_FST 'f' /* CMD+PLAY/SAVE+FST for Syll and Phonotactic constraints*/
+#endif
+
+#define PICODATA_ITEMINFO2_CMD_TO_UNKNOWN 255
+
+/* iteminfo2 for start/end commands */
+#define PICODATA_ITEMINFO2_CMD_START 's'
+#define PICODATA_ITEMINFO2_CMD_END 'e'
+
+/* iteminfo2 for speed/pitch/volume commands */
+#define PICODATA_ITEMINFO2_CMD_ABSOLUTE 'a'
+#define PICODATA_ITEMINFO2_CMD_RELATIVE 'r'
+
+/* len for CMD item could be >= 0 */
+/* ------------------------- TOKEN item type ---------------------------- */
+/* iteminfo1: simple token type : */
+#define PICODATA_ITEMINFO1_TOKTYPE_SPACE 'W'
+#define PICODATA_ITEMINFO1_TOKTYPE_LETTERV 'V'
+#define PICODATA_ITEMINFO1_TOKTYPE_LETTER 'L'
+#define PICODATA_ITEMINFO1_TOKTYPE_DIGIT 'D'
+#define PICODATA_ITEMINFO1_TOKTYPE_SEQ 'S'
+#define PICODATA_ITEMINFO1_TOKTYPE_CHAR 'C'
+#define PICODATA_ITEMINFO1_TOKTYPE_BEGIN 'B'
+#define PICODATA_ITEMINFO1_TOKTYPE_END 'E'
+#define PICODATA_ITEMINFO1_TOKTYPE_UNDEFINED 'U'
+/* iteminfo2 : token subtype */
+/* len for WORDTOK item is ALWAYS > 0, if len==0 an error should be raised */
+
+/**
+ * @addtogroup picodata
+ *
+ * ------------------------- WORDGRAPH item type ----------------------------
+ * - iteminfo1 : POS and multi-POS values defined in lingware
+ * - iteminfo2 : not applicable
+ * - len for WORDGRAPH item is ALWAYS > 0, if len==0 an error should be raised
+ * (currently picopr may produce empty WORDGRAPH that is eliminated by picowa)
+ * \n------------------------- WORDINDEX item type ----------------------------
+ * - iteminfo1 : POS and multi-POS values defined in lingware
+ * - iteminfo2 : not applicable
+ * - len for WORDINDEX item is ALWAYS > 0, if len==0 an error should be raised
+ * \n------------------------- WORDPHON item type ----------------------------
+ * - iteminfo1 : POS values defined in lingware
+ * - iteminfo2 : Uses PICODATA_ACC0 .. ACC4
+ * -len WORDPHON item is ALWAYS > 0, if len==0 an error should be raised
+ * \n------------------------- SYLLPHON item type ----------------------------
+ * - iteminfo1 : not applicable
+ * - iteminfo2 : Uses PICODATA_ACC0 .. ACC4
+ * - len for SYLLPHON item is ALWAYS > 0, if len==0 an error should be raised
+ * \n------------------------- PHONE item type (PRODUCED BY PAM)-----------------
+ * - iteminfo1 : phonId : the phonetic identity of the phone
+ * - iteminfo2 : n_S_P_Phone : number of states per phoneme
+ * - len for PHON item is ALWAYS > 0, if len==0 an error should be raised
+ * \n------------------------- FRAME_PAR item type (PRODUCED BY CEP) --------
+ * - iteminfo1 : format (float, fixed)
+ * - iteminfo2 : vector size
+ * - len for FRAME_PAR item is ALWAYS > 0, if len==0 an error should be raised
+ * \n------------------------- FRAME item type (PRODUCED BY SIG) -----------
+ * - iteminfo1 : number of samples per frame
+ * - iteminfo2 : number of bytes per sample
+ * - len for FRAME item is ALWAYS > 0, if len==0 an error should be raised
+ *
+ */
+#define PICODATA_ITEMINFO1_FRAME_PAR_DATA_FORMAT_FIXED '\x78' /* 120 'x' fixed point */
+#define PICODATA_ITEMINFO1_FRAME_PAR_DATA_FORMAT_FLOAT '\x66' /* 102 'f' floating point */
+
+/* ***************************************************************
+ * items: CharBuffer functions *
+ *****************************************************************/
+
+/* gets a single item (head and content) from a CharBuffer in buf;
+ blenmax is the max length (in number of bytes) of buf; blen is
+ set to the number of bytes gotten in buf; return values:
+ PICO_OK <- one item gotten
+ PICO_EOF <- no item available, cb is empty
+ PICO_EXC_BUF_UNDERFLOW <- cb not empty, but no valid item
+ PICO_EXC_BUF_OVERFLOW <- buf not large enough
+*/
+pico_status_t picodata_cbGetItem(register picodata_CharBuffer this,
+ picoos_uint8 *buf, const picoos_uint16 blenmax,
+ picoos_uint16 *blen);
+
+/* gets the speech data (without item head) from a CharBuffer in buf;
+ blenmax is the max length (in number of bytes) of buf; blen is
+ set to the number of bytes gotten in buf; return values:
+ PICO_OK <- speech data of one item gotten
+ PICO_EOF <- no item available, cb is empty
+ PICO_EXC_BUF_UNDERFLOW <- cb not empty, but no valid item
+ PICO_EXC_BUF_OVERFLOW <- buf not large enough
+*/
+pico_status_t picodata_cbGetSpeechData(register picodata_CharBuffer this,
+ picoos_uint8 *buf, const picoos_uint16 blenmax,
+ picoos_uint16 *blen);
+
+/* puts a single item (head and content) to a CharBuffer; clenmax is
+ the max length (in number of bytes) accessible in content; clen is
+ set to the number of bytes put from content; return values:
+ PICO_OK <- one item put
+ PICO_EXC_BUF_UNDERFLOW <- no valid item in buf
+ PICO_EXC_BUF_OVERFLOW <- cb not large enough
+*/
+pico_status_t picodata_cbPutItem(register picodata_CharBuffer this,
+ const picoos_uint8 *buf, const picoos_uint16 blenmax,
+ picoos_uint16 *blen);
+
+/* unsafe, just for measuring purposes */
+picoos_uint8 picodata_cbGetFrontItemType(register picodata_CharBuffer this);
+
+/* ***************************************************************
+ * items: support function *
+ *****************************************************************/
+
+/* checks, whether item of type 'ch' is a valid item type */
+picoos_uint8 is_valid_itemtype(const picoos_uint8 ch);
+
+/* gets from buf a single item, values in head set and item content
+ copied to content; blenmax and clenmax are the max lengths (in
+ number of bytes) accessible in buf and content; clen is set to the
+ number of bytes gotten in content; return values:
+ PICO_OK <- all ok
+ PICO_EXC_BUF_UNDERFLOW <- blenmax problem, or no valid item
+ PICO_EXC_BUF_OVERFLOW <- overflow in content
+*/
+pico_status_t picodata_get_itemparts_nowarn(
+ const picoos_uint8 *buf, const picoos_uint16 blenmax,
+ picodata_itemhead_t *head, picoos_uint8 *content,
+ const picoos_uint16 clenmax, picoos_uint16 *clen);
+
+/* gets from buf a single item, values in head set and item content
+ copied to content; blenmax and clenmax are the max lengths (in
+ number of bytes) accessible in buf and content; clen is set to the
+ number of bytes gotten in content; return values:
+ PICO_OK <- all ok
+ PICO_EXC_BUF_UNDERFLOW <- blenmax problem, or no valid item
+ PICO_EXC_BUF_OVERFLOW <- overflow in content
+*/
+pico_status_t picodata_get_itemparts(
+ const picoos_uint8 *buf, const picoos_uint16 blenmax,
+ picodata_itemhead_t *head, picoos_uint8 *content,
+ const picoos_uint16 clenmax, picoos_uint16 *clen);
+
+/* puts a single item to buf; values in head and content copied to
+ buf; clenmax is the max length (in number of bytes) accessible in
+ content; blenmax is the max length (bytes) accessible in buf; blen
+ is set to the number of bytes put to buf; return values:
+ PICO_OK <- all ok
+ PICO_EXC_BUF_UNDERFLOW <- clenmax problem, or no valid item
+ PICO_EXC_BUF_OVERFLOW <- overflow in buf
+*/
+pico_status_t picodata_put_itemparts(const picodata_itemhead_t *head,
+ const picoos_uint8 *content, const picoos_uint16 clenmax,
+ picoos_uint8 *buf, const picoos_uint16 blenmax, picoos_uint16 *blen);
+
+/* gets from buf info of a single item, values in head are set and
+ content is set to the start of content in buf (not copied!);
+ content is set to NULL if the content length is 0; blenmax is the
+ max lengths (in number of bytes) accessible in buf; return values:
+ PICO_OK <- all ok
+ PICO_EXC_BUF_UNDERFLOW <- blenmax problem, or no valid item
+*/
+pico_status_t picodata_get_iteminfo(
+ picoos_uint8 *buf, const picoos_uint16 blenmax,
+ picodata_itemhead_t *head, picoos_uint8 **content);
+
+/* copies the item in inbuf to outbuf after first checking if there is
+ a valid item in inbuf; inlenmax and outlenmax are the max length
+ (in number of byte) accessible in the buffers); in *numb the total
+ number of bytes copied to outbuf (incl. header) is returned; return
+ values:
+ PICO_OK <- item copied
+ PICO_EXC_BUF_OVERFLOW <- overflow in outbuf
+ PICO_ERR_OTHER <- no valid item in inbuf
+*/
+pico_status_t picodata_copy_item(const picoos_uint8 *inbuf,
+ const picoos_uint16 inlenmax, picoos_uint8 *outbuf,
+ const picoos_uint16 outlenmax, picoos_uint16 *numb);
+
+/* sets the info1 field in the header contained in the item in buf;
+ return values:
+ PICO_OK <- all ok
+ PICO_EXC_BUF_UNDERFLOW <- underflow in buf
+*/
+pico_status_t picodata_set_iteminfo1(picoos_uint8 *buf,
+ const picoos_uint16 blenmax, const picoos_uint8 info);
+
+/* sets the info2 field in the header contained in the item in buf;
+ return values:
+ PICO_OK <- all ok
+ PICO_EXC_BUF_UNDERFLOW <- underflow in buf
+*/
+pico_status_t picodata_set_iteminfo2(picoos_uint8 *buf,
+ const picoos_uint16 blenmax, const picoos_uint8 info);
+
+/* sets the len field in the header contained in the item in buf;
+ return values:
+ PICO_OK <- all ok
+ PICO_EXC_BUF_UNDERFLOW <- underflow in buf
+*/
+pico_status_t picodata_set_itemlen(picoos_uint8 *buf,
+ const picoos_uint16 blenmax, const picoos_uint8 len);
+
+/* check item validity and return TRUE if valid; return FALSE if
+ invalid; ilenmax is the max index to be used in item
+*/
+picoos_uint8 picodata_is_valid_item(const picoos_uint8 *item,
+ const picoos_uint16 ilenmax);
+
+/* return TRUE if head is a valid item head, FALSE otherwise */
+picoos_uint8 picodata_is_valid_itemhead(const picodata_itemhead_t *head);
+
+
+/* ***************************************************************
+ * ProcessingUnit *
+ *****************************************************************/
+/* public */
+
+#define PICODATA_MAX_ITEMSIZE (picoos_uint16) (PICODATA_ITEM_HEADSIZE + 256)
+
+/* different buffer sizes per processing unit */
+#define PICODATA_BUFSIZE_DEFAULT (picoos_uint16) PICODATA_MAX_ITEMSIZE
+#define PICODATA_BUFSIZE_TEXT (picoos_uint16) 1 * PICODATA_BUFSIZE_DEFAULT
+#define PICODATA_BUFSIZE_TOK (picoos_uint16) 2 * PICODATA_BUFSIZE_DEFAULT
+#define PICODATA_BUFSIZE_PR (picoos_uint16) 2 * PICODATA_BUFSIZE_DEFAULT
+#define PICODATA_BUFSIZE_WA (picoos_uint16) 2 * PICODATA_BUFSIZE_DEFAULT
+#define PICODATA_BUFSIZE_SA (picoos_uint16) 2 * PICODATA_BUFSIZE_DEFAULT
+#define PICODATA_BUFSIZE_ACPH (picoos_uint16) 2 * PICODATA_BUFSIZE_DEFAULT
+#define PICODATA_BUFSIZE_SPHO (picoos_uint16) 4 * PICODATA_BUFSIZE_DEFAULT
+#define PICODATA_BUFSIZE_PAM (picoos_uint16) 4 * PICODATA_BUFSIZE_DEFAULT
+#define PICODATA_BUFSIZE_CEP (picoos_uint16) 16 * PICODATA_BUFSIZE_DEFAULT
+#define PICODATA_BUFSIZE_SIG (picoos_uint16) 16 * PICODATA_BUFSIZE_DEFAULT
+#define PICODATA_BUFSIZE_SINK (picoos_uint16) 1 * PICODATA_BUFSIZE_DEFAULT
+
+/* different types of processing units */
+typedef enum picodata_putype {
+ PICODATA_PUTYPE_TEXT, /* text */
+ PICODATA_PUTYPE_TOK, /* tokenizer output */
+ PICODATA_PUTYPE_PR, /* preprocessor output */
+ PICODATA_PUTYPE_WA, /* word analysis */
+ PICODATA_PUTYPE_SA, /* sentence analysis */
+ PICODATA_PUTYPE_ACPH, /* accentuation and phrasing */
+ PICODATA_PUTYPE_SPHO, /* sentence phonology (textana postproc) */
+ PICODATA_PUTYPE_PAM, /* phonetics to acoustics mapper processing unit */
+ PICODATA_PUTYPE_CEP, /* cepstral smoothing processing unit */
+ PICODATA_PUTYPE_SIG, /* signal generation processing unit*/
+ PICODATA_PUTYPE_SINK /* item sink unit*/
+} picodata_putype_t;
+
+picoos_uint16 picodata_get_default_buf_size (picodata_putype_t puType);
+
+/* result values returned from the pu->puStep() methode */
+typedef enum picodata_step_result {
+ PICODATA_PU_ERROR,
+ /* PICODATA_PU_EMPTY, *//* reserved (no internal data to be processed) */
+ PICODATA_PU_IDLE, /* need more input to process internal data */
+ PICODATA_PU_BUSY, /* processing internal data */
+ PICODATA_PU_ATOMIC, /* same as pu_busy, but wants to get next time slot (while in an "atomar" operation) */
+ PICODATA_PU_OUT_FULL /* can't proceed because output is full. (next time slot to be assigned to pu's output's consumer) */
+} picodata_step_result_t;
+
+typedef struct picodata_processing_unit * picodata_ProcessingUnit;
+
+picodata_ProcessingUnit picodata_newProcessingUnit(
+ picoos_MemoryManager mm,
+ picoos_Common common,
+ picodata_CharBuffer cbIn,
+ picodata_CharBuffer cbOut,
+ picorsrc_Voice voice);
+
+void picodata_disposeProcessingUnit(
+ picoos_MemoryManager mm,
+ picodata_ProcessingUnit * this);
+
+picodata_CharBuffer picodata_getCbIn(picodata_ProcessingUnit this);
+picodata_CharBuffer picodata_getCbOut(picodata_ProcessingUnit this);
+pico_status_t picodata_setCbIn(picodata_ProcessingUnit this, picodata_CharBuffer cbIn);
+pico_status_t picodata_setCbOut(picodata_ProcessingUnit this, picodata_CharBuffer cbOut);
+
+/* protected */
+typedef pico_status_t (* picodata_puInitializeMethod) (register picodata_ProcessingUnit this);
+typedef pico_status_t (* picodata_puTerminateMethod) (register picodata_ProcessingUnit this);
+typedef picodata_step_result_t (* picodata_puStepMethod) (register picodata_ProcessingUnit this, picoos_int16 mode, picoos_uint16 * numBytesOutput);
+typedef pico_status_t (* picodata_puSubDeallocateMethod) (register picodata_ProcessingUnit this, picoos_MemoryManager mm);
+
+typedef struct picodata_processing_unit
+{
+ /* public */
+ picodata_puInitializeMethod initialize;
+ picodata_puStepMethod step;
+ picodata_puTerminateMethod terminate;
+ picorsrc_Voice voice;
+
+ /* protected */
+ picoos_Common common;
+ picodata_CharBuffer cbIn, cbOut;
+ picodata_puSubDeallocateMethod subDeallocate;
+ void * subObj;
+
+} picodata_processing_unit_t;
+
+/* currently, only wav input and output is supported */
+#define PICODATA_PUTYPE_TEXT_OUTPUT_EXTENSION (picoos_uchar*)".txt"
+#define PICODATA_PUTYPE_TOK_INPUT_EXTENSION PICODATA_PUTYPE_TEXT_OUTPUT_EXTENSION
+#define PICODATA_PUTYPE_TOK_OUTPUT_EXTENSION (picoos_uchar*)".tok"
+#define PICODATA_PUTYPE_PR_INPUT_EXTENSION PICODATA_PUTYPE_TOK_OUTPUT_EXTENSION
+#define PICODATA_PUTYPE_PR_OUTPUT_EXTENSION (picoos_uchar*)".pr"
+#define PICODATA_PUTYPE_WA_INPUT_EXTENSION PICODATA_PUTYPE_PR_OUTPUT_EXTENSION
+#define PICODATA_PUTYPE_WA_OUTPUT_EXTENSION (picoos_uchar*)".wa"
+#define PICODATA_PUTYPE_SA_INPUT_EXTENSION PICODATA_PUTYPE_WA_OUTPUT_EXTENSION
+#define PICODATA_PUTYPE_SA_OUTPUT_EXTENSION (picoos_uchar*)".sa"
+#define PICODATA_PUTYPE_ACPH_INPUT_EXTENSION PICODATA_PUTYPE_SA_OUTPUT_EXTENSION
+#define PICODATA_PUTYPE_ACPH_OUTPUT_EXTENSION (picoos_uchar*)".acph"
+#define PICODATA_PUTYPE_SPHO_INPUT_EXTENSION PICODATA_PUTYPE_ACPH_OUTPUT_EXTENSION
+#define PICODATA_PUTYPE_SPHO_OUTPUT_EXTENSION (picoos_uchar*)".spho"
+#define PICODATA_PUTYPE_PAM_INPUT_EXTENSION PICODATA_PUTYPE_SPHO_OUTPUT_EXTENSION
+#define PICODATA_PUTYPE_PAM_OUTPUT_EXTENSION (picoos_uchar*)".pam"
+#define PICODATA_PUTYPE_CEP_INPUT_EXTENSION PICODATA_PUTYPE_PAM_OUTPUT_EXTENSION
+#define PICODATA_PUTYPE_CEP_OUTPUT_EXTENSION (picoos_uchar*)".cep"
+#define PICODATA_PUTYPE_SIG_INPUT_EXTENSION PICODATA_PUTYPE_CEP_OUTPUT_EXTENSION /*PP 11.7.08*/
+#define PICODATA_PUTYPE_SIG_OUTPUT_EXTENSION (picoos_uchar*)".sig"
+#define PICODATA_PUTYPE_SINK_INPUT_EXTENSION PICODATA_PUTYPE_SIG_OUTPUT_EXTENSION
+
+/*wav input is for play wav files in sig */
+#define PICODATA_PUTYPE_WAV_INPUT_EXTENSION (picoos_uchar*)".wav" /*PP 11.7.08*/
+
+/*wav output is for saving wav (binary) files in sig*/
+#define PICODATA_PUTYPE_WAV_OUTPUT_EXTENSION (picoos_uchar*)".wav" /*PP 14.7.08*/
+
+/* ***************************************************************
+ * auxiliary routines *
+ *****************************************************************/
+
+picoos_uint8 picodata_getPuTypeFromExtension(picoos_uchar * filename, picoos_bool input);
+
+#define PICODATA_XSAMPA (picoos_uchar *)"xsampa"
+#define PICODATA_SAMPA (picoos_uchar *)"sampa"
+#define PICODATA_SVOXPA (picoos_uchar *)"svoxpa"
+
+/*----------------------------------------------------------*/
+/** @brief maps an input phone string to its internal representation
+ *
+ * @param transducer initialized SimpleTransducer
+ * @param xsampa_parser fst converting xsampa char input to xsampa ids
+ * @param svoxpa_parser
+ * @param xsampa2svoxpa_mapper
+ * @param inputPhones input phone string in alphabet 'alphabet'
+ * @param alphabet input alphabet
+ * @retval outputPhoneIds output phone string in internal representation
+ * @param maxOutputPhoneIds
+ * @return PICO_OK=mapping done, PICO_ERR_OTHER:unknown alphabet, unknown phones
+ */
+/*---------------------------------------------------------*/
+pico_status_t picodata_mapPAStrToPAIds(
+ picotrns_SimpleTransducer transducer,
+ picoos_Common common,
+ picokfst_FST xsampa_parser,
+ picokfst_FST svoxpa_parser,
+ picokfst_FST xsampa2svoxpa_mapper,
+ picoos_uchar * inputPhones,
+ picoos_uchar * alphabet,
+ picoos_uint8 * outputPhoneIds,
+ picoos_int32 maxOutputPhoneIds);
+
+/* number of binary digits after the comma for fixed-point calculation */
+#define PICODATA_PRECISION 10
+/* constant 0.5 in PICODATA_PRECISION */
+#define PICODATA_PREC_HALF 512
+
+void picodata_transformDurations(
+ picoos_uint8 frame_duration_exp,
+ picoos_int8 array_length,
+ picoos_uint8 * inout,
+ const picoos_uint16 * weight, /* integer weights */
+ picoos_int16 mintarget, /* minimum target duration in ms */
+ picoos_int16 maxtarget, /* maximum target duration in ms */
+ picoos_int16 facttarget, /* factor to be multiplied with original length to get the target
+ the factor is fixed-point with precision PRECISION, i.e.
+ the factor as float would be facttarget / PRECISION_FACT
+ if factor is 0, only min/max are considered */
+ picoos_int16 * dur_rest /* in/out, rest in ms */
+ );
+
+
+
+/* ***************************************************************
+ * For Debugging only *
+ *****************************************************************/
+
+#if defined (PICO_DEBUG)
+
+/* convert (pretty print) item head 'head' and put output in 'str',
+ strsize is the maximum length of 'str' in bytes */
+picoos_char * picodata_head_to_string(const picodata_itemhead_t *head,
+ picoos_char * str, picoos_uint16 strsize);
+
+/* put 'pref6ch' (max. 6 char prefix) and a pretty print output of
+ 'item' in 'str', strlenmax is the maximum length of 'str' in
+ bytes */
+void picodata_info_item(const picoknow_KnowledgeBase kb,
+ const picoos_uint8 *pref6ch,
+ const picoos_uint8 *item,
+ const picoos_uint16 itemlenmax,
+ const picoos_char *filterfn);
+
+
+#define PICODATA_INFO_ITEM(kb, pref, item, itemlenmax) \
+ PICODBG_INFO_CTX(); \
+ picodata_info_item(kb, pref, item, itemlenmax, (picoos_char *)__FILE__)
+
+
+
+#else
+
+#define PICODATA_INFO_ITEM(kb, pref, item, itemlenmax)
+
+#endif
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /*PICODATA_H_*/
diff --git a/lib/picodbg.c b/lib/picodbg.c
new file mode 100644
index 0000000..8fa753b
--- /dev/null
+++ b/lib/picodbg.c
@@ -0,0 +1,438 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picodbg.c
+ *
+ * Provides functions and macros to debug the Pico system and to trace
+ * the execution of its code.
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ */
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+#if defined(PICO_DEBUG)
+
+/* Two variants of colored console output are implemented:
+ COLOR_MODE_WINDOWS
+ uses the Windows API function SetConsoleTextAttribute
+ COLOR_MODE_ANSI
+ uses ANSI escape codes */
+#if defined(_WIN32)
+#define COLOR_MODE_WINDOWS
+#else
+#define COLOR_MODE_ANSI
+#endif
+
+
+#include <stdio.h>
+#include <stdlib.h>
+
+#include <stdarg.h>
+#include <string.h>
+
+#include "picodbg.h"
+
+
+/* Maximum length of a formatted tracing message */
+#define MAX_MESSAGE_LEN 999
+
+/* Maximum length of contextual information */
+#define MAX_CONTEXT_LEN 499
+
+/* Maximum length of filename filter */
+#define MAX_FILTERFN_LEN 16
+
+/* Delimiter used in debug messages */
+#define MSG_DELIM "|"
+
+/* Standard output file for debug messages */
+#define STDDBG stdout /* or stderr */
+
+/* Default setup */
+#define PICODBG_DEFAULT_LEVEL PICODBG_LOG_LEVEL_WARN
+#define PICODBG_DEFAULT_FILTERFN ""
+#define PICODBG_DEFAULT_FORMAT \
+ (PICODBG_SHOW_LEVEL | PICODBG_SHOW_SRCNAME | PICODBG_SHOW_FUNCTION)
+#define PICODBG_DEFAULT_COLOR 1
+
+
+/* Current log level */
+static int logLevel = PICODBG_DEFAULT_LEVEL;
+
+/* Current log filter (filename) */
+static char logFilterFN[MAX_FILTERFN_LEN + 1];
+
+/* Current log file or NULL if no log file is set */
+static FILE *logFile = NULL;
+
+/* Current output format */
+static int logFormat = PICODBG_DEFAULT_FORMAT;
+
+/* Color mode for console output (0 : disable colors, != 0 : enable colors */
+static int optColor = 0;
+
+/* Buffer for context information */
+static char ctxbuf[MAX_CONTEXT_LEN + 1];
+
+/* Buffer to format tracing messages */
+static char msgbuf[MAX_MESSAGE_LEN + 1];
+
+
+/* *** Support for colored text output to console *****/
+
+
+/* Console text colors */
+enum color_t {
+ /* order matches Windows color codes */
+ ColorBlack,
+ ColorBlue,
+ ColorGreen,
+ ColorCyan,
+ ColorRed,
+ ColorPurple,
+ ColorBrown,
+ ColorLightGray,
+ ColorDarkGray,
+ ColorLightBlue,
+ ColorLightGreen,
+ ColorLightCyan,
+ ColorLightRed,
+ ColorLightPurple,
+ ColorYellow,
+ ColorWhite
+};
+
+
+static enum color_t picodbg_getLevelColor(int level)
+{
+ switch (level) {
+ case PICODBG_LOG_LEVEL_ERROR: return ColorLightRed;
+ case PICODBG_LOG_LEVEL_WARN : return ColorYellow;
+ case PICODBG_LOG_LEVEL_INFO : return ColorGreen;
+ case PICODBG_LOG_LEVEL_DEBUG: return ColorLightGray;
+ case PICODBG_LOG_LEVEL_TRACE: return ColorDarkGray;
+ }
+ return ColorWhite;
+}
+
+
+#if defined(COLOR_MODE_WINDOWS)
+
+#define WIN32_LEAN_AND_MEAN
+#include <windows.h>
+
+static int picodbg_setTextAttr(FILE *stream, int attr)
+{
+ HANDLE hConsole;
+
+ if (stream == stdout) {
+ hConsole = GetStdHandle(STD_OUTPUT_HANDLE);
+ } else if (stream == stderr) {
+ hConsole = GetStdHandle(STD_ERROR_HANDLE);
+ } else {
+ hConsole = INVALID_HANDLE_VALUE;
+ }
+
+ if (hConsole != INVALID_HANDLE_VALUE) {
+ /* do nothing if console output is redirected to a file */
+ if (GetFileType(hConsole) == FILE_TYPE_CHAR) {
+ CONSOLE_SCREEN_BUFFER_INFO csbi;
+ GetConsoleScreenBufferInfo(hConsole, &csbi);
+ SetConsoleTextAttribute(hConsole, (WORD) attr);
+ return (int) csbi.wAttributes;
+ }
+ }
+
+ return 0;
+}
+
+#elif defined(COLOR_MODE_ANSI)
+
+static int picodbg_setTextAttr(FILE *stream, int attr)
+{
+ const char *c = "";
+
+ if (attr == -1) {
+ c = "0";
+ } else switch (attr) {
+ case ColorBlack: c = "0;30"; break;
+ case ColorRed: c = "0;31"; break;
+ case ColorGreen: c = "0;32"; break;
+ case ColorBrown: c = "0;33"; break;
+ case ColorBlue: c = "0;34"; break;
+ case ColorPurple: c = "0;35"; break;
+ case ColorCyan: c = "0;36"; break;
+ case ColorLightGray: c = "0;37"; break;
+ case ColorDarkGray: c = "1;30"; break;
+ case ColorLightRed: c = "1;31"; break;
+ case ColorLightGreen: c = "1;32"; break;
+ case ColorYellow: c = "1;33"; break;
+ case ColorLightBlue: c = "1;34"; break;
+ case ColorLightPurple: c = "1;35"; break;
+ case ColorLightCyan: c = "1;36"; break;
+ case ColorWhite: c = "1;37"; break;
+ }
+
+ fprintf(stream, "\x1b[%sm", c);
+ return -1;
+}
+
+#else
+
+static int picodbg_setTextAttr(FILE *stream, int attr)
+{
+ /* avoid 'unreferenced formal parameter' */
+ (void) stream;
+ (void) attr;
+ return 0;
+}
+
+#endif
+
+
+/* *** Auxiliary routines *****/
+
+
+static const char *picodbg_fileTitle(const char *file)
+{
+ const char *name = file, *str = file;
+
+ /* try to extract file name without path in a platform independent
+ way, i.e., skip all chars preceding path separator chars like
+ '/' (Unix, MacOSX), '\' (Windows, DOS), and ':' (MacOS9) */
+ while (*str) {
+ if ((*str == '\\') || (*str == '/') || (*str == ':')) {
+ name = str + 1;
+ }
+ str++;
+ }
+
+ return name;
+}
+
+
+static void picodbg_logToStream(int level, int donewline,
+ const char *context, const char *msg)
+{
+ int oldAttr = 0;
+
+ if (optColor) {
+ oldAttr = picodbg_setTextAttr(STDDBG, picodbg_getLevelColor(level));
+ }
+
+ fprintf(STDDBG, "%s%s", context, msg);
+ if (donewline) fprintf(STDDBG, "\n");
+ if (logFile != NULL) {
+ fprintf(logFile, "%s%s", context, msg);
+ if (donewline) fprintf(logFile, "\n");
+ }
+
+ if (optColor) {
+ picodbg_setTextAttr(STDDBG, oldAttr);
+ }
+}
+
+
+/* *** Exported routines *****/
+
+
+void picodbg_initialize(int level)
+{
+ logLevel = level;
+ strcpy(logFilterFN, PICODBG_DEFAULT_FILTERFN);
+ logFile = NULL;
+ logFormat = PICODBG_DEFAULT_FORMAT;
+ optColor = PICODBG_DEFAULT_COLOR;
+ PICODBG_ASSERT_RANGE(level, 0, PICODBG_LOG_LEVEL_TRACE);
+}
+
+
+void picodbg_terminate()
+{
+ if (logFile != NULL) {
+ fclose(logFile);
+ }
+
+ logLevel = 0;
+ logFile = NULL;
+}
+
+
+void picodbg_setLogLevel(int level)
+{
+ PICODBG_ASSERT_RANGE(level, 0, PICODBG_LOG_LEVEL_TRACE);
+ logLevel = level;
+}
+
+
+void picodbg_setLogFilterFN(const char *name)
+{
+ strcpy(logFilterFN, name);
+}
+
+
+void picodbg_setLogFile(const char *name)
+{
+ if (logFile != NULL) {
+ fclose(logFile);
+ }
+
+ if ((name != NULL) && (strlen(name) > 0)) {
+ logFile = fopen(name, "wt");
+ } else {
+ logFile = NULL;
+ }
+}
+
+
+void picodbg_enableColors(int flag)
+{
+ optColor = (flag != 0);
+}
+
+
+void picodbg_setOutputFormat(unsigned int format)
+{
+ logFormat = format;
+}
+
+
+const char *picodbg_varargs(const char *format, ...)
+{
+ int len;
+
+ va_list argptr;
+ va_start(argptr, format);
+
+ len = vsprintf(msgbuf, format, argptr);
+ PICODBG_ASSERT_RANGE(len, 0, MAX_MESSAGE_LEN);
+
+ return msgbuf;
+}
+
+
+void picodbg_log(int level, int donewline, const char *file, int line,
+ const char *func, const char *msg)
+{
+ char cb[MAX_CONTEXT_LEN + 1];
+
+ PICODBG_ASSERT_RANGE(level, 0, PICODBG_LOG_LEVEL_TRACE);
+
+ if ((level <= logLevel) &&
+ ((strlen(logFilterFN) == 0) || !strcmp(logFilterFN, picodbg_fileTitle(file)))) {
+ /* compose output format string */
+ strcpy(ctxbuf, "*** ");
+ if (logFormat & PICODBG_SHOW_LEVEL) {
+ switch (level) {
+ case PICODBG_LOG_LEVEL_ERROR:
+ strcat(ctxbuf, "error" MSG_DELIM);
+ break;
+ case PICODBG_LOG_LEVEL_WARN:
+ strcat(ctxbuf, "warn " MSG_DELIM);
+ break;
+ case PICODBG_LOG_LEVEL_INFO:
+ strcat(ctxbuf, "info " MSG_DELIM);
+ break;
+ case PICODBG_LOG_LEVEL_DEBUG:
+ strcat(ctxbuf, "debug" MSG_DELIM);
+ break;
+ case PICODBG_LOG_LEVEL_TRACE:
+ strcat(ctxbuf, "trace" MSG_DELIM);
+ break;
+ default:
+ break;
+ }
+ }
+ if (logFormat & PICODBG_SHOW_DATE) {
+ /* nyi */
+ }
+ if (logFormat & PICODBG_SHOW_TIME) {
+ /* nyi */
+ }
+ if (logFormat & PICODBG_SHOW_SRCNAME) {
+ sprintf(cb, "%-10s", picodbg_fileTitle(file));
+ strcat(ctxbuf, cb);
+ if (logFormat & PICODBG_SHOW_SRCLINE) {
+ sprintf(cb, "(%d)", line);
+ strcat(ctxbuf, cb);
+ }
+ strcat(ctxbuf, MSG_DELIM);
+ }
+ if (logFormat & PICODBG_SHOW_FUNCTION) {
+ if (strlen(func) > 0) {
+ sprintf(cb, "%-18s", func);
+ strcat(ctxbuf, cb);
+ strcat(ctxbuf, MSG_DELIM);
+ }
+ }
+
+ picodbg_logToStream(level, donewline, ctxbuf, msg);
+ }
+}
+
+
+void picodbg_log_msg(int level, const char *file, const char *msg)
+{
+ PICODBG_ASSERT_RANGE(level, 0, PICODBG_LOG_LEVEL_TRACE);
+
+ if ((level <= logLevel) &&
+ ((strlen(logFilterFN) == 0) || !strcmp(logFilterFN, picodbg_fileTitle(file)))) {
+ picodbg_logToStream(level, 0, "", msg);
+ }
+}
+
+
+void picodbg_assert(const char *file, int line, const char *func, const char *expr)
+{
+ if (strlen(func) > 0) {
+ fprintf(STDDBG, "assertion failed: %s, file %s, function %s, line %d",
+ expr, picodbg_fileTitle(file), func, line);
+ } else {
+ fprintf(STDDBG, "assertion failed: %s, file %s, line %d",
+ expr, picodbg_fileTitle(file), line);
+ }
+ picodbg_terminate();
+ abort();
+}
+
+
+#else
+
+/* To prevent warning about "translation unit is empty" when
+ diagnostic output is disabled. */
+static void picodbg_dummy(void) {
+ picodbg_dummy();
+}
+
+#endif /* defined(PICO_DEBUG) */
+
+#ifdef __cplusplus
+}
+#endif
+
+
+/* end */
diff --git a/lib/picodbg.h b/lib/picodbg.h
new file mode 100644
index 0000000..490d3b5
--- /dev/null
+++ b/lib/picodbg.h
@@ -0,0 +1,311 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ *
+ * @file picodbg.h
+ *
+ * Provides functions and macros to debug the Pico system and to trace
+ * the execution of its code.
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ */
+
+/**
+ * @addtogroup picodbg
+ * ---------------------------------------------------\n
+ * <b> Pico Debug Support </b>\n
+ * ---------------------------------------------------\n
+ * GENERAL REMARKS
+ * ---------------
+ * This module provides a set of macros to help debug the Pico system.
+ * The usage of macros allows for completely removing debug code from
+ * the binaries delivered to customers. To enable diagnostic output
+ * the preprocessor symbol PICO_DEBUG has to be defined.
+ *
+ * By using global variables (to store the current log level etc.)
+ * this module violates a basic Pico design principle!
+ *
+ * Justification:\n
+ * - going without global data would reduce the functionality
+ * of this module considerably (e.g., log level could not be
+ * changed at runtime etc.)
+ * - at the moment, the only known system interdicting global
+ * variables is Symbian; but even there global variables are
+ * possible by using thread-local storage
+ * - allocating global data on the heap would require to pass
+ * a handle to this memory block to all routines of this
+ * module which in turn implies that _every_ function in the
+ * Pico system would require a pointer to some global data;
+ * obviously, this would be very awkward
+ *
+ * Furthermore, this module uses the non-standardized but handy
+ * __FUNCTION__ macro. It expands to the name of the enclosing
+ * function. For compilers not supporting this macro simply
+ * define __FUNCTION__ as an empty string.
+ *
+ *
+ * INITIALIZATION/TERMINATION\n
+ * --------------------------\n
+ * Before using any debug macros, this module has to be initialized
+ * by calling PICODBG_INITIALIZE(). If the routines are not needed
+ * anymore, PICODBG_TERMINATE() has to be called to terminate the
+ * module (e.g., to close the log file).
+ *
+ *
+ * TRACING\n
+ * -------\n
+ * Each tracing message is associated with a log level which describes
+ * its severity. The following levels are supported:
+ * - Trace - Very detailed log messages, potentially of a high
+ * frequency and volume
+ * - Debug - Less detailed and/or less frequent debugging messages
+ * - Info - Informational messages
+ * - Warn - Warnings which don't appear to the Pico user
+ * - Error - Error messages
+ *
+ * Tracing messages can use the well-known printf format specification.
+ * But because variadic macros (macros with a variable no. arguments)
+ * are not commonly supported by compilers a little trick is used
+ * which requires the format string and its arguments to be enclosed
+ * in double parenthesis:
+ *
+ * - PICODBG_INFO(("hello, world!"));
+ * - PICODBG_TRACE(("argc=%d", argc));
+ * ...
+ *
+ * Each tracing message is expected to be a single line of text. Some
+ * contextual information (e.g., log level, time and date, source file
+ * and line number) and a newline are automatically added. The output
+ * format can be customized by a call to PICODBG_SET_OUTPUT_FORMAT().
+ *
+ * Sample output:
+ * - *** info|2008-04-03|14:51:36|dbgdemo.c(15)|hello world
+ * - *** trace|2008-04-03|14:51:36|dbgdemo.c(16)|argc=2
+ * - ...
+ *
+ * To compose a tracing message line consisting of, e.g. the elements
+ * of an array, on the Info level two additional macros shown in the
+ * following example are provided:
+ *
+ * PICODBG_INFO_CTX();\n
+ * for (i = 0; i < len; i++)\n
+ * ...some calc with arr and i\n
+ * PICODBG_INFO_MSG((" %d", arr[i]));\n
+ * }\n
+ * PICODBG_INFO_MSG(("\n"));\n
+ *
+ * Colored output of tracing messages helps to capture severe problems
+ * quickly. This feature is supported on the Windows platform and on
+ * platforms supporting ANSI escape codes. PICODBG_ENABLE_COLORS() lets
+ * you turn on and off colored output.
+ *
+ *
+ * FILTERING\n
+ * ---------\n
+ * By calling PICODBG_SET_LOG_LEVEL() the log level may be changed at
+ * any time to increase/decrease the amount of debugging output.
+ *
+ * By calling PICODBG_SET_LOG_FILTERFN() the log filter may be changed
+ * at any time to change the source file name being used as filter for
+ * log messages (ie. only tracing info of the specified file will be
+ * logged). To disable the file name based filter set the filter file
+ * name to an empty string.
+ *
+ * Future version of this module might provide further filtering
+ * possibilities (e.g., filtering based on function names * etc.).
+ *
+ *
+ * LOGGING\n
+ * -------\n
+ * By default, tracing messages are output to the console (stderr).
+ * This allows for separating diagnostic output from other console
+ * output to stdout. In addition, tracing messages may be saved to
+ * a file by calling PICODBG_SET_LOG_FILE().
+ * Currently, file output is the only additional output target; but
+ * on embedded systems, more output targets may be required (e.g.,
+ * sending output to a serial port or over the network).
+ *
+ *
+ * ASSERTIONS\n
+ * ----------\n
+ * To support the 'design/programming by contract' paradigm, this
+ * module also provides assertions. PICODBG_ASSERT(expr) evualuates
+ * an expression and, when the result is false, prints a diagnostic
+ * message and aborts the program.
+ *
+ *
+ * FUTURE EXTENSIONS\n
+ * -----------------\n
+ * - advanced tracing functions to dump complex data
+ * - debug memory allocation that can be used to assist in
+ * finding memory problems
+ */
+
+
+#if !defined(__PICODBG_H__)
+#define __PICODBG_H__
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* Not all compilers support the __FUNCTION__ macro */
+#if !defined(__FUNCTION__) && !defined(__GNUC__)
+#define __FUNCTION__ ""
+#endif
+
+
+/* Log levels sorted by severity */
+#define PICODBG_LOG_LEVEL_ERROR 1
+#define PICODBG_LOG_LEVEL_WARN 2
+#define PICODBG_LOG_LEVEL_INFO 3
+#define PICODBG_LOG_LEVEL_DEBUG 4
+#define PICODBG_LOG_LEVEL_TRACE 5
+
+/* Output format flags */
+#define PICODBG_SHOW_LEVEL 0x0001
+#define PICODBG_SHOW_DATE 0x0002
+#define PICODBG_SHOW_TIME 0x0004
+#define PICODBG_SHOW_SRCNAME 0x0008
+#define PICODBG_SHOW_SRCLINE 0x0010
+#define PICODBG_SHOW_SRCALL (PICODBG_SHOW_SRCNAME | PICODBG_SHOW_SRCLINE)
+#define PICODBG_SHOW_FUNCTION 0x0020
+#define PICODBG_SHOW_POS (PICODBG_SHOW_SRCALL | PICODBG_SHOW_FUNCTION)
+
+/* definition of PICO_DEBUG enables debugging code */
+#if defined(PICO_DEBUG)
+
+#define PICODBG_INITIALIZE(level) \
+ picodbg_initialize(level)
+
+#define PICODBG_TERMINATE() \
+ picodbg_terminate()
+
+#define PICODBG_SET_LOG_LEVEL(level) \
+ picodbg_setLogLevel(level)
+
+#define PICODBG_SET_LOG_FILTERFN(name) \
+ picodbg_setLogFilterFN(name)
+
+#define PICODBG_SET_LOG_FILE(name) \
+ picodbg_setLogFile(name)
+
+#define PICODBG_ENABLE_COLORS(flag) \
+ picodbg_enableColors(flag)
+
+#define PICODBG_SET_OUTPUT_FORMAT(format) \
+ picodbg_setOutputFormat(format)
+
+
+#define PICODBG_ASSERT(expr) \
+ for (;!(expr);picodbg_assert(__FILE__, __LINE__, __FUNCTION__, #expr))
+
+#define PICODBG_ASSERT_RANGE(val, min, max) \
+ PICODBG_ASSERT(((val) >= (min)) && ((val) <= (max)))
+
+
+#define PICODBG_LOG(level, msg) \
+ picodbg_log(level, 1, __FILE__, __LINE__, __FUNCTION__, picodbg_varargs msg)
+
+#define PICODBG_ERROR(msg) \
+ PICODBG_LOG(PICODBG_LOG_LEVEL_ERROR, msg)
+
+#define PICODBG_WARN(msg) \
+ PICODBG_LOG(PICODBG_LOG_LEVEL_WARN, msg)
+
+#define PICODBG_INFO(msg) \
+ PICODBG_LOG(PICODBG_LOG_LEVEL_INFO, msg)
+
+#define PICODBG_DEBUG(msg) \
+ PICODBG_LOG(PICODBG_LOG_LEVEL_DEBUG, msg)
+
+#define PICODBG_TRACE(msg) \
+ PICODBG_LOG(PICODBG_LOG_LEVEL_TRACE, msg)
+
+
+#define PICODBG_INFO_CTX() \
+ picodbg_log(PICODBG_LOG_LEVEL_INFO, 0, __FILE__, __LINE__, __FUNCTION__, "")
+
+#define PICODBG_INFO_MSG(msg) \
+ picodbg_log_msg(PICODBG_LOG_LEVEL_INFO, __FILE__, picodbg_varargs msg)
+
+#define PICODBG_INFO_MSG_F(filterfn, msg) \
+ picodbg_log_msg(PICODBG_LOG_LEVEL_INFO, (const char *)filterfn, picodbg_varargs msg)
+
+
+
+/* helper routines; should NOT be used directly! */
+
+void picodbg_initialize(int level);
+void picodbg_terminate();
+
+void picodbg_setLogLevel(int level);
+void picodbg_setLogFilterFN(const char *name);
+void picodbg_setLogFile(const char *name);
+void picodbg_enableColors(int flag);
+void picodbg_setOutputFormat(unsigned int format);
+
+const char *picodbg_varargs(const char *format, ...);
+
+void picodbg_log(int level, int donewline, const char *file, int line,
+ const char *func, const char *msg);
+void picodbg_assert(const char *file, int line, const char *func,
+ const char *expr);
+
+void picodbg_log_msg(int level, const char *file, const char *msg);
+
+
+#else /* release version; omit debugging code */
+
+#define PICODBG_INITIALIZE(level)
+#define PICODBG_TERMINATE()
+#define PICODBG_SET_LOG_LEVEL(level)
+#define PICODBG_SET_LOG_FILTERFN(name)
+#define PICODBG_SET_LOG_FILE(name)
+#define PICODBG_ENABLE_COLORS(flag)
+#define PICODBG_SET_OUTPUT_FORMAT(format)
+
+#define PICODBG_ASSERT(expr)
+#define PICODBG_ASSERT_RANGE(val, min, max)
+
+#define PICODBG_LOG(level, msg)
+#define PICODBG_ERROR(msg)
+#define PICODBG_WARN(msg)
+#define PICODBG_INFO(msg)
+#define PICODBG_DEBUG(msg)
+#define PICODBG_TRACE(msg)
+
+#define PICODBG_INFO_CTX()
+#define PICODBG_INFO_MSG(msg)
+#define PICODBG_INFO_MSG_F(filterfn, msg)
+
+
+#endif /* defined(PICO_DEBUG) */
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif /* !defined(__PICODBG_H__) */
diff --git a/lib/picodefs.h b/lib/picodefs.h
new file mode 100644
index 0000000..6613e84
--- /dev/null
+++ b/lib/picodefs.h
@@ -0,0 +1,173 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picodefs.h
+ *
+ * SVOX Pico definitions
+ * (SVOX Pico version 1.0 and later)
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ */
+
+
+#ifndef PICODEFS_H_
+#define PICODEFS_H_
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+
+/* ********************************************************************/
+/* SVOX Pico limits */
+/* ********************************************************************/
+/* maximum size of a voice name, including the terminating null
+ character */
+#define PICO_MAX_VOICE_NAME_SIZE 32
+
+/* maximum size of a resource name, incl. the terminating null
+ character */
+#define PICO_MAX_RESOURCE_NAME_SIZE 32
+
+/* maximum size of a data path name, incl. the terminating null
+ character */
+#define PICO_MAX_DATAPATH_NAME_SIZE 128
+
+/* maximum size of a file name, incl. the terminating null
+ character */
+#define PICO_MAX_FILE_NAME_SIZE 64
+
+/* maximum number of resources */
+#define PICO_MAX_NUM_RESOURCES 64
+
+/* maximum number of voice definitions */
+#define PICO_MAX_NUM_VOICE_DEFINITIONS 64
+
+/* maximum number of resources per voice */
+#define PICO_MAX_NUM_RSRC_PER_VOICE 16
+
+/* maximum length of foreign header prepended to PICO resource files
+ (header length must be a multiple of 4 bytes) */
+#define PICO_MAX_FOREIGN_HEADER_LEN 64
+
+
+
+/* ********************************************************************/
+/* SVOX PICO status codes */
+/* ********************************************************************/
+
+typedef signed int pico_Status;
+
+
+/* Okay ***************************************************************/
+/* functions return PICO_OK if all is okay */
+
+#define PICO_OK (pico_Status) 0
+
+
+/* Exceptions and error codes *****************************************/
+
+/* in case of exceptional events and errors (e.g. unexpected user
+ input) that disrupt the normal flow of operation, PICO_EXC_* or
+ PICO_ERR_* are returned. */
+
+#define PICO_EXC_NUMBER_FORMAT (pico_Status) -10
+#define PICO_EXC_MAX_NUM_EXCEED (pico_Status) -11
+#define PICO_EXC_NAME_CONFLICT (pico_Status) -12
+#define PICO_EXC_NAME_UNDEFINED (pico_Status) -13
+#define PICO_EXC_NAME_ILLEGAL (pico_Status) -14
+
+/* buffer interaction */
+#define PICO_EXC_BUF_OVERFLOW (pico_Status) -20
+#define PICO_EXC_BUF_UNDERFLOW (pico_Status) -21
+#define PICO_EXC_BUF_IGNORE (pico_Status) -22
+
+/* internal memory handling */
+#define PICO_EXC_OUT_OF_MEM (pico_Status) -30
+
+/* files */
+#define PICO_EXC_CANT_OPEN_FILE (pico_Status) -40
+#define PICO_EXC_UNEXPECTED_FILE_TYPE (pico_Status) -41
+#define PICO_EXC_FILE_CORRUPT (pico_Status) -42
+#define PICO_EXC_FILE_NOT_FOUND (pico_Status) -43
+
+/* resources */
+#define PICO_EXC_RESOURCE_BUSY (pico_Status) -50
+#define PICO_EXC_RESOURCE_MISSING (pico_Status) -51
+
+/* knowledge bases */
+#define PICO_EXC_KB_MISSING (pico_Status) -60
+
+/* runtime exceptions */
+#define PICO_ERR_NULLPTR_ACCESS (pico_Status) -100
+#define PICO_ERR_INVALID_HANDLE (pico_Status) -101
+#define PICO_ERR_INVALID_ARGUMENT (pico_Status) -102
+#define PICO_ERR_INDEX_OUT_OF_RANGE (pico_Status) -103
+
+/* other errors ("external" errors, e.g. hardware failure). */
+#define PICO_ERR_OTHER (pico_Status) -999
+
+
+/* Warnings ***********************************************************/
+
+/* general */
+#define PICO_WARN_INCOMPLETE (pico_Status) 10
+#define PICO_WARN_FALLBACK (pico_Status) 11
+#define PICO_WARN_OTHER (pico_Status) 19
+
+/* resources */
+#define PICO_WARN_KB_OVERWRITE (pico_Status) 50
+#define PICO_WARN_RESOURCE_DOUBLE_LOAD (pico_Status) 51
+
+/* classifiers */
+#define PICO_WARN_INVECTOR (pico_Status) 60
+#define PICO_WARN_CLASSIFICATION (pico_Status) 61
+#define PICO_WARN_OUTVECTOR (pico_Status) 62
+
+/* processing units */
+#define PICO_WARN_PU_IRREG_ITEM (pico_Status) 70
+#define PICO_WARN_PU_DISCARD_BUF (pico_Status) 71
+
+
+
+/* ********************************************************************/
+/* Engine getData return values */
+/* ********************************************************************/
+
+#define PICO_STEP_IDLE (pico_Status) 200
+#define PICO_STEP_BUSY (pico_Status) 201
+
+#define PICO_STEP_ERROR (pico_Status) -200
+
+
+
+/* ********************************************************************/
+/* Engine getData outDataType values */
+/* ********************************************************************/
+
+/* 16 bit PCM samples, native endianness of platform */
+#define PICO_DATA_PCM_16BIT (pico_Int16) 1
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif /*PICODEFS_H_*/
diff --git a/lib/picodsp.h b/lib/picodsp.h
new file mode 100644
index 0000000..67457f8
--- /dev/null
+++ b/lib/picodsp.h
@@ -0,0 +1,112 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picodsp.h
+ *
+ * Include file for DSP related data types and constants in Pico
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#ifndef PICODSP_H_
+#define PICODSP_H_
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+/*----------------------------CONSTANTS ----------------------*/
+/*Normalization factors used at the start and at the end of the sig*/
+#define PICODSP_START_FLOAT_NORM 0.41f
+#define PICODSP_ENVSPEC_K1 0.5f
+#define PICODSP_ENVSPEC_K2 2
+#define PICODSP_GETEXC_K1 1024
+#define PICODSP_FIXRESP_NORM 4096.0f
+#define PICODSP_END_FLOAT_NORM 1.5f*16.0f
+#define PICODSP_FIX_SCALE1 0x4000000
+#define PICODSP_FIX_SCALE2 0x4000
+#define PICODSP_SHIFT_FACT1 10
+#define PICODSP_SHIFT_FACT2 16
+#define PICODSP_SHIFT_FACT3 12
+#define PICODSP_SHIFT_FACT4 1
+#define PICODSP_SHIFT_FACT5 18
+#define PICODSP_SHIFT_FACT6 9
+#define PICOSIG_NORM1 9.14f /100.0f /*normalization factor*/
+#define PICOSIG_MAXAMP (32767)
+#define PICOSIG_MINAMP (-32768)
+#define PICODSP_M_PI 3.14159265358979323846
+#define PICODSP_MAX_EX 32
+#define PICODSP_WGT_SHIFT (0x20000000)
+#define PICODSP_N_RAND_TABLE (760)
+#define PICODSP_COS_TABLE_LEN (512)
+#define PICODSP_COS_TABLE_LEN2 (1024)
+#define PICODSP_COS_TABLE_LEN4 (2048)
+#define PICODSP_PI_SHIFT (4) /* -log2(PICODSP_COS_TABLE_LEN2/0x4000) */
+
+#define PICODSP_V_CUTOFF_FREQ 4500
+#define PICODSP_UV_CUTOFF_FREQ 300
+#define PICODSP_SAMP_FREQ 16000
+#define PICODSP_FREQ_WARP_FACT 0.42f
+
+/*----------------------------CEP/PHASE CONSTANTS----------------------------*/
+#define PICODSP_CEPORDER 25
+#define PICODSP_PHASEORDER 72
+#define CEPST_BUFF_SIZE 3
+#define PHASE_BUFF_SIZE 5
+/*----------------------------FFT CONSTANTS----------------------------*/
+#define PICODSP_FFTSIZE (256)
+
+#define PICODSP_H_FFTSIZE (PICODSP_FFTSIZE/2)
+
+#define PICODSP_DISPLACE PICODSP_FFTSIZE/4
+
+#define PICODSP_H_FFTSIZE (PICODSP_FFTSIZE/2)
+#define PICODSP_HFFTSIZE_P1 (PICODSP_H_FFTSIZE+1)
+
+#define FAST_DEVICE(aCount, aAction) \
+{ \
+ int count_ = (aCount); \
+ int times_ = (count_ + 7) >> 3; \
+ switch (count_ & 7){ \
+ case 0: do { aAction; \
+ case 7: aAction; \
+ case 6: aAction; \
+ case 5: aAction; \
+ case 4: aAction; \
+ case 3: aAction; \
+ case 2: aAction; \
+ case 1: aAction; \
+ } while (--times_ > 0); \
+} \
+}
+/*------------------------------------------------------------------------------------------
+ Fast Exp Approximation now remapped to a function in picoos
+ -----------------------------------------------------------------------------------------*/
+#define EXP(y) picoos_quick_exp(y)
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /*PICODSP_H_*/
diff --git a/lib/picoextapi.c b/lib/picoextapi.c
new file mode 100644
index 0000000..4c5d30c
--- /dev/null
+++ b/lib/picoextapi.c
@@ -0,0 +1,228 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picoextapi.c
+ *
+ * API extension for development use
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+#include "picodefs.h"
+#include "picoos.h"
+#include "picoctrl.h"
+#include "picodbg.h"
+#include "picoapi.h"
+#include "picoextapi.h"
+#include "picoapid.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+/* this is not used anymore. For the picosh banner, set
+ * progv.progVers in picosh.c instead. */
+#define PICO_VERSION_INFO (picoos_char *)"invalid"
+
+
+extern pico_Status pico_initialize_priv(
+ void *memory,
+ const pico_Uint32 size,
+ pico_Int16 enableMemProt,
+ pico_System *system);
+
+
+/* System initialization and termination functions ****************************/
+
+
+PICO_FUNC picoext_initialize(
+ void *memory,
+ const pico_Uint32 size,
+ pico_Int16 enableMemProt,
+ pico_System *outSystem
+ )
+{
+ return pico_initialize_priv(memory, size, enableMemProt, outSystem);
+}
+
+
+/* System and lingware inspection functions ***********************************/
+
+/* @todo : not supported yet */
+PICO_FUNC picoext_getVersionInfo(
+ pico_Retstring outInfo,
+ const pico_Int16 outInfoMaxLen
+ )
+{
+ if (outInfo == NULL) {
+ return PICO_ERR_NULLPTR_ACCESS;
+ }
+ picoos_strlcpy((picoos_char *) outInfo, PICO_VERSION_INFO, outInfoMaxLen);
+ return PICO_OK;
+}
+
+
+/* Debugging/testing support functions *****************************************/
+
+
+PICO_FUNC picoext_setTraceLevel(
+ pico_System system,
+ pico_Int32 level
+ )
+{
+ if (NULL == system) {
+ return PICO_ERR_NULLPTR_ACCESS;
+ }
+ if (level < 0) {
+ level = 0;
+ }
+ if (level > PICODBG_LOG_LEVEL_TRACE) {
+ level = PICODBG_LOG_LEVEL_TRACE;
+ }
+ PICODBG_SET_LOG_LEVEL(level);
+ return PICO_OK;
+}
+
+
+PICO_FUNC picoext_setTraceFilterFN(
+ pico_System system,
+ const pico_Char *name
+ )
+{
+
+ if (NULL == system) {
+ return PICO_ERR_NULLPTR_ACCESS;
+ }
+ name = name; /*PP 13.10.08 : fix warning "var not used in this function"*/
+ PICODBG_SET_LOG_FILTERFN((const char *)name);
+ return PICO_OK;
+}
+
+
+PICO_FUNC picoext_setLogFile(
+ pico_System system,
+ const pico_Char *name
+ )
+{
+ if (NULL == system) {
+ return PICO_ERR_NULLPTR_ACCESS;
+ }
+ name = name; /*PP 13.10.08 : fix warning "var not used in this function"*/
+ PICODBG_SET_LOG_FILE((const char *) name);
+ return PICO_OK;
+}
+
+
+/* Memory usage ***************************************************************/
+
+
+pico_Status getMemUsage(
+ picoos_Common common,
+ picoos_bool resetIncremental,
+ picoos_int32 *usedBytes,
+ picoos_int32 *incrUsedBytes,
+ picoos_int32 *maxUsedBytes
+ )
+{
+ pico_Status status = PICO_OK;
+
+ if (common == NULL) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else {
+ picoos_emReset(common->em);
+ picoos_getMemUsage(common->mm, resetIncremental, usedBytes, incrUsedBytes, maxUsedBytes);
+ status = picoos_emGetExceptionCode(common->em);
+ }
+
+ return status;
+}
+
+
+PICO_FUNC picoext_getSystemMemUsage(
+ pico_System system,
+ pico_Int16 resetIncremental,
+ pico_Int32 *outUsedBytes,
+ pico_Int32 *outIncrUsedBytes,
+ pico_Int32 *outMaxUsedBytes
+ )
+{
+ pico_Status status = PICO_OK;
+
+ if (!is_valid_system_handle(system)) {
+ status = PICO_ERR_INVALID_HANDLE;
+ } else if ((outUsedBytes == NULL) || (outIncrUsedBytes == NULL) || (outMaxUsedBytes == NULL)) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else {
+ picoos_Common common = pico_sysGetCommon(system);
+ status = getMemUsage(common, resetIncremental != 0, outUsedBytes, outIncrUsedBytes, outMaxUsedBytes);
+ }
+
+ return status;
+}
+
+
+PICO_FUNC picoext_getEngineMemUsage(
+ pico_Engine engine,
+ pico_Int16 resetIncremental,
+ pico_Int32 *outUsedBytes,
+ pico_Int32 *outIncrUsedBytes,
+ pico_Int32 *outMaxUsedBytes
+ )
+{
+ pico_Status status = PICO_OK;
+
+ if (!picoctrl_isValidEngineHandle((picoctrl_Engine) engine)) {
+ status = PICO_ERR_INVALID_HANDLE;
+ } else if ((outUsedBytes == NULL) || (outIncrUsedBytes == NULL) || (outMaxUsedBytes == NULL)) {
+ status = PICO_ERR_NULLPTR_ACCESS;
+ } else {
+ picoos_Common common = picoctrl_engGetCommon((picoctrl_Engine) engine);
+ status = getMemUsage(common, resetIncremental != 0, outUsedBytes, outIncrUsedBytes, outMaxUsedBytes);
+ }
+
+ return status;
+}
+
+PICO_FUNC picoext_getLastScheduledPU(
+ pico_Engine engine
+ )
+{
+ pico_Status status = PICO_OK;
+ status = picoctrl_getLastScheduledPU((picoctrl_Engine) engine);
+ return status;
+}
+
+PICO_FUNC picoext_getLastProducedItemType(
+ pico_Engine engine
+ )
+{
+ pico_Status status = PICO_OK;
+ status = picoctrl_getLastProducedItemType((picoctrl_Engine) engine);
+ return status;
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+/* end */
diff --git a/lib/picoextapi.h b/lib/picoextapi.h
new file mode 100644
index 0000000..1768394
--- /dev/null
+++ b/lib/picoextapi.h
@@ -0,0 +1,163 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picoextapi.h
+ *
+ * API extensions for development use
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#ifndef PICOEXTAPI_H_
+#define PICOEXTAPI_H_
+
+#include "picodefs.h"
+#include "picodbg.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* ****************************************************************************/
+/* Things that might be added to picoapi later but should not appear there */
+/* for the time being */
+/* ****************************************************************************/
+
+/* String type for Unicode text input *****************************************/
+
+/* Unicode encodings supported by PICO. */
+
+#define PICO_STRENC_UTF8 0
+#define PICO_STRENC_UTF16 1
+
+/* An UTF-8 string must point to a byte array, terminated by a null character
+ ('\0'). An UTF-16 string must point to a contiguous array of 16-bit units
+ (in native byte ordering), terminated by a 0. */
+
+typedef char *PICO_STRING_UTF8;
+typedef pico_Uint16 *PICO_STRING_UTF16;
+
+/* Generic pointer to a Unicode string, encoded either as UTF-8 or UTF-16.
+ The application must make sure that for each 'PICO_STRING_PTR' it provides
+ an argument of type 'PICO_STRING_UTF8' or 'PICO_STRING_UTF16' (or of a type
+ compatible to one of these types). */
+
+typedef void *PICO_STRING_PTR;
+
+
+/* ****************************************************************************/
+/* System-level API functions */
+/* ****************************************************************************/
+
+/* System initialization and termination functions ****************************/
+
+/* Same as pico_initialize, but allows to enable memory protection
+ functionality for testing purposes (enableMemProt != 0). */
+
+PICO_FUNC picoext_initialize(
+ void *memory,
+ const pico_Uint32 size,
+ pico_Int16 enableMemProt,
+ pico_System *outSystem
+ );
+
+
+/* System and lingware inspection functions ***********************************/
+
+/* Returns version information of the current Pico engine. */
+
+PICO_FUNC picoext_getVersionInfo(
+ pico_Retstring outInfo,
+ const pico_Int16 outInfoMaxLen
+ );
+
+/* Returns unique resource name */
+
+/*
+PICO_FUNC picoext_getResourceName(
+ pico_Resource resource,
+ pico_Retstring outInfo
+ );
+*/
+
+/* Debugging/testing support functions *****************************************/
+
+/* Sets tracing level. Increasing amounts of information is displayed
+ at each level. */
+
+PICO_FUNC picoext_setTraceLevel(
+ pico_System system,
+ pico_Int32 level
+ );
+
+/* Sets trace filtering. Limits tracing output to tracing information
+ resulting from the source file name being filtered. */
+
+PICO_FUNC picoext_setTraceFilterFN(
+ pico_System system,
+ const pico_Char *name
+ );
+
+/* Enables logging of debug output to log file 'name'. If 'name' is NULL
+ or an empty string, logging is disabled. */
+
+PICO_FUNC picoext_setLogFile(
+ pico_System system,
+ const pico_Char *name
+ );
+
+
+/* Memory usage ***************************************************************/
+
+PICO_FUNC picoext_getSystemMemUsage(
+ pico_System system,
+ pico_Int16 resetIncremental,
+ pico_Int32 *outUsedBytes,
+ pico_Int32 *outIncrUsedBytes,
+ pico_Int32 *outMaxUsedBytes
+ );
+
+PICO_FUNC picoext_getEngineMemUsage(
+ pico_Engine engine,
+ pico_Int16 resetIncremental,
+ pico_Int32 *outUsedBytes,
+ pico_Int32 *outIncrUsedBytes,
+ pico_Int32 *outMaxUsedBytes
+ );
+
+PICO_FUNC picoext_getLastScheduledPU(
+ pico_Engine engine
+ );
+
+PICO_FUNC picoext_getLastProducedItemType(
+ pico_Engine engine
+ );
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif /* PICOEXTAPI_H_ */
diff --git a/lib/picofftsg.c b/lib/picofftsg.c
new file mode 100644
index 0000000..dcca018
--- /dev/null
+++ b/lib/picofftsg.c
@@ -0,0 +1,3274 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picofftsg.c
+ *
+ * FFT/DCT related data types, constants and functions in Pico
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+*/
+
+#include "picoos.h"
+#include "picofftsg.h"
+#include "picodbg.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/**
+ * @addtogroup picofft
+ * ---------------------------------------------------\n
+ * <b> Fast Fourier/Cosine/Sine Transform </b>\n
+ * Adapted from http://www.kurims.kyoto-u.ac.jp/~ooura/fft.html (Copyright Takuya OOURA, 1996-2001)\n
+ * ---------------------------------------------------\n
+
+ Overall features
+ - dimension :one
+ - data length :power of 2
+ - decimation :frequency
+ - radix :split-radix
+ - data :inplace
+ - table :not use
+
+ functions
+ - cdft: Complex Discrete Fourier Transform
+ - rdft: Real Discrete Fourier Transform
+ - ddct: Discrete Cosine Transform
+ - ddst: Discrete Sine Transform
+ - dfct: Cosine Transform of RDFT (Real Symmetric DFT)
+ - dfst: Sine Transform of RDFT (Real Anti-symmetric DFT)
+
+ function prototypes
+ - void cdft(picoos_int32, picoos_int32, PICOFFTSG_FFTTYPE *);
+ - void rdft(picoos_int32, picoos_int32, PICOFFTSG_FFTTYPE *);
+ - void ddct(picoos_int32, picoos_int32, PICOFFTSG_FFTTYPE *);
+ - void ddst(picoos_int32, picoos_int32, PICOFFTSG_FFTTYPE *);
+ - void dfct(picoos_int32, PICOFFTSG_FFTTYPE *);
+ - void dfst(picoos_int32, PICOFFTSG_FFTTYPE *);
+
+ <b>Complex DFT (Discrete Fourier Transform)</b>
+
+ [definition]
+ - <case1>
+ - X[k] = sum_j=0^n-1 x[j]*exp(2*pi*i*j*k/n), 0<=k<n
+ - <case2>
+ - X[k] = sum_j=0^n-1 x[j]*exp(-2*pi*i*j*k/n), 0<=k<n
+ - (notes: sum_j=0^n-1 is a summation from j=0 to n-1)
+
+ [usage]
+ - <case1>
+ - cdft(2*n, 1, a);
+ - <case2>
+ - cdft(2*n, -1, a);
+
+ [parameters]
+ - 2*n :data length (picoos_int32)
+ - n >= 1, n = power of 2
+ - a[0...2*n-1] :input/output data (PICOFFTSG_FFTTYPE *)
+ - input data
+ - a[2*j] = Re(x[j]),
+ - a[2*j+1] = Im(x[j]), 0<=j<n
+ - output data
+ - a[2*k] = Re(X[k]),
+ - a[2*k+1] = Im(X[k]), 0<=k<n
+
+ [remark]
+ - Inverse of cdft(2*n, -1, a); is
+ -cdft(2*n, 1, a);
+ - for (j = 0; j <= 2 * n - 1; j++) {
+ - a[j] *= 1.0 / n;
+ - }
+
+
+ <b> Real DFT / Inverse of Real DFT </b>
+
+ [definition]
+ - <case1> RDFT
+ - R[k] = sum_j=0^n-1 a[j]*cos(2*pi*j*k/n), 0<=k<=n/2
+ - I[k] = sum_j=0^n-1 a[j]*sin(2*pi*j*k/n), 0<k<n/2
+ - <case2> IRDFT (excluding scale)
+ - a[k] = (R[0] + R[n/2]*cos(pi*k))/2 +
+ - sum_j=1^n/2-1 R[j]*cos(2*pi*j*k/n) +
+ - sum_j=1^n/2-1 I[j]*sin(2*pi*j*k/n), 0<=k<n
+
+ [usage]
+ - <case1>
+ - rdft(n, 1, a);
+ - <case2>
+ - rdft(n, -1, a);
+
+ [parameters]
+ - n :data length (picoos_int32)
+ - n >= 2, n = power of 2
+ - a[0...n-1] :input/output data (PICOFFTSG_FFTTYPE *)
+ - <case1>
+ - output data
+ - a[2*k] = R[k], 0<=k<n/2
+ - a[2*k+1] = I[k], 0<k<n/2
+ - a[1] = R[n/2]
+ - <case2>
+ - input data
+ - a[2*j] = R[j], 0<=j<n/2
+ - a[2*j+1] = I[j], 0<j<n/2
+ - a[1] = R[n/2]
+
+ [remark]
+ - Inverse of rdft(n, 1, a); is
+ - rdft(n, -1, a);
+ - for (j = 0; j <= n - 1; j++) {
+ - a[j] *= 2.0 / n;
+ -}
+
+
+ <b> DCT (Discrete Cosine Transform) / Inverse of DCT</b>
+
+ [definition]
+ - <case1> IDCT (excluding scale)
+ - C[k] = sum_j=0^n-1 a[j]*cos(pi*j*(k+1/2)/n), 0<=k<n
+ - <case2> DCT
+ - C[k] = sum_j=0^n-1 a[j]*cos(pi*(j+1/2)*k/n), 0<=k<n
+
+ [usage]
+ - <case1>
+ - ddct(n, 1, a);
+ - <case2>
+ - ddct(n, -1, a);
+
+ [parameters]
+ - n :data length (picoos_int32)
+ - n >= 2, n = power of 2
+ - a[0...n-1] :input/output data (PICOFFTSG_FFTTYPE *)
+ - output data
+ - a[k] = C[k], 0<=k<n
+
+ [remark]
+ - Inverse of ddct(n, -1, a); is
+ - a[0] *= 0.5;
+ - ddct(n, 1, a);
+ - for (j = 0; j <= n - 1; j++) {
+ - a[j] *= 2.0 / n;
+ - }
+
+ <b> DST (Discrete Sine Transform) / Inverse of DST</b>
+
+ [definition]
+ - <case1> IDST (excluding scale)
+ - S[k] = sum_j=1^n A[j]*sin(pi*j*(k+1/2)/n), 0<=k<n
+ - <case2> DST
+ - S[k] = sum_j=0^n-1 a[j]*sin(pi*(j+1/2)*k/n), 0<k<=n
+
+ [usage]
+ - <case1>
+ - ddst(n, 1, a);
+ - <case2>
+ - ddst(n, -1, a);
+
+ [parameters]
+ - n :data length (picoos_int32)
+ - n >= 2, n = power of 2
+ - a[0...n-1] :input/output data (PICOFFTSG_FFTTYPE *)
+ - <case1>
+ - input data
+ - a[j] = A[j], 0<j<n
+ - a[0] = A[n]
+ - output data
+ - a[k] = S[k], 0<=k<n
+ - <case2>
+ - output data
+ - a[k] = S[k], 0<k<n
+ - a[0] = S[n]
+
+ [remark]
+ - Inverse of ddst(n, -1, a); is
+ - a[0] *= 0.5;
+ - ddst(n, 1, a);
+ - for (j = 0; j <= n - 1; j++) {
+ - a[j] *= 2.0 / n;
+ - }
+
+ <b> Cosine Transform of RDFT (Real Symmetric DFT)</b>
+
+ [definition]
+ - C[k] = sum_j=0^n a[j]*cos(pi*j*k/n), 0<=k<=n
+
+ [usage]
+ - dfct(n, a);
+
+ [parameters]
+ - n :data length - 1 (picoos_int32)
+ - n >= 2, n = power of 2
+ - a[0...n] :input/output data (PICOFFTSG_FFTTYPE *)
+
+ - output data
+ - a[k] = C[k], 0<=k<=n
+
+ [remark]
+ - Inverse of a[0] *= 0.5; a[n] *= 0.5; dfct(n, a); is
+ - a[0] *= 0.5;
+ - a[n] *= 0.5;
+ - dfct(n, a);
+ - for (j = 0; j <= n; j++) {
+ - a[j] *= 2.0 / n;
+ - }
+
+ <b> Sine Transform of RDFT (Real Anti-symmetric DFT)</b>
+
+ [definition]
+ - S[k] = sum_j=1^n-1 a[j]*sin(pi*j*k/n), 0<k<n
+
+ [usage]
+ - dfst(n, a);
+
+ [parameters]
+ - n :data length + 1 (picoos_int32)
+ - n >= 2, n = power of 2
+ - a[0...n-1] :input/output data (PICOFFTSG_FFTTYPE *)
+ - output data
+ - a[k] = S[k], 0<k<n
+ - (a[0] is used for work area)
+
+ [remark]
+ - Inverse of dfst(n, a); is
+ - dfst(n, a);
+ - for (j = 1; j <= n - 1; j++) {
+ - a[j] *= 2.0 / n;
+ - }
+
+*/
+
+/* fixed point multiplier for weights */
+#define PICODSP_WGT_SHIFT (0x20000000) /* 2^29 */
+#define PICOFFTSG_WGT_SHIFT2 (0x10000000) /* PICODSP_WGT_SHIFT/2 */
+#define PICOFFTSG_WGT_N_SHIFT (29) /* 2^29 */
+/* fixed point known constants */
+#ifndef WR5000 /* cos(M_PI_2*0.5000) */
+#define WR5000 (PICOFFTSG_FFTTYPE)(0.707106781186547524400844362104849039284835937688*PICODSP_WGT_SHIFT)
+#ifndef WR2500 /* cos(M_PI_2*0.2500) */
+#define WR2500 (PICOFFTSG_FFTTYPE)(0.923879532511286756128183189396788286822416625863*PICODSP_WGT_SHIFT)
+#endif
+#ifndef WI2500 /* sin(M_PI_2*0.2500) */
+#define WI2500 (PICOFFTSG_FFTTYPE)(0.382683432365089771728459984030398866761344562485*PICODSP_WGT_SHIFT)
+#endif
+#ifndef WR1250 /* cos(M_PI_2*0.1250) */
+#define WR1250 (PICOFFTSG_FFTTYPE)(0.980785280403230449126182236134239036973933730893*PICODSP_WGT_SHIFT)
+#endif
+#ifndef WI1250 /* sin(M_PI_2*0.1250) */
+#define WI1250 (PICOFFTSG_FFTTYPE)(0.195090322016128267848284868477022240927691617751*PICODSP_WGT_SHIFT)
+#endif
+#ifndef WR3750 /* cos(M_PI_2*0.3750) */
+#define WR3750 (PICOFFTSG_FFTTYPE)(0.831469612302545237078788377617905756738560811987*PICODSP_WGT_SHIFT)
+#endif
+#ifndef WI3750 /* sin(M_PI_2*0.3750) */
+#define WI3750 (PICOFFTSG_FFTTYPE)(0.555570233019602224742830813948532874374937190754*PICODSP_WGT_SHIFT)
+#endif
+
+#else
+
+ /*#ifndef M_PI_2
+ #define M_PI_2 (PICOFFTSG_FFTTYPE)1.570796326794896619231321691639751442098584699687
+ #endif*/
+#ifndef WR5000 /* cos(M_PI_2*0.5000) */
+#define WR5000 (PICOFFTSG_FFTTYPE)0.707106781186547524400844362104849039284835937688
+#endif
+#ifndef WR2500 /* cos(M_PI_2*0.2500) */
+#define WR2500 (PICOFFTSG_FFTTYPE)0.923879532511286756128183189396788286822416625863
+#endif
+#ifndef WI2500 /* sin(M_PI_2*0.2500) */
+#define WI2500 (PICOFFTSG_FFTTYPE)0.382683432365089771728459984030398866761344562485
+#endif
+#ifndef WR1250 /* cos(M_PI_2*0.1250) */
+#define WR1250 (PICOFFTSG_FFTTYPE)0.980785280403230449126182236134239036973933730893
+#endif
+#ifndef WI1250 /* sin(M_PI_2*0.1250) */
+#define WI1250 (PICOFFTSG_FFTTYPE)0.195090322016128267848284868477022240927691617751
+#endif
+#ifndef WR3750 /* cos(M_PI_2*0.3750) */
+#define WR3750 (PICOFFTSG_FFTTYPE)0.831469612302545237078788377617905756738560811987
+#endif
+#ifndef WI3750 /* sin(M_PI_2*0.3750) */
+#define WI3750 (PICOFFTSG_FFTTYPE)0.555570233019602224742830813948532874374937190754
+#endif
+
+#endif
+
+#ifndef CDFT_LOOP_DIV /* control of the CDFT's speed & tolerance */
+#define CDFT_LOOP_DIV 32
+#define CDFT_LOOP_DIV_4 128
+#endif
+
+#ifndef RDFT_LOOP_DIV /* control of the RDFT's speed & tolerance */
+#define RDFT_LOOP_DIV 64
+#define RDFT_LOOP_DIV4 256
+#define RDFT_LOOP_DIV_4 256
+#endif
+
+#ifndef DCST_LOOP_DIV /* control of the DCT,DST's speed & tolerance */
+#define DCST_LOOP_DIV 64
+#define DCST_LOOP_DIV2 128
+#endif
+
+
+#define POW1 (0x1)
+#define POW2 (0x2)
+#define POW3 (0x4)
+#define POW4 (0x8)
+#define POW5 (0x10)
+#define POW6 (0x20)
+#define POW7 (0x40)
+#define POW8 (0x80)
+#define POW9 (0x100)
+#define POW10 (0x200)
+#define POW11 (0x400)
+#define POW12 (0x800)
+#define POW13 (0x1000)
+#define POW14 (0x2000)
+#define POW15 (0x4000)
+#define POW16 (0x8000)
+#define POW17 (0x10000)
+#define POW18 (0x20000)
+#define POW19 (0x40000)
+#define POW20 (0x80000)
+#define POW21 (0x100000)
+#define POW22 (0x200000)
+#define POW23 (0x400000)
+#define POW24 (0x800000)
+#define POW25 (0x1000000)
+#define POW26 (0x2000000)
+#define POW27 (0x4000000)
+#define POW28 (0x8000000)
+#define POW29 (0x10000000)
+#define POW30 (0x20000000)
+#define POW31 (0x40000000)
+/**************
+ * useful macros
+ ************** */
+#define picofftsg_highestBitPos(x) (x==0?0:(x>=POW17?(x>=POW25?(x>=POW29?(x>=POW31?31:(x>=POW30?30:29)):(x>=POW27?(x>=POW28?28:27):(x>=POW26?26:25))):(x>=POW21?(x>=POW23?(x>=POW24?24:23):(x>=POW22?22:21)):(x>=POW19?(x>=POW20?20:19):(x>=POW18?18:17)))):(x>=POW9?(x>=POW13?(x>=POW15?(x>=POW16?16:15):(x>=POW14?14:13)):(x>=POW11?(x>=POW12?12:11):(x>=POW10?10:9))):(x>=POW5?(x>=POW7?(x>=POW8?8:7):(x>=POW6?6:5)):(x>=POW3?(x>=POW4?4:3):(x>=POW2?2:1))))))
+#define picofftsg_highestBit(x) (x==0?0:(x<0?(zz=-x,(zz>=POW17?(zz>=POW25?(zz>=POW29?(zz>=POW31?31:(zz>=POW30?30:29)):(zz>=POW27?(zz>=POW28?28:27):(zz>=POW26?26:25))):(zz>=POW21?(zz>=POW23?(zz>=POW24?24:23):(zz>=POW22?22:21)):(zz>=POW19?(zz>=POW20?20:19):(zz>=POW18?18:17)))):(zz>=POW9?(zz>=POW13?(zz>=POW15?(zz>=POW16?16:15):(zz>=POW14?14:13)):(zz>=POW11?(zz>=POW12?12:11):(zz>=POW10?10:9))):(zz>=POW5?(zz>=POW7?(zz>=POW8?8:7):(zz>=POW6?6:5)):(zz>=POW3?(zz>=POW4?4:3):(zz>=POW2?2:1)))))):(x>=POW17?(x>=POW25?(x>=POW29?(x>=POW31?31:(x>=POW30?30:29)):(x>=POW27?(x>=POW28?28:27):(x>=POW26?26:25))):(x>=POW21?(x>=POW23?(x>=POW24?24:23):(x>=POW22?22:21)):(x>=POW19?(x>=POW20?20:19):(x>=POW18?18:17)))):(x>=POW9?(x>=POW13?(x>=POW15?(x>=POW16?16:15):(x>=POW14?14:13)):(x>=POW11?(x>=POW12?12:11):(x>=POW10?10:9))):(x>=POW5?(x>=POW7?(x>=POW8?8:7):(x>=POW6?6:5)):(x>=POW3?(x>=POW4?4:3):(x>=POW2?2:1)))))))
+#define Mult_W_W picofftsg_mult_w_w
+
+
+/* ***********************************************************************************************/
+/* forward declarations */
+/* ***********************************************************************************************/
+static PICOFFTSG_FFTTYPE picofftsg_mult_w_w(PICOFFTSG_FFTTYPE x1, PICOFFTSG_FFTTYPE y1);
+static PICOFFTSG_FFTTYPE picofftsg_mult_w_a(PICOFFTSG_FFTTYPE x1, PICOFFTSG_FFTTYPE y1);
+
+
+static void cftfsub(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+static void cftbsub(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+static void rftfsub(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+static void rftbsub(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+
+static void cftfsub(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+static void cftbsub(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+static void rftfsub(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+static void rftbsub(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+static void dctsub(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+static void dctsub4(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+
+static void ddct(picoos_int32 n, picoos_int32 isgn, PICOFFTSG_FFTTYPE *a);
+static void bitrv1(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+
+static void bitrv2(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+static void bitrv216(PICOFFTSG_FFTTYPE *a);
+static void bitrv208(PICOFFTSG_FFTTYPE *a);
+static void cftmdl1(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+static void cftrec4(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+static void cftleaf(picoos_int32 n, picoos_int32 isplt, PICOFFTSG_FFTTYPE *a);
+static void cftfx41(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+static void cftf161(PICOFFTSG_FFTTYPE *a);
+static void cftf081(PICOFFTSG_FFTTYPE *a);
+static void cftf040(PICOFFTSG_FFTTYPE *a);
+static void cftx020(PICOFFTSG_FFTTYPE *a);
+
+void bitrv2conj(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+void bitrv216neg(PICOFFTSG_FFTTYPE *a);
+void bitrv208neg(PICOFFTSG_FFTTYPE *a);
+void cftb1st(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+void cftrec4(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+void cftleaf(picoos_int32 n, picoos_int32 isplt, PICOFFTSG_FFTTYPE *a);
+void cftfx41(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+void cftf161(PICOFFTSG_FFTTYPE *a);
+void cftf081(PICOFFTSG_FFTTYPE *a);
+void cftb040(PICOFFTSG_FFTTYPE *a);
+void cftx020(PICOFFTSG_FFTTYPE *a);
+
+static picoos_int32 cfttree(picoos_int32 n, picoos_int32 j, picoos_int32 k, PICOFFTSG_FFTTYPE *a);
+static void cftleaf(picoos_int32 n, picoos_int32 isplt, PICOFFTSG_FFTTYPE *a);
+static void cftmdl1(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+
+static void cftmdl1(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+static void cftmdl2(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+
+static void cftmdl1(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+static void cftmdl2(picoos_int32 n, PICOFFTSG_FFTTYPE *a);
+static void cftf161(PICOFFTSG_FFTTYPE *a);
+static void cftf162(PICOFFTSG_FFTTYPE *a);
+static void cftf081(PICOFFTSG_FFTTYPE *a);
+static void cftf082(PICOFFTSG_FFTTYPE *a);
+
+static void cftf161(PICOFFTSG_FFTTYPE *a);
+static void cftf162(PICOFFTSG_FFTTYPE *a);
+static void cftf081(PICOFFTSG_FFTTYPE *a);
+static void cftf082(PICOFFTSG_FFTTYPE *a);
+
+/* ***********************************************************************************************/
+/* Exported functions */
+/* ***********************************************************************************************/
+void rdft(picoos_int32 n, picoos_int32 isgn, PICOFFTSG_FFTTYPE *a)
+{
+ PICOFFTSG_FFTTYPE xi;
+
+ if (isgn >= 0) {
+ if (n > 4) {
+ cftfsub(n, a);
+ rftfsub(n, a);
+ } else if (n == 4) {
+ cftfsub(n, a);
+ }
+ xi = a[0] - a[1];
+ a[0] += a[1];
+ a[1] = xi;
+ } else {
+ a[1] = (a[0] - a[1]) / 2;
+ a[0] -= a[1];
+ if (n > 4) {
+ rftbsub(n, a);
+ cftbsub(n, a);
+ } else if (n == 4) {
+ cftbsub(n, a);
+ }
+ }
+
+}
+
+
+picoos_single norm_result(picoos_int32 m2, PICOFFTSG_FFTTYPE *tmpX, PICOFFTSG_FFTTYPE *norm_window)
+{
+ picoos_int16 nI;
+ PICOFFTSG_FFTTYPE a,b, E;
+
+ E = (picoos_int32)0;
+ for (nI=0; nI<m2; nI++) {
+ a = (norm_window[nI]>>18) * ((tmpX[nI]>0) ? tmpX[nI]>>11 : -((-tmpX[nI])>>11));
+ tmpX[nI] = a;
+ b = (a>=0?a:-a) >> 18;
+ E += (b*b);
+ }
+
+ if (E>0) {
+ return (picoos_single)sqrt((double)E/16.0)/m2;
+ }
+ else {
+ return 0.0;
+ }
+}
+
+void ddct(picoos_int32 n, picoos_int32 isgn, PICOFFTSG_FFTTYPE *a)
+{
+ picoos_int32 j;
+ PICOFFTSG_FFTTYPE xr;
+
+ if (isgn < 0) {
+ xr = a[n - 1];
+ for (j = n - 2; j >= 2; j -= 2) {
+ a[j + 1] = a[j] - a[j - 1];
+ a[j] += a[j - 1];
+ }
+ a[1] = a[0] - xr;
+ a[0] += xr;
+ if (n > 4) {
+ rftbsub(n, a);
+ cftbsub(n, a);
+ } else if (n == 4) {
+ cftbsub(n, a);
+ }
+ }
+ if (n > 4) {
+ dctsub(n, a);
+ } else {
+ dctsub4(n, a);
+ }
+ if (isgn >= 0) {
+ if (n > 4) {
+ cftfsub(n, a);
+ rftfsub(n, a);
+ } else if (n == 4) {
+ cftfsub(n, a);
+ }
+
+
+ xr = a[0] - a[1];
+ a[0] += a[1];
+ for (j = 2; j < n; j += 2) {
+ a[j - 1] = a[j] - a[j + 1];
+ a[j] += a[j + 1];
+ }
+ a[n - 1] = xr;
+ }
+}
+
+void dfct_nmf(picoos_int32 n, picoos_int32 *a)
+{
+ picoos_int32 j, k, m, mh;
+ PICOFFTSG_FFTTYPE xr, xi, yr, yi, an;
+ PICOFFTSG_FFTTYPE *aj, *ak, *amj, *amk;
+
+ m = n >> 1;
+ for (j = 0; j < m; j++) {
+ k = n - j;
+ xr = a[j] + a[k];
+ a[j] -= a[k];
+ a[k] = xr;
+ }
+ an = a[n];
+ while (m >= 2) {
+ ddct(m, 1, a);
+ if (m > 2) {
+ bitrv1(m, a);
+ }
+ mh = m >> 1;
+ xi = a[m];
+ a[m] = a[0];
+ a[0] = an - xi;
+ an += xi;
+ k = m-1;
+ aj = a + 1; ak = a + k; amj = aj + m; amk = ak + m;
+ for (j = 1; j < mh; j++, aj++, ak--, amj++, amk--) {
+ xr = *amk;
+ xi = *amj;
+ yr = *aj;
+ yi = *ak;
+ *amj = yr;
+ *amk = yi;
+ *aj = xr - xi;
+ *ak = xr + xi;
+ }
+ xr = *aj;
+ *aj = *amj;
+ *amj = xr;
+
+ m = mh;
+ }
+
+ xi = a[1];
+ a[1] = a[0];
+ a[0] = an + xi;
+ a[n] = an - xi;
+ if (n > 2) {
+ bitrv1(n, a);
+ }
+
+}
+
+/* ***********************************************************************************************/
+/* internal routines */
+/* ***********************************************************************************************/
+/*
+ mult two numbers which are guaranteed to be in the range -1..1
+ shift right as little as possible before mult, and the rest after the mult
+ Also, shift bigger number more - lose less accuracy
+ */
+static PICOFFTSG_FFTTYPE picofftsg_mult_w_w(PICOFFTSG_FFTTYPE x1, PICOFFTSG_FFTTYPE y1)
+{
+ PICOFFTSG_FFTTYPE x, y;
+ x = x1>=0 ? x1>>15 : -((-x1)>>15);
+ y = y1>=0 ? y1>>14 : -((-y1)>>14);
+ return x * y;
+}
+
+static PICOFFTSG_FFTTYPE picofftsg_mult_w_a(PICOFFTSG_FFTTYPE x1, PICOFFTSG_FFTTYPE y1)
+{
+ PICOFFTSG_FFTTYPE x, y;
+
+
+ x = x1>=0 ? x1>>15 : -((-x1)>>15);
+ y = y1>=0 ? y1>>14 : -((-y1)>>14);
+ return x * y;
+}
+
+static void cftfsub(picoos_int32 n, PICOFFTSG_FFTTYPE *a)
+{
+
+ if (n > 8) {
+ if (n > 32) {
+ cftmdl1(n, a);
+ if (n > 512) {
+ cftrec4(n, a);
+ } else if (n > 128) {
+ cftleaf(n, 1, a);
+ } else {
+ cftfx41(n, a);
+ }
+ bitrv2(n, a);
+ } else if (n == 32) {
+ cftf161(a);
+ bitrv216(a);
+ } else {
+ cftf081(a);
+ bitrv208(a);
+ }
+ } else if (n == 8) {
+ cftf040(a);
+ } else if (n == 4) {
+ cftx020(a);
+ }
+}
+
+
+void cftbsub(picoos_int32 n, PICOFFTSG_FFTTYPE *a)
+{
+ if (n > 8) {
+ if (n > 32) {
+ cftb1st(n, a);
+ if (n > 512) {
+ cftrec4(n, a);
+ } else if (n > 128) {
+ cftleaf(n, 1, a);
+ } else {
+ cftfx41(n, a);
+ }
+ bitrv2conj(n, a);
+ } else if (n == 32) {
+ cftf161(a);
+ bitrv216neg(a);
+ } else {
+ cftf081(a);
+ bitrv208neg(a);
+ }
+ } else if (n == 8) {
+ cftb040(a);
+ } else if (n == 4) {
+ cftx020(a);
+ }
+}
+
+/* **************************************************************************************************/
+
+/* **************************************************************************************************/
+void bitrv2(picoos_int32 n, PICOFFTSG_FFTTYPE *a)
+{
+ picoos_int32 j0, k0, j1, k1, l, m, i, j, k, nh, m2;
+ PICOFFTSG_FFTTYPE xr, xi, yr, yi;
+
+ m = 4;
+ for (l = n >> 2; l > 8; l >>= 2) {
+ m <<= 1;
+ }
+ m2 = m + m;
+ nh = n >> 1;
+ if (l == 8) {
+ j0 = 0;
+ for (k0 = 0; k0 < m; k0 += 4) {
+ k = k0;
+ for (j = j0; j < j0 + k0; j += 4) {
+ xr = a[j];
+ xi = a[j + 1];
+ yr = a[k];
+ yi = a[k + 1];
+ a[j] = yr;
+ a[j + 1] = yi;
+ a[k] = xr;
+ a[k + 1] = xi;
+ j1 = j + m;
+ k1 = k + m2;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += m;
+ k1 -= m;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += m;
+ k1 += m2;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += nh;
+ k1 += 2;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= m;
+ k1 -= m2;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= m;
+ k1 += m;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= m;
+ k1 -= m2;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += 2;
+ k1 += nh;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += m;
+ k1 += m2;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += m;
+ k1 -= m;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += m;
+ k1 += m2;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= nh;
+ k1 -= 2;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= m;
+ k1 -= m2;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= m;
+ k1 += m;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= m;
+ k1 -= m2;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ for (i = nh >> 1; i > (k ^= i); i >>= 1) {
+ /* Avoid warning*/
+ };
+ }
+ k1 = j0 + k0;
+ j1 = k1 + 2;
+ k1 += nh;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += m;
+ k1 += m2;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += m;
+ k1 -= m;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= 2;
+ k1 -= nh;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += nh + 2;
+ k1 += nh + 2;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= nh - m;
+ k1 += m2 - 2;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ for (i = nh >> 1; i > (j0 ^= i); i >>= 1) {
+ /* Avoid warning */
+ }
+ }
+ } else {
+ j0 = 0;
+ for (k0 = 0; k0 < m; k0 += 4) {
+ k = k0;
+ for (j = j0; j < j0 + k0; j += 4) {
+ xr = a[j];
+ xi = a[j + 1];
+ yr = a[k];
+ yi = a[k + 1];
+ a[j] = yr;
+ a[j + 1] = yi;
+ a[k] = xr;
+ a[k + 1] = xi;
+ j1 = j + m;
+ k1 = k + m;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += nh;
+ k1 += 2;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= m;
+ k1 -= m;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += 2;
+ k1 += nh;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += m;
+ k1 += m;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= nh;
+ k1 -= 2;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= m;
+ k1 -= m;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ for (i = nh >> 1; i > (k ^= i); i >>= 1){
+ /* Avoid warning */
+ }
+ }
+ k1 = j0 + k0;
+ j1 = k1 + 2;
+ k1 += nh;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += m;
+ k1 += m;
+ xr = a[j1];
+ xi = a[j1 + 1];
+ yr = a[k1];
+ yi = a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ for (i = nh >> 1; i > (j0 ^= i); i >>= 1){
+ /* Avoid warning */
+ }
+ }
+ }
+}
+
+
+void bitrv2conj(picoos_int32 n, PICOFFTSG_FFTTYPE *a)
+{
+ picoos_int32 j0, k0, j1, k1, l, m, i, j, k, nh, m2;
+ PICOFFTSG_FFTTYPE xr, xi, yr, yi;
+
+
+ m = 4;
+ for (l = n >> 2; l > 8; l >>= 2) {
+ m <<= 1;
+ }
+ m2 = m + m;
+ nh = n >> 1;
+ if (l == 8) {
+ j0 = 0;
+ for (k0 = 0; k0 < m; k0 += 4) {
+ k = k0;
+ for (j = j0; j < j0 + k0; j += 4) {
+ xr = a[j];
+ xi = -a[j + 1];
+ yr = a[k];
+ yi = -a[k + 1];
+ a[j] = yr;
+ a[j + 1] = yi;
+ a[k] = xr;
+ a[k + 1] = xi;
+ j1 = j + m;
+ k1 = k + m2;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += m;
+ k1 -= m;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += m;
+ k1 += m2;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += nh;
+ k1 += 2;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= m;
+ k1 -= m2;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= m;
+ k1 += m;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= m;
+ k1 -= m2;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += 2;
+ k1 += nh;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += m;
+ k1 += m2;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += m;
+ k1 -= m;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += m;
+ k1 += m2;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= nh;
+ k1 -= 2;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= m;
+ k1 -= m2;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= m;
+ k1 += m;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= m;
+ k1 -= m2;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ for (i = nh >> 1; i > (k ^= i); i >>= 1) {
+ /* Avoid warning */
+ }
+ }
+ k1 = j0 + k0;
+ j1 = k1 + 2;
+ k1 += nh;
+ a[j1 - 1] = -a[j1 - 1];
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ a[k1 + 3] = -a[k1 + 3];
+ j1 += m;
+ k1 += m2;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += m;
+ k1 -= m;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= 2;
+ k1 -= nh;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += nh + 2;
+ k1 += nh + 2;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= nh - m;
+ k1 += m2 - 2;
+ a[j1 - 1] = -a[j1 - 1];
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ a[k1 + 3] = -a[k1 + 3];
+ for (i = nh >> 1; i > (j0 ^= i); i >>= 1) { /* Avoid warning*/
+
+ }
+ }
+ } else {
+ j0 = 0;
+ for (k0 = 0; k0 < m; k0 += 4) {
+ k = k0;
+ for (j = j0; j < j0 + k0; j += 4) {
+ xr = a[j];
+ xi = -a[j + 1];
+ yr = a[k];
+ yi = -a[k + 1];
+ a[j] = yr;
+ a[j + 1] = yi;
+ a[k] = xr;
+ a[k + 1] = xi;
+ j1 = j + m;
+ k1 = k + m;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += nh;
+ k1 += 2;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= m;
+ k1 -= m;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += 2;
+ k1 += nh;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 += m;
+ k1 += m;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= nh;
+ k1 -= 2;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ j1 -= m;
+ k1 -= m;
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ for (i = nh >> 1; i > (k ^= i); i >>= 1) {
+ /* Avoid warning*/
+ }
+ }
+ k1 = j0 + k0;
+ j1 = k1 + 2;
+ k1 += nh;
+ a[j1 - 1] = -a[j1 - 1];
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ a[k1 + 3] = -a[k1 + 3];
+ j1 += m;
+ k1 += m;
+ a[j1 - 1] = -a[j1 - 1];
+ xr = a[j1];
+ xi = -a[j1 + 1];
+ yr = a[k1];
+ yi = -a[k1 + 1];
+ a[j1] = yr;
+ a[j1 + 1] = yi;
+ a[k1] = xr;
+ a[k1 + 1] = xi;
+ a[k1 + 3] = -a[k1 + 3];
+ for (i = nh >> 1; i > (j0 ^= i); i >>= 1) {
+ /* Avoid warning*/
+ }
+ }
+ }
+}
+
+
+void bitrv216(PICOFFTSG_FFTTYPE *a)
+{
+ PICOFFTSG_FFTTYPE x1r, x1i, x2r, x2i, x3r, x3i, x4r, x4i,
+ x5r, x5i, x7r, x7i, x8r, x8i, x10r, x10i,
+ x11r, x11i, x12r, x12i, x13r, x13i, x14r, x14i;
+
+ x1r = a[2];
+ x1i = a[3];
+ x2r = a[4];
+ x2i = a[5];
+ x3r = a[6];
+ x3i = a[7];
+ x4r = a[8];
+ x4i = a[9];
+ x5r = a[10];
+ x5i = a[11];
+ x7r = a[14];
+ x7i = a[15];
+ x8r = a[16];
+ x8i = a[17];
+ x10r = a[20];
+ x10i = a[21];
+ x11r = a[22];
+ x11i = a[23];
+ x12r = a[24];
+ x12i = a[25];
+ x13r = a[26];
+ x13i = a[27];
+ x14r = a[28];
+ x14i = a[29];
+ a[2] = x8r;
+ a[3] = x8i;
+ a[4] = x4r;
+ a[5] = x4i;
+ a[6] = x12r;
+ a[7] = x12i;
+ a[8] = x2r;
+ a[9] = x2i;
+ a[10] = x10r;
+ a[11] = x10i;
+ a[14] = x14r;
+ a[15] = x14i;
+ a[16] = x1r;
+ a[17] = x1i;
+ a[20] = x5r;
+ a[21] = x5i;
+ a[22] = x13r;
+ a[23] = x13i;
+ a[24] = x3r;
+ a[25] = x3i;
+ a[26] = x11r;
+ a[27] = x11i;
+ a[28] = x7r;
+ a[29] = x7i;
+}
+
+
+void bitrv216neg(PICOFFTSG_FFTTYPE *a)
+{
+ PICOFFTSG_FFTTYPE x1r, x1i, x2r, x2i, x3r, x3i, x4r, x4i,
+ x5r, x5i, x6r, x6i, x7r, x7i, x8r, x8i,
+ x9r, x9i, x10r, x10i, x11r, x11i, x12r, x12i,
+ x13r, x13i, x14r, x14i, x15r, x15i;
+
+ x1r = a[2];
+ x1i = a[3];
+ x2r = a[4];
+ x2i = a[5];
+ x3r = a[6];
+ x3i = a[7];
+ x4r = a[8];
+ x4i = a[9];
+ x5r = a[10];
+ x5i = a[11];
+ x6r = a[12];
+ x6i = a[13];
+ x7r = a[14];
+ x7i = a[15];
+ x8r = a[16];
+ x8i = a[17];
+ x9r = a[18];
+ x9i = a[19];
+ x10r = a[20];
+ x10i = a[21];
+ x11r = a[22];
+ x11i = a[23];
+ x12r = a[24];
+ x12i = a[25];
+ x13r = a[26];
+ x13i = a[27];
+ x14r = a[28];
+ x14i = a[29];
+ x15r = a[30];
+ x15i = a[31];
+ a[2] = x15r;
+ a[3] = x15i;
+ a[4] = x7r;
+ a[5] = x7i;
+ a[6] = x11r;
+ a[7] = x11i;
+ a[8] = x3r;
+ a[9] = x3i;
+ a[10] = x13r;
+ a[11] = x13i;
+ a[12] = x5r;
+ a[13] = x5i;
+ a[14] = x9r;
+ a[15] = x9i;
+ a[16] = x1r;
+ a[17] = x1i;
+ a[18] = x14r;
+ a[19] = x14i;
+ a[20] = x6r;
+ a[21] = x6i;
+ a[22] = x10r;
+ a[23] = x10i;
+ a[24] = x2r;
+ a[25] = x2i;
+ a[26] = x12r;
+ a[27] = x12i;
+ a[28] = x4r;
+ a[29] = x4i;
+ a[30] = x8r;
+ a[31] = x8i;
+}
+
+
+void bitrv208(PICOFFTSG_FFTTYPE *a)
+{
+ PICOFFTSG_FFTTYPE x1r, x1i, x3r, x3i, x4r, x4i, x6r, x6i;
+
+ x1r = a[2];
+ x1i = a[3];
+ x3r = a[6];
+ x3i = a[7];
+ x4r = a[8];
+ x4i = a[9];
+ x6r = a[12];
+ x6i = a[13];
+ a[2] = x4r;
+ a[3] = x4i;
+ a[6] = x6r;
+ a[7] = x6i;
+ a[8] = x1r;
+ a[9] = x1i;
+ a[12] = x3r;
+ a[13] = x3i;
+}
+
+
+void bitrv208neg(PICOFFTSG_FFTTYPE *a)
+{
+ PICOFFTSG_FFTTYPE x1r, x1i, x2r, x2i, x3r, x3i, x4r, x4i,
+ x5r, x5i, x6r, x6i, x7r, x7i;
+
+ x1r = a[2];
+ x1i = a[3];
+ x2r = a[4];
+ x2i = a[5];
+ x3r = a[6];
+ x3i = a[7];
+ x4r = a[8];
+ x4i = a[9];
+ x5r = a[10];
+ x5i = a[11];
+ x6r = a[12];
+ x6i = a[13];
+ x7r = a[14];
+ x7i = a[15];
+ a[2] = x7r;
+ a[3] = x7i;
+ a[4] = x3r;
+ a[5] = x3i;
+ a[6] = x5r;
+ a[7] = x5i;
+ a[8] = x1r;
+ a[9] = x1i;
+ a[10] = x6r;
+ a[11] = x6i;
+ a[12] = x2r;
+ a[13] = x2i;
+ a[14] = x4r;
+ a[15] = x4i;
+}
+
+
+void bitrv1(picoos_int32 n, PICOFFTSG_FFTTYPE *a)
+{
+ picoos_int32 j0, k0, j1, k1, l, m, i, j, k, nh;
+ PICOFFTSG_FFTTYPE x;
+ nh = n >> 1;
+ x = a[1];
+ a[1] = a[nh];
+ a[nh] = x;
+ m = 2;
+ for (l = n >> 2; l > 2; l >>= 2) {
+ m <<= 1;
+ }
+ if (l == 2) {
+ j1 = m + 1;
+ k1 = m + nh;
+ x = a[j1];
+ a[j1] = a[k1];
+ a[k1] = x;
+ j0 = 0;
+ for (k0 = 2; k0 < m; k0 += 2) {
+ for (i = nh >> 1; i > (j0 ^= i); i >>= 1) {
+ /* Avoid warning*/
+ }
+ k = k0;
+ for (j = j0; j < j0 + k0; j += 2) {
+ x = a[j];
+ a[j] = a[k];
+ a[k] = x;
+ j1 = j + m;
+ k1 = k + m;
+ x = a[j1];
+ a[j1] = a[k1];
+ a[k1] = x;
+ j1 += nh;
+ k1++;
+ x = a[j1];
+ a[j1] = a[k1];
+ a[k1] = x;
+ j1 -= m;
+ k1 -= m;
+ x = a[j1];
+ a[j1] = a[k1];
+ a[k1] = x;
+ j1++;
+ k1 += nh;
+ x = a[j1];
+ a[j1] = a[k1];
+ a[k1] = x;
+ j1 += m;
+ k1 += m;
+ x = a[j1];
+ a[j1] = a[k1];
+ a[k1] = x;
+ j1 -= nh;
+ k1--;
+ x = a[j1];
+ a[j1] = a[k1];
+ a[k1] = x;
+ j1 -= m;
+ k1 -= m;
+ x = a[j1];
+ a[j1] = a[k1];
+ a[k1] = x;
+ for (i = nh >> 1; i > (k ^= i); i >>= 1) {
+ /* Avoid warning*/
+ }
+ }
+ k1 = j0 + k0;
+ j1 = k1 + 1;
+ k1 += nh;
+ x = a[j1];
+ a[j1] = a[k1];
+ a[k1] = x;
+ j1 += m;
+ k1 += m;
+ x = a[j1];
+ a[j1] = a[k1];
+ a[k1] = x;
+ }
+ } else {
+ j0 = 0;
+ for (k0 = 2; k0 < m; k0 += 2) {
+ for (i = nh >> 1; i > (j0 ^= i); i >>= 1) {
+ /* Avoid warning*/
+ }
+ k = k0;
+ for (j = j0; j < j0 + k0; j += 2) {
+ x = a[j];
+ a[j] = a[k];
+ a[k] = x;
+ j1 = j + nh;
+ k1 = k + 1;
+ x = a[j1];
+ a[j1] = a[k1];
+ a[k1] = x;
+ j1++;
+ k1 += nh;
+ x = a[j1];
+ a[j1] = a[k1];
+ a[k1] = x;
+ j1 -= nh;
+ k1--;
+ x = a[j1];
+ a[j1] = a[k1];
+ a[k1] = x;
+ for (i = nh >> 1; i > (k ^= i); i >>= 1) {
+ /* Avoid warning*/
+ }
+ }
+ k1 = j0 + k0;
+ j1 = k1 + 1;
+ k1 += nh;
+ x = a[j1];
+ a[j1] = a[k1];
+ a[k1] = x;
+ }
+ }
+}
+
+
+/* **************************************************************************************************/
+
+/* **************************************************************************************************/
+
+void cftb1st(picoos_int32 n, PICOFFTSG_FFTTYPE *a)
+{
+ picoos_int32 i, i0, j, j0, j1, j2, j3, m, mh;
+ PICOFFTSG_FFTTYPE wk1r, wk1i, wk3r, wk3i,
+ wd1r, wd1i, wd3r, wd3i, ss1, ss3;
+ PICOFFTSG_FFTTYPE x0r, x0i, x1r, x1i, x2r, x2i, x3r, x3i;
+
+ mh = n >> 3;
+ m = 2 * mh;
+ j1 = m;
+ j2 = j1 + m;
+ j3 = j2 + m;
+ x0r = a[0] + a[j2];
+ x0i = -a[1] - a[j2 + 1];
+ x1r = a[0] - a[j2];
+ x1i = -a[1] + a[j2 + 1];
+ x2r = a[j1] + a[j3];
+ x2i = a[j1 + 1] + a[j3 + 1];
+ x3r = a[j1] - a[j3];
+ x3i = a[j1 + 1] - a[j3 + 1];
+ a[0] = x0r + x2r;
+ a[1] = x0i - x2i;
+ a[j1] = x0r - x2r;
+ a[j1 + 1] = x0i + x2i;
+ a[j2] = x1r + x3i;
+ a[j2 + 1] = x1i + x3r;
+ a[j3] = x1r - x3i;
+ a[j3 + 1] = x1i - x3r;
+ wd1r = PICODSP_WGT_SHIFT;
+ wd1i = 0;
+ wd3r = PICODSP_WGT_SHIFT;
+ wd3i = 0;
+
+ wk1r = (PICOFFTSG_FFTTYPE) (0.998795449734 *PICODSP_WGT_SHIFT);
+ wk1i = (PICOFFTSG_FFTTYPE) (0.049067676067 *PICODSP_WGT_SHIFT);
+ ss1 = (PICOFFTSG_FFTTYPE) (0.098135352135 *PICODSP_WGT_SHIFT);
+ wk3i = (PICOFFTSG_FFTTYPE) (-0.146730467677 *PICODSP_WGT_SHIFT);
+ wk3r = (PICOFFTSG_FFTTYPE) (0.989176511765 *PICODSP_WGT_SHIFT);
+ ss3 = (PICOFFTSG_FFTTYPE) (-0.293460935354 *PICODSP_WGT_SHIFT);
+
+ i = 0;
+ for (;;) {
+ i0 = i + CDFT_LOOP_DIV_4;
+ if (i0 > mh - 4) {
+ i0 = mh - 4;
+ }
+ for (j = i + 2; j < i0; j += 4) {
+
+ wd1r -= Mult_W_W(ss1, wk1i);
+ wd1i += Mult_W_W(ss1, wk1r);
+ wd3r -= Mult_W_W(ss3, wk3i);
+ wd3i += Mult_W_W(ss3, wk3r);
+
+ j1 = j + m;
+ j2 = j1 + m;
+ j3 = j2 + m;
+ x0r = a[j] + a[j2];
+ x0i = -a[j + 1] - a[j2 + 1];
+ x1r = a[j] - a[j2];
+ x1i = -a[j + 1] + a[j2 + 1];
+ x2r = a[j1] + a[j3];
+ x2i = a[j1 + 1] + a[j3 + 1];
+ x3r = a[j1] - a[j3];
+ x3i = a[j1 + 1] - a[j3 + 1];
+ a[j] = x0r + x2r;
+ a[j + 1] = x0i - x2i;
+ a[j1] = x0r - x2r;
+ a[j1 + 1] = x0i + x2i;
+ x0r = x1r + x3i;
+ x0i = x1i + x3r;
+ a[j2] = Mult_W_W(wk1r, x0r) - Mult_W_W(wk1i, x0i);
+ a[j2 + 1] = Mult_W_W(wk1r, x0i) + Mult_W_W(wk1i, x0r);
+ x0r = x1r - x3i;
+ x0i = x1i - x3r;
+ a[j3] = Mult_W_W(wk3r, x0r) + Mult_W_W(wk3i, x0i);
+ a[j3 + 1] = Mult_W_W(wk3r, x0i) - Mult_W_W(wk3i, x0r);
+ x0r = a[j + 2] + a[j2 + 2];
+ x0i = -a[j + 3] - a[j2 + 3];
+ x1r = a[j + 2] - a[j2 + 2];
+ x1i = -a[j + 3] + a[j2 + 3];
+ x2r = a[j1 + 2] + a[j3 + 2];
+ x2i = a[j1 + 3] + a[j3 + 3];
+ x3r = a[j1 + 2] - a[j3 + 2];
+ x3i = a[j1 + 3] - a[j3 + 3];
+ a[j + 2] = x0r + x2r;
+ a[j + 3] = x0i - x2i;
+ a[j1 + 2] = x0r - x2r;
+ a[j1 + 3] = x0i + x2i;
+ x0r = x1r + x3i;
+ x0i = x1i + x3r;
+ a[j2 + 2] = Mult_W_W(wd1r, x0r) - Mult_W_W(wd1i, x0i);
+ a[j2 + 3] = Mult_W_W(wd1r, x0i) + Mult_W_W(wd1i, x0r);
+ x0r = x1r - x3i;
+ x0i = x1i - x3r;
+ a[j3 + 2] = Mult_W_W(wd3r, x0r) + Mult_W_W(wd3i, x0i);
+ a[j3 + 3] = Mult_W_W(wd3r, x0i) - Mult_W_W(wd3i, x0r);
+ j0 = m - j;
+ j1 = j0 + m;
+ j2 = j1 + m;
+ j3 = j2 + m;
+ x0r = a[j0] + a[j2];
+ x0i = -a[j0 + 1] - a[j2 + 1];
+ x1r = a[j0] - a[j2];
+ x1i = -a[j0 + 1] + a[j2 + 1];
+ x2r = a[j1] + a[j3];
+ x2i = a[j1 + 1] + a[j3 + 1];
+ x3r = a[j1] - a[j3];
+ x3i = a[j1 + 1] - a[j3 + 1];
+ a[j0] = x0r + x2r;
+ a[j0 + 1] = x0i - x2i;
+ a[j1] = x0r - x2r;
+ a[j1 + 1] = x0i + x2i;
+ x0r = x1r + x3i;
+ x0i = x1i + x3r;
+ a[j2] = Mult_W_W(wk1i, x0r) - Mult_W_W(wk1r, x0i);
+ a[j2 + 1] = Mult_W_W(wk1i, x0i) + Mult_W_W(wk1r, x0r);
+ x0r = x1r - x3i;
+ x0i = x1i - x3r;
+ a[j3] = Mult_W_W(wk3i, x0r) + Mult_W_W(wk3r, x0i);
+ a[j3 + 1] = Mult_W_W(wk3i, x0i) - Mult_W_W(wk3r, x0r);
+ x0r = a[j0 - 2] + a[j2 - 2];
+ x0i = -a[j0 - 1] - a[j2 - 1];
+ x1r = a[j0 - 2] - a[j2 - 2];
+ x1i = -a[j0 - 1] + a[j2 - 1];
+ x2r = a[j1 - 2] + a[j3 - 2];
+ x2i = a[j1 - 1] + a[j3 - 1];
+ x3r = a[j1 - 2] - a[j3 - 2];
+ x3i = a[j1 - 1] - a[j3 - 1];
+ a[j0 - 2] = x0r + x2r;
+ a[j0 - 1] = x0i - x2i;
+ a[j1 - 2] = x0r - x2r;
+ a[j1 - 1] = x0i + x2i;
+ x0r = x1r + x3i;
+ x0i = x1i + x3r;
+ a[j2 - 2] = Mult_W_W(wd1i, x0r) - Mult_W_W(wd1r, x0i);
+ a[j2 - 1] = Mult_W_W(wd1i, x0i) + Mult_W_W(wd1r, x0r);
+ x0r = x1r - x3i;
+ x0i = x1i - x3r;
+ a[j3 - 2] = Mult_W_W(wd3i, x0r) + Mult_W_W(wd3r, x0i);
+ a[j3 - 1] = Mult_W_W(wd3i, x0i) - Mult_W_W(wd3r, x0r);
+ wk1r -= Mult_W_W(ss1, wd1i);
+ wk1i += Mult_W_W(ss1, wd1r);
+ wk3r -= Mult_W_W(ss3, wd3i);
+ wk3i += Mult_W_W(ss3, wd3r);
+ }
+ if (i0 == mh - 4) {
+ break;
+ }
+ }
+ wd1r = WR5000;
+ j0 = mh;
+ j1 = j0 + m;
+ j2 = j1 + m;
+ j3 = j2 + m;
+ x0r = a[j0 - 2] + a[j2 - 2];
+ x0i = -a[j0 - 1] - a[j2 - 1];
+ x1r = a[j0 - 2] - a[j2 - 2];
+ x1i = -a[j0 - 1] + a[j2 - 1];
+ x2r = a[j1 - 2] + a[j3 - 2];
+ x2i = a[j1 - 1] + a[j3 - 1];
+ x3r = a[j1 - 2] - a[j3 - 2];
+ x3i = a[j1 - 1] - a[j3 - 1];
+ a[j0 - 2] = x0r + x2r;
+ a[j0 - 1] = x0i - x2i;
+ a[j1 - 2] = x0r - x2r;
+ a[j1 - 1] = x0i + x2i;
+ x0r = x1r + x3i;
+ x0i = x1i + x3r;
+ a[j2 - 2] = Mult_W_W(wk1r, x0r) - Mult_W_W(wk1i, x0i);
+ a[j2 - 1] = Mult_W_W(wk1r, x0i) + Mult_W_W(wk1i, x0r);
+ x0r = x1r - x3i;
+ x0i = x1i - x3r;
+ a[j3 - 2] = Mult_W_W(wk3r, x0r) + Mult_W_W(wk3i, x0i);
+ a[j3 - 1] = Mult_W_W(wk3r, x0i) - Mult_W_W(wk3i, x0r);
+ x0r = a[j0] + a[j2];
+ x0i = -a[j0 + 1] - a[j2 + 1];
+ x1r = a[j0] - a[j2];
+ x1i = -a[j0 + 1] + a[j2 + 1];
+ x2r = a[j1] + a[j3];
+ x2i = a[j1 + 1] + a[j3 + 1];
+ x3r = a[j1] - a[j3];
+ x3i = a[j1 + 1] - a[j3 + 1];
+ a[j0] = x0r + x2r;
+ a[j0 + 1] = x0i - x2i;
+ a[j1] = x0r - x2r;
+ a[j1 + 1] = x0i + x2i;
+ x0r = x1r + x3i;
+ x0i = x1i + x3r;
+ a[j2] = picofftsg_mult_w_a(wd1r, (x0r - x0i));
+ a[j2 + 1] = picofftsg_mult_w_a(wd1r, (x0i + x0r));
+ x0r = x1r - x3i;
+ x0i = x1i - x3r;
+ a[j3] = -picofftsg_mult_w_a(wd1r, (x0r + x0i));
+ a[j3 + 1] = -picofftsg_mult_w_a(wd1r, (x0i - x0r));
+ x0r = a[j0 + 2] + a[j2 + 2];
+ x0i = -a[j0 + 3] - a[j2 + 3];
+ x1r = a[j0 + 2] - a[j2 + 2];
+ x1i = -a[j0 + 3] + a[j2 + 3];
+ x2r = a[j1 + 2] + a[j3 + 2];
+ x2i = a[j1 + 3] + a[j3 + 3];
+ x3r = a[j1 + 2] - a[j3 + 2];
+ x3i = a[j1 + 3] - a[j3 + 3];
+ a[j0 + 2] = x0r + x2r;
+ a[j0 + 3] = x0i - x2i;
+ a[j1 + 2] = x0r - x2r;
+ a[j1 + 3] = x0i + x2i;
+ x0r = x1r + x3i;
+ x0i = x1i + x3r;
+ a[j2 + 2] = Mult_W_W(wk1i, x0r) - Mult_W_W(wk1r, x0i);
+ a[j2 + 3] = Mult_W_W(wk1i, x0i) + Mult_W_W(wk1r, x0r);
+ x0r = x1r - x3i;
+ x0i = x1i - x3r;
+ a[j3 + 2] = Mult_W_W(wk3i, x0r) + Mult_W_W(wk3r, x0i);
+ a[j3 + 3] = Mult_W_W(wk3i, x0i) - Mult_W_W(wk3r, x0r);
+}
+
+void cftrec4(picoos_int32 n, PICOFFTSG_FFTTYPE *a)
+{
+ picoos_int32 isplt, j, k, m;
+
+ m = n;
+ while (m > 512) {
+ m >>= 2;
+ cftmdl1(m, &a[n - m]);
+ }
+ cftleaf(m, 1, &a[n - m]);
+ k = 0;
+ for (j = n - m; j > 0; j -= m) {
+ k++;
+ isplt = cfttree(m, j, k, a);
+ cftleaf(m, isplt, &a[j - m]);
+ }
+}
+
+
+picoos_int32 cfttree(picoos_int32 n, picoos_int32 j, picoos_int32 k, PICOFFTSG_FFTTYPE *a)
+{
+ picoos_int32 i, isplt, m;
+
+ if ((k & 3) != 0) {
+ isplt = k & 1;
+ if (isplt != 0) {
+ cftmdl1(n, &a[j - n]);
+ } else {
+ cftmdl2(n, &a[j - n]);
+ }
+ } else {
+ m = n;
+ for (i = k; (i & 3) == 0; i >>= 2) {
+ m <<= 2;
+ }
+ isplt = i & 1;
+ if (isplt != 0) {
+ while (m > 128) {
+ cftmdl1(m, &a[j - m]);
+ m >>= 2;
+ }
+ } else {
+ while (m > 128) {
+ cftmdl2(m, &a[j - m]);
+ m >>= 2;
+ }
+ }
+ }
+ return isplt;
+}
+
+
+void cftleaf(picoos_int32 n, picoos_int32 isplt, PICOFFTSG_FFTTYPE *a)
+{
+
+ if (n == 512) {
+ cftmdl1(128, a);
+ cftf161(a);
+ cftf162(&a[32]);
+ cftf161(&a[64]);
+ cftf161(&a[96]);
+ cftmdl2(128, &a[128]);
+ cftf161(&a[128]);
+ cftf162(&a[160]);
+ cftf161(&a[192]);
+ cftf162(&a[224]);
+ cftmdl1(128, &a[256]);
+ cftf161(&a[256]);
+ cftf162(&a[288]);
+ cftf161(&a[320]);
+ cftf161(&a[352]);
+ if (isplt != 0) {
+ cftmdl1(128, &a[384]);
+ cftf161(&a[480]);
+ } else {
+ cftmdl2(128, &a[384]);
+ cftf162(&a[480]);
+ }
+ cftf161(&a[384]);
+ cftf162(&a[416]);
+ cftf161(&a[448]);
+ } else {
+ cftmdl1(64, a);
+ cftf081(a);
+ cftf082(&a[16]);
+ cftf081(&a[32]);
+ cftf081(&a[48]);
+ cftmdl2(64, &a[64]);
+ cftf081(&a[64]);
+ cftf082(&a[80]);
+ cftf081(&a[96]);
+ cftf082(&a[112]);
+ cftmdl1(64, &a[128]);
+ cftf081(&a[128]);
+ cftf082(&a[144]);
+ cftf081(&a[160]);
+ cftf081(&a[176]);
+ if (isplt != 0) {
+ cftmdl1(64, &a[192]);
+ cftf081(&a[240]);
+ } else {
+ cftmdl2(64, &a[192]);
+ cftf082(&a[240]);
+ }
+ cftf081(&a[192]);
+ cftf082(&a[208]);
+ cftf081(&a[224]);
+ }
+}
+
+
+void cftmdl1(picoos_int32 n, PICOFFTSG_FFTTYPE *a)
+{
+ picoos_int32 i, i0, j, j0, j1, j2, j3, m, mh;
+ PICOFFTSG_FFTTYPE wk1r, wk1i, wk3r, wk3i,
+ wd1r, wd1i, wd3r, wd3i, ss1, ss3;
+ PICOFFTSG_FFTTYPE x0r, x0i, x1r, x1i, x2r, x2i, x3r, x3i;
+
+ mh = n >> 3;
+ m = 2 * mh;
+ j1 = m;
+ j2 = j1 + m;
+ j3 = j2 + m;
+ x0r = a[0] + a[j2];
+ x0i = a[1] + a[j2 + 1];
+ x1r = a[0] - a[j2];
+ x1i = a[1] - a[j2 + 1];
+ x2r = a[j1] + a[j3];
+ x2i = a[j1 + 1] + a[j3 + 1];
+ x3r = a[j1] - a[j3];
+ x3i = a[j1 + 1] - a[j3 + 1];
+ a[0] = x0r + x2r;
+ a[1] = x0i + x2i;
+ a[j1] = x0r - x2r;
+ a[j1 + 1] = x0i - x2i;
+ a[j2] = x1r - x3i;
+ a[j2 + 1] = x1i + x3r;
+ a[j3] = x1r + x3i;
+ a[j3 + 1] = x1i - x3r;
+ wd1r = (PICOFFTSG_FFTTYPE)(PICODSP_WGT_SHIFT);
+ wd1i = 0;
+ wd3r = (PICOFFTSG_FFTTYPE)(PICODSP_WGT_SHIFT);
+ wd3i = 0;
+ wk1r = (PICOFFTSG_FFTTYPE) (0.980785250664 *PICODSP_WGT_SHIFT);
+ wk1i = (PICOFFTSG_FFTTYPE) (0.195090323687 *PICODSP_WGT_SHIFT);
+ ss1 = (PICOFFTSG_FFTTYPE) (0.390180647373 *PICODSP_WGT_SHIFT);
+ wk3i = (PICOFFTSG_FFTTYPE) (-0.555570185184 *PICODSP_WGT_SHIFT);
+ wk3r = (PICOFFTSG_FFTTYPE) (0.831469595432 *PICODSP_WGT_SHIFT);
+ ss3 = (PICOFFTSG_FFTTYPE) (-1.111140370369 *PICODSP_WGT_SHIFT);
+
+ i = 0;
+ for (;;) {
+ i0 = i + CDFT_LOOP_DIV_4;
+ if (i0 > mh - 4) {
+ i0 = mh - 4;
+ }
+ for (j = i + 2; j < i0; j += 4) {
+ wd1r -= Mult_W_W(ss1, wk1i);
+ wd1i += Mult_W_W(ss1, wk1r);
+ wd3r -= Mult_W_W(ss3, wk3i);
+ wd3i += Mult_W_W(ss3, wk3r);
+ j1 = j + m;
+ j2 = j1 + m;
+ j3 = j2 + m;
+ x0r = a[j] + a[j2];
+ x0i = a[j + 1] + a[j2 + 1];
+ x1r = a[j] - a[j2];
+ x1i = a[j + 1] - a[j2 + 1];
+ x2r = a[j1] + a[j3];
+ x2i = a[j1 + 1] + a[j3 + 1];
+ x3r = a[j1] - a[j3];
+ x3i = a[j1 + 1] - a[j3 + 1];
+ a[j] = x0r + x2r;
+ a[j + 1] = x0i + x2i;
+ a[j1] = x0r - x2r;
+ a[j1 + 1] = x0i - x2i;
+ x0r = x1r - x3i;
+ x0i = x1i + x3r;
+ a[j2] = Mult_W_W(wk1r, x0r) - Mult_W_W(wk1i, x0i);
+ a[j2 + 1] = Mult_W_W(wk1r, x0i) + Mult_W_W(wk1i, x0r);
+ x0r = x1r + x3i;
+ x0i = x1i - x3r;
+ a[j3] = Mult_W_W(wk3r, x0r) + Mult_W_W(wk3i, x0i);
+ a[j3 + 1] = Mult_W_W(wk3r, x0i) - Mult_W_W(wk3i, x0r);
+ x0r = a[j + 2] + a[j2 + 2];
+ x0i = a[j + 3] + a[j2 + 3];
+ x1r = a[j + 2] - a[j2 + 2];
+ x1i = a[j + 3] - a[j2 + 3];
+ x2r = a[j1 + 2] + a[j3 + 2];
+ x2i = a[j1 + 3] + a[j3 + 3];
+ x3r = a[j1 + 2] - a[j3 + 2];
+ x3i = a[j1 + 3] - a[j3 + 3];
+ a[j + 2] = x0r + x2r;
+ a[j + 3] = x0i + x2i;
+ a[j1 + 2] = x0r - x2r;
+ a[j1 + 3] = x0i - x2i;
+ x0r = x1r - x3i;
+ x0i = x1i + x3r;
+ a[j2 + 2] = Mult_W_W(wd1r, x0r) - Mult_W_W(wd1i, x0i);
+ a[j2 + 3] = Mult_W_W(wd1r, x0i) + Mult_W_W(wd1i, x0r);
+ x0r = x1r + x3i;
+ x0i = x1i - x3r;
+ a[j3 + 2] = Mult_W_W(wd3r, x0r) + Mult_W_W(wd3i, x0i);
+ a[j3 + 3] = Mult_W_W(wd3r, x0i) - Mult_W_W(wd3i, x0r);
+ j0 = m - j;
+ j1 = j0 + m;
+ j2 = j1 + m;
+ j3 = j2 + m;
+ x0r = a[j0] + a[j2];
+ x0i = a[j0 + 1] + a[j2 + 1];
+ x1r = a[j0] - a[j2];
+ x1i = a[j0 + 1] - a[j2 + 1];
+ x2r = a[j1] + a[j3];
+ x2i = a[j1 + 1] + a[j3 + 1];
+ x3r = a[j1] - a[j3];
+ x3i = a[j1 + 1] - a[j3 + 1];
+ a[j0] = x0r + x2r;
+ a[j0 + 1] = x0i + x2i;
+ a[j1] = x0r - x2r;
+ a[j1 + 1] = x0i - x2i;
+ x0r = x1r - x3i;
+ x0i = x1i + x3r;
+ a[j2] = Mult_W_W(wk1i, x0r) - Mult_W_W(wk1r, x0i);
+ a[j2 + 1] = Mult_W_W(wk1i, x0i) + Mult_W_W(wk1r, x0r);
+ x0r = x1r + x3i;
+ x0i = x1i - x3r;
+ a[j3] = Mult_W_W(wk3i, x0r) + Mult_W_W(wk3r, x0i);
+ a[j3 + 1] = Mult_W_W(wk3i, x0i) - Mult_W_W(wk3r, x0r);
+ x0r = a[j0 - 2] + a[j2 - 2];
+ x0i = a[j0 - 1] + a[j2 - 1];
+ x1r = a[j0 - 2] - a[j2 - 2];
+ x1i = a[j0 - 1] - a[j2 - 1];
+ x2r = a[j1 - 2] + a[j3 - 2];
+ x2i = a[j1 - 1] + a[j3 - 1];
+ x3r = a[j1 - 2] - a[j3 - 2];
+ x3i = a[j1 - 1] - a[j3 - 1];
+ a[j0 - 2] = x0r + x2r;
+ a[j0 - 1] = x0i + x2i;
+ a[j1 - 2] = x0r - x2r;
+ a[j1 - 1] = x0i - x2i;
+ x0r = x1r - x3i;
+ x0i = x1i + x3r;
+ a[j2 - 2] = Mult_W_W(wd1i, x0r) - Mult_W_W(wd1r, x0i);
+ a[j2 - 1] = Mult_W_W(wd1i, x0i) + Mult_W_W(wd1r, x0r);
+ x0r = x1r + x3i;
+ x0i = x1i - x3r;
+ a[j3 - 2] = Mult_W_W(wd3i, x0r) + Mult_W_W(wd3r, x0i);
+ a[j3 - 1] = Mult_W_W(wd3i, x0i) - Mult_W_W(wd3r, x0r);
+ wk1r -= Mult_W_W(ss1, wd1i);
+ wk1i += Mult_W_W(ss1, wd1r);
+ wk3r -= Mult_W_W(ss3, wd3i);
+ wk3i += Mult_W_W(ss3, wd3r);
+ }
+ if (i0 == mh - 4) {
+ break;
+ }
+ }
+ wd1r = WR5000;
+ j0 = mh;
+ j1 = j0 + m;
+ j2 = j1 + m;
+ j3 = j2 + m;
+ x0r = a[j0 - 2] + a[j2 - 2];
+ x0i = a[j0 - 1] + a[j2 - 1];
+ x1r = a[j0 - 2] - a[j2 - 2];
+ x1i = a[j0 - 1] - a[j2 - 1];
+ x2r = a[j1 - 2] + a[j3 - 2];
+ x2i = a[j1 - 1] + a[j3 - 1];
+ x3r = a[j1 - 2] - a[j3 - 2];
+ x3i = a[j1 - 1] - a[j3 - 1];
+ a[j0 - 2] = x0r + x2r;
+ a[j0 - 1] = x0i + x2i;
+ a[j1 - 2] = x0r - x2r;
+ a[j1 - 1] = x0i - x2i;
+ x0r = x1r - x3i;
+ x0i = x1i + x3r;
+ a[j2 - 2] = Mult_W_W(wk1r, x0r) - Mult_W_W(wk1i, x0i);
+ a[j2 - 1] = Mult_W_W(wk1r, x0i) + Mult_W_W(wk1i, x0r);
+ x0r = x1r + x3i;
+ x0i = x1i - x3r;
+ a[j3 - 2] = Mult_W_W(wk3r, x0r) + Mult_W_W(wk3i, x0i);
+ a[j3 - 1] = Mult_W_W(wk3r, x0i) - Mult_W_W(wk3i, x0r);
+ x0r = a[j0] + a[j2];
+ x0i = a[j0 + 1] + a[j2 + 1];
+ x1r = a[j0] - a[j2];
+ x1i = a[j0 + 1] - a[j2 + 1];
+ x2r = a[j1] + a[j3];
+ x2i = a[j1 + 1] + a[j3 + 1];
+ x3r = a[j1] - a[j3];
+ x3i = a[j1 + 1] - a[j3 + 1];
+ a[j0] = x0r + x2r;
+ a[j0 + 1] = x0i + x2i;
+ a[j1] = x0r - x2r;
+ a[j1 + 1] = x0i - x2i;
+ x0r = x1r - x3i;
+ x0i = x1i + x3r;
+ a[j2] = picofftsg_mult_w_a(wd1r, (x0r - x0i));
+ a[j2 + 1] = picofftsg_mult_w_a(wd1r, (x0i + x0r));
+ x0r = x1r + x3i;
+ x0i = x1i - x3r;
+ a[j3] = -picofftsg_mult_w_a(wd1r, (x0r + x0i));
+ a[j3 + 1] = -picofftsg_mult_w_a(wd1r, (x0i - x0r));
+ x0r = a[j0 + 2] + a[j2 + 2];
+ x0i = a[j0 + 3] + a[j2 + 3];
+ x1r = a[j0 + 2] - a[j2 + 2];
+ x1i = a[j0 + 3] - a[j2 + 3];
+ x2r = a[j1 + 2] + a[j3 + 2];
+ x2i = a[j1 + 3] + a[j3 + 3];
+ x3r = a[j1 + 2] - a[j3 + 2];
+ x3i = a[j1 + 3] - a[j3 + 3];
+ a[j0 + 2] = x0r + x2r;
+ a[j0 + 3] = x0i + x2i;
+ a[j1 + 2] = x0r - x2r;
+ a[j1 + 3] = x0i - x2i;
+ x0r = x1r - x3i;
+ x0i = x1i + x3r;
+ a[j2 + 2] = Mult_W_W(wk1i, x0r) - Mult_W_W(wk1r, x0i);
+ a[j2 + 3] = Mult_W_W(wk1i, x0i) + Mult_W_W(wk1r, x0r);
+ x0r = x1r + x3i;
+ x0i = x1i - x3r;
+ a[j3 + 2] = Mult_W_W(wk3i, x0r) + Mult_W_W(wk3r, x0i);
+ a[j3 + 3] = Mult_W_W(wk3i, x0i) - Mult_W_W(wk3r, x0r);
+}
+
+
+void cftmdl2(picoos_int32 n, PICOFFTSG_FFTTYPE *a)
+{
+ picoos_int32 i, i0, j, j0, j1, j2, j3, m, mh;
+ PICOFFTSG_FFTTYPE wn4r, wk1r, wk1i, wk3r, wk3i,
+ wl1r, wl1i, wl3r, wl3i, wd1r, wd1i, wd3r, wd3i,
+ we1r, we1i, we3r, we3i, ss1, ss3;
+ PICOFFTSG_FFTTYPE x0r, x0i, x1r, x1i, x2r, x2i, x3r, x3i, y0r, y0i, y2r, y2i;
+
+ mh = n >> 3;
+ m = 2 * mh;
+ wn4r = WR5000;
+ j1 = m;
+ j2 = j1 + m;
+ j3 = j2 + m;
+ x0r = a[0] - a[j2 + 1];
+ x0i = a[1] + a[j2];
+ x1r = a[0] + a[j2 + 1];
+ x1i = a[1] - a[j2];
+ x2r = a[j1] - a[j3 + 1];
+ x2i = a[j1 + 1] + a[j3];
+ x3r = a[j1] + a[j3 + 1];
+ x3i = a[j1 + 1] - a[j3];
+ y0r = picofftsg_mult_w_a(wn4r, (x2r - x2i));
+ y0i = picofftsg_mult_w_a(wn4r, (x2i + x2r));
+ a[0] = x0r + y0r;
+ a[1] = x0i + y0i;
+ a[j1] = x0r - y0r;
+ a[j1 + 1] = x0i - y0i;
+ y0r = picofftsg_mult_w_a(wn4r, (x3r - x3i));
+ y0i = picofftsg_mult_w_a(wn4r, (x3i + x3r));
+ a[j2] = x1r - y0i;
+ a[j2 + 1] = x1i + y0r;
+ a[j3] = x1r + y0i;
+ a[j3 + 1] = x1i - y0r;
+ wl1r = PICODSP_WGT_SHIFT;
+ wl1i = 0;
+ wl3r = PICODSP_WGT_SHIFT;
+ wl3i = 0;
+ we1r = wn4r;
+ we1i = wn4r;
+ we3r = -wn4r;
+ we3i = -wn4r;
+
+ wk1r = (PICOFFTSG_FFTTYPE)(0.995184719563 *PICODSP_WGT_SHIFT);
+ wk1i = (PICOFFTSG_FFTTYPE)(0.098017141223 *PICODSP_WGT_SHIFT);
+ wd1r = (PICOFFTSG_FFTTYPE)(0.634393274784 *PICODSP_WGT_SHIFT);
+ wd1i = (PICOFFTSG_FFTTYPE)(0.773010432720 *PICODSP_WGT_SHIFT);
+ ss1 = (PICOFFTSG_FFTTYPE)(0.196034282446 *PICODSP_WGT_SHIFT);
+ wk3i = (PICOFFTSG_FFTTYPE)(-0.290284663439 *PICODSP_WGT_SHIFT);
+ wk3r = (PICOFFTSG_FFTTYPE)(0.956940352917 *PICODSP_WGT_SHIFT);
+ ss3 = (PICOFFTSG_FFTTYPE)(-0.580569326878 *PICODSP_WGT_SHIFT);
+ wd3r = (PICOFFTSG_FFTTYPE)(-0.881921231747 *PICODSP_WGT_SHIFT);
+ wd3i = (PICOFFTSG_FFTTYPE)(-0.471396744251 *PICODSP_WGT_SHIFT);
+
+ i = 0;
+ for (;;) {
+ i0 = i + 4 * CDFT_LOOP_DIV;
+ if (i0 > mh - 4) {
+ i0 = mh - 4;
+ }
+ for (j = i + 2; j < i0; j += 4) {
+ wl1r -= Mult_W_W(ss1, wk1i);
+ wl1i += Mult_W_W(ss1, wk1r);
+ wl3r -= Mult_W_W(ss3, wk3i);
+ wl3i += Mult_W_W(ss3, wk3r);
+ we1r -= Mult_W_W(ss1, wd1i);
+ we1i += Mult_W_W(ss1, wd1r);
+ we3r -= Mult_W_W(ss3, wd3i);
+ we3i += Mult_W_W(ss3, wd3r);
+ j1 = j + m;
+ j2 = j1 + m;
+ j3 = j2 + m;
+ x0r = a[j] - a[j2 + 1];
+ x0i = a[j + 1] + a[j2];
+ x1r = a[j] + a[j2 + 1];
+ x1i = a[j + 1] - a[j2];
+ x2r = a[j1] - a[j3 + 1];
+ x2i = a[j1 + 1] + a[j3];
+ x3r = a[j1] + a[j3 + 1];
+ x3i = a[j1 + 1] - a[j3];
+ y0r = Mult_W_W(wk1r, x0r) - Mult_W_W(wk1i, x0i);
+ y0i = Mult_W_W(wk1r, x0i) + Mult_W_W(wk1i, x0r);
+ y2r = Mult_W_W(wd1r, x2r) - Mult_W_W(wd1i, x2i);
+ y2i = Mult_W_W(wd1r, x2i) + Mult_W_W(wd1i, x2r);
+ a[j] = y0r + y2r;
+ a[j + 1] = y0i + y2i;
+ a[j1] = y0r - y2r;
+ a[j1 + 1] = y0i - y2i;
+ y0r = Mult_W_W(wk3r, x1r) + Mult_W_W(wk3i, x1i);
+ y0i = Mult_W_W(wk3r, x1i) - Mult_W_W(wk3i, x1r);
+ y2r = Mult_W_W(wd3r, x3r) + Mult_W_W(wd3i, x3i);
+ y2i = Mult_W_W(wd3r, x3i) - Mult_W_W(wd3i, x3r);
+ a[j2] = y0r + y2r;
+ a[j2 + 1] = y0i + y2i;
+ a[j3] = y0r - y2r;
+ a[j3 + 1] = y0i - y2i;
+ x0r = a[j + 2] - a[j2 + 3];
+ x0i = a[j + 3] + a[j2 + 2];
+ x1r = a[j + 2] + a[j2 + 3];
+ x1i = a[j + 3] - a[j2 + 2];
+ x2r = a[j1 + 2] - a[j3 + 3];
+ x2i = a[j1 + 3] + a[j3 + 2];
+ x3r = a[j1 + 2] + a[j3 + 3];
+ x3i = a[j1 + 3] - a[j3 + 2];
+ y0r = Mult_W_W(wl1r, x0r) - Mult_W_W(wl1i, x0i);
+ y0i = Mult_W_W(wl1r, x0i) + Mult_W_W(wl1i, x0r);
+ y2r = Mult_W_W(we1r, x2r) - Mult_W_W(we1i, x2i);
+ y2i = Mult_W_W(we1r, x2i) + Mult_W_W(we1i, x2r);
+ a[j + 2] = y0r + y2r;
+ a[j + 3] = y0i + y2i;
+ a[j1 + 2] = y0r - y2r;
+ a[j1 + 3] = y0i - y2i;
+ y0r = Mult_W_W(wl3r, x1r) + Mult_W_W(wl3i, x1i);
+ y0i = Mult_W_W(wl3r, x1i) - Mult_W_W(wl3i, x1r);
+ y2r = Mult_W_W(we3r, x3r) + Mult_W_W(we3i, x3i);
+ y2i = Mult_W_W(we3r, x3i) - Mult_W_W(we3i, x3r);
+ a[j2 + 2] = y0r + y2r;
+ a[j2 + 3] = y0i + y2i;
+ a[j3 + 2] = y0r - y2r;
+ a[j3 + 3] = y0i - y2i;
+ j0 = m - j;
+ j1 = j0 + m;
+ j2 = j1 + m;
+ j3 = j2 + m;
+ x0r = a[j0] - a[j2 + 1];
+ x0i = a[j0 + 1] + a[j2];
+ x1r = a[j0] + a[j2 + 1];
+ x1i = a[j0 + 1] - a[j2];
+ x2r = a[j1] - a[j3 + 1];
+ x2i = a[j1 + 1] + a[j3];
+ x3r = a[j1] + a[j3 + 1];
+ x3i = a[j1 + 1] - a[j3];
+ y0r = Mult_W_W(wd1i, x0r) - Mult_W_W(wd1r, x0i);
+ y0i = Mult_W_W(wd1i, x0i) + Mult_W_W(wd1r, x0r);
+ y2r = Mult_W_W(wk1i, x2r) - Mult_W_W(wk1r, x2i);
+ y2i = Mult_W_W(wk1i, x2i) + Mult_W_W(wk1r, x2r);
+ a[j0] = y0r + y2r;
+ a[j0 + 1] = y0i + y2i;
+ a[j1] = y0r - y2r;
+ a[j1 + 1] = y0i - y2i;
+ y0r = Mult_W_W(wd3i, x1r) + Mult_W_W(wd3r, x1i);
+ y0i = Mult_W_W(wd3i, x1i) - Mult_W_W(wd3r, x1r);
+ y2r = Mult_W_W(wk3i, x3r) + Mult_W_W(wk3r, x3i);
+ y2i = Mult_W_W(wk3i, x3i) - Mult_W_W(wk3r, x3r);
+ a[j2] = y0r + y2r;
+ a[j2 + 1] = y0i + y2i;
+ a[j3] = y0r - y2r;
+ a[j3 + 1] = y0i - y2i;
+ x0r = a[j0 - 2] - a[j2 - 1];
+ x0i = a[j0 - 1] + a[j2 - 2];
+ x1r = a[j0 - 2] + a[j2 - 1];
+ x1i = a[j0 - 1] - a[j2 - 2];
+ x2r = a[j1 - 2] - a[j3 - 1];
+ x2i = a[j1 - 1] + a[j3 - 2];
+ x3r = a[j1 - 2] + a[j3 - 1];
+ x3i = a[j1 - 1] - a[j3 - 2];
+ y0r = Mult_W_W(we1i, x0r) - Mult_W_W(we1r, x0i);
+ y0i = Mult_W_W(we1i, x0i) + Mult_W_W(we1r, x0r);
+ y2r = Mult_W_W(wl1i, x2r) - Mult_W_W(wl1r, x2i);
+ y2i = Mult_W_W(wl1i, x2i) + Mult_W_W(wl1r, x2r);
+ a[j0 - 2] = y0r + y2r;
+ a[j0 - 1] = y0i + y2i;
+ a[j1 - 2] = y0r - y2r;
+ a[j1 - 1] = y0i - y2i;
+ y0r = Mult_W_W(we3i, x1r) + Mult_W_W(we3r, x1i);
+ y0i = Mult_W_W(we3i, x1i) - Mult_W_W(we3r, x1r);
+ y2r = Mult_W_W(wl3i, x3r) + Mult_W_W(wl3r, x3i);
+ y2i = Mult_W_W(wl3i, x3i) - Mult_W_W(wl3r, x3r);
+ a[j2 - 2] = y0r + y2r;
+ a[j2 - 1] = y0i + y2i;
+ a[j3 - 2] = y0r - y2r;
+ a[j3 - 1] = y0i - y2i;
+ wk1r -= Mult_W_W(ss1, wl1i);
+ wk1i += Mult_W_W(ss1, wl1r);
+ wk3r -= Mult_W_W(ss3, wl3i);
+ wk3i += Mult_W_W(ss3, wl3r);
+ wd1r -= Mult_W_W(ss1, we1i);
+ wd1i += Mult_W_W(ss1, we1r);
+ wd3r -= Mult_W_W(ss3, we3i);
+ wd3i += Mult_W_W(ss3, we3r);
+ }
+ if (i0 == mh - 4) {
+ break;
+ }
+ }
+ wl1r = WR2500;
+ wl1i = WI2500;
+ j0 = mh;
+ j1 = j0 + m;
+ j2 = j1 + m;
+ j3 = j2 + m;
+ x0r = a[j0 - 2] - a[j2 - 1];
+ x0i = a[j0 - 1] + a[j2 - 2];
+ x1r = a[j0 - 2] + a[j2 - 1];
+ x1i = a[j0 - 1] - a[j2 - 2];
+ x2r = a[j1 - 2] - a[j3 - 1];
+ x2i = a[j1 - 1] + a[j3 - 2];
+ x3r = a[j1 - 2] + a[j3 - 1];
+ x3i = a[j1 - 1] - a[j3 - 2];
+ y0r = Mult_W_W(wk1r, x0r) - Mult_W_W(wk1i, x0i);
+ y0i = Mult_W_W(wk1r, x0i) + Mult_W_W(wk1i, x0r);
+ y2r = Mult_W_W(wd1r, x2r) - Mult_W_W(wd1i, x2i);
+ y2i = Mult_W_W(wd1r, x2i) + Mult_W_W(wd1i, x2r);
+ a[j0 - 2] = y0r + y2r;
+ a[j0 - 1] = y0i + y2i;
+ a[j1 - 2] = y0r - y2r;
+ a[j1 - 1] = y0i - y2i;
+ y0r = Mult_W_W(wk3r, x1r) + Mult_W_W(wk3i, x1i);
+ y0i = Mult_W_W(wk3r, x1i) - Mult_W_W(wk3i, x1r);
+ y2r = Mult_W_W(wd3r, x3r) + Mult_W_W(wd3i, x3i);
+ y2i = Mult_W_W(wd3r, x3i) - Mult_W_W(wd3i, x3r);
+ a[j2 - 2] = y0r + y2r;
+ a[j2 - 1] = y0i + y2i;
+ a[j3 - 2] = y0r - y2r;
+ a[j3 - 1] = y0i - y2i;
+ x0r = a[j0] - a[j2 + 1];
+ x0i = a[j0 + 1] + a[j2];
+ x1r = a[j0] + a[j2 + 1];
+ x1i = a[j0 + 1] - a[j2];
+ x2r = a[j1] - a[j3 + 1];
+ x2i = a[j1 + 1] + a[j3];
+ x3r = a[j1] + a[j3 + 1];
+ x3i = a[j1 + 1] - a[j3];
+ y0r = Mult_W_W(wl1r, x0r) - Mult_W_W(wl1i, x0i);
+ y0i = Mult_W_W(wl1r, x0i) + Mult_W_W(wl1i, x0r);
+ y2r = Mult_W_W(wl1i, x2r) - Mult_W_W(wl1r, x2i);
+ y2i = Mult_W_W(wl1i, x2i) + Mult_W_W(wl1r, x2r);
+ a[j0] = y0r + y2r;
+ a[j0 + 1] = y0i + y2i;
+ a[j1] = y0r - y2r;
+ a[j1 + 1] = y0i - y2i;
+ y0r = Mult_W_W(wl1i, x1r) - Mult_W_W(wl1r, x1i);
+ y0i = Mult_W_W(wl1i, x1i) + Mult_W_W(wl1r, x1r);
+ y2r = Mult_W_W(wl1r, x3r) - Mult_W_W(wl1i, x3i);
+ y2i = Mult_W_W(wl1r, x3i) + Mult_W_W(wl1i, x3r);
+ a[j2] = y0r - y2r;
+ a[j2 + 1] = y0i - y2i;
+ a[j3] = y0r + y2r;
+ a[j3 + 1] = y0i + y2i;
+ x0r = a[j0 + 2] - a[j2 + 3];
+ x0i = a[j0 + 3] + a[j2 + 2];
+ x1r = a[j0 + 2] + a[j2 + 3];
+ x1i = a[j0 + 3] - a[j2 + 2];
+ x2r = a[j1 + 2] - a[j3 + 3];
+ x2i = a[j1 + 3] + a[j3 + 2];
+ x3r = a[j1 + 2] + a[j3 + 3];
+ x3i = a[j1 + 3] - a[j3 + 2];
+ y0r = Mult_W_W(wd1i, x0r) - Mult_W_W(wd1r, x0i);
+ y0i = Mult_W_W(wd1i, x0i) + Mult_W_W(wd1r, x0r);
+ y2r = Mult_W_W(wk1i, x2r) - Mult_W_W(wk1r, x2i);
+ y2i = Mult_W_W(wk1i, x2i) + Mult_W_W(wk1r, x2r);
+ a[j0 + 2] = y0r + y2r;
+ a[j0 + 3] = y0i + y2i;
+ a[j1 + 2] = y0r - y2r;
+ a[j1 + 3] = y0i - y2i;
+ y0r = Mult_W_W(wd3i, x1r) + Mult_W_W(wd3r, x1i);
+ y0i = Mult_W_W(wd3i, x1i) - Mult_W_W(wd3r, x1r);
+ y2r = Mult_W_W(wk3i, x3r) + Mult_W_W(wk3r, x3i);
+ y2i = Mult_W_W(wk3i, x3i) - Mult_W_W(wk3r, x3r);
+ a[j2 + 2] = y0r + y2r;
+ a[j2 + 3] = y0i + y2i;
+ a[j3 + 2] = y0r - y2r;
+ a[j3 + 3] = y0i - y2i;
+}
+
+
+void cftfx41(picoos_int32 n, PICOFFTSG_FFTTYPE *a)
+{
+
+ if (n == 128) {
+ cftf161(a);
+ cftf162(&a[32]);
+ cftf161(&a[64]);
+ cftf161(&a[96]);
+ } else {
+ cftf081(a);
+ cftf082(&a[16]);
+ cftf081(&a[32]);
+ cftf081(&a[48]);
+ }
+}
+
+
+void cftf161(PICOFFTSG_FFTTYPE *a)
+{
+ PICOFFTSG_FFTTYPE wn4r, wk1r, wk1i,
+ x0r, x0i, x1r, x1i, x2r, x2i, x3r, x3i,
+ y0r, y0i, y1r, y1i, y2r, y2i, y3r, y3i,
+ y4r, y4i, y5r, y5i, y6r, y6i, y7r, y7i,
+ y8r, y8i, y9r, y9i, y10r, y10i, y11r, y11i,
+ y12r, y12i, y13r, y13i, y14r, y14i, y15r, y15i;
+
+ wn4r = WR5000;
+ wk1r = WR2500;
+ wk1i = WI2500;
+ x0r = a[0] + a[16];
+ x0i = a[1] + a[17];
+ x1r = a[0] - a[16];
+ x1i = a[1] - a[17];
+ x2r = a[8] + a[24];
+ x2i = a[9] + a[25];
+ x3r = a[8] - a[24];
+ x3i = a[9] - a[25];
+ y0r = x0r + x2r;
+ y0i = x0i + x2i;
+ y4r = x0r - x2r;
+ y4i = x0i - x2i;
+ y8r = x1r - x3i;
+ y8i = x1i + x3r;
+ y12r = x1r + x3i;
+ y12i = x1i - x3r;
+ x0r = a[2] + a[18];
+ x0i = a[3] + a[19];
+ x1r = a[2] - a[18];
+ x1i = a[3] - a[19];
+ x2r = a[10] + a[26];
+ x2i = a[11] + a[27];
+ x3r = a[10] - a[26];
+ x3i = a[11] - a[27];
+ y1r = x0r + x2r;
+ y1i = x0i + x2i;
+ y5r = x0r - x2r;
+ y5i = x0i - x2i;
+ x0r = x1r - x3i;
+ x0i = x1i + x3r;
+ y9r = Mult_W_W(wk1r, x0r) - Mult_W_W(wk1i, x0i);
+ y9i = Mult_W_W(wk1r, x0i) + Mult_W_W(wk1i, x0r);
+ x0r = x1r + x3i;
+ x0i = x1i - x3r;
+ y13r = Mult_W_W(wk1i, x0r) - Mult_W_W(wk1r, x0i);
+ y13i = Mult_W_W(wk1i, x0i) + Mult_W_W(wk1r, x0r);
+ x0r = a[4] + a[20];
+ x0i = a[5] + a[21];
+ x1r = a[4] - a[20];
+ x1i = a[5] - a[21];
+ x2r = a[12] + a[28];
+ x2i = a[13] + a[29];
+ x3r = a[12] - a[28];
+ x3i = a[13] - a[29];
+ y2r = x0r + x2r;
+ y2i = x0i + x2i;
+ y6r = x0r - x2r;
+ y6i = x0i - x2i;
+ x0r = x1r - x3i;
+ x0i = x1i + x3r;
+ y10r = picofftsg_mult_w_a(wn4r, (x0r - x0i));
+ y10i = picofftsg_mult_w_a(wn4r, (x0i + x0r));
+ x0r = x1r + x3i;
+ x0i = x1i - x3r;
+ y14r = picofftsg_mult_w_a(wn4r, (x0r + x0i));
+ y14i = picofftsg_mult_w_a(wn4r, (x0i - x0r));
+ x0r = a[6] + a[22];
+ x0i = a[7] + a[23];
+ x1r = a[6] - a[22];
+ x1i = a[7] - a[23];
+ x2r = a[14] + a[30];
+ x2i = a[15] + a[31];
+ x3r = a[14] - a[30];
+ x3i = a[15] - a[31];
+ y3r = x0r + x2r;
+ y3i = x0i + x2i;
+ y7r = x0r - x2r;
+ y7i = x0i - x2i;
+ x0r = x1r - x3i;
+ x0i = x1i + x3r;
+ y11r = Mult_W_W(wk1i, x0r) - Mult_W_W(wk1r, x0i);
+ y11i = Mult_W_W(wk1i, x0i) + Mult_W_W(wk1r, x0r);
+ x0r = x1r + x3i;
+ x0i = x1i - x3r;
+ y15r = Mult_W_W(wk1r, x0r) - Mult_W_W(wk1i, x0i);
+ y15i = Mult_W_W(wk1r, x0i) + Mult_W_W(wk1i, x0r);
+ x0r = y12r - y14r;
+ x0i = y12i - y14i;
+ x1r = y12r + y14r;
+ x1i = y12i + y14i;
+ x2r = y13r - y15r;
+ x2i = y13i - y15i;
+ x3r = y13r + y15r;
+ x3i = y13i + y15i;
+ a[24] = x0r + x2r;
+ a[25] = x0i + x2i;
+ a[26] = x0r - x2r;
+ a[27] = x0i - x2i;
+ a[28] = x1r - x3i;
+ a[29] = x1i + x3r;
+ a[30] = x1r + x3i;
+ a[31] = x1i - x3r;
+ x0r = y8r + y10r;
+ x0i = y8i + y10i;
+ x1r = y8r - y10r;
+ x1i = y8i - y10i;
+ x2r = y9r + y11r;
+ x2i = y9i + y11i;
+ x3r = y9r - y11r;
+ x3i = y9i - y11i;
+ a[16] = x0r + x2r;
+ a[17] = x0i + x2i;
+ a[18] = x0r - x2r;
+ a[19] = x0i - x2i;
+ a[20] = x1r - x3i;
+ a[21] = x1i + x3r;
+ a[22] = x1r + x3i;
+ a[23] = x1i - x3r;
+ x0r = y5r - y7i;
+ x0i = y5i + y7r;
+ x2r = picofftsg_mult_w_a(wn4r, (x0r - x0i));
+ x2i = picofftsg_mult_w_a(wn4r, (x0i + x0r));
+ x0r = y5r + y7i;
+ x0i = y5i - y7r;
+ x3r = picofftsg_mult_w_a(wn4r, (x0r - x0i));
+ x3i = picofftsg_mult_w_a(wn4r, (x0i + x0r));
+ x0r = y4r - y6i;
+ x0i = y4i + y6r;
+ x1r = y4r + y6i;
+ x1i = y4i - y6r;
+ a[8] = x0r + x2r;
+ a[9] = x0i + x2i;
+ a[10] = x0r - x2r;
+ a[11] = x0i - x2i;
+ a[12] = x1r - x3i;
+ a[13] = x1i + x3r;
+ a[14] = x1r + x3i;
+ a[15] = x1i - x3r;
+ x0r = y0r + y2r;
+ x0i = y0i + y2i;
+ x1r = y0r - y2r;
+ x1i = y0i - y2i;
+ x2r = y1r + y3r;
+ x2i = y1i + y3i;
+ x3r = y1r - y3r;
+ x3i = y1i - y3i;
+ a[0] = x0r + x2r;
+ a[1] = x0i + x2i;
+ a[2] = x0r - x2r;
+ a[3] = x0i - x2i;
+ a[4] = x1r - x3i;
+ a[5] = x1i + x3r;
+ a[6] = x1r + x3i;
+ a[7] = x1i - x3r;
+}
+
+
+void cftf162(PICOFFTSG_FFTTYPE *a)
+{
+ PICOFFTSG_FFTTYPE wn4r, wk1r, wk1i, wk2r, wk2i, wk3r, wk3i,
+ x0r, x0i, x1r, x1i, x2r, x2i,
+ y0r, y0i, y1r, y1i, y2r, y2i, y3r, y3i,
+ y4r, y4i, y5r, y5i, y6r, y6i, y7r, y7i,
+ y8r, y8i, y9r, y9i, y10r, y10i, y11r, y11i,
+ y12r, y12i, y13r, y13i, y14r, y14i, y15r, y15i;
+
+ wn4r = WR5000;
+ wk1r = WR1250;
+ wk1i = WI1250;
+ wk2r = WR2500;
+ wk2i = WI2500;
+ wk3r = WR3750;
+ wk3i = WI3750;
+ x1r = a[0] - a[17];
+ x1i = a[1] + a[16];
+ x0r = a[8] - a[25];
+ x0i = a[9] + a[24];
+ x2r = picofftsg_mult_w_a(wn4r, (x0r - x0i));
+ x2i = picofftsg_mult_w_a(wn4r, (x0i + x0r));
+ y0r = x1r + x2r;
+ y0i = x1i + x2i;
+ y4r = x1r - x2r;
+ y4i = x1i - x2i;
+ x1r = a[0] + a[17];
+ x1i = a[1] - a[16];
+ x0r = a[8] + a[25];
+ x0i = a[9] - a[24];
+ x2r = picofftsg_mult_w_a(wn4r, (x0r - x0i));
+ x2i = picofftsg_mult_w_a(wn4r, (x0i + x0r));
+ y8r = x1r - x2i;
+ y8i = x1i + x2r;
+ y12r = x1r + x2i;
+ y12i = x1i - x2r;
+ x0r = a[2] - a[19];
+ x0i = a[3] + a[18];
+ x1r = Mult_W_W(wk1r, x0r) - Mult_W_W(wk1i, x0i);
+ x1i = Mult_W_W(wk1r, x0i) + Mult_W_W(wk1i, x0r);
+ x0r = a[10] - a[27];
+ x0i = a[11] + a[26];
+ x2r = Mult_W_W(wk3i, x0r) - Mult_W_W(wk3r, x0i);
+ x2i = Mult_W_W(wk3i, x0i) + Mult_W_W(wk3r, x0r);
+ y1r = x1r + x2r;
+ y1i = x1i + x2i;
+ y5r = x1r - x2r;
+ y5i = x1i - x2i;
+ x0r = a[2] + a[19];
+ x0i = a[3] - a[18];
+ x1r = Mult_W_W(wk3r, x0r) - Mult_W_W(wk3i, x0i);
+ x1i = Mult_W_W(wk3r, x0i) + Mult_W_W(wk3i, x0r);
+ x0r = a[10] + a[27];
+ x0i = a[11] - a[26];
+ x2r = Mult_W_W(wk1r, x0r) + Mult_W_W(wk1i, x0i);
+ x2i = Mult_W_W(wk1r, x0i) - Mult_W_W(wk1i, x0r);
+ y9r = x1r - x2r;
+ y9i = x1i - x2i;
+ y13r = x1r + x2r;
+ y13i = x1i + x2i;
+ x0r = a[4] - a[21];
+ x0i = a[5] + a[20];
+ x1r = Mult_W_W(wk2r, x0r) - Mult_W_W(wk2i, x0i);
+ x1i = Mult_W_W(wk2r, x0i) + Mult_W_W(wk2i, x0r);
+ x0r = a[12] - a[29];
+ x0i = a[13] + a[28];
+ x2r = Mult_W_W(wk2i, x0r) - Mult_W_W(wk2r, x0i);
+ x2i = Mult_W_W(wk2i, x0i) + Mult_W_W(wk2r, x0r);
+ y2r = x1r + x2r;
+ y2i = x1i + x2i;
+ y6r = x1r - x2r;
+ y6i = x1i - x2i;
+ x0r = a[4] + a[21];
+ x0i = a[5] - a[20];
+ x1r = Mult_W_W(wk2i, x0r) - Mult_W_W(wk2r, x0i);
+ x1i = Mult_W_W(wk2i, x0i) + Mult_W_W(wk2r, x0r);
+ x0r = a[12] + a[29];
+ x0i = a[13] - a[28];
+ x2r = Mult_W_W(wk2r, x0r) - Mult_W_W(wk2i, x0i);
+ x2i = Mult_W_W(wk2r, x0i) + Mult_W_W(wk2i, x0r);
+ y10r = x1r - x2r;
+ y10i = x1i - x2i;
+ y14r = x1r + x2r;
+ y14i = x1i + x2i;
+ x0r = a[6] - a[23];
+ x0i = a[7] + a[22];
+ x1r = Mult_W_W(wk3r, x0r) - Mult_W_W(wk3i, x0i);
+ x1i = Mult_W_W(wk3r, x0i) + Mult_W_W(wk3i, x0r);
+ x0r = a[14] - a[31];
+ x0i = a[15] + a[30];
+ x2r = Mult_W_W(wk1i, x0r) - Mult_W_W(wk1r, x0i);
+ x2i = Mult_W_W(wk1i, x0i) + Mult_W_W(wk1r, x0r);
+ y3r = x1r + x2r;
+ y3i = x1i + x2i;
+ y7r = x1r - x2r;
+ y7i = x1i - x2i;
+ x0r = a[6] + a[23];
+ x0i = a[7] - a[22];
+ x1r = Mult_W_W(wk1i, x0r) + Mult_W_W(wk1r, x0i);
+ x1i = Mult_W_W(wk1i, x0i) - Mult_W_W(wk1r, x0r);
+ x0r = a[14] + a[31];
+ x0i = a[15] - a[30];
+ x2r = Mult_W_W(wk3i, x0r) - Mult_W_W(wk3r, x0i);
+ x2i = Mult_W_W(wk3i, x0i) + Mult_W_W(wk3r, x0r);
+ y11r = x1r + x2r;
+ y11i = x1i + x2i;
+ y15r = x1r - x2r;
+ y15i = x1i - x2i;
+ x1r = y0r + y2r;
+ x1i = y0i + y2i;
+ x2r = y1r + y3r;
+ x2i = y1i + y3i;
+ a[0] = x1r + x2r;
+ a[1] = x1i + x2i;
+ a[2] = x1r - x2r;
+ a[3] = x1i - x2i;
+ x1r = y0r - y2r;
+ x1i = y0i - y2i;
+ x2r = y1r - y3r;
+ x2i = y1i - y3i;
+ a[4] = x1r - x2i;
+ a[5] = x1i + x2r;
+ a[6] = x1r + x2i;
+ a[7] = x1i - x2r;
+ x1r = y4r - y6i;
+ x1i = y4i + y6r;
+ x0r = y5r - y7i;
+ x0i = y5i + y7r;
+ x2r = picofftsg_mult_w_a(wn4r, (x0r - x0i));
+ x2i = picofftsg_mult_w_a(wn4r, (x0i + x0r));
+ a[8] = x1r + x2r;
+ a[9] = x1i + x2i;
+ a[10] = x1r - x2r;
+ a[11] = x1i - x2i;
+ x1r = y4r + y6i;
+ x1i = y4i - y6r;
+ x0r = y5r + y7i;
+ x0i = y5i - y7r;
+ x2r = picofftsg_mult_w_a(wn4r, (x0r - x0i));
+ x2i = picofftsg_mult_w_a(wn4r, (x0i + x0r));
+ a[12] = x1r - x2i;
+ a[13] = x1i + x2r;
+ a[14] = x1r + x2i;
+ a[15] = x1i - x2r;
+ x1r = y8r + y10r;
+ x1i = y8i + y10i;
+ x2r = y9r - y11r;
+ x2i = y9i - y11i;
+ a[16] = x1r + x2r;
+ a[17] = x1i + x2i;
+ a[18] = x1r - x2r;
+ a[19] = x1i - x2i;
+ x1r = y8r - y10r;
+ x1i = y8i - y10i;
+ x2r = y9r + y11r;
+ x2i = y9i + y11i;
+ a[20] = x1r - x2i;
+ a[21] = x1i + x2r;
+ a[22] = x1r + x2i;
+ a[23] = x1i - x2r;
+ x1r = y12r - y14i;
+ x1i = y12i + y14r;
+ x0r = y13r + y15i;
+ x0i = y13i - y15r;
+ x2r = picofftsg_mult_w_a(wn4r, (x0r - x0i));
+ x2i = picofftsg_mult_w_a(wn4r, (x0i + x0r));
+ a[24] = x1r + x2r;
+ a[25] = x1i + x2i;
+ a[26] = x1r - x2r;
+ a[27] = x1i - x2i;
+ x1r = y12r + y14i;
+ x1i = y12i - y14r;
+ x0r = y13r - y15i;
+ x0i = y13i + y15r;
+ x2r = picofftsg_mult_w_a(wn4r, (x0r - x0i));
+ x2i = picofftsg_mult_w_a(wn4r, (x0i + x0r));
+ a[28] = x1r - x2i;
+ a[29] = x1i + x2r;
+ a[30] = x1r + x2i;
+ a[31] = x1i - x2r;
+}
+
+
+void cftf081(PICOFFTSG_FFTTYPE *a)
+{
+ PICOFFTSG_FFTTYPE wn4r, x0r, x0i, x1r, x1i, x2r, x2i, x3r, x3i,
+ y0r, y0i, y1r, y1i, y2r, y2i, y3r, y3i,
+ y4r, y4i, y5r, y5i, y6r, y6i, y7r, y7i;
+
+ wn4r = WR5000;
+ x0r = a[0] + a[8];
+ x0i = a[1] + a[9];
+ x1r = a[0] - a[8];
+ x1i = a[1] - a[9];
+ x2r = a[4] + a[12];
+ x2i = a[5] + a[13];
+ x3r = a[4] - a[12];
+ x3i = a[5] - a[13];
+ y0r = x0r + x2r;
+ y0i = x0i + x2i;
+ y2r = x0r - x2r;
+ y2i = x0i - x2i;
+ y1r = x1r - x3i;
+ y1i = x1i + x3r;
+ y3r = x1r + x3i;
+ y3i = x1i - x3r;
+ x0r = a[2] + a[10];
+ x0i = a[3] + a[11];
+ x1r = a[2] - a[10];
+ x1i = a[3] - a[11];
+ x2r = a[6] + a[14];
+ x2i = a[7] + a[15];
+ x3r = a[6] - a[14];
+ x3i = a[7] - a[15];
+ y4r = x0r + x2r;
+ y4i = x0i + x2i;
+ y6r = x0r - x2r;
+ y6i = x0i - x2i;
+ x0r = x1r - x3i;
+ x0i = x1i + x3r;
+ x2r = x1r + x3i;
+ x2i = x1i - x3r;
+ y5r = picofftsg_mult_w_a(wn4r, (x0r - x0i));
+ y5i = picofftsg_mult_w_a(wn4r, (x0r + x0i));
+ y7r = picofftsg_mult_w_a(wn4r, (x2r - x2i));
+ y7i = picofftsg_mult_w_a(wn4r, (x2r + x2i));
+ a[8] = y1r + y5r;
+ a[9] = y1i + y5i;
+ a[10] = y1r - y5r;
+ a[11] = y1i - y5i;
+ a[12] = y3r - y7i;
+ a[13] = y3i + y7r;
+ a[14] = y3r + y7i;
+ a[15] = y3i - y7r;
+ a[0] = y0r + y4r;
+ a[1] = y0i + y4i;
+ a[2] = y0r - y4r;
+ a[3] = y0i - y4i;
+ a[4] = y2r - y6i;
+ a[5] = y2i + y6r;
+ a[6] = y2r + y6i;
+ a[7] = y2i - y6r;
+}
+
+
+void cftf082(PICOFFTSG_FFTTYPE *a)
+{
+ PICOFFTSG_FFTTYPE wn4r, wk1r, wk1i, x0r, x0i, x1r, x1i,
+ y0r, y0i, y1r, y1i, y2r, y2i, y3r, y3i,
+ y4r, y4i, y5r, y5i, y6r, y6i, y7r, y7i;
+
+ wn4r = WR5000;
+ wk1r = WR2500;
+ wk1i = WI2500;
+ y0r = a[0] - a[9];
+ y0i = a[1] + a[8];
+ y1r = a[0] + a[9];
+ y1i = a[1] - a[8];
+ x0r = a[4] - a[13];
+ x0i = a[5] + a[12];
+ y2r = picofftsg_mult_w_a(wn4r, (x0r - x0i));
+ y2i = picofftsg_mult_w_a(wn4r, (x0i + x0r));
+ x0r = a[4] + a[13];
+ x0i = a[5] - a[12];
+ y3r = picofftsg_mult_w_a(wn4r, (x0r - x0i));
+ y3i = picofftsg_mult_w_a(wn4r, (x0i + x0r));
+ x0r = a[2] - a[11];
+ x0i = a[3] + a[10];
+ y4r = Mult_W_W(wk1r, x0r) - Mult_W_W(wk1i, x0i);
+ y4i = Mult_W_W(wk1r, x0i) + Mult_W_W(wk1i, x0r);
+ x0r = a[2] + a[11];
+ x0i = a[3] - a[10];
+ y5r = Mult_W_W(wk1i, x0r) - Mult_W_W(wk1r, x0i);
+ y5i = Mult_W_W(wk1i, x0i) + Mult_W_W(wk1r, x0r);
+ x0r = a[6] - a[15];
+ x0i = a[7] + a[14];
+ y6r = Mult_W_W(wk1i, x0r) - Mult_W_W(wk1r, x0i);
+ y6i = Mult_W_W(wk1i, x0i) + Mult_W_W(wk1r, x0r);
+ x0r = a[6] + a[15];
+ x0i = a[7] - a[14];
+ y7r = Mult_W_W(wk1r, x0r) - Mult_W_W(wk1i, x0i);
+ y7i = Mult_W_W(wk1r, x0i) + Mult_W_W(wk1i, x0r);
+ x0r = y0r + y2r;
+ x0i = y0i + y2i;
+ x1r = y4r + y6r;
+ x1i = y4i + y6i;
+ a[0] = x0r + x1r;
+ a[1] = x0i + x1i;
+ a[2] = x0r - x1r;
+ a[3] = x0i - x1i;
+ x0r = y0r - y2r;
+ x0i = y0i - y2i;
+ x1r = y4r - y6r;
+ x1i = y4i - y6i;
+ a[4] = x0r - x1i;
+ a[5] = x0i + x1r;
+ a[6] = x0r + x1i;
+ a[7] = x0i - x1r;
+ x0r = y1r - y3i;
+ x0i = y1i + y3r;
+ x1r = y5r - y7r;
+ x1i = y5i - y7i;
+ a[8] = x0r + x1r;
+ a[9] = x0i + x1i;
+ a[10] = x0r - x1r;
+ a[11] = x0i - x1i;
+ x0r = y1r + y3i;
+ x0i = y1i - y3r;
+ x1r = y5r + y7r;
+ x1i = y5i + y7i;
+ a[12] = x0r - x1i;
+ a[13] = x0i + x1r;
+ a[14] = x0r + x1i;
+ a[15] = x0i - x1r;
+}
+
+
+void cftf040(PICOFFTSG_FFTTYPE *a)
+{
+ PICOFFTSG_FFTTYPE x0r, x0i, x1r, x1i, x2r, x2i, x3r, x3i;
+
+ x0r = a[0] + a[4];
+ x0i = a[1] + a[5];
+ x1r = a[0] - a[4];
+ x1i = a[1] - a[5];
+ x2r = a[2] + a[6];
+ x2i = a[3] + a[7];
+ x3r = a[2] - a[6];
+ x3i = a[3] - a[7];
+ a[0] = x0r + x2r;
+ a[1] = x0i + x2i;
+ a[2] = x1r - x3i;
+ a[3] = x1i + x3r;
+ a[4] = x0r - x2r;
+ a[5] = x0i - x2i;
+ a[6] = x1r + x3i;
+ a[7] = x1i - x3r;
+}
+
+
+void cftb040(PICOFFTSG_FFTTYPE *a)
+{
+ PICOFFTSG_FFTTYPE x0r, x0i, x1r, x1i, x2r, x2i, x3r, x3i;
+
+ x0r = a[0] + a[4];
+ x0i = a[1] + a[5];
+ x1r = a[0] - a[4];
+ x1i = a[1] - a[5];
+ x2r = a[2] + a[6];
+ x2i = a[3] + a[7];
+ x3r = a[2] - a[6];
+ x3i = a[3] - a[7];
+ a[0] = x0r + x2r;
+ a[1] = x0i + x2i;
+ a[2] = x1r + x3i;
+ a[3] = x1i - x3r;
+ a[4] = x0r - x2r;
+ a[5] = x0i - x2i;
+ a[6] = x1r - x3i;
+ a[7] = x1i + x3r;
+}
+
+
+void cftx020(PICOFFTSG_FFTTYPE *a)
+{
+ PICOFFTSG_FFTTYPE x0r, x0i;
+
+ x0r = a[0] - a[2];
+ x0i = a[1] - a[3];
+ a[0] += a[2];
+ a[1] += a[3];
+ a[2] = x0r;
+ a[3] = x0i;
+}
+
+
+void rftfsub(picoos_int32 n, PICOFFTSG_FFTTYPE *a)
+{
+ picoos_int32 i, i0, j, k;
+ PICOFFTSG_FFTTYPE w1r, w1i, wkr, wki, wdr, wdi, ss, xr, xi, yr, yi;
+
+ wkr = 0;
+ wki = 0;
+
+ switch (n) {
+ case 8 :
+ wdi=(PICOFFTSG_FFTTYPE)(0.353553414345*PICODSP_WGT_SHIFT); wdr=(PICOFFTSG_FFTTYPE)(0.146446630359*PICODSP_WGT_SHIFT);
+ w1r=(PICOFFTSG_FFTTYPE)(0.707106709480*PICODSP_WGT_SHIFT); w1i=(PICOFFTSG_FFTTYPE)(0.707106828690*PICODSP_WGT_SHIFT);
+ ss =(PICOFFTSG_FFTTYPE)(1.414213657379*PICODSP_WGT_SHIFT); break;
+ case 16 :
+ wdi=(PICOFFTSG_FFTTYPE)(0.191341713071*PICODSP_WGT_SHIFT); wdr=(PICOFFTSG_FFTTYPE)(0.038060232997*PICODSP_WGT_SHIFT);
+ w1r=(PICOFFTSG_FFTTYPE)(0.923879504204*PICODSP_WGT_SHIFT); w1i=(PICOFFTSG_FFTTYPE)(0.382683426142*PICODSP_WGT_SHIFT);
+ ss =(PICOFFTSG_FFTTYPE)(0.765366852283*PICODSP_WGT_SHIFT); break;
+ case 32 :
+ wdi=(PICOFFTSG_FFTTYPE)(0.097545161843*PICODSP_WGT_SHIFT); wdr=(PICOFFTSG_FFTTYPE)(0.009607359767*PICODSP_WGT_SHIFT);
+ w1r=(PICOFFTSG_FFTTYPE)(0.980785250664*PICODSP_WGT_SHIFT); w1i=(PICOFFTSG_FFTTYPE)(0.195090323687*PICODSP_WGT_SHIFT);
+ ss =(PICOFFTSG_FFTTYPE)(0.390180647373*PICODSP_WGT_SHIFT); break;
+ case 64 :
+ wdi=(PICOFFTSG_FFTTYPE)(0.049008570611*PICODSP_WGT_SHIFT); wdr=(PICOFFTSG_FFTTYPE)(0.002407636726*PICODSP_WGT_SHIFT);
+ w1r=(PICOFFTSG_FFTTYPE)(0.995184719563*PICODSP_WGT_SHIFT); w1i=(PICOFFTSG_FFTTYPE)(0.098017141223*PICODSP_WGT_SHIFT);
+ ss =(PICOFFTSG_FFTTYPE)(0.196034282446*PICODSP_WGT_SHIFT); break;
+ default :
+ wdr = 0; wdi = 0; ss = 0;
+ break;
+ }
+
+ i = n >> 1;
+ for (;;) {
+ i0 = i - RDFT_LOOP_DIV_4;
+ if (i0 < 4) {
+ i0 = 4;
+ }
+ for (j = i - 4; j >= i0; j -= 4) {
+ k = n - j;
+ xr = a[j + 2] - a[k - 2];
+ xi = a[j + 3] + a[k - 1];
+ yr = Mult_W_W(wdr, xr) - Mult_W_W(wdi, xi);
+ yi = Mult_W_W(wdr, xi) + Mult_W_W(wdi, xr);
+ a[j + 2] -= yr;
+ a[j + 3] -= yi;
+ a[k - 2] += yr;
+ a[k - 1] -= yi;
+ wkr += Mult_W_W(ss, wdi);
+ wki += picofftsg_mult_w_w(ss, (PICOFFTSG_WGT_SHIFT2 - wdr));
+ xr = a[j] - a[k];
+ xi = a[j + 1] + a[k + 1];
+ yr = Mult_W_W(wkr, xr) - Mult_W_W(wki, xi);
+ yi = Mult_W_W(wkr, xi) + Mult_W_W(wki, xr);
+ a[j] -= yr;
+ a[j + 1] -= yi;
+ a[k] += yr;
+ a[k + 1] -= yi;
+ wdr += Mult_W_W(ss, wki);
+ wdi += picofftsg_mult_w_w(ss, (PICOFFTSG_WGT_SHIFT2 - wkr));
+ }
+ if (i0 == 4) {
+ break;
+ }
+ }
+
+ xr = a[2] - a[n - 2];
+ xi = a[3] + a[n - 1];
+ yr = Mult_W_W(wdr, xr) - Mult_W_W(wdi, xi);
+ yi = Mult_W_W(wdr, xi) + Mult_W_W(wdi, xr);
+
+ a[2] -= yr;
+ a[3] -= yi;
+ a[n - 2] += yr;
+ a[n - 1] -= yi;
+
+
+}
+
+
+void rftbsub(picoos_int32 n, PICOFFTSG_FFTTYPE *a)
+{
+ picoos_int32 i, i0, j, k;
+ PICOFFTSG_FFTTYPE w1r, w1i, wkr, wki, wdr, wdi, ss, xr, xi, yr, yi;
+ wkr = 0;
+ wki = 0;
+ wdi=(PICOFFTSG_FFTTYPE)(0.012270614505*PICODSP_WGT_SHIFT);
+ wdr=(PICOFFTSG_FFTTYPE)(0.000150590655*PICODSP_WGT_SHIFT);
+ w1r=(PICOFFTSG_FFTTYPE)(0.999698817730*PICODSP_WGT_SHIFT);
+ w1i=(PICOFFTSG_FFTTYPE)(0.024541229010*PICODSP_WGT_SHIFT);
+ ss=(PICOFFTSG_FFTTYPE)(0.049082458019*PICODSP_WGT_SHIFT);
+
+ i = n >> 1;
+ for (;;) {
+ i0 = i - RDFT_LOOP_DIV4;
+ if (i0 < 4) {
+ i0 = 4;
+ }
+ for (j = i - 4; j >= i0; j -= 4) {
+ k = n - j;
+ xr = a[j + 2] - a[k - 2];
+ xi = a[j + 3] + a[k - 1];
+ yr = Mult_W_W(wdr, xr) + Mult_W_W(wdi, xi);
+ yi = Mult_W_W(wdr, xi) - Mult_W_W(wdi, xr);
+ a[j + 2] -= yr;
+ a[j + 3] -= yi;
+ a[k - 2] += yr;
+ a[k - 1] -= yi;
+ wkr += Mult_W_W(ss, wdi);
+ wki += picofftsg_mult_w_w(ss, (PICOFFTSG_WGT_SHIFT2 - wdr));
+ xr = a[j] - a[k];
+ xi = a[j + 1] + a[k + 1];
+ yr = Mult_W_W(wkr, xr) + Mult_W_W(wki, xi);
+ yi = Mult_W_W(wkr, xi) - Mult_W_W(wki, xr);
+ a[j] -= yr;
+ a[j + 1] -= yi;
+ a[k] += yr;
+ a[k + 1] -= yi;
+ wdr += Mult_W_W(ss, wki);
+ wdi += picofftsg_mult_w_w(ss, (PICOFFTSG_WGT_SHIFT2 - wkr));
+ }
+ if (i0 == 4) {
+ break;
+ }
+ }
+ xr = a[2] - a[n - 2];
+ xi = a[3] + a[n - 1];
+ yr = Mult_W_W(wdr, xr) + Mult_W_W(wdi, xi);
+ yi = Mult_W_W(wdr, xi) - Mult_W_W(wdi, xr);
+ a[2] -= yr;
+ a[3] -= yi;
+ a[n - 2] += yr;
+ a[n - 1] -= yi;
+}
+
+
+void dctsub(picoos_int32 n, PICOFFTSG_FFTTYPE *a)
+{
+ picoos_int32 i, i0, j, k, m;
+ PICOFFTSG_FFTTYPE w1r, w1i, wkr, wki, wdr, wdi, ss, xr, xi, yr, yi;
+ wkr = (PICOFFTSG_FFTTYPE)(0.5*PICODSP_WGT_SHIFT);
+ wki = (PICOFFTSG_FFTTYPE)(0.5*PICODSP_WGT_SHIFT);
+
+ switch (n) {
+ case 8 : wdi=(PICOFFTSG_FFTTYPE)(0.587937772274*PICODSP_WGT_SHIFT);
+ wdr=(PICOFFTSG_FFTTYPE)(0.392847478390*PICODSP_WGT_SHIFT); w1r=(PICOFFTSG_FFTTYPE)(0.980785250664*PICODSP_WGT_SHIFT);
+ w1i=(PICOFFTSG_FFTTYPE)(0.195090323687*PICODSP_WGT_SHIFT); ss =(PICOFFTSG_FFTTYPE)(0.390180647373*PICODSP_WGT_SHIFT); break;
+ case 16 : wdi=(PICOFFTSG_FFTTYPE)(0.546600937843*PICODSP_WGT_SHIFT);
+ wdr=(PICOFFTSG_FFTTYPE)(0.448583781719*PICODSP_WGT_SHIFT); w1r=(PICOFFTSG_FFTTYPE)(0.995184719563*PICODSP_WGT_SHIFT);
+ w1i=(PICOFFTSG_FFTTYPE)(0.098017141223*PICODSP_WGT_SHIFT); ss =(PICOFFTSG_FFTTYPE)(0.196034282446*PICODSP_WGT_SHIFT); break;
+ case 32 : wdi=(PICOFFTSG_FFTTYPE)(0.523931562901*PICODSP_WGT_SHIFT);
+ wdr=(PICOFFTSG_FFTTYPE)(0.474863886833*PICODSP_WGT_SHIFT); w1r=(PICOFFTSG_FFTTYPE)(0.998795449734*PICODSP_WGT_SHIFT);
+ w1i=(PICOFFTSG_FFTTYPE)(0.049067676067*PICODSP_WGT_SHIFT); ss =(PICOFFTSG_FFTTYPE)(0.098135352135*PICODSP_WGT_SHIFT); break;
+ case 64 : wdi=(PICOFFTSG_FFTTYPE)(0.512120008469*PICODSP_WGT_SHIFT);
+ wdr=(PICOFFTSG_FFTTYPE)(0.487578809261*PICODSP_WGT_SHIFT); w1r=(PICOFFTSG_FFTTYPE)(0.999698817730*PICODSP_WGT_SHIFT);
+ w1i=(PICOFFTSG_FFTTYPE)(0.024541229010*PICODSP_WGT_SHIFT); ss =(PICOFFTSG_FFTTYPE)(0.049082458019*PICODSP_WGT_SHIFT); break;
+ default:
+ wdr = 0; wdi = 0; ss = 0; break;
+ }
+
+ m = n >> 1;
+ i = 0;
+ for (;;) {
+ i0 = i + DCST_LOOP_DIV2;
+ if (i0 > m - 2) {
+ i0 = m - 2;
+ }
+ for (j = i + 2; j <= i0; j += 2) {
+ k = n - j;
+ xr = picofftsg_mult_w_a(wdi, a[j - 1]) - picofftsg_mult_w_a(wdr, a[k + 1]);
+ xi = picofftsg_mult_w_a(wdr, a[j - 1]) + picofftsg_mult_w_a(wdi, a[k + 1]);
+ wkr -= Mult_W_W(ss, wdi);
+ wki += Mult_W_W(ss, wdr);
+ yr = Mult_W_W(wki, a[j]) - Mult_W_W(wkr, a[k]);
+ yi = Mult_W_W(wkr, a[j]) + Mult_W_W(wki, a[k]);
+ wdr -= Mult_W_W(ss, wki);
+ wdi += Mult_W_W(ss, wkr);
+ a[k + 1] = xr;
+ a[k] = yr;
+ a[j - 1] = xi;
+ a[j] = yi;
+ }
+ if (i0 == m - 2) {
+ break;
+ }
+ }
+ xr = picofftsg_mult_w_a(wdi, a[m - 1]) - picofftsg_mult_w_a(wdr, a[m + 1]);
+ a[m - 1] = picofftsg_mult_w_a(wdr, a[m - 1]) + picofftsg_mult_w_a(wdi, a[m + 1]);
+ a[m + 1] = xr;
+ a[m] = Mult_W_W(WR5000, a[m]);
+}
+
+
+void dctsub4(picoos_int32 n, PICOFFTSG_FFTTYPE *a)
+{
+ picoos_int32 m;
+ PICOFFTSG_FFTTYPE wki, wdr, wdi, xr;
+
+ wki = WR5000;
+ m = n >> 1;
+ if (m == 2) {
+ wdr = Mult_W_W(wki, WI2500);
+ wdi = Mult_W_W(wki, WR2500);
+ xr = Mult_W_W(wdi, a[1]) - Mult_W_W(wdr, a[3]);
+ a[1] = Mult_W_W(wdr, a[1]) + Mult_W_W(wdi, a[3]);
+ a[3] = xr;
+ }
+ a[m] = Mult_W_W(wki, a[m]);
+}
+
+#ifdef __cplusplus
+}
+#endif
diff --git a/lib/picofftsg.h b/lib/picofftsg.h
new file mode 100644
index 0000000..d9c6a1a
--- /dev/null
+++ b/lib/picofftsg.h
@@ -0,0 +1,54 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picofftsg.h
+ *
+ * include file for FFT/DCT related data types, constants and functions in Pico
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+*/
+#ifndef PICOFFTSG_H_
+#define PCOFFTSG_H_
+
+#include "picoos.h"
+#include "picodsp.h"
+
+
+#ifdef __cplusplus
+extern "C" {
+
+#endif
+#if 0
+}
+#endif
+
+#define PICOFFTSG_FFTTYPE picoos_int32
+
+extern void rdft(int n, int isgn, PICOFFTSG_FFTTYPE *a);
+extern void dfct(int n, float *a, int VAL_SHIFT);
+extern void dfct_nmf(int n, int *a);
+extern float norm_result(int m2, PICOFFTSG_FFTTYPE *tmpX, PICOFFTSG_FFTTYPE *norm_window);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
diff --git a/lib/picokdbg.c b/lib/picokdbg.c
new file mode 100644
index 0000000..c0336ba
--- /dev/null
+++ b/lib/picokdbg.c
@@ -0,0 +1,173 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picokdbg.c
+ *
+ * debug support knowledge base
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#include "picoos.h"
+#include "picoknow.h"
+#include "picodbg.h"
+#include "picokdbg.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+#if defined(PICO_DEBUG)
+
+/**
+ * @addtogroup picokdbg
+
+ * <b> Pico Debug Support for knowledge base </b>\n
+ *
+
+ * @b Phones
+
+ * overview of binary file format for dbg kb:
+
+ dbg-kb = phonesyms
+
+ phonesyms = {PHONESYM8}=256
+
+ PHONESYM6: 8 bytes, symbol name (must be 0 terminated), the
+ corresponding ID corresponds to the offset in the
+ phonesyms array
+*/
+
+/* maximum length of phonesym string including terminating 0 */
+#define KDBG_PHONESYMLEN_MAX 8
+
+
+typedef struct kdbg_subobj *kdbg_SubObj;
+
+typedef struct kdbg_subobj {
+ picoos_uint8 *phonesyms;
+} kdbg_subobj_t;
+
+
+static pico_status_t kdbgInitialize(register picoknow_KnowledgeBase this,
+ picoos_Common common) {
+ kdbg_subobj_t *kdbg;
+
+ PICODBG_DEBUG(("start"));
+
+ if (NULL == this || NULL == this->subObj) {
+ PICODBG_DEBUG(("2nd check failed"));
+ return picoos_emRaiseException(common->em, PICO_ERR_OTHER, NULL, NULL);
+ }
+ kdbg = (kdbg_subobj_t *)this->subObj;
+ kdbg->phonesyms = this->base;
+ return PICO_OK;
+}
+
+
+static pico_status_t kdbgSubObjDeallocate(register picoknow_KnowledgeBase this,
+ picoos_MemoryManager mm) {
+ if (NULL != this) {
+ picoos_deallocate(mm, (void *) &this->subObj);
+ }
+ return PICO_OK;
+}
+
+
+pico_status_t picokdbg_specializeDbgKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common) {
+ if (NULL == this) {
+ PICODBG_INFO(("no debug symbols loaded"));
+ return PICO_OK;
+ }
+ this->subDeallocate = kdbgSubObjDeallocate;
+ this->subObj = picoos_allocate(common->mm, sizeof(kdbg_subobj_t));
+ if (NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM,
+ NULL, NULL);
+ }
+ return kdbgInitialize(this, common);
+}
+
+
+picokdbg_Dbg picokdbg_getDbg(picoknow_KnowledgeBase this) {
+ if (NULL == this) {
+ return NULL;
+ } else {
+ return (picokdbg_Dbg)this->subObj;
+ }
+}
+
+
+/* Dbg methods */
+
+picoos_uint8 picokdbg_getPhoneId(const picokdbg_Dbg this,
+ const picoos_char *phsym) {
+ kdbg_subobj_t *kdbg;
+ picoos_uint16 i;
+
+ if (this == NULL)
+ return 0;
+
+ kdbg = (kdbg_subobj_t *)this;
+ /* sequential search */
+ for (i = 0; i < 256; i++) {
+ if (!picoos_strcmp(phsym,
+ (picoos_char *)&(kdbg->phonesyms[i * KDBG_PHONESYMLEN_MAX])))
+ return (picoos_uint8)i;
+ }
+ return 0;
+}
+
+
+picoos_char *picokdbg_getPhoneSym(const picokdbg_Dbg this,
+ const picoos_uint8 phid) {
+ kdbg_subobj_t *kdbg;
+
+ if (this == NULL)
+ return NULL;
+
+ kdbg = (kdbg_subobj_t *)this;
+ return (picoos_char *)&(kdbg->phonesyms[phid * KDBG_PHONESYMLEN_MAX]);
+}
+
+
+
+#else
+
+/* To prevent warning about "translation unit is empty" when
+ diagnostic output is disabled. */
+static void picokdbg_dummy(void) {
+ picokdbg_dummy();
+}
+
+
+#endif /* defined(PICO_DEBUG) */
+
+#ifdef __cplusplus
+}
+#endif
+
+
+/* end */
diff --git a/lib/picokdbg.h b/lib/picokdbg.h
new file mode 100644
index 0000000..c17942f
--- /dev/null
+++ b/lib/picokdbg.h
@@ -0,0 +1,82 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picokdbg.h
+ *
+ * debug support knowledge base
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *- 0.1, 08.05.2008, MRi - initial version
+ *
+ */
+
+#ifndef PICOKDBG_H_
+#define PICOKDBG_H_
+
+
+#include "picoos.h"
+#include "picoknow.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* ************************************************************/
+/* Dbg type and functions */
+/* ************************************************************/
+
+/**
+ * to be used by picorsrc only
+ */
+pico_status_t picokdbg_specializeDbgKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common);
+
+typedef struct picokdbg_dbg *picokdbg_Dbg;
+
+/**
+ * return kb Phones for usage in PU
+ */
+picokdbg_Dbg picokdbg_getDbg(picoknow_KnowledgeBase this);
+
+
+/* phone ID - phone symbol conversion functions */
+
+/**
+ * return phone ID for phone symbol 'phsym' which must be 0 terminated
+ */
+picoos_uint8 picokdbg_getPhoneId(const picokdbg_Dbg dbg,
+ const picoos_char *phsym);
+
+/**
+ * return pointer to phone symbol for phone ID phid
+ */
+picoos_char *picokdbg_getPhoneSym(const picokdbg_Dbg dbg,
+ const picoos_uint8 phid);
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif /*PICOKDBG_H_*/
diff --git a/lib/picokdt.c b/lib/picokdt.c
new file mode 100644
index 0000000..54e36ac
--- /dev/null
+++ b/lib/picokdt.c
@@ -0,0 +1,2642 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picokdt.c
+ *
+ * knowledge handling for decision trees
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#include "picoos.h"
+#include "picodbg.h"
+#include "picobase.h"
+#include "picoknow.h"
+#include "picodata.h"
+#include "picokdt.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* ************************************************************/
+/* decision tree */
+/* ************************************************************/
+
+/**
+ * @addtogroup picokdt
+ * ---------------------------------------------------\n
+ * <b> Pico KDT support </b>\n
+ * ---------------------------------------------------\n
+ overview extended binary tree file:
+ - dt consists of optional attribute mapping tables and a non-empty
+ tree part
+ - using the attribute mapping tables an attribute value as used
+ throughout the TTS can be mapped to its smaller representation
+ used in the tree
+ - multi-byte values always little endian
+
+ -------------------------------------------------------------------
+ - bin-file, decision tree knowledge base in binary form
+
+ - dt-kb = header inputmaptables outputmaptables tree
+
+
+ - header = INPMAPTABLEPOS2 OUTMAPTABLEPOS2 TREEPOS2
+
+ - INPMAPTABLEPOS2: two bytes, equals offest in number of bytes from
+ the start of kb to the start of input map tables,
+ may not be 0
+ - OUTMAPTABLEPOS2: two bytes, equals offest in number of bytes from
+ the start of kb to the start of outtables,
+ may not be 0
+ - TREEPOS2: two bytes, equals offest in number of bytes from the
+ start of kb to the start of the tree
+
+
+ - inputmaptables = maptables
+ - outputmaptables = maptables
+ - maptables = NRMAPTABLES1 {maptable}=NRMAPTABLES1
+ - maptable = LENTABLE2 TABLETYPE1 ( bytemaptable
+ | wordmaptable
+ | graphinmaptable
+ | bytetovarmaptable )
+ - bytemaptable (in or out, usage varies) = NRBYTES2 {BYTE1}=NRBYTES2
+ - wordmaptable (in or out, usage varies) = NRWORDS2 {WORD2}=NRWORDS2
+ - graphinmaptable (in only) = NRGRAPHS2 {GRAPH1:4}=NRGRAPHS2
+ - bytetovarmaptable (out only) = NRINBYTES2 outvarsearchind
+ outvaroutputs
+ - outvarsearchind = {OUTVAROFFSET2}=NRINBYTES2
+ - outvaroutputs = {VARVALID1:}=NRINBYTES2
+
+ - bytemaptable: fixed size, *Map*Fixed \n
+ - wordmaptable: fixed size, *Map*Fixed \n
+ - graphinmaptable: search value is variable size (UTF8 grapheme), \n
+ value to be mapped to is fixed size, one byte \n
+ - bytetovarmaptable: search value is fixed size, one byte, values \n
+ to be mapped to are of variable size (e.g. several \n
+ phones) \n
+
+ - NRMAPTABLES1: one byte representing the number of map tables
+ - LENTABLE2: two bytes, equals offset to the next table (or next
+ part of kb, e.g. tree),
+ if LENTABLE2 = 3, and
+ TABLETYPE1 = EMPTY -> empty table, no mapping to be done
+ - TABLETYPE1: one byte, type of map table (byte, word, or graph=utf8)
+ - NRBYTES2: two bytes, number of bytes following in the table (one
+ would be okay, to simplify some implementation also set
+ to 2)
+ - BYTE1: one btye, the sequence is used to determine the values
+ being mapped to, starting with 0
+ - NRWORDS2: two bytes, number of words (two btyes) following in the table
+ - WORD2: two bytes, the sequence is used to determine the values
+ being mapped to, starting with 0
+ - NRGRAPHS2: two bytes, number of graphemes encoded in UTF8 following
+ in table
+ - GRAPH1:4: one to four bytes, UTF8 representation of a grapheme, the
+ sequence of graphemes is used to determine the value being
+ mapped to, starting with 0, the length information is
+ encoded in UTF8, no need for extra length info
+ - NRINBYTES2: two bytes, number of single byte IDs the tree can produce
+ - OUTVAROFFSET2: two bytes, offset from the start of the
+ outvaroutputs to the start of the following output
+ phone ID group, ie. the first outvaroffset is the
+ offset to the start of the second PHONEID
+ group. Using the previous outvaroffset (or the start
+ of the outvaroutputs) the start and lenth of the
+ PHONEID group can be determined and we can get the
+ sequence of output values we map the chunk value to
+ - VARVALID1:: one to several bytes, one byte each for an output phone ID
+
+ - tree = treenodeinfos TREEBODYSIZE4 treebody
+ - treenodeinfos = NRVFIELDS1 vfields NRATTRIBUTES1 NRQFIELDS1 qfields
+ - vfields = {VFIELD1}=NRVFIELDS1
+ - qfields = {QFIELD1}=NRATTRIBUTES1xNRQFIELDS1
+ - treebody = "cf. code"
+
+ - TREEBODYSIZE4: four bytes, size of treebody in number of bytes
+ - NRVFIELDS1: one byte, number of node properties in the following
+ vector (predefined and fixed sequence of properties)
+ - VFIELD1: number of bits used to represent a node property
+ - NRATTRIBUTES1: one byte, number of attributes (rows) in the
+ following matrix
+ - NRQFIELDS1: one byte, number (columns) of question-dependent node
+ properties per attribute in the following matrix
+ (predefined and fixed sequence of properties)
+ - QFIELD1: number of bits used to represent a question-dependent
+ property in the matrix
+
+
+ - Currently,
+ - NRVFIELDS1 is fixed at 2 for all trees, ie.
+ - vfields = 2 aVFIELD1 bVFIELD1
+ - aVFIELD1: nr of bits for questions
+ - bVFIELD1: nr of bits for decisions
+
+ - NRQFIELDS1 is fixed at 5 for all trees, ie. \n
+ - qfields = NRATTRIBUTES1 5 aQFIELD1 bQFIELD1 cQFIELD1 dQFIELD1 eQFIELD1 \n
+ - aQFIELD1: nr of bits for fork count \n
+ - bQFIELD1: nr of bits for start position for subsets \n
+ - cQFIELD1: nr of bits for group size \n
+ - dQFIELD1: nr of bits for offset to reach output \n
+ - eQFIELD1: nr of bits for threshold (if continuous node) \n
+*/
+
+
+/* ************************************************************/
+/* decision tree data defines */
+/* may not be changed with current implementation */
+/* ************************************************************/
+
+/* maptables fields */
+#define PICOKDT_MTSPOS_NRMAPTABLES 0
+
+/* position of first byte of first maptable (for omt the only table */
+#define PICOKDT_MTPOS_START 1
+
+/* maptable fields */
+#define PICOKDT_MTPOS_LENTABLE 0
+#define PICOKDT_MTPOS_TABLETYPE 2
+#define PICOKDT_MTPOS_NUMBER 3
+#define PICOKDT_MTPOS_MAPSTART 5
+
+/* treenodeinfos fields */
+#define PICOKDT_NIPOS_NRVFIELDS 0
+#define PICOKDT_NIPOS_NRATTS 3
+#define PICOKDT_NIPOS_NRQFIELDS 4
+
+/* fixed treenodeinfos number of fields */
+#define PICOKDT_NODEINFO_NRVFIELDS 2
+#define PICOKDT_NODEINFO_NRQFIELDS 5
+
+/* fixed number of bits used */
+#define PICOKDT_NODETYPE_NRBITS 2
+#define PICOKDT_SUBSETTYPE_NRBITS 2
+#define PICOKDT_ISDECIDE_NRBITS 1
+
+/* number of inpmaptables for each tree. Since we have a possibly
+ empty input map table for each att, currently these values must be
+ equal to PICOKDT_NRATT* */
+typedef enum {
+ PICOKDT_NRINPMT_POSP = 12,
+ PICOKDT_NRINPMT_POSD = 7,
+ PICOKDT_NRINPMT_G2P = 16,
+ PICOKDT_NRINPMT_PHR = 8,
+ PICOKDT_NRINPMT_ACC = 13,
+ PICOKDT_NRINPMT_PAM = 60
+} kdt_nrinpmaptables_t;
+
+/* number of outmaptables for each tree, at least one, possibly empty,
+ output map table for each tree */
+typedef enum {
+ PICOKDT_NROUTMT_POSP = 1,
+ PICOKDT_NROUTMT_POSD = 1,
+ PICOKDT_NROUTMT_G2P = 1,
+ PICOKDT_NROUTMT_PHR = 1,
+ PICOKDT_NROUTMT_ACC = 1,
+ PICOKDT_NROUTMT_PAM = 1
+} kdt_nroutmaptables_t;
+
+/* maptable types */
+typedef enum {
+ PICOKDT_MTTYPE_EMPTY = 0,
+ PICOKDT_MTTYPE_BYTE = 1,
+ PICOKDT_MTTYPE_WORD = 2,
+ PICOKDT_MTTYPE_GRAPH = 3,
+ PICOKDT_MTTYPE_BYTETOVAR = 4
+} kdt_mttype_t;
+
+
+/* ************************************************************/
+/* decision tree types and loading */
+/* ************************************************************/
+/* object : Dt*KnowledgeBase
+ * shortcut : kdt*
+ * derived from : picoknow_KnowledgeBase
+ */
+
+/* subobj shared by all decision trees */
+typedef struct {
+ picokdt_kdttype_t type;
+ picoos_uint8 *inpmaptable;
+ picoos_uint8 *outmaptable;
+ picoos_uint8 *tree;
+ picoos_uint32 beg_offset[128]; /* for efficiency */
+
+ /* tree-internal details for faster processing */
+ picoos_uint8 *vfields;
+ picoos_uint8 *qfields;
+ picoos_uint8 nrattributes;
+ picoos_uint8 *treebody;
+ /*picoos_uint8 nrvfields;*/ /* fix PICOKDT_NODEINFO_NRVFIELDS */
+ /*picoos_uint8 nrqfields;*/ /* fix PICOKDT_NODEINFO_NRQFIELDS */
+
+ /* direct output vector (no output mapping) */
+ picoos_uint8 dset; /* TRUE if class set, FALSE otherwise */
+ picoos_uint16 dclass;
+} kdt_subobj_t;
+
+/* subobj specific for each decision tree type */
+typedef struct {
+ kdt_subobj_t dt;
+ picoos_uint16 invec[PICOKDT_NRATT_POSP]; /* input vector */
+ picoos_uint8 inveclen; /* nr of ele set in invec; must be =nrattributes */
+} kdtposp_subobj_t;
+
+typedef struct {
+ kdt_subobj_t dt;
+ picoos_uint16 invec[PICOKDT_NRATT_POSD]; /* input vector */
+ picoos_uint8 inveclen; /* nr of ele set in invec; must be =nrattributes */
+} kdtposd_subobj_t;
+
+typedef struct {
+ kdt_subobj_t dt;
+ picoos_uint16 invec[PICOKDT_NRATT_G2P]; /* input vector */
+ picoos_uint8 inveclen; /* nr of ele set in invec; must be =nrattributes */
+} kdtg2p_subobj_t;
+
+typedef struct {
+ kdt_subobj_t dt;
+ picoos_uint16 invec[PICOKDT_NRATT_PHR]; /* input vector */
+ picoos_uint8 inveclen; /* nr of ele set in invec; must be =nrattributes */
+} kdtphr_subobj_t;
+
+typedef struct {
+ kdt_subobj_t dt;
+ picoos_uint16 invec[PICOKDT_NRATT_ACC]; /* input vector */
+ picoos_uint8 inveclen; /* nr of ele set in invec; must be =nrattributes */
+} kdtacc_subobj_t;
+
+typedef struct {
+ kdt_subobj_t dt;
+ picoos_uint16 invec[PICOKDT_NRATT_PAM]; /* input vector */
+ picoos_uint8 inveclen; /* nr of ele set in invec; must be =nrattributes */
+} kdtpam_subobj_t;
+
+
+static pico_status_t kdtDtInitialize(register picoknow_KnowledgeBase this,
+ picoos_Common common,
+ kdt_subobj_t *dtp) {
+ picoos_uint16 inppos;
+ picoos_uint16 outpos;
+ picoos_uint16 treepos;
+ picoos_uint32 curpos = 0, pos;
+ picoos_uint16 lentable;
+ picoos_uint16 i;
+ picoos_uint8 imtnr;
+
+ PICODBG_DEBUG(("start"));
+
+ /* get inmap, outmap, tree offsets */
+ if ((PICO_OK == picoos_read_mem_pi_uint16(this->base, &curpos, &inppos))
+ && (PICO_OK == picoos_read_mem_pi_uint16(this->base, &curpos, &outpos))
+ && (PICO_OK == picoos_read_mem_pi_uint16(this->base, &curpos,
+ &treepos))) {
+
+ /* all pos are mandatory, verify */
+ if (inppos && outpos && treepos) {
+ dtp->inpmaptable = this->base + inppos;
+ dtp->outmaptable = this->base + outpos;
+ dtp->tree = this->base + treepos;
+ /* precalc beg offset table */
+ imtnr=dtp->inpmaptable[0];
+ pos=1;
+ dtp->beg_offset[0] = 1;
+ for (i = 0; i < imtnr; i++) {
+ lentable = ((picoos_uint16)(dtp->inpmaptable[pos+1])) << 8 |
+ dtp->inpmaptable[pos];
+ pos += lentable;
+ dtp->beg_offset[i+1] = pos;
+ }
+ } else {
+ dtp->inpmaptable = NULL;
+ dtp->outmaptable = NULL;
+ dtp->tree = NULL;
+ PICODBG_ERROR(("invalid kb position info"));
+ return picoos_emRaiseException(common->em, PICO_EXC_FILE_CORRUPT,
+ NULL, NULL);
+ }
+
+ /* nr of outmaptables is equal 1 for all trees, verify */
+ if (dtp->outmaptable[PICOKDT_MTSPOS_NRMAPTABLES] != 1) {
+ PICODBG_ERROR(("wrong number of outmaptables"));
+ return picoos_emRaiseException(common->em, PICO_EXC_FILE_CORRUPT,
+ NULL, NULL);
+ }
+
+ /* check if this is an empty table, ie. len == 3 */
+ if ((dtp->outmaptable[PICOKDT_MTPOS_START + PICOKDT_MTPOS_LENTABLE]
+ == 3)
+ && (dtp->outmaptable[PICOKDT_MTPOS_START + PICOKDT_MTPOS_LENTABLE
+ + 1] == 0)) {
+ /* verify that this is supposed to be an empty table and
+ set outmaptable to NULL if so */
+ if (dtp->outmaptable[PICOKDT_MTPOS_START + PICOKDT_MTPOS_TABLETYPE]
+ == PICOKDT_MTTYPE_EMPTY) {
+ dtp->outmaptable = NULL;
+ } else {
+ PICODBG_ERROR(("table length vs. type problem"));
+ return picoos_emRaiseException(common->em,
+ PICO_EXC_FILE_CORRUPT,
+ NULL, NULL);
+ }
+ }
+
+ dtp->vfields = dtp->tree + 1;
+ dtp->qfields = dtp->tree + PICOKDT_NODEINFO_NRVFIELDS + 3;
+ dtp->nrattributes = dtp->tree[PICOKDT_NIPOS_NRATTS];
+ dtp->treebody = dtp->qfields + 4 +
+ (dtp->nrattributes * PICOKDT_NODEINFO_NRQFIELDS); /* TREEBODYSIZE4*/
+
+ /*dtp->nrvfields = dtp->tree[PICOKDT_NIPOS_NRVFIELDS]; <- is fix */
+ /*dtp->nrqfields = dtp->tree[PICOKDT_NIPOS_NRQFIELDS]; <- is fix */
+ /* verify that nrvfields ad nrqfields are correct */
+ if ((PICOKDT_NODEINFO_NRVFIELDS != dtp->tree[PICOKDT_NIPOS_NRVFIELDS]) ||
+ (PICOKDT_NODEINFO_NRQFIELDS != dtp->tree[PICOKDT_NIPOS_NRQFIELDS])) {
+ PICODBG_ERROR(("problem with nr of vfields (%d) or qfields (%d)",
+ dtp->tree[PICOKDT_NIPOS_NRVFIELDS],
+ dtp->tree[PICOKDT_NIPOS_NRQFIELDS]));
+ return picoos_emRaiseException(common->em, PICO_EXC_FILE_CORRUPT,
+ NULL, NULL);
+ }
+ dtp->dset = 0;
+ dtp->dclass = 0;
+ PICODBG_DEBUG(("tree init: nratt: %d, posomt: %d, postree: %d",
+ dtp->nrattributes, (dtp->outmaptable - dtp->inpmaptable),
+ (dtp->tree - dtp->inpmaptable)));
+ return PICO_OK;
+ } else {
+ PICODBG_ERROR(("problem reading kb in memory"));
+ return picoos_emRaiseException(common->em, PICO_EXC_FILE_CORRUPT,
+ NULL, NULL);
+ }
+}
+
+
+static pico_status_t kdtDtCheck(register picoknow_KnowledgeBase this,
+ picoos_Common common,
+ kdt_subobj_t *dtp,
+ kdt_nratt_t nratt,
+ kdt_nrinpmaptables_t nrinpmt,
+ kdt_nroutmaptables_t nroutmt,
+ kdt_mttype_t mttype) {
+ /* check nr attributes */
+ /* check nr inpmaptables */
+ /* check nr outmaptables */
+ /* check outmaptable is word type */
+ if ((nratt != dtp->nrattributes)
+ || (dtp->inpmaptable == NULL)
+ || (dtp->outmaptable == NULL)
+ || (dtp->inpmaptable[PICOKDT_MTSPOS_NRMAPTABLES] != nrinpmt)
+ || (dtp->outmaptable[PICOKDT_MTSPOS_NRMAPTABLES] != nroutmt)
+ || (dtp->outmaptable[PICOKDT_MTPOS_START+PICOKDT_MTPOS_TABLETYPE]
+ != mttype)) {
+ PICODBG_ERROR(("check failed, nratt %d, nrimt %d, nromt %d, omttype %d",
+ dtp->nrattributes,
+ dtp->inpmaptable[PICOKDT_MTSPOS_NRMAPTABLES],
+ dtp->outmaptable[PICOKDT_MTSPOS_NRMAPTABLES],
+ dtp->outmaptable[PICOKDT_MTPOS_START +
+ PICOKDT_MTPOS_TABLETYPE]));
+ return picoos_emRaiseException(common->em, PICO_EXC_FILE_CORRUPT,
+ NULL, NULL);
+ }
+ return PICO_OK;
+}
+
+
+
+static pico_status_t kdtPosPInitialize(register picoknow_KnowledgeBase this,
+ picoos_Common common) {
+ pico_status_t status;
+ kdtposp_subobj_t *dtposp;
+ kdt_subobj_t *dt;
+ picoos_uint8 i;
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ dtposp = (kdtposp_subobj_t *)this->subObj;
+ dt = &(dtposp->dt);
+ dt->type = PICOKDT_KDTTYPE_POSP;
+ if ((status = kdtDtInitialize(this, common, dt)) != PICO_OK) {
+ return status;
+ }
+ if ((status = kdtDtCheck(this, common, dt, PICOKDT_NRATT_POSP,
+ PICOKDT_NRINPMT_POSP, PICOKDT_NROUTMT_POSP,
+ PICOKDT_MTTYPE_WORD)) != PICO_OK) {
+ return status;
+ }
+
+ /* init specialized subobj part */
+ for (i = 0; i < PICOKDT_NRATT_POSP; i++) {
+ dtposp->invec[i] = 0;
+ }
+ dtposp->inveclen = 0;
+ PICODBG_DEBUG(("posp tree initialized"));
+ return PICO_OK;
+}
+
+
+static pico_status_t kdtPosDInitialize(register picoknow_KnowledgeBase this,
+ picoos_Common common) {
+ pico_status_t status;
+ kdtposd_subobj_t *dtposd;
+ kdt_subobj_t *dt;
+ picoos_uint8 i;
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ dtposd = (kdtposd_subobj_t *)this->subObj;
+ dt = &(dtposd->dt);
+ dt->type = PICOKDT_KDTTYPE_POSD;
+ if ((status = kdtDtInitialize(this, common, dt)) != PICO_OK) {
+ return status;
+ }
+ if ((status = kdtDtCheck(this, common, dt, PICOKDT_NRATT_POSD,
+ PICOKDT_NRINPMT_POSD, PICOKDT_NROUTMT_POSD,
+ PICOKDT_MTTYPE_WORD)) != PICO_OK) {
+ return status;
+ }
+
+ /* init spezialized subobj part */
+ for (i = 0; i < PICOKDT_NRATT_POSD; i++) {
+ dtposd->invec[i] = 0;
+ }
+ dtposd->inveclen = 0;
+ PICODBG_DEBUG(("posd tree initialized"));
+ return PICO_OK;
+}
+
+
+static pico_status_t kdtG2PInitialize(register picoknow_KnowledgeBase this,
+ picoos_Common common) {
+ pico_status_t status;
+ kdtg2p_subobj_t *dtg2p;
+ kdt_subobj_t *dt;
+ picoos_uint8 i;
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ dtg2p = (kdtg2p_subobj_t *)this->subObj;
+ dt = &(dtg2p->dt);
+ dt->type = PICOKDT_KDTTYPE_G2P;
+ if ((status = kdtDtInitialize(this, common, dt)) != PICO_OK) {
+ return status;
+ }
+
+ if ((status = kdtDtCheck(this, common, dt, PICOKDT_NRATT_G2P,
+ PICOKDT_NRINPMT_G2P, PICOKDT_NROUTMT_G2P,
+ PICOKDT_MTTYPE_BYTETOVAR)) != PICO_OK) {
+ return status;
+ }
+
+ /* init spezialized subobj part */
+ for (i = 0; i < PICOKDT_NRATT_G2P; i++) {
+ dtg2p->invec[i] = 0;
+ }
+ dtg2p->inveclen = 0;
+ PICODBG_DEBUG(("g2p tree initialized"));
+ return PICO_OK;
+}
+
+
+static pico_status_t kdtPhrInitialize(register picoknow_KnowledgeBase this,
+ picoos_Common common) {
+ pico_status_t status;
+ kdtphr_subobj_t *dtphr;
+ kdt_subobj_t *dt;
+ picoos_uint8 i;
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ dtphr = (kdtphr_subobj_t *)this->subObj;
+ dt = &(dtphr->dt);
+ dt->type = PICOKDT_KDTTYPE_PHR;
+ if ((status = kdtDtInitialize(this, common,dt)) != PICO_OK) {
+ return status;
+ }
+
+ if ((status = kdtDtCheck(this, common, dt, PICOKDT_NRATT_PHR,
+ PICOKDT_NRINPMT_PHR, PICOKDT_NROUTMT_PHR,
+ PICOKDT_MTTYPE_WORD)) != PICO_OK) {
+ return status;
+ }
+
+ /* init spezialized subobj part */
+ for (i = 0; i < PICOKDT_NRATT_PHR; i++) {
+ dtphr->invec[i] = 0;
+ }
+ dtphr->inveclen = 0;
+ PICODBG_DEBUG(("phr tree initialized"));
+ return PICO_OK;
+}
+
+
+static pico_status_t kdtAccInitialize(register picoknow_KnowledgeBase this,
+ picoos_Common common) {
+ pico_status_t status;
+ kdtacc_subobj_t *dtacc;
+ kdt_subobj_t *dt;
+ picoos_uint8 i;
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ dtacc = (kdtacc_subobj_t *)this->subObj;
+ dt = &(dtacc->dt);
+ dt->type = PICOKDT_KDTTYPE_ACC;
+ if ((status = kdtDtInitialize(this, common, dt)) != PICO_OK) {
+ return status;
+ }
+
+ if ((status = kdtDtCheck(this, common, dt, PICOKDT_NRATT_ACC,
+ PICOKDT_NRINPMT_ACC, PICOKDT_NROUTMT_ACC,
+ PICOKDT_MTTYPE_WORD)) != PICO_OK) {
+ return status;
+ }
+
+ /* init spezialized subobj part */
+ for (i = 0; i < PICOKDT_NRATT_ACC; i++) {
+ dtacc->invec[i] = 0;
+ }
+ dtacc->inveclen = 0;
+ PICODBG_DEBUG(("acc tree initialized"));
+ return PICO_OK;
+}
+
+
+static pico_status_t kdtPamInitialize(register picoknow_KnowledgeBase this,
+ picoos_Common common) {
+ pico_status_t status;
+ kdtpam_subobj_t *dtpam;
+ kdt_subobj_t *dt;
+ picoos_uint8 i;
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ dtpam = (kdtpam_subobj_t *)this->subObj;
+ dt = &(dtpam->dt);
+ dt->type = PICOKDT_KDTTYPE_PAM;
+ if ((status = kdtDtInitialize(this, common, dt)) != PICO_OK) {
+ return status;
+ }
+
+ if ((status = kdtDtCheck(this, common, dt, PICOKDT_NRATT_PAM,
+ PICOKDT_NRINPMT_PAM, PICOKDT_NROUTMT_PAM,
+ PICOKDT_MTTYPE_WORD)) != PICO_OK) {
+ return status;
+ }
+
+ /* init spezialized subobj part */
+ for (i = 0; i < PICOKDT_NRATT_PAM; i++) {
+ dtpam->invec[i] = 0;
+ }
+ dtpam->inveclen = 0;
+ PICODBG_DEBUG(("pam tree initialized"));
+ return PICO_OK;
+}
+
+
+static pico_status_t kdtSubObjDeallocate(register picoknow_KnowledgeBase this,
+ picoos_MemoryManager mm) {
+ if (NULL != this) {
+ picoos_deallocate(mm, (void *) &this->subObj);
+ }
+ return PICO_OK;
+}
+
+
+/* we don't offer a specialized constructor for a *KnowledgeBase but
+ * instead a "specializer" of an allready existing generic
+ * picoknow_KnowledgeBase */
+
+pico_status_t picokdt_specializeDtKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common,
+ const picokdt_kdttype_t kdttype) {
+ pico_status_t status;
+
+ if (NULL == this) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ this->subDeallocate = kdtSubObjDeallocate;
+ switch (kdttype) {
+ case PICOKDT_KDTTYPE_POSP:
+ this->subObj = picoos_allocate(common->mm,sizeof(kdtposp_subobj_t));
+ if (NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM,
+ NULL, NULL);
+ }
+ status = kdtPosPInitialize(this, common);
+ break;
+ case PICOKDT_KDTTYPE_POSD:
+ this->subObj = picoos_allocate(common->mm,sizeof(kdtposd_subobj_t));
+ if (NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM,
+ NULL, NULL);
+ }
+ status = kdtPosDInitialize(this, common);
+ break;
+ case PICOKDT_KDTTYPE_G2P:
+ this->subObj = picoos_allocate(common->mm,sizeof(kdtg2p_subobj_t));
+ if (NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM,
+ NULL, NULL);
+ }
+ status = kdtG2PInitialize(this, common);
+ break;
+ case PICOKDT_KDTTYPE_PHR:
+ this->subObj = picoos_allocate(common->mm,sizeof(kdtphr_subobj_t));
+ if (NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM,
+ NULL, NULL);
+ }
+ status = kdtPhrInitialize(this, common);
+ break;
+ case PICOKDT_KDTTYPE_ACC:
+ this->subObj = picoos_allocate(common->mm,sizeof(kdtacc_subobj_t));
+ if (NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM,
+ NULL, NULL);
+ }
+ status = kdtAccInitialize(this, common);
+ break;
+ case PICOKDT_KDTTYPE_PAM:
+ this->subObj = picoos_allocate(common->mm,sizeof(kdtpam_subobj_t));
+ if (NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM,
+ NULL, NULL);
+ }
+ status = kdtPamInitialize(this, common);
+ break;
+ default:
+ return picoos_emRaiseException(common->em, PICO_ERR_OTHER,
+ NULL, NULL);
+ }
+
+ if (status != PICO_OK) {
+ picoos_deallocate(common->mm, (void *) &this->subObj);
+ return picoos_emRaiseException(common->em, status, NULL, NULL);
+ }
+ return PICO_OK;
+}
+
+
+/* ************************************************************/
+/* decision tree getDt* */
+/* ************************************************************/
+
+picokdt_DtPosP picokdt_getDtPosP(picoknow_KnowledgeBase this) {
+ return ((NULL == this) ? NULL : ((picokdt_DtPosP) this->subObj));
+}
+
+picokdt_DtPosD picokdt_getDtPosD(picoknow_KnowledgeBase this) {
+ return ((NULL == this) ? NULL : ((picokdt_DtPosD) this->subObj));
+}
+
+picokdt_DtG2P picokdt_getDtG2P (picoknow_KnowledgeBase this) {
+ return ((NULL == this) ? NULL : ((picokdt_DtG2P) this->subObj));
+}
+
+picokdt_DtPHR picokdt_getDtPHR (picoknow_KnowledgeBase this) {
+ return ((NULL == this) ? NULL : ((picokdt_DtPHR) this->subObj));
+}
+
+picokdt_DtACC picokdt_getDtACC (picoknow_KnowledgeBase this) {
+ return ((NULL == this) ? NULL : ((picokdt_DtACC) this->subObj));
+}
+
+picokdt_DtPAM picokdt_getDtPAM (picoknow_KnowledgeBase this) {
+ return ((NULL == this) ? NULL : ((picokdt_DtPAM) this->subObj));
+}
+
+
+
+/* ************************************************************/
+/* decision tree support functions, tree */
+/* ************************************************************/
+
+
+typedef enum {
+ eQuestion = 0, /* index to #bits to identify question */
+ eDecide = 1 /* index to #bits to identify decision */
+} kdt_vfields_ind_t;
+
+typedef enum {
+ eForkCount = 0, /* index to #bits for number of forks */
+ eBitNo = 1, /* index to #bits for index of 1st element */
+ eBitCount = 2, /* index to #bits for size of the group */
+ eJump = 3, /* index to #bits for offset to reach output node */
+ eCut = 4 /* for contin. node: #bits for threshold checked */
+} kdt_qfields_ind_t;
+
+typedef enum {
+ eNTerminal = 0,
+ eNBinary = 1,
+ eNContinuous = 2,
+ eNDiscrete = 3
+} kdt_nodetypes_t;
+
+typedef enum {
+ eOneValue = 0,
+ eTwoValues = 1,
+ eWithoutBitMask = 2,
+ eBitMask = 3
+} kdt_subsettypes_t;
+
+
+/* Name : kdt_jump
+ Function: maps the iJump offset to byte + bit coordinates
+ Input : iJump absolute bit offset (0..(nr-bytes-treebody)*8)
+ Output : iByteNo the first byte containing the bits to extract
+ (0..(nr-bytes-treebody))
+ iBitNo the first bit to be extracted (0..7)
+ Returns : void
+ Notes : updates the iByteNo + iBitNo fields
+*/
+static void kdt_jump(const picoos_uint32 iJump,
+ picoos_uint32 *iByteNo,
+ picoos_int8 *iBitNo) {
+ picoos_uint32 iByteSize;
+
+ iByteSize = (iJump / 8 );
+ *iBitNo = (iJump - (iByteSize * 8)) + (7 - *iBitNo);
+ *iByteNo += iByteSize;
+ if (*iBitNo >= 8) {
+ (*iByteNo)++;
+ *iBitNo = 15 - *iBitNo;
+ } else {
+ *iBitNo = 7 - *iBitNo;
+ }
+}
+
+
+/* replaced inline for speedup */
+/* Name : kdtIsVal
+ Function: Returns the binary value of the bit pointed to by iByteNo, iBitNo
+ Input : iByteNo ofsset to the byte containing the bits to extract
+ (0..sizeof(treebody))
+ iBitNo ofsset to the first bit to be extracted (0..7)
+ Returns : 0/1 depending on the bit pointed to
+*/
+/*
+static picoos_uint8 kdtIsVal(register kdt_subobj_t *this,
+ picoos_uint32 iByteNo,
+ picoos_int8 iBitNo) {
+ return ((this->treebody[iByteNo] & ((1)<<iBitNo)) > 0);
+}
+*/
+
+
+/* @todo : consider replacing inline for speedup */
+
+/* Name : kdtGetQFieldsVal (was: m_QuestDependentFields)
+ Function: gets a byte from qfields
+ Input : this handle to a dt subobj
+ attind index of the attribute
+ qind index of the byte to be read
+ Returns : the requested byte
+ Notes : check that attind < this->nrattributes needed before calling
+ this function!
+*/
+static picoos_uint8 kdtGetQFieldsVal(register kdt_subobj_t *this,
+ const picoos_uint8 attind,
+ const kdt_qfields_ind_t qind) {
+ /* check of qind done in initialize and (for some compilers) with typing */
+ /* check of attind needed before calling this function */
+ return this->qfields[(attind * PICOKDT_NODEINFO_NRQFIELDS) + qind];
+}
+
+
+/* Name : kdtGetShiftVal (was: get_shift_value)
+ Function: returns the (treebody) value pointed to by iByteNo, iBitNo,
+ and with size iSize
+ Input : this reference to the processing unit struct
+ iSize number of bits to be extracted (0..N)
+ iByteNo ofsset to the byte containing the bits to extract
+ (0..sizeof(treebody))
+ iBitNo ofsset to the first bit to be extracted (0..7)
+ Returns : the value requested (if size==0 --> 0 is returned)
+*/
+/*
+static picoos_uint32 orig_kdtGetShiftVal(register kdt_subobj_t *this,
+ const picoos_int16 iSize,
+ picoos_uint32 *iByteNo,
+ picoos_int8 *iBitNo) {
+ picoos_uint32 iVal;
+ picoos_int16 i;
+
+ iVal = 0;
+ for (i = iSize-1; i >= 0; i--) {
+ if ( (this->treebody[*iByteNo] & ((1)<<(*iBitNo))) > 0) {
+ iVal |= ( (1) << i );
+ }
+ (*iBitNo)--;
+ if (*iBitNo < 0) {
+ *iBitNo = 7;
+ (*iByteNo)++;
+ }
+ }
+ return iVal;
+}
+*/
+/* refactor */
+static picoos_uint32 kdtGetShiftVal(register kdt_subobj_t *this,
+ const picoos_int16 iSize, picoos_uint32 *iByteNo, picoos_int8 *iBitNo)
+{
+ picoos_uint32 v, b, iVal;
+ picoos_int16 i, j, len;
+ picoos_uint8 val;
+
+ if (iSize < 4) {
+ iVal = 0;
+ for (i = iSize - 1; i >= 0; i--) {
+ /* no check that *iByteNo is within valid treebody range */
+ if ((this->treebody[*iByteNo] & ((1) << (*iBitNo))) > 0) {
+ iVal |= ((1) << i);
+ }
+ (*iBitNo)--;
+ if (*iBitNo < 0) {
+ *iBitNo = 7;
+ (*iByteNo)++;
+ }
+ }
+ return iVal;
+ }
+
+ b = *iByteNo;
+ j = *iBitNo;
+ len = iSize;
+ *iBitNo = j - iSize;
+ v = 0;
+ while (*iBitNo < 0) {
+ *iBitNo += 8;
+ (*iByteNo)++;
+ }
+
+ val = this->treebody[b++];
+ if (j < 7) {
+ switch (j) {
+ case 0:
+ val &= 0x01;
+ break;
+ case 1:
+ val &= 0x03;
+ break;
+ case 2:
+ val &= 0x07;
+ break;
+ case 3:
+ val &= 0x0f;
+ break;
+ case 4:
+ val &= 0x1f;
+ break;
+ case 5:
+ val &= 0x3f;
+ break;
+ case 6:
+ val &= 0x7f;
+ break;
+ }
+ }
+ len -= j + 1;
+ if (len < 0) {
+ val >>= -len;
+ }
+ v = val;
+ while (len > 0) {
+ if (len >= 8) {
+ j = 8;
+ } else {
+ j = len;
+ }
+ v <<= j;
+ val = this->treebody[b++];
+ if (j < 8) {
+ switch (j) {
+ case 1:
+ val &= 0x80;
+ val >>= 7;
+ break;
+ case 2:
+ val &= 0xc0;
+ val >>= 6;
+ break;
+ case 3:
+ val &= 0xe0;
+ val >>= 5;
+ break;
+ case 4:
+ val &= 0xf0;
+ val >>= 4;
+ break;
+ case 5:
+ val &= 0xf8;
+ val >>= 3;
+ break;
+ case 6:
+ val &= 0xfc;
+ val >>= 2;
+ break;
+ case 7:
+ val &= 0xfe;
+ val >>= 1;
+ break;
+ }
+ }
+ v |= val;
+ len -= j;
+ }
+ return v;
+}
+
+
+/* Name : kdtAskTree
+ Function: Tree Traversal routine
+ Input : iByteNo ofsset to the first byte containing the bits
+ to extract (0..sizeof(treebody))
+ iBitNo ofsset to the first bit to be extracted (0..7)
+ Returns : >0 continue, no solution yet found
+ =0 solution found
+ <0 error, no solution found
+ Notes :
+*/
+static picoos_int8 kdtAskTree(register kdt_subobj_t *this,
+ picoos_uint16 *invec,
+ const kdt_nratt_t invecmax,
+ picoos_uint32 *iByteNo,
+ picoos_int8 *iBitNo) {
+ picoos_uint32 iNodeType;
+ picoos_uint8 iQuestion;
+ picoos_int32 iVal;
+ picoos_int32 iForks;
+ picoos_int32 iID;
+
+ picoos_int32 iCut, iSubsetType, iBitPos, iBitCount, iPos, iJump, iDecision;
+ picoos_int32 i;
+ picoos_char iIsDecide;
+
+ PICODBG_TRACE(("start"));
+
+ /* get node type, value should be in kdt_nodetype_t range */
+ iNodeType = kdtGetShiftVal(this, PICOKDT_NODETYPE_NRBITS, iByteNo, iBitNo);
+ PICODBG_TRACE(("iNodeType: %d", iNodeType));
+
+ /* get attribute to be used in question, check if in range, and get val */
+ /* check of vfields argument done in initialize */
+ iQuestion = kdtGetShiftVal(this, this->vfields[eQuestion], iByteNo, iBitNo);
+ if ((iQuestion < this->nrattributes) && (iQuestion < invecmax)) {
+ iVal = invec[iQuestion];
+ } else {
+ this->dset = FALSE;
+ PICODBG_TRACE(("invalid question"));
+ return -1; /* iQuestion invalid */
+ }
+ iForks = 0;
+ iID = -1;
+ PICODBG_TRACE(("iQuestion: %d", iQuestion));
+
+ switch (iNodeType) {
+ case eNBinary: {
+ iForks = 2;
+ iID = iVal;
+ break;
+ }
+ case eNContinuous: {
+ iForks = 2;
+ iID = 1;
+ iCut = kdtGetShiftVal(this, kdtGetQFieldsVal(this, iQuestion, eCut),
+ iByteNo, iBitNo); /*read the threshold*/
+ if (iVal <= iCut) {
+ iID = 0;
+ }
+ break;
+ }
+ case eNDiscrete: {
+ iForks =
+ kdtGetShiftVal(this,
+ kdtGetQFieldsVal(this, iQuestion, eForkCount),
+ iByteNo, iBitNo);
+
+ for (i = 0; i < iForks-1; i++) {
+ iSubsetType =
+ kdtGetShiftVal(this, PICOKDT_SUBSETTYPE_NRBITS,
+ iByteNo, iBitNo);
+
+ switch (iSubsetType) {
+ case eOneValue: {
+ if (iID > -1) {
+ kdt_jump(kdtGetQFieldsVal(this, iQuestion, eBitNo),
+ iByteNo, iBitNo);
+ break;
+ }
+ iBitPos =
+ kdtGetShiftVal(this,
+ kdtGetQFieldsVal(this, iQuestion,
+ eBitNo),
+ iByteNo, iBitNo);
+ if (iVal == iBitPos) {
+ iID = i;
+ }
+ break;
+ }
+ case eTwoValues: {
+ if (iID > -1) {
+ kdt_jump((kdtGetQFieldsVal(this, iQuestion, eBitNo) +
+ kdtGetQFieldsVal(this, iQuestion, eBitCount)),
+ iByteNo, iBitNo);
+ break;
+ }
+
+ iBitPos =
+ kdtGetShiftVal(this, kdtGetQFieldsVal(this, iQuestion,
+ eBitNo),
+ iByteNo, iBitNo);
+ iBitCount =
+ kdtGetShiftVal(this, kdtGetQFieldsVal(this, iQuestion,
+ eBitCount),
+ iByteNo, iBitNo);
+ if ((iVal == iBitPos) || (iVal == iBitCount)) {
+ iID = i;
+ }
+ break;
+ }
+ case eWithoutBitMask: {
+ if (iID > -1) {
+ kdt_jump((kdtGetQFieldsVal(this, iQuestion, eBitNo) +
+ kdtGetQFieldsVal(this, iQuestion, eBitCount)),
+ iByteNo, iBitNo);
+ break;
+ }
+
+ iBitPos =
+ kdtGetShiftVal(this, kdtGetQFieldsVal(this, iQuestion,
+ eBitNo),
+ iByteNo, iBitNo);
+ iBitCount =
+ kdtGetShiftVal(this, kdtGetQFieldsVal(this, iQuestion,
+ eBitCount),
+ iByteNo, iBitNo);
+ if ((iVal >= iBitPos) && (iVal < (iBitPos + iBitCount))) {
+ iID = i;
+ }
+ break;
+ }
+ case eBitMask: {
+ iBitPos = 0;
+ if (iID > -1) {
+ kdt_jump(kdtGetQFieldsVal(this, iQuestion, eBitNo),
+ iByteNo, iBitNo);
+ } else {
+ iBitPos =
+ kdtGetShiftVal(this,
+ kdtGetQFieldsVal(this, iQuestion,
+ eBitNo),
+ iByteNo, iBitNo);
+ }
+
+ iBitCount =
+ kdtGetShiftVal(this,
+ kdtGetQFieldsVal(this, iQuestion,
+ eBitCount),
+ iByteNo, iBitNo);
+ if (iID > -1) {
+ kdt_jump(iBitCount, iByteNo, iBitNo);
+ break;
+ }
+
+ if ((iVal >= iBitPos) && (iVal < (iBitPos + iBitCount))) {
+ iPos = iVal - iBitPos;
+ kdt_jump((iVal - iBitPos), iByteNo, iBitNo);
+ /* if (kdtIsVal(this, *iByteNo, *iBitNo))*/
+ if ((this->treebody[*iByteNo] & ((1)<<(*iBitNo))) > 0) {
+ iID = i;
+ }
+ kdt_jump((iBitCount - (iVal-iBitPos)), iByteNo, iBitNo);
+ } else {
+ kdt_jump(iBitCount, iByteNo, iBitNo);
+ }
+ break;
+ }/*end case eBitMask*/
+ }/*end switch (iSubsetType)*/
+ }/*end for ( i = 0; i < iForks-1; i++ ) */
+
+ /*default tree branch*/
+ if (-1 == iID) {
+ iID = iForks-1;
+ }
+ break;
+ }/*end case eNDiscrete*/
+ }/*end switch (iNodeType)*/
+
+ for (i = 0; i < iForks; i++) {
+ iIsDecide = kdtGetShiftVal(this, PICOKDT_ISDECIDE_NRBITS, iByteNo, iBitNo);
+
+ PICODBG_TRACE(("doing forks: %d", i));
+
+ if (!iIsDecide) {
+ if (iID == i) {
+ iJump =
+ kdtGetShiftVal(this, kdtGetQFieldsVal(this, iQuestion, eJump),
+ iByteNo, iBitNo);
+ kdt_jump(iJump, iByteNo, iBitNo);
+ this->dset = FALSE;
+ return 1; /* to be continued, no solution yet found */
+ } else {
+ kdt_jump(kdtGetQFieldsVal(this, iQuestion, eJump),
+ iByteNo, iBitNo);
+ }
+ } else {
+ if (iID == i) {
+ /* check of vfields argument done in initialize */
+ iDecision = kdtGetShiftVal(this, this->vfields[eDecide],
+ iByteNo, iBitNo);
+ this->dclass = iDecision;
+ this->dset = TRUE;
+ return 0; /* solution found */
+ } else {
+ /* check of vfields argument done in initialize */
+ kdt_jump(this->vfields[eDecide], iByteNo, iBitNo);
+ }
+ }/*end if (!iIsDecide)*/
+ }/*end for (i = 0; i < iForks; i++ )*/
+
+ this->dset = FALSE;
+ PICODBG_TRACE(("problem determining class"));
+ return -1; /* solution not found, problem determining a class */
+}
+
+
+
+/* ************************************************************/
+/* decision tree support functions, mappings */
+/* ************************************************************/
+
+
+/* size==1 -> MapInByte, size==2 -> MapInWord,
+ size determined from table type contained in kb.
+ if the inmaptable is empty, outval = inval */
+
+static picoos_uint8 kdtMapInFixed(const kdt_subobj_t *dt,
+ const picoos_uint8 imtnr,
+ const picoos_uint16 inval,
+ picoos_uint16 *outval,
+ picoos_uint16 *outfallbackval) {
+ picoos_uint8 size;
+ picoos_uint32 pos;
+ picoos_uint16 lentable;
+ picoos_uint16 posbound;
+ picoos_uint16 i;
+
+ *outval = 0;
+ *outfallbackval = 0;
+
+ size = 0;
+ pos = 0;
+
+ /* check what can be checked */
+ if (imtnr >= dt->inpmaptable[pos++]) { /* outside tablenr range? */
+ PICODBG_ERROR(("check failed: nrtab: %d, imtnr: %d",
+ dt->inpmaptable[pos-1], imtnr));
+ return FALSE;
+ }
+
+ /* go forward to the needed tablenr */
+ if (imtnr > 0) {
+ pos = dt->beg_offset[imtnr];
+ }
+
+ /* get length */
+ lentable = ((picoos_uint16)(dt->inpmaptable[pos+1])) << 8 |
+ dt->inpmaptable[pos];
+ posbound = pos + lentable;
+ pos += 2;
+
+ /* check type of table and set size */
+ if (dt->inpmaptable[pos] == PICOKDT_MTTYPE_EMPTY) {
+ /* empty table no mapping needed */
+ PICODBG_TRACE(("empty table: %d", imtnr));
+ *outval = inval;
+ return TRUE;
+ } else if (dt->inpmaptable[pos] == PICOKDT_MTTYPE_BYTE) {
+ size = 1;
+ } else if (dt->inpmaptable[pos] == PICOKDT_MTTYPE_WORD) {
+ size = 2;
+ } else {
+ /* wrong table type */
+ PICODBG_ERROR(("wrong table type %d", dt->inpmaptable[pos]));
+ return FALSE;
+ }
+ pos++;
+
+ /* set fallback value in case of failed mapping, and set upper bound pos */
+ *outfallbackval = ((picoos_uint16)(dt->inpmaptable[pos+1])) << 8 |
+ dt->inpmaptable[pos];
+ pos += 2;
+
+ /* size must be 1 or 2 here, keep 'redundant' so save time */
+ if (size == 1) {
+ for (i = 0; (i < *outfallbackval) && (pos < posbound); i++) {
+ if (inval == dt->inpmaptable[pos]) {
+ *outval = i;
+ PICODBG_TRACE(("s1 %d in %d -> out %d", imtnr, inval, *outval));
+ return TRUE;
+ }
+ pos++;
+ }
+ } else if (size == 2) {
+ posbound--;
+ for (i = 0; (i < *outfallbackval) && (pos < posbound); i++) {
+ if (inval == (((picoos_uint16)(dt->inpmaptable[pos+1])) << 8 |
+ dt->inpmaptable[pos])) {
+ *outval = i;
+ PICODBG_TRACE(("s2 %d in %d -> out %d", imtnr, inval, *outval));
+ return TRUE;
+ }
+ pos += 2;
+ }
+ } else {
+ /* impossible size */
+ PICODBG_ERROR(("wrong size %d", size));
+ return FALSE;
+ }
+
+ PICODBG_DEBUG(("no mapping found, fallback: %d", *outfallbackval));
+ return FALSE;
+}
+
+
+static picoos_uint8 kdtMapInGraph(const kdt_subobj_t *dt,
+ const picoos_uint8 imtnr,
+ const picoos_uint8 *inval,
+ const picoos_uint8 invalmaxlen,
+ picoos_uint16 *outval,
+ picoos_uint16 *outfallbackval) {
+ picoos_uint8 ilen;
+ picoos_uint8 tlen;
+ picoos_uint8 cont;
+ picoos_uint32 pos;
+ picoos_uint16 lentable;
+ picoos_uint16 posbound;
+ picoos_uint16 i;
+ picoos_uint8 j;
+
+ *outfallbackval = 0;
+
+ pos = 0;
+ /* check what can be checked */
+ if ((imtnr >= dt->inpmaptable[pos++]) || /* outside tablenr range? */
+ (invalmaxlen == 0) || /* too short? */
+ ((ilen = picobase_det_utf8_length(inval[0])) == 0) || /* invalid? */
+ (ilen > invalmaxlen)) { /* not accessible? */
+ PICODBG_ERROR(("check failed: nrtab: %d, imtnr: %d, invalmaxlen: %d, "
+ "ilen: %d",
+ dt->inpmaptable[pos-1], imtnr, invalmaxlen, ilen));
+ return FALSE;
+ }
+
+ /* go forward to the needed tablenr */
+ for (i = 0; i < imtnr; i++) {
+ lentable = ((picoos_uint16)(dt->inpmaptable[pos+1])) << 8 |
+ dt->inpmaptable[pos];
+ pos += lentable;
+ }
+
+ /* get length and check type of inpmaptable */
+ lentable = ((picoos_uint16)(dt->inpmaptable[pos+1])) << 8 |
+ dt->inpmaptable[pos];
+ posbound = pos + lentable;
+ pos += 2;
+
+#if defined(PICO_DEBUG)
+ if (1) {
+ int id;
+ PICODBG_TRACE(("imtnr %d", imtnr));
+ for (id = pos-2; id < posbound; id++) {
+ PICODBG_TRACE(("imtbyte pos %d, %c %d", id - (pos-2),
+ dt->inpmaptable[id], dt->inpmaptable[id]));
+ }
+ }
+#endif
+
+ /* check type of table */
+ if (dt->inpmaptable[pos] != PICOKDT_MTTYPE_GRAPH) {
+ /* empty table does not make sense for graph */
+ /* wrong table type */
+ PICODBG_ERROR(("wrong table type"));
+ return FALSE;
+ }
+ pos++;
+
+ /* set fallback value in case of failed mapping, and set upper bound pos */
+ *outfallbackval = ((picoos_uint16)(dt->inpmaptable[pos+1])) << 8 |
+ dt->inpmaptable[pos];
+ pos += 2;
+
+ /* sequential search */
+ for (i = 0; (i < *outfallbackval) && (pos < posbound); i++) {
+ tlen = picobase_det_utf8_length(dt->inpmaptable[pos]);
+ if ((pos + tlen) > posbound) {
+ PICODBG_ERROR(("trying outside imt, posb: %d, pos: %d, tlen: %d",
+ posbound, pos, tlen));
+ return FALSE;
+ }
+ if (ilen == tlen) {
+ cont = TRUE;
+ for (j = 0; cont && (j < ilen); j++) {
+ if (dt->inpmaptable[pos + j] != inval[j]) {
+ cont = FALSE;
+ }
+ }
+ if (cont && (j == ilen)) { /* match found */
+ *outval = i;
+ PICODBG_TRACE(("found mapval, posb %d, pos %d, i %d, tlen %d",
+ posbound, pos, i, tlen));
+ return TRUE;
+ }
+ }
+ pos += tlen;
+ }
+ PICODBG_DEBUG(("outside imt %d, posb/pos/i: %d/%d/%d, fallback: %d",
+ imtnr, posbound, pos, i, *outfallbackval));
+ return FALSE;
+}
+
+
+/* size==1 -> MapOutByte, size==2 -> MapOutWord */
+static picoos_uint8 kdtMapOutFixed(const kdt_subobj_t *dt,
+ const picoos_uint16 inval,
+ picoos_uint16 *outval) {
+ picoos_uint8 size;
+ picoos_uint16 nr;
+
+ /* no check of lentable vs. nr in initialize done */
+
+ size = 0;
+
+ /* type */
+ nr = dt->outmaptable[PICOKDT_MTPOS_START + PICOKDT_MTPOS_TABLETYPE];
+
+ /* check type of table and set size */
+ if (nr == PICOKDT_MTTYPE_EMPTY) {
+ /* empty table no mapping needed */
+ PICODBG_TRACE(("empty table"));
+ *outval = inval;
+ return TRUE;
+ } else if (nr == PICOKDT_MTTYPE_BYTE) {
+ size = 1;
+ } else if (nr == PICOKDT_MTTYPE_WORD) {
+ size = 2;
+ } else {
+ /* wrong table type */
+ PICODBG_ERROR(("wrong table type %d", nr));
+ return FALSE;
+ }
+
+ /* number of mapvalues */
+ nr = ((picoos_uint16)(dt->outmaptable[PICOKDT_MTPOS_START +
+ PICOKDT_MTPOS_NUMBER + 1])) << 8
+ | dt->outmaptable[PICOKDT_MTPOS_START + PICOKDT_MTPOS_NUMBER];
+
+ if (inval < nr) {
+ if (size == 1) {
+ *outval = dt->outmaptable[PICOKDT_MTPOS_START +
+ PICOKDT_MTPOS_MAPSTART + (size * inval)];
+ } else {
+ *outval = ((picoos_uint16)(dt->outmaptable[PICOKDT_MTPOS_START +
+ PICOKDT_MTPOS_MAPSTART + (size * inval) + 1])) << 8
+ | dt->outmaptable[PICOKDT_MTPOS_START +
+ PICOKDT_MTPOS_MAPSTART + (size * inval)];
+ }
+ return TRUE;
+ } else {
+ *outval = 0;
+ return FALSE;
+ }
+}
+
+
+/* size==1 -> ReverseMapOutByte, size==2 -> ReverseMapOutWord */
+/* outmaptable also used to map from decoded tree output domain to
+ direct tree output domain */
+static picoos_uint8 kdtReverseMapOutFixed(const kdt_subobj_t *dt,
+ const picoos_uint16 inval,
+ picoos_uint16 *outval,
+ picoos_uint16 *outfallbackval) {
+ picoos_uint8 size;
+ picoos_uint32 pos;
+ picoos_uint16 lentable;
+ picoos_uint16 posbound;
+ picoos_uint16 i;
+
+ /* no check of lentable vs. nr in initialize done */
+
+ size = 0;
+ pos = 0;
+ *outval = 0;
+ *outfallbackval = 0;
+
+ if (dt->outmaptable == NULL) {
+ /* empty table no mapping needed */
+ PICODBG_TRACE(("empty table"));
+ *outval = inval;
+ return TRUE;
+ }
+
+ /* check what can be checked */
+ if (dt->outmaptable[pos++] != 1) { /* only one omt possible */
+ PICODBG_ERROR(("check failed: nrtab: %d", dt->outmaptable[pos-1]));
+ return FALSE;
+ }
+
+ /* get length */
+ lentable = ((picoos_uint16)(dt->outmaptable[pos+1])) << 8 |
+ dt->outmaptable[pos];
+ posbound = pos + lentable;
+ pos += 2;
+
+ /* check type of table and set size */
+ /* if (dt->outmaptable[pos] == PICOKDT_MTTYPE_EMPTY), in
+ ...Initialize the omt is set to NULL if not existing, checked
+ above */
+
+ if (dt->outmaptable[pos] == PICOKDT_MTTYPE_BYTE) {
+ size = 1;
+ } else if (dt->outmaptable[pos] == PICOKDT_MTTYPE_WORD) {
+ size = 2;
+ } else {
+ /* wrong table type */
+ PICODBG_ERROR(("wrong table type %d", dt->outmaptable[pos]));
+ return FALSE;
+ }
+ pos++;
+
+ /* set fallback value in case of failed mapping, and set upper bound pos */
+ *outfallbackval = ((picoos_uint16)(dt->outmaptable[pos+1])) << 8 |
+ dt->outmaptable[pos];
+ pos += 2;
+
+ /* size must be 1 or 2 here, keep 'redundant' so save time */
+ if (size == 1) {
+ for (i = 0; (i < *outfallbackval) && (pos < posbound); i++) {
+ if (inval == dt->outmaptable[pos]) {
+ *outval = i;
+ PICODBG_TRACE(("s1 inval %d -> outval %d", inval, *outval));
+ return TRUE;
+ }
+ pos++;
+ }
+ } else if (size == 2) {
+ posbound--;
+ for (i = 0; (i < *outfallbackval) && (pos < posbound); i++) {
+ if (inval == (((picoos_uint16)(dt->outmaptable[pos+1])) << 8 |
+ dt->outmaptable[pos])) {
+ *outval = i;
+ PICODBG_TRACE(("s2 inval %d -> outval %d", inval, *outval));
+ return TRUE;
+ }
+ pos += 2;
+ }
+ } else {
+ /* impossible size */
+ PICODBG_ERROR(("wrong size %d", size));
+ return FALSE;
+ }
+
+ PICODBG_DEBUG(("no mapping found, fallback: %d", *outfallbackval));
+ return FALSE;
+}
+
+
+picoos_uint8 picokdt_dtPosDreverseMapOutFixed(const picokdt_DtPosD this,
+ const picoos_uint16 inval,
+ picoos_uint16 *outval,
+ picoos_uint16 *outfallbackval) {
+
+ kdtposd_subobj_t * dtposd = (kdtposd_subobj_t *)this;
+ kdt_subobj_t * dt = &(dtposd->dt);
+ return kdtReverseMapOutFixed(dt,inval, outval, outfallbackval);
+}
+
+/* not yet impl. size==1 -> MapOutByteToVar,
+ fix: size==2 -> MapOutWordToVar */
+static picoos_uint8 kdtMapOutVar(const kdt_subobj_t *dt,
+ const picoos_uint16 inval,
+ picoos_uint8 *nr,
+ picoos_uint16 *outval,
+ const picoos_uint16 outvalmaxlen) {
+ picoos_uint16 pos;
+ picoos_uint16 off2ind;
+ picoos_uint16 lentable;
+ picoos_uint16 nrinbytes;
+ picoos_uint8 size;
+ picoos_uint16 offset1;
+ picoos_uint16 i;
+
+ if (dt->outmaptable == NULL) {
+ /* empty table not possible */
+ PICODBG_ERROR(("no table found"));
+ return FALSE;
+ }
+
+ /* nr of tables == 1 already checked in *Initialize, no need here, go
+ directly to position 1 */
+ pos = 1;
+
+ /* get length of table */
+ lentable = (((picoos_uint16)(dt->outmaptable[pos + 1])) << 8 |
+ dt->outmaptable[pos]);
+ pos += 2;
+
+ /* check table type */
+ if (dt->outmaptable[pos] != PICOKDT_MTTYPE_BYTETOVAR) {
+ /* wrong table type */
+ PICODBG_ERROR(("wrong table type %d", dt->outmaptable[pos]));
+ return FALSE;
+ }
+ size = 2;
+ pos++;
+
+ /* get nr of ele in maptable (= nr of possible invals) */
+ nrinbytes = (((picoos_uint16)(dt->outmaptable[pos+1])) << 8 |
+ dt->outmaptable[pos]);
+ pos += 2;
+
+ /* check what's checkable */
+ if (nrinbytes == 0) {
+ PICODBG_ERROR(("table with length zero"));
+ return FALSE;
+ } else if (inval >= nrinbytes) {
+ PICODBG_ERROR(("inval %d outside valid range %d", inval, nrinbytes));
+ return FALSE;
+ }
+
+ PICODBG_TRACE(("inval %d, lentable %d, nrinbytes %d, pos %d", inval,
+ lentable, nrinbytes, pos));
+
+ /* set off2ind to the position of the start of offset2-val */
+ /* offset2 points to start of next ele */
+ off2ind = pos + (size*inval);
+
+ /* get number of output values, offset2 - offset1 */
+ if (inval == 0) {
+ offset1 = 0;
+ } else {
+ offset1 = (((picoos_uint16)(dt->outmaptable[off2ind - 1])) << 8 |
+ dt->outmaptable[off2ind - 2]);
+ }
+ *nr = (((picoos_uint16)(dt->outmaptable[off2ind + 1])) << 8 |
+ dt->outmaptable[off2ind]) - offset1;
+
+ PICODBG_TRACE(("offset1 %d, nr %d, pos %d", offset1, *nr, pos));
+
+ /* set pos to position of 1st value being mapped to */
+ pos += (size * nrinbytes) + offset1;
+
+ if ((pos + *nr - 1) > lentable) {
+ /* outside table, should not happen */
+ PICODBG_ERROR(("problem with table index, pos %d, nr %d, len %d",
+ pos, *nr, lentable));
+ return FALSE;
+ }
+ if (*nr > outvalmaxlen) {
+ /* not enough space in outval */
+ PICODBG_ERROR(("overflow in outval, %d > %d", *nr, outvalmaxlen));
+ return FALSE;
+ }
+
+ /* finally, copy outmap result to outval */
+ for (i = 0; i < *nr; i++) {
+ outval[i] = dt->outmaptable[pos++];
+ }
+ return TRUE;
+}
+
+
+
+/* ************************************************************/
+/* decision tree POS prediction (PosP) functions */
+/* ************************************************************/
+
+/* number of prefix and suffix graphemes used to construct the input vector */
+#define KDT_POSP_NRGRAPHPREFATT 4
+#define KDT_POSP_NRGRAPHSUFFATT 6
+#define KDT_POSP_NRGRAPHATT 10
+
+/* positions of specgraph and nrgraphs attributes */
+#define KDT_POSP_SPECGRAPHATTPOS 10
+#define KDT_POSP_NRGRAPHSATTPOS 11
+
+
+/* construct PosP input vector
+
+ PosP invec: 12 elements
+
+ prefix 0-3 prefix graphemes (encoded using tree inpmaptable 0-3)
+ suffix 4-9 suffix graphemes (encoded using tree inpmaptable 4-9)
+ isspecchar 10 is a special grapheme (e.g. hyphen) inside the word (0/1)?
+ nr-utf-graphs 11 number of graphemes (ie. UTF8 chars)
+
+ if there are less than 10 graphemes, each grapheme is used only
+ once, with the suffix having higher priority, ie. elements 0-9 are
+ filled as follows:
+
+ #graph
+ 1 0 0 0 0 0 0 0 0 0 1
+ 2 0 0 0 0 0 0 0 0 1 2
+ 3 0 0 0 0 0 0 0 1 2 3
+ 4 0 0 0 0 0 0 1 2 3 4
+ 5 0 0 0 0 0 1 2 3 4 5
+ 6 0 0 0 0 1 2 3 4 5 6
+ 7 1 0 0 0 2 3 4 5 6 7
+ 8 1 2 0 0 3 4 5 6 7 8
+ 9 1 2 3 0 4 5 6 7 8 9
+ 10 1 2 3 4 5 6 7 8 9 10
+ 11 1 2 3 4 6 7 8 9 10 11
+ ...
+
+ 1-6: Fill chbuf
+ 7-10: front to invec 1st part, remove front, add rear
+ >10: remove front, add rear
+ no more graph ->
+ while chbuflen>0:
+ add rear to the last empty slot in 2nd part of invec, remove rear
+*/
+
+
+picoos_uint8 picokdt_dtPosPconstructInVec(const picokdt_DtPosP this,
+ const picoos_uint8 *graph,
+ const picoos_uint16 graphlen,
+ const picoos_uint8 specgraphflag) {
+ kdtposp_subobj_t *dtposp;
+
+ /* utf8 circular char buffer, used as restricted input deque */
+ /* 2nd part of graph invec has KDT_POSP_NRGRAPHSUFFATT elements, */
+ /* max of UTF8_MAXLEN bytes per utf8 char */
+ picoos_uint8 chbuf[KDT_POSP_NRGRAPHSUFFATT][PICOBASE_UTF8_MAXLEN];
+ picoos_uint8 chbrear; /* next free pos */
+ picoos_uint8 chbfront; /* next read pos */
+ picoos_uint8 chblen; /* empty=0; full=KDT_POSP_NRGRAPHSUFFATT */
+
+ picoos_uint16 poscg; /* position of current graph (= utf8 char) */
+ picoos_uint16 lencg = 0; /* length of current grapheme */
+ picoos_uint16 nrutfg; /* number of utf graphemes */
+ picoos_uint8 invecpos; /* next element to add in invec */
+ picoos_uint16 fallback; /* fallback value for failed graph encodings */
+ picoos_uint8 i;
+
+ dtposp = (kdtposp_subobj_t *)this;
+ chbrear = 0;
+ chbfront = 0;
+ chblen = 0;
+ poscg = 0;
+ nrutfg = 0;
+ invecpos = 0;
+
+ PICODBG_DEBUG(("graphlen %d", graphlen));
+
+ /* not needed, since all elements are set
+ for (i = 0; i < PICOKDT_NRATT_POSP; i++) {
+ dtposp->invec[i] = '\x63';
+ }
+ */
+
+ dtposp->inveclen = 0;
+
+ while ((poscg < graphlen) &&
+ ((lencg = picobase_det_utf8_length(graph[poscg])) > 0)) {
+ if (chblen >= KDT_POSP_NRGRAPHSUFFATT) { /* chbuf full */
+ if (invecpos < KDT_POSP_NRGRAPHPREFATT) { /* prefix not full */
+ /* att-encode front utf graph and add in invec */
+ if (!kdtMapInGraph(&(dtposp->dt), invecpos,
+ chbuf[chbfront], PICOBASE_UTF8_MAXLEN,
+ &(dtposp->invec[invecpos]),
+ &fallback)) {
+ if (fallback) {
+ dtposp->invec[invecpos] = fallback;
+ } else {
+ return FALSE;
+ }
+ }
+ invecpos++;
+ }
+ /* remove front utf graph */
+ chbfront++;
+ chbfront %= KDT_POSP_NRGRAPHSUFFATT;
+ chblen--;
+ }
+ /* add current utf graph to chbuf */
+ for (i=0; i<lencg; i++) {
+ chbuf[chbrear][i] = graph[poscg++];
+ }
+ if (i < PICOBASE_UTF8_MAXLEN) {
+ chbuf[chbrear][i] = '\0';
+ }
+ chbrear++;
+ chbrear %= KDT_POSP_NRGRAPHSUFFATT;
+ chblen++;
+ /* increase utf graph count */
+ nrutfg++;
+ }
+
+ if ((lencg == 0) || (chblen == 0)) {
+ return FALSE;
+ } else if (chblen > 0) {
+
+ while (invecpos < KDT_POSP_NRGRAPHPREFATT) { /* fill up prefix */
+ if (!kdtMapInGraph(&(dtposp->dt), invecpos,
+ PICOKDT_OUTSIDEGRAPH_DEFSTR,
+ PICOKDT_OUTSIDEGRAPH_DEFLEN,
+ &(dtposp->invec[invecpos]), &fallback)) {
+ if (fallback) {
+ dtposp->invec[invecpos] = fallback;
+ } else {
+ return FALSE;
+ }
+ }
+ invecpos++;
+ }
+
+ for (i = (KDT_POSP_NRGRAPHATT - 1);
+ i >= KDT_POSP_NRGRAPHPREFATT; i--) {
+ if (chblen > 0) {
+ if (chbrear == 0) {
+ chbrear = KDT_POSP_NRGRAPHSUFFATT - 1;
+ } else {
+ chbrear--;
+ }
+ if (!kdtMapInGraph(&(dtposp->dt), i, chbuf[chbrear],
+ PICOBASE_UTF8_MAXLEN,
+ &(dtposp->invec[i]), &fallback)) {
+ if (fallback) {
+ dtposp->invec[i] = fallback;
+ } else {
+ return FALSE;
+ }
+ }
+ chblen--;
+ } else {
+ if (!kdtMapInGraph(&(dtposp->dt), i,
+ PICOKDT_OUTSIDEGRAPH_DEFSTR,
+ PICOKDT_OUTSIDEGRAPH_DEFLEN,
+ &(dtposp->invec[i]), &fallback)) {
+ if (fallback) {
+ dtposp->invec[i] = fallback;
+ } else {
+ return FALSE;
+ }
+ }
+ }
+ }
+
+ /* set isSpecChar attribute, reuse var i */
+ i = (specgraphflag ? 1 : 0);
+ if (!kdtMapInFixed(&(dtposp->dt), KDT_POSP_SPECGRAPHATTPOS, i,
+ &(dtposp->invec[KDT_POSP_SPECGRAPHATTPOS]),
+ &fallback)) {
+ if (fallback) {
+ dtposp->invec[KDT_POSP_SPECGRAPHATTPOS] = fallback;
+ } else {
+ return FALSE;
+ }
+ }
+
+ /* set nrGraphs attribute */
+ if (!kdtMapInFixed(&(dtposp->dt), KDT_POSP_NRGRAPHSATTPOS, nrutfg,
+ &(dtposp->invec[KDT_POSP_NRGRAPHSATTPOS]),
+ &fallback)) {
+ if (fallback) {
+ dtposp->invec[KDT_POSP_NRGRAPHSATTPOS] = fallback;
+ } else {
+ return FALSE;
+ }
+ }
+ PICODBG_DEBUG(("posp-invec: [%d,%d,%d,%d|%d,%d,%d,%d,%d,%d|%d|%d]",
+ dtposp->invec[0], dtposp->invec[1], dtposp->invec[2],
+ dtposp->invec[3], dtposp->invec[4], dtposp->invec[5],
+ dtposp->invec[6], dtposp->invec[7], dtposp->invec[8],
+ dtposp->invec[9], dtposp->invec[10],
+ dtposp->invec[11], dtposp->invec[12]));
+ dtposp->inveclen = PICOKDT_NRINPMT_POSP;
+ return TRUE;
+ }
+
+ return FALSE;
+}
+
+
+picoos_uint8 picokdt_dtPosPclassify(const picokdt_DtPosP this) {
+ picoos_uint32 iByteNo;
+ picoos_int8 iBitNo;
+ picoos_int8 rv;
+ kdtposp_subobj_t *dtposp;
+ kdt_subobj_t *dt;
+
+ dtposp = (kdtposp_subobj_t *)this;
+ dt = &(dtposp->dt);
+ iByteNo = 0;
+ iBitNo = 7;
+ while ((rv = kdtAskTree(dt, dtposp->invec, PICOKDT_NRATT_POSP,
+ &iByteNo, &iBitNo)) > 0) {
+ PICODBG_TRACE(("asking tree"));
+ }
+ PICODBG_DEBUG(("done: %d", dt->dclass));
+ return ((rv == 0) && dt->dset);
+}
+
+
+picoos_uint8 picokdt_dtPosPdecomposeOutClass(const picokdt_DtPosP this,
+ picokdt_classify_result_t *dtres) {
+ kdtposp_subobj_t *dtposp;
+ picoos_uint16 val;
+
+ dtposp = (kdtposp_subobj_t *)this;
+
+ if (dtposp->dt.dset &&
+ kdtMapOutFixed(&(dtposp->dt), dtposp->dt.dclass, &val)) {
+ dtres->set = TRUE;
+ dtres->class = val;
+ return TRUE;
+ } else {
+ dtres->set = FALSE;
+ return FALSE;
+ }
+}
+
+
+
+/* ************************************************************/
+/* decision tree POS disambiguation (PosD) functions */
+/* ************************************************************/
+
+
+picoos_uint8 picokdt_dtPosDconstructInVec(const picokdt_DtPosD this,
+ const picoos_uint16 * input) {
+ kdtposd_subobj_t *dtposd;
+ picoos_uint8 i;
+ picoos_uint16 fallback = 0;
+
+ dtposd = (kdtposd_subobj_t *)this;
+ dtposd->inveclen = 0;
+
+ PICODBG_DEBUG(("in: [%d,%d,%d|%d|%d,%d,%d]",
+ input[0], input[1], input[2],
+ input[3], input[4], input[5],
+ input[6]));
+ for (i = 0; i < PICOKDT_NRATT_POSD; i++) {
+
+ /* do the imt mapping for all inval */
+ if (!kdtMapInFixed(&(dtposd->dt), i, input[i],
+ &(dtposd->invec[i]), &fallback)) {
+ if (fallback) {
+ PICODBG_DEBUG(("*** using fallback for input mapping: %i -> %i", input[i], fallback));
+ dtposd->invec[i] = fallback;
+ } else {
+ PICODBG_ERROR(("problem doing input mapping"));
+ return FALSE;
+ }
+ }
+ }
+
+ PICODBG_DEBUG(("out: [%d,%d,%d|%d|%d,%d,%d]",
+ dtposd->invec[0], dtposd->invec[1], dtposd->invec[2],
+ dtposd->invec[3], dtposd->invec[4], dtposd->invec[5],
+ dtposd->invec[6]));
+ dtposd->inveclen = PICOKDT_NRINPMT_POSD;
+ return TRUE;
+}
+
+
+picoos_uint8 picokdt_dtPosDclassify(const picokdt_DtPosD this,
+ picoos_uint16 *treeout) {
+ picoos_uint32 iByteNo;
+ picoos_int8 iBitNo;
+ picoos_int8 rv;
+ kdtposd_subobj_t *dtposd;
+ kdt_subobj_t *dt;
+
+ dtposd = (kdtposd_subobj_t *)this;
+ dt = &(dtposd->dt);
+ iByteNo = 0;
+ iBitNo = 7;
+ while ((rv = kdtAskTree(dt, dtposd->invec, PICOKDT_NRATT_POSD,
+ &iByteNo, &iBitNo)) > 0) {
+ PICODBG_TRACE(("asking tree"));
+ }
+ PICODBG_DEBUG(("done: %d", dt->dclass));
+ if ((rv == 0) && dt->dset) {
+ *treeout = dt->dclass;
+ return TRUE;
+ } else {
+ return FALSE;
+ }
+}
+
+
+/* decompose the tree output and return the class in dtres
+ dtres: POS classification result
+ returns: TRUE if okay, FALSE otherwise
+*/
+picoos_uint8 picokdt_dtPosDdecomposeOutClass(const picokdt_DtPosD this,
+ picokdt_classify_result_t *dtres) {
+ kdtposd_subobj_t *dtposd;
+ picoos_uint16 val;
+
+ dtposd = (kdtposd_subobj_t *)this;
+
+ if (dtposd->dt.dset &&
+ kdtMapOutFixed(&(dtposd->dt), dtposd->dt.dclass, &val)) {
+ dtres->set = TRUE;
+ dtres->class = val;
+ return TRUE;
+ } else {
+ dtres->set = FALSE;
+ return FALSE;
+ }
+}
+
+
+
+/* ************************************************************/
+/* decision tree grapheme-to-phoneme (G2P) functions */
+/* ************************************************************/
+
+
+/* get the nr'th (starting at 0) utf char in utfgraph */
+static picoos_uint8 kdtGetUTF8char(const picoos_uint8 *utfgraph,
+ const picoos_uint16 graphlen,
+ const picoos_uint16 nr,
+ picoos_uint8 *utf8char) {
+ picoos_uint16 i;
+ picoos_uint32 pos;
+
+ pos = 0;
+ for (i = 0; i < nr; i++) {
+ if (!picobase_get_next_utf8charpos(utfgraph, graphlen, &pos)) {
+ return FALSE;
+ }
+ }
+ return picobase_get_next_utf8char(utfgraph, graphlen, &pos, utf8char);
+}
+
+/* determine the utfchar count (starting at 1) of the utfchar starting at pos */
+static picoos_uint16 kdtGetUTF8Nr(const picoos_uint8 *utfgraph,
+ const picoos_uint16 graphlen,
+ const picoos_uint16 pos) {
+ picoos_uint32 postmp;
+ picoos_uint16 count;
+
+ count = 0;
+ postmp = 0;
+ while ((postmp <= pos) && (count < graphlen)) {
+ if (!picobase_get_next_utf8charpos(utfgraph, graphlen, &postmp)) {
+ PICODBG_ERROR(("invalid utf8 string, count: %d, pos: %d, post: %d",
+ count, pos, postmp));
+ return count + 1;
+ }
+ count++;
+ }
+ return count;
+}
+
+
+picoos_uint8 picokdt_dtG2PconstructInVec(const picokdt_DtG2P this,
+ const picoos_uint8 *graph,
+ const picoos_uint16 graphlen,
+ const picoos_uint8 count,
+ const picoos_uint8 pos,
+ const picoos_uint8 nrvow,
+ const picoos_uint8 ordvow,
+ picoos_uint8 *primstressflag,
+ const picoos_uint16 phonech1,
+ const picoos_uint16 phonech2,
+ const picoos_uint16 phonech3) {
+ kdtg2p_subobj_t *dtg2p;
+ picoos_uint16 fallback = 0;
+ picoos_uint8 iAttr;
+ picoos_uint8 utf8char[PICOBASE_UTF8_MAXLEN + 1];
+ picoos_uint16 inval;
+ picoos_int16 cinv;
+ picoos_uint8 retval;
+ picoos_int32 utfgraphlen;
+ picoos_uint16 utfcount;
+
+ dtg2p = (kdtg2p_subobj_t *)this;
+ retval = TRUE;
+ inval = 0;
+
+ PICODBG_TRACE(("in: [%d,%d,%d|%d,%d|%d|%d,%d,%d]", graphlen, count, pos,
+ nrvow, ordvow, *primstressflag, phonech1, phonech2,
+ phonech3));
+
+ dtg2p->inveclen = 0;
+
+ /* many speed-ups possible */
+
+ /* graph attributes */
+ /* count > = <= count
+ iAttr lowbound eow upbound delta
+ 0 4 4 graphlen 5
+ 1 3 3 graphlen 4
+ 2 2 2 graphlen 3
+ 3 1 1 graphlen 2
+ 4 0 - graphlen 1
+
+ 5 0 graphlen graphlen-1 0
+ 6 0 graphlen-1 graphlen-2 -1
+ 7 0 graphlen-2 graphlen-3 -2
+ 8 0 graphlen-3 graphlen-4 -3
+ */
+
+ /* graph attributes left (context -4/-3/-2/-1) and current, MapInGraph */
+
+ utfgraphlen = picobase_utf8_length(graph, graphlen);
+ if (utfgraphlen <= 0) {
+ utfgraphlen = 0;
+ }
+ utfcount = kdtGetUTF8Nr(graph, graphlen, count);
+
+ cinv = 4;
+ for (iAttr = 0; iAttr < 5; iAttr++) {
+ if ((utfcount > cinv) && (utfcount <= utfgraphlen)) {
+
+/* utf8char[0] = graph[count - cinv - 1];*/
+ if (!kdtGetUTF8char(graph, graphlen, utfcount-cinv-1,
+ utf8char)) {
+ PICODBG_WARN(("problem getting UTF char %d", utfcount-cinv-1));
+ utf8char[0] = PICOKDT_OUTSIDEGRAPH_DEFCH;
+ utf8char[1] = '\0';
+ }
+ } else {
+ if ((utfcount == cinv) && (iAttr != 4)) {
+ utf8char[0] = PICOKDT_OUTSIDEGRAPH_EOW_DEFCH;
+ } else {
+ utf8char[0] = PICOKDT_OUTSIDEGRAPH_DEFCH;
+ }
+ utf8char[1] = '\0';
+ }
+
+ if (!kdtMapInGraph(&(dtg2p->dt), iAttr,
+ utf8char, PICOBASE_UTF8_MAXLEN,
+ &(dtg2p->invec[iAttr]),
+ &fallback)) {
+ if (fallback) {
+ dtg2p->invec[iAttr] = fallback;
+ } else {
+ PICODBG_WARN(("setting attribute %d to zero", iAttr));
+ dtg2p->invec[iAttr] = 0;
+ retval = FALSE;
+ }
+ }
+ PICODBG_TRACE(("invec %d %c", iAttr, utf8char[0]));
+ cinv--;
+ }
+
+ /* graph attributes right (context 1/2/3/4), MapInGraph */
+ cinv = utfgraphlen;
+ for (iAttr = 5; iAttr < 9; iAttr++) {
+ if ((utfcount > 0) && (utfcount <= (cinv - 1))) {
+/* utf8char[0] = graph[count + graphlen - cinv];*/
+ if (!kdtGetUTF8char(graph, graphlen, utfcount+utfgraphlen-cinv,
+ utf8char)) {
+ PICODBG_WARN(("problem getting UTF char %d",
+ utfcount+utfgraphlen-cinv-1));
+ utf8char[0] = PICOKDT_OUTSIDEGRAPH_DEFCH;
+ utf8char[1] = '\0';
+ }
+ } else {
+ if (utfcount == cinv) {
+ utf8char[0] = PICOKDT_OUTSIDEGRAPH_EOW_DEFCH;
+ utf8char[1] = '\0';
+ } else {
+ utf8char[0] = PICOKDT_OUTSIDEGRAPH_DEFCH;
+ utf8char[1] = '\0';
+ }
+ }
+ if (!kdtMapInGraph(&(dtg2p->dt), iAttr,
+ utf8char, PICOBASE_UTF8_MAXLEN,
+ &(dtg2p->invec[iAttr]),
+ &fallback)) {
+ if (fallback) {
+ dtg2p->invec[iAttr] = fallback;
+ } else {
+ PICODBG_WARN(("setting attribute %d to zero", iAttr));
+ dtg2p->invec[iAttr] = 0;
+ retval = FALSE;
+ }
+ }
+ PICODBG_TRACE(("invec %d %c", iAttr, utf8char[0]));
+ cinv--;
+ }
+
+ /* other attributes, MapInFixed */
+ for (iAttr = 9; iAttr < PICOKDT_NRATT_G2P; iAttr++) {
+ switch (iAttr) {
+ case 9: /* word POS, Fix1 */
+ inval = pos;
+ break;
+ case 10: /* nr of vowel-like graphs in word, if vowel, Fix2 */
+ inval = nrvow;
+ break;
+ case 11: /* order of current vowel-like graph in word, Fix2 */
+ inval = ordvow;
+ break;
+ case 12: /* primary stress mark, Fix2 */
+ if (*primstressflag == 1) {
+ /*already set previously*/
+ inval = 1;
+ } else {
+ inval = 0;
+ }
+ break;
+ case 13: /* phone chunk right context +1, Hist */
+ inval = phonech1;
+ break;
+ case 14: /* phone chunk right context +2, Hist */
+ inval = phonech2;
+ break;
+ case 15: /* phone chunk right context +3, Hist */
+ inval = phonech3;
+ break;
+ }
+
+ PICODBG_TRACE(("invec %d %d", iAttr, inval));
+
+ if (!kdtMapInFixed(&(dtg2p->dt), iAttr, inval,
+ &(dtg2p->invec[iAttr]), &fallback)) {
+ if (fallback) {
+ dtg2p->invec[iAttr] = fallback;
+ } else {
+ PICODBG_WARN(("setting attribute %d to zero", iAttr));
+ dtg2p->invec[iAttr] = 0;
+ retval = FALSE;
+ }
+ }
+ }
+
+ PICODBG_TRACE(("out: [%d,%d%,%d,%d|%d|%d,%d,%d,%d|%d,%d,%d,%d|"
+ "%d,%d,%d]", dtg2p->invec[0], dtg2p->invec[1],
+ dtg2p->invec[2], dtg2p->invec[3], dtg2p->invec[4],
+ dtg2p->invec[5], dtg2p->invec[6], dtg2p->invec[7],
+ dtg2p->invec[8], dtg2p->invec[9], dtg2p->invec[10],
+ dtg2p->invec[11], dtg2p->invec[12], dtg2p->invec[13],
+ dtg2p->invec[14], dtg2p->invec[15]));
+
+ dtg2p->inveclen = PICOKDT_NRINPMT_G2P;
+ return retval;
+}
+
+
+
+
+picoos_uint8 picokdt_dtG2Pclassify(const picokdt_DtG2P this,
+ picoos_uint16 *treeout) {
+ picoos_uint32 iByteNo;
+ picoos_int8 iBitNo;
+ picoos_int8 rv;
+ kdtg2p_subobj_t *dtg2p;
+ kdt_subobj_t *dt;
+
+ dtg2p = (kdtg2p_subobj_t *)this;
+ dt = &(dtg2p->dt);
+ iByteNo = 0;
+ iBitNo = 7;
+ while ((rv = kdtAskTree(dt, dtg2p->invec, PICOKDT_NRATT_G2P,
+ &iByteNo, &iBitNo)) > 0) {
+ PICODBG_TRACE(("asking tree"));
+ }
+ PICODBG_TRACE(("done: %d", dt->dclass));
+ if ((rv == 0) && dt->dset) {
+ *treeout = dt->dclass;
+ return TRUE;
+ } else {
+ return FALSE;
+ }
+}
+
+
+
+picoos_uint8 picokdt_dtG2PdecomposeOutClass(const picokdt_DtG2P this,
+ picokdt_classify_vecresult_t *dtvres) {
+ kdtg2p_subobj_t *dtg2p;
+
+ dtg2p = (kdtg2p_subobj_t *)this;
+
+ if (dtg2p->dt.dset &&
+ kdtMapOutVar(&(dtg2p->dt), dtg2p->dt.dclass, &(dtvres->nr),
+ dtvres->classvec, PICOKDT_MAXSIZE_OUTVEC)) {
+ return TRUE;
+ } else {
+ dtvres->nr = 0;
+ return FALSE;
+ }
+ return TRUE;
+}
+
+
+
+/* ************************************************************/
+/* decision tree phrasing (PHR) functions */
+/* ************************************************************/
+
+picoos_uint8 picokdt_dtPHRconstructInVec(const picokdt_DtPHR this,
+ const picoos_uint8 pre2,
+ const picoos_uint8 pre1,
+ const picoos_uint8 src,
+ const picoos_uint8 fol1,
+ const picoos_uint8 fol2,
+ const picoos_uint16 nrwordspre,
+ const picoos_uint16 nrwordsfol,
+ const picoos_uint16 nrsyllsfol) {
+ kdtphr_subobj_t *dtphr;
+ picoos_uint8 i;
+ picoos_uint16 inval = 0;
+ picoos_uint16 fallback = 0;
+
+ dtphr = (kdtphr_subobj_t *)this;
+ PICODBG_DEBUG(("in: [%d,%d|%d|%d,%d|%d,%d,%d]",
+ pre2, pre1, src, fol1, fol2,
+ nrwordspre, nrwordsfol, nrsyllsfol));
+ dtphr->inveclen = 0;
+
+ for (i = 0; i < PICOKDT_NRATT_PHR; i++) {
+ switch (i) {
+ case 0: inval = pre2; break;
+ case 1: inval = pre1; break;
+ case 2: inval = src; break;
+ case 3: inval = fol1; break;
+ case 4: inval = fol2; break;
+ case 5: inval = nrwordspre; break;
+ case 6: inval = nrwordsfol; break;
+ case 7: inval = nrsyllsfol; break;
+ default:
+ PICODBG_ERROR(("size mismatch"));
+ return FALSE;
+ break;
+ }
+
+ /* do the imt mapping for all inval */
+ if (!kdtMapInFixed(&(dtphr->dt), i, inval,
+ &(dtphr->invec[i]), &fallback)) {
+ if (fallback) {
+ dtphr->invec[i] = fallback;
+ } else {
+ PICODBG_ERROR(("problem doing input mapping"));
+ return FALSE;
+ }
+ }
+ }
+
+ PICODBG_DEBUG(("out: [%d,%d|%d|%d,%d|%d,%d,%d]",
+ dtphr->invec[0], dtphr->invec[1], dtphr->invec[2],
+ dtphr->invec[3], dtphr->invec[4], dtphr->invec[5],
+ dtphr->invec[6], dtphr->invec[7]));
+ dtphr->inveclen = PICOKDT_NRINPMT_PHR;
+ return TRUE;
+}
+
+
+picoos_uint8 picokdt_dtPHRclassify(const picokdt_DtPHR this) {
+ picoos_uint32 iByteNo;
+ picoos_int8 iBitNo;
+ picoos_int8 rv;
+ kdtphr_subobj_t *dtphr;
+ kdt_subobj_t *dt;
+
+ dtphr = (kdtphr_subobj_t *)this;
+ dt = &(dtphr->dt);
+ iByteNo = 0;
+ iBitNo = 7;
+ while ((rv = kdtAskTree(dt, dtphr->invec, PICOKDT_NRATT_PHR,
+ &iByteNo, &iBitNo)) > 0) {
+ PICODBG_TRACE(("asking tree"));
+ }
+ PICODBG_DEBUG(("done: %d", dt->dclass));
+ return ((rv == 0) && dt->dset);
+}
+
+
+picoos_uint8 picokdt_dtPHRdecomposeOutClass(const picokdt_DtPHR this,
+ picokdt_classify_result_t *dtres) {
+ kdtphr_subobj_t *dtphr;
+ picoos_uint16 val;
+
+ dtphr = (kdtphr_subobj_t *)this;
+
+ if (dtphr->dt.dset &&
+ kdtMapOutFixed(&(dtphr->dt), dtphr->dt.dclass, &val)) {
+ dtres->set = TRUE;
+ dtres->class = val;
+ return TRUE;
+ } else {
+ dtres->set = FALSE;
+ return FALSE;
+ }
+}
+
+
+
+/* ************************************************************/
+/* decision tree phono-acoustical model (PAM) functions */
+/* ************************************************************/
+
+picoos_uint8 picokdt_dtPAMconstructInVec(const picokdt_DtPAM this,
+ const picoos_uint8 *vec,
+ const picoos_uint8 veclen) {
+ kdtpam_subobj_t *dtpam;
+ picoos_uint8 i;
+ picoos_uint16 fallback = 0;
+
+ dtpam = (kdtpam_subobj_t *)this;
+
+ PICODBG_TRACE(("in0: %d %d %d %d %d %d %d %d %d %d",
+ vec[0], vec[1], vec[2], vec[3], vec[4],
+ vec[5], vec[6], vec[7], vec[8], vec[9]));
+ PICODBG_TRACE(("in1: %d %d %d %d %d %d %d %d %d %d",
+ vec[10], vec[11], vec[12], vec[13], vec[14],
+ vec[15], vec[16], vec[17], vec[18], vec[19]));
+ PICODBG_TRACE(("in2: %d %d %d %d %d %d %d %d %d %d",
+ vec[20], vec[21], vec[22], vec[23], vec[24],
+ vec[25], vec[26], vec[27], vec[28], vec[29]));
+ PICODBG_TRACE(("in3: %d %d %d %d %d %d %d %d %d %d",
+ vec[30], vec[31], vec[32], vec[33], vec[34],
+ vec[35], vec[36], vec[37], vec[38], vec[39]));
+ PICODBG_TRACE(("in4: %d %d %d %d %d %d %d %d %d %d",
+ vec[40], vec[41], vec[42], vec[43], vec[44],
+ vec[45], vec[46], vec[47], vec[48], vec[49]));
+ PICODBG_TRACE(("in5: %d %d %d %d %d %d %d %d %d %d",
+ vec[50], vec[51], vec[52], vec[53], vec[54],
+ vec[55], vec[56], vec[57], vec[58], vec[59]));
+
+ dtpam->inveclen = 0;
+
+ /* check veclen */
+ if (veclen != PICOKDT_NRINPMT_PAM) {
+ PICODBG_ERROR(("wrong number of input vector elements"));
+ return FALSE;
+ }
+
+ for (i = 0; i < PICOKDT_NRATT_PAM; i++) {
+
+ /* do the imt mapping for all vec eles */
+ if (!kdtMapInFixed(&(dtpam->dt), i, vec[i],
+ &(dtpam->invec[i]), &fallback)) {
+ if (fallback) {
+ dtpam->invec[i] = fallback;
+ } else {
+ PICODBG_ERROR(("problem doing input mapping, %d %d", i,vec[i]));
+ return FALSE;
+ }
+ }
+ }
+
+ PICODBG_TRACE(("in0: %d %d %d %d %d %d %d %d %d %d",
+ dtpam->invec[0], dtpam->invec[1], dtpam->invec[2],
+ dtpam->invec[3], dtpam->invec[4], dtpam->invec[5],
+ dtpam->invec[6], dtpam->invec[7], dtpam->invec[8],
+ dtpam->invec[9]));
+ PICODBG_TRACE(("in1: %d %d %d %d %d %d %d %d %d %d",
+ dtpam->invec[10], dtpam->invec[11], dtpam->invec[12],
+ dtpam->invec[13], dtpam->invec[14], dtpam->invec[15],
+ dtpam->invec[16], dtpam->invec[17], dtpam->invec[18],
+ dtpam->invec[19]));
+ PICODBG_TRACE(("in2: %d %d %d %d %d %d %d %d %d %d",
+ dtpam->invec[20], dtpam->invec[21], dtpam->invec[22],
+ dtpam->invec[23], dtpam->invec[24], dtpam->invec[25],
+ dtpam->invec[26], dtpam->invec[27], dtpam->invec[28],
+ dtpam->invec[29]));
+ PICODBG_TRACE(("in3: %d %d %d %d %d %d %d %d %d %d",
+ dtpam->invec[30], dtpam->invec[31], dtpam->invec[32],
+ dtpam->invec[33], dtpam->invec[34], dtpam->invec[35],
+ dtpam->invec[36], dtpam->invec[37], dtpam->invec[38],
+ dtpam->invec[39]));
+ PICODBG_TRACE(("in4: %d %d %d %d %d %d %d %d %d %d",
+ dtpam->invec[40], dtpam->invec[41], dtpam->invec[42],
+ dtpam->invec[43], dtpam->invec[44], dtpam->invec[45],
+ dtpam->invec[46], dtpam->invec[47], dtpam->invec[48],
+ dtpam->invec[49]));
+ PICODBG_TRACE(("in5: %d %d %d %d %d %d %d %d %d %d",
+ dtpam->invec[50], dtpam->invec[51], dtpam->invec[52],
+ dtpam->invec[53], dtpam->invec[54], dtpam->invec[55],
+ dtpam->invec[56], dtpam->invec[57], dtpam->invec[58],
+ dtpam->invec[59]));
+
+ dtpam->inveclen = PICOKDT_NRINPMT_PAM;
+ return TRUE;
+}
+
+
+picoos_uint8 picokdt_dtPAMclassify(const picokdt_DtPAM this) {
+ picoos_uint32 iByteNo;
+ picoos_int8 iBitNo;
+ picoos_int8 rv;
+ kdtpam_subobj_t *dtpam;
+ kdt_subobj_t *dt;
+
+ dtpam = (kdtpam_subobj_t *)this;
+ dt = &(dtpam->dt);
+ iByteNo = 0;
+ iBitNo = 7;
+ while ((rv = kdtAskTree(dt, dtpam->invec, PICOKDT_NRATT_PAM,
+ &iByteNo, &iBitNo)) > 0) {
+ PICODBG_TRACE(("asking tree"));
+ }
+ PICODBG_DEBUG(("done: %d", dt->dclass));
+ return ((rv == 0) && dt->dset);
+}
+
+
+picoos_uint8 picokdt_dtPAMdecomposeOutClass(const picokdt_DtPAM this,
+ picokdt_classify_result_t *dtres) {
+ kdtpam_subobj_t *dtpam;
+ picoos_uint16 val;
+
+ dtpam = (kdtpam_subobj_t *)this;
+
+ if (dtpam->dt.dset &&
+ kdtMapOutFixed(&(dtpam->dt), dtpam->dt.dclass, &val)) {
+ dtres->set = TRUE;
+ dtres->class = val;
+ return TRUE;
+ } else {
+ dtres->set = FALSE;
+ return FALSE;
+ }
+}
+
+
+
+/* ************************************************************/
+/* decision tree accentuation (ACC) functions */
+/* ************************************************************/
+
+picoos_uint8 picokdt_dtACCconstructInVec(const picokdt_DtACC this,
+ const picoos_uint8 pre2,
+ const picoos_uint8 pre1,
+ const picoos_uint8 src,
+ const picoos_uint8 fol1,
+ const picoos_uint8 fol2,
+ const picoos_uint16 hist1,
+ const picoos_uint16 hist2,
+ const picoos_uint16 nrwordspre,
+ const picoos_uint16 nrsyllspre,
+ const picoos_uint16 nrwordsfol,
+ const picoos_uint16 nrsyllsfol,
+ const picoos_uint16 footwordsfol,
+ const picoos_uint16 footsyllsfol) {
+ kdtacc_subobj_t *dtacc;
+ picoos_uint8 i;
+ picoos_uint16 inval = 0;
+ picoos_uint16 fallback = 0;
+
+ dtacc = (kdtacc_subobj_t *)this;
+ PICODBG_DEBUG(("in: [%d,%d,%d,%d,%d|%d,%d|%d,%d,%d,%d|%d,%d]",
+ pre2, pre1, src, fol1, fol2, hist1, hist2,
+ nrwordspre, nrsyllspre, nrwordsfol, nrsyllsfol,
+ footwordsfol, footsyllsfol));
+ dtacc->inveclen = 0;
+
+ for (i = 0; i < PICOKDT_NRATT_ACC; i++) {
+ switch (i) {
+ case 0: inval = pre2; break;
+ case 1: inval = pre1; break;
+ case 2: inval = src; break;
+ case 3: inval = fol1; break;
+ case 4: inval = fol2; break;
+ case 5: inval = hist1; break;
+ case 6: inval = hist2; break;
+ case 7: inval = nrwordspre; break;
+ case 8: inval = nrsyllspre; break;
+ case 9: inval = nrwordsfol; break;
+ case 10: inval = nrsyllsfol; break;
+ case 11: inval = footwordsfol; break;
+ case 12: inval = footsyllsfol; break;
+ default:
+ PICODBG_ERROR(("size mismatch"));
+ return FALSE;
+ break;
+ }
+
+ if (((i == 5) || (i == 6)) && (inval == PICOKDT_HISTORY_ZERO)) {
+ /* in input to this function the HISTORY_ZERO is used to
+ mark the no-value-available case. For sparsity reasons
+ this was not used in the training. For
+ no-value-available cases, instead, do reverse out
+ mapping of ACC0 to get tree domain for ACC0 */
+ if (!kdtReverseMapOutFixed(&(dtacc->dt), PICODATA_ACC0,
+ &inval, &fallback)) {
+ if (fallback) {
+ inval = fallback;
+ } else {
+ PICODBG_ERROR(("problem doing reverse output mapping"));
+ return FALSE;
+ }
+ }
+ }
+
+ /* do the imt mapping for all inval */
+ if (!kdtMapInFixed(&(dtacc->dt), i, inval,
+ &(dtacc->invec[i]), &fallback)) {
+ if (fallback) {
+ dtacc->invec[i] = fallback;
+ } else {
+ PICODBG_ERROR(("problem doing input mapping"));
+ return FALSE;
+ }
+ }
+ }
+
+ PICODBG_DEBUG(("out: [%d,%d,%d,%d,%d|%d,%d|%d,%d,%d,%d|%d,%d]",
+ dtacc->invec[0], dtacc->invec[1], dtacc->invec[2],
+ dtacc->invec[3], dtacc->invec[4], dtacc->invec[5],
+ dtacc->invec[6], dtacc->invec[7], dtacc->invec[8],
+ dtacc->invec[9], dtacc->invec[10], dtacc->invec[11],
+ dtacc->invec[12]));
+ dtacc->inveclen = PICOKDT_NRINPMT_ACC;
+ return TRUE;
+}
+
+
+picoos_uint8 picokdt_dtACCclassify(const picokdt_DtACC this,
+ picoos_uint16 *treeout) {
+ picoos_uint32 iByteNo;
+ picoos_int8 iBitNo;
+ picoos_int8 rv;
+ kdtacc_subobj_t *dtacc;
+ kdt_subobj_t *dt;
+
+ dtacc = (kdtacc_subobj_t *)this;
+ dt = &(dtacc->dt);
+ iByteNo = 0;
+ iBitNo = 7;
+ while ((rv = kdtAskTree(dt, dtacc->invec, PICOKDT_NRATT_ACC,
+ &iByteNo, &iBitNo)) > 0) {
+ PICODBG_TRACE(("asking tree"));
+ }
+ PICODBG_TRACE(("done: %d", dt->dclass));
+ if ((rv == 0) && dt->dset) {
+ *treeout = dt->dclass;
+ return TRUE;
+ } else {
+ return FALSE;
+ }
+}
+
+
+picoos_uint8 picokdt_dtACCdecomposeOutClass(const picokdt_DtACC this,
+ picokdt_classify_result_t *dtres) {
+ kdtacc_subobj_t *dtacc;
+ picoos_uint16 val;
+
+ dtacc = (kdtacc_subobj_t *)this;
+
+ if (dtacc->dt.dset &&
+ kdtMapOutFixed(&(dtacc->dt), dtacc->dt.dclass, &val)) {
+ dtres->set = TRUE;
+ dtres->class = val;
+ return TRUE;
+ } else {
+ dtres->set = FALSE;
+ return FALSE;
+ }
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+
+/* end */
diff --git a/lib/picokdt.h b/lib/picokdt.h
new file mode 100644
index 0000000..3ef973c
--- /dev/null
+++ b/lib/picokdt.h
@@ -0,0 +1,465 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picokdt.h
+ *
+ * knowledge handling for decision trees
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#ifndef PICOKDT_H_
+#define PICOKDT_H_
+
+#include "picoos.h"
+#include "picoknow.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* ************************************************************/
+/*
+ Several specialized decision trees kb are provided by this
+ knowledge handling module:
+
+ - Part of speech prediction decision tree: ...kdt_PosP
+ - Part of speech disambiguation decision tree: ...kdt_PosD
+ - Grapheme-to-phoneme decision tree: ...kdt_G2P
+ - Phrasing decision tree: ...kdt_PHR
+ - Accentuation decision tree: ...kdt_ACC
+ these 5 tree types may be unified in the future to a single type
+
+ - Phono-acoustical model trees: ...kdt_PAM
+ (actually 11 trees, but all have the same characteristics and
+ are instances of the same class)
+*/
+/* ************************************************************/
+
+
+/* ************************************************************/
+/* defines and functions to create specialized kb, */
+/* to be used by picorsrc only */
+/* ************************************************************/
+
+typedef enum {
+ PICOKDT_KDTTYPE_POSP,
+ PICOKDT_KDTTYPE_POSD,
+ PICOKDT_KDTTYPE_G2P,
+ PICOKDT_KDTTYPE_PHR,
+ PICOKDT_KDTTYPE_ACC,
+ PICOKDT_KDTTYPE_PAM
+} picokdt_kdttype_t;
+
+pico_status_t picokdt_specializeDtKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common,
+ const picokdt_kdttype_t type);
+
+
+/* ************************************************************/
+/* decision tree types (opaque) and get Tree functions */
+/* ************************************************************/
+
+/* decision tree types */
+typedef struct picokdt_dtposp * picokdt_DtPosP;
+typedef struct picokdt_dtposd * picokdt_DtPosD;
+typedef struct picokdt_dtg2p * picokdt_DtG2P;
+typedef struct picokdt_dtphr * picokdt_DtPHR;
+typedef struct picokdt_dtacc * picokdt_DtACC;
+typedef struct picokdt_dtpam * picokdt_DtPAM;
+
+/* return kb decision tree for usage in PU */
+picokdt_DtPosP picokdt_getDtPosP(picoknow_KnowledgeBase this);
+picokdt_DtPosD picokdt_getDtPosD(picoknow_KnowledgeBase this);
+picokdt_DtG2P picokdt_getDtG2P (picoknow_KnowledgeBase this);
+picokdt_DtPHR picokdt_getDtPHR (picoknow_KnowledgeBase this);
+picokdt_DtACC picokdt_getDtACC (picoknow_KnowledgeBase this);
+picokdt_DtPAM picokdt_getDtPAM (picoknow_KnowledgeBase this);
+
+
+/* number of attributes (= input vector size) for each tree type */
+typedef enum {
+ PICOKDT_NRATT_POSP = 12,
+ PICOKDT_NRATT_POSD = 7,
+ PICOKDT_NRATT_G2P = 16,
+ PICOKDT_NRATT_PHR = 8,
+ PICOKDT_NRATT_ACC = 13,
+ PICOKDT_NRATT_PAM = 60
+} kdt_nratt_t;
+
+
+/* ************************************************************/
+/* decision tree classification result type */
+/* ************************************************************/
+
+typedef struct {
+ picoos_uint8 set; /* TRUE if class set, FALSE otherwise */
+ picoos_uint16 class;
+} picokdt_classify_result_t;
+
+
+/* maximum number of output values the tree output is mapped to */
+#define PICOKDT_MAXSIZE_OUTVEC 8
+
+typedef struct {
+ picoos_uint8 nr; /* 0 if no class set, nr of values set otherwise */
+ picoos_uint16 classvec[PICOKDT_MAXSIZE_OUTVEC];
+} picokdt_classify_vecresult_t;
+
+
+/* ************************************************************/
+/* decision tree functions */
+/* ************************************************************/
+
+/* constructInVec:
+ for every tree type there is a constructInVec function to construct
+ the size-optimized input vector for the tree using the input map
+ tables that are part of the decistion tree knowledge base. The
+ constructed input vector is stored in the tree object (this->invec
+ and this->inveclen) and will be used in the following call to the
+ classify function.
+
+ classify:
+ for every tree type there is a classify function to apply the
+ decision tree to the previously constructed input vector. The
+ size-optimized, encoded output is stored in the tree object
+ (this->outval) and will be used in the following call to the
+ decompose function. Where needed (hitory attribute) the direct tree
+ output is returned by the classify function in a variable.
+
+ decomposeOutClass:
+ for every tree type there is a decompose function to decompose the
+ size-optimized, encoded tree output and map it to the outside the
+ tree usable class value.
+*/
+
+
+/* ************************************************************/
+/* decision tree defines */
+/* ************************************************************/
+
+/* to construct the input vectors several hard-coded values are used
+ to handle attributes that, at the given position, are outside the
+ context. */
+
+/* graph attributes: values to be used if the graph attribute is
+ outside the grapheme string (ie. word) */
+#define PICOKDT_OUTSIDEGRAPH_DEFCH (picoos_uint8)'\x30' /* ascii "0" */
+#define PICOKDT_OUTSIDEGRAPH_DEFSTR (picoos_uint8 *)"\x30" /* ascii "0" */
+#define PICOKDT_OUTSIDEGRAPH_DEFLEN 1
+
+/* graph attributes (special case for g2p): values to be used if the
+ graph attribute is directly outside the grapheme string (ie. at the
+ word boundary word). Use PICOKDT_OUTSIDEGRAPH_DEF* if further
+ outside. */
+#define PICOKDT_OUTSIDEGRAPH_EOW_DEFCH (picoos_uint8)'\x31' /* ascii "1" */
+#define PICOKDT_OUTSIDEGRAPH_EOW_DEFSTR (picoos_uint8 *)"\x31" /* ascii "1" */
+#define PICOKDT_OUTSIDEGRAPH_EOW_DEFLEN 1
+
+/* byte and word type attributes: value to be used if a byte or word
+ attribute is outside the context, e.g. for POS */
+#define PICOKDT_EPSILON 7
+
+/* byte and word type attributes: for attribute with history info a
+ 'zero' value is needed when starting the sequence of predictions.
+ Use the following value to initialize history. Note that the direct
+ tree outputs (not mapped with output map table) of previous
+ predictions need to be used when constructing the input vector for
+ a following prediction. This direct tree output will then be mapped
+ together with the rest of the input vector by the input map
+ table. */
+#define PICOKDT_HISTORY_ZERO 30000
+
+
+/* ************************************************************/
+/* decision tree POS prediction (PosP) functions */
+/* ************************************************************/
+
+/* construct a POS prediction input vector
+ tree input vector: 0-3 prefix UTF8 graphemes
+ 4-9 suffex UTF8 graphemes
+ 10 special grapheme existence flag (TRUE/FALSE)
+ 11 number of graphemes
+ graph: the grapheme string of the word for wich POS will be predicted
+ graphlen: length of graph in number of bytes
+ specgraphflag: existence of a special grapheme boolean
+ returns: TRUE if okay, FALSE otherwise
+ note: use PICOKDT_OUTSIDEGRAPH* for att values outside context
+*/
+picoos_uint8 picokdt_dtPosPconstructInVec(const picokdt_DtPosP this,
+ const picoos_uint8 *graph,
+ const picoos_uint16 graphlen,
+ const picoos_uint8 specgraphflag);
+
+
+/* classify a previously constructed input vector using tree 'this'
+ returns: TRUE if okay, FALSE otherwise
+*/
+picoos_uint8 picokdt_dtPosPclassify(const picokdt_DtPosP this);
+
+/* decompose the tree output and return the class in dtres
+ dtres: POS or POSgroup ID classification result
+ returns: TRUE if okay, FALSE otherwise
+*/
+picoos_uint8 picokdt_dtPosPdecomposeOutClass(const picokdt_DtPosP this,
+ picokdt_classify_result_t *dtres);
+
+
+/* ************************************************************/
+/* decision tree POS disambiguation (PosD) functions */
+/* ************************************************************/
+
+/* construct a POS disambiguation input vector (run in left-to-right mode)
+ tree input vector: 0-2 POS or POSgroup for each of the three previous words
+ 3 POSgroup for current word
+ 4-6 POS or POSgroup (can be history) for each of
+ the three following words
+ pre3 - pre1: POSgroup or POS for the previous three words
+ src: POSgroup of current word (if unique POS no posdisa possible)
+ fol1 - fol3: POS or history for the following three words (the more
+ complicated the better... :-( NEEDS TO BE uint16
+ ishist1-ishist3: flag to indicate if fol1-3 are predicted tree
+ output values (history) or the HISTORY_ZERO (TRUE)
+ or an already unambiguous POS (FALSE)
+ returns: TRUE if okay, FALSE otherwise
+ note: use PICOKDT_EPSILON for att values outside context,
+ if POS in fol* unique use this POS instead of real
+ history, use reverse output mapping in these cases
+*/
+picoos_uint8 picokdt_dtPosDconstructInVec(const picokdt_DtPosD this,
+ const picoos_uint16 * input);
+
+
+/* classify a previously constructed input vector using tree 'this'
+ treeout: direct tree output value
+ returns: TRUE if okay, FALSE otherwise
+*/
+picoos_uint8 picokdt_dtPosDclassify(const picokdt_DtPosD this,
+ picoos_uint16 *treeout);
+
+/* decompose the tree output and return the class in dtres
+ dtres: POS classification result
+ returns: TRUE if okay, FALSE otherwise
+*/
+picoos_uint8 picokdt_dtPosDdecomposeOutClass(const picokdt_DtPosD this,
+ picokdt_classify_result_t *dtres);
+
+/* convert (unique) POS index into corresponding tree output index */
+picoos_uint8 picokdt_dtPosDreverseMapOutFixed(const picokdt_DtPosD this,
+ const picoos_uint16 inval,
+ picoos_uint16 *outval,
+ picoos_uint16 *outfallbackval);
+
+/* ************************************************************/
+/* decision tree grapheme-to-phoneme (G2P) functions */
+/* ************************************************************/
+
+/* construct a G2P input vector (run in right-to-left mode)
+ tree input vector: 0-8 the 4 previous, current, and 4 following graphemes
+ 9 POS
+ 10-11 vowel count and vowel ID
+ 12 primary stress flag (TRUE/FALSE)
+ 13-15 the three following phones predicted
+ graph: the grapheme string used to determine invec[0:8]
+ graphlen: length of graph in number of bytes
+ count: the grapheme number for which invec will be constructed [0..]
+ pos: the part of speech of the word
+ nrvow number of vowel-like graphemes in graph if vowel,
+ set to 0 otherwise
+ ordvow order of 'count' vowel in graph if vowel,
+ set to 0 otherwise
+ primstressflag: flag indicating if primary stress was already predicted
+ phonech1-3: the three following phon chunks predicted (right-to-left)
+ returns: TRUE if okay, FALSE otherwise
+*/
+picoos_uint8 picokdt_dtG2PconstructInVec(const picokdt_DtG2P this,
+ const picoos_uint8 *graph,
+ const picoos_uint16 graphlen,
+ const picoos_uint8 count,
+ const picoos_uint8 pos,
+ const picoos_uint8 nrvow,
+ const picoos_uint8 ordvow,
+ picoos_uint8 *primstressflag,
+ const picoos_uint16 phonech1,
+ const picoos_uint16 phonech2,
+ const picoos_uint16 phonech3);
+
+/* classify a previously constructed input vector using tree 'this'
+ treeout: direct tree output value
+ returns: TRUE if okay, FALSE otherwise
+*/
+picoos_uint8 picokdt_dtG2Pclassify(const picokdt_DtG2P this,
+ picoos_uint16 *treeout);
+
+/* decompose the tree output and return the class vector in dtvres
+ dtvres: phones vector classification result
+ returns: TRUE if okay, FALSE otherwise
+*/
+picoos_uint8 picokdt_dtG2PdecomposeOutClass(const picokdt_DtG2P this,
+ picokdt_classify_vecresult_t *dtvres);
+
+
+/* ************************************************************/
+/* decision tree phrasing (PHR) functions */
+/* ************************************************************/
+
+/* construct a PHR input vector (run in right-to-left mode)
+ tree input vector: 0-1 POS for each of the two previous words
+ 2 POS for current word
+ 3-4 POS for each of the two following words
+ 5 nr words left
+ 6 nr words right
+ 7 nr syllables right
+ pre2 - pre1: POS for the previous two words
+ src: POS of current word
+ fol1 - fol2: POS for the following two words
+ nrwordspre: number of words left (previous) of current word
+ nrwordsfol: number of words right (following) of current word,
+ incl. current word, up to next BOUND (also
+ considering previously predicted PHR2/3)
+ nrsyllsfol: number of syllables right (following) of current word,
+ incl. syllables of current word, up to next BOUND
+ (also considering previously predicted PHR2/3)
+ returns: TRUE if okay, FALSE otherwise
+ note: use PICOKDT_EPSILON for att values outside context
+*/
+picoos_uint8 picokdt_dtPHRconstructInVec(const picokdt_DtPHR this,
+ const picoos_uint8 pre2,
+ const picoos_uint8 pre1,
+ const picoos_uint8 src,
+ const picoos_uint8 fol1,
+ const picoos_uint8 fol2,
+ const picoos_uint16 nrwordspre,
+ const picoos_uint16 nrwordsfol,
+ const picoos_uint16 nrsyllsfol);
+
+/* classify a previously constructed input vector using tree 'this'
+ returns: TRUE if okay, FALSE otherwise
+*/
+picoos_uint8 picokdt_dtPHRclassify(const picokdt_DtPHR this);
+
+/* decompose the tree output and return the class vector in dtres
+ dtres: phrasing classification result
+ returns: TRUE if okay, FALSE otherwise
+*/
+picoos_uint8 picokdt_dtPHRdecomposeOutClass(const picokdt_DtPHR this,
+ picokdt_classify_result_t *dtres);
+
+
+/* ************************************************************/
+/* decision tree accentuation (ACC) functions */
+/* ************************************************************/
+
+/* construct an ACC input vector (run in right-to-left mode)
+ tree input vector: 0-1 POS for each of the two previous words
+ 2 POS for current word
+ 3-4 POS for each of the two following words
+ 5-6 history values (already predicted following)
+ 7 nr words left (previous) to any bound
+ 8 nr syllables left to any bound
+ 9 nr words right (following) to any bound
+ 10 nr syllables right to any bound
+ 11 nr words right to predicted "1" prominence (foot)
+ 12 nr syllables right to predicted "1" prominence (foot)
+ pre2 - pre1: POS for the previous two words
+ src: POS of current word
+ fol1 - fol2: POS for the following two words
+ hist1 - hist2: previously predicted ACC values
+ nrwordspre: number of words left (previous) of current word
+ nrsyllspre: number of syllables left (previous) of current word,
+ incl. initial non-prim stress syllables of current word
+ nrwordsfol: number of words right (following) of current word,
+ incl. current word, up to next BOUND (any strength != 0)
+ nrsyllsfol: number of syllables right (following) of current word,
+ incl. syllables of current word starting with prim. stress
+ syllable
+ footwordsfol: nr of words to the following prominence '1'
+ footsyllspre: nr of syllables to the previous prominence '1'
+ returns: TRUE if okay, FALSE otherwise
+ note: use PICOKDT_EPSILON for att 0-4 values outside context
+*/
+picoos_uint8 picokdt_dtACCconstructInVec(const picokdt_DtACC this,
+ const picoos_uint8 pre2,
+ const picoos_uint8 pre1,
+ const picoos_uint8 src,
+ const picoos_uint8 fol1,
+ const picoos_uint8 fol2,
+ const picoos_uint16 hist1,
+ const picoos_uint16 hist2,
+ const picoos_uint16 nrwordspre,
+ const picoos_uint16 nrsyllspre,
+ const picoos_uint16 nrwordsfol,
+ const picoos_uint16 nrsyllsfol,
+ const picoos_uint16 footwordsfol,
+ const picoos_uint16 footsyllsfol);
+
+/* classify a previously constructed input vector using tree 'this'
+ treeout: direct tree output value
+ returns: TRUE if okay, FALSE otherwise
+*/
+picoos_uint8 picokdt_dtACCclassify(const picokdt_DtACC this,
+ picoos_uint16 *treeout);
+
+/* decompose the tree output and return the class vector in dtres
+ dtres: phrasing classification result
+ returns: TRUE if okay, FALSE otherwise
+*/
+picoos_uint8 picokdt_dtACCdecomposeOutClass(const picokdt_DtACC this,
+ picokdt_classify_result_t *dtres);
+
+
+/* ************************************************************/
+/* decision tree phono-acoustical model (PAM) functions */
+/* ************************************************************/
+
+/* construct a Pam input vector and store the tree-specific encoded
+ input vector in the tree object.
+ vec: tree input vector, 60 single-byte-sized attributes
+ veclen: length of vec in number of bytes
+ returns: TRUE if okay, FALSE otherwise
+*/
+picoos_uint8 picokdt_dtPAMconstructInVec(const picokdt_DtPAM this,
+ const picoos_uint8 *vec,
+ const picoos_uint8 veclen);
+
+/* classify a previously constructed input vector using tree 'this'
+ returns: TRUE if okay, FALSE otherwise
+*/
+picoos_uint8 picokdt_dtPAMclassify(const picokdt_DtPAM this);
+
+/* decompose the tree output and return the class in dtres
+ dtres: phones vector classification result
+ returns: TRUE if okay, FALSE otherwise
+*/
+picoos_uint8 picokdt_dtPAMdecomposeOutClass(const picokdt_DtPAM this,
+ picokdt_classify_result_t *dtres);
+
+#ifdef __cplusplus
+}
+#endif
+
+
+
+#endif /*PICOKDT_H_*/
diff --git a/lib/picokfst.c b/lib/picokfst.c
new file mode 100644
index 0000000..560709c
--- /dev/null
+++ b/lib/picokfst.c
@@ -0,0 +1,438 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picokfst.c
+ *
+ * FST knowledge loading and access
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+#include "picoos.h"
+#include "picodbg.h"
+#include "picoknow.h"
+#include "picokfst.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+#define FileHdrSize 4 /* size of FST file header */
+
+
+
+/* ************************************************************/
+/* function to create specialized kb, */
+/* to be used by picorsrc only */
+/* ************************************************************/
+
+/** object : FSTKnowledgeBase
+ * shortcut : kfst
+ * derived from : picoknow_KnowledgeBase
+ */
+
+typedef struct kfst_subobj * kfst_SubObj;
+
+typedef struct kfst_subobj{
+ picoos_uint8 * fstStream; /* the byte stream base address */
+ picoos_int32 hdrLen; /* length of file header */
+ picoos_int32 transductionMode; /* transduction mode to be used for FST */
+ picoos_int32 nrClasses; /* nr of pair/transition classes in FST; class is in [1..nrClasses] */
+ picoos_int32 nrStates; /* nr of states in FST; state is in [1..nrState] */
+ picoos_int32 termClass; /* pair class of terminator symbol pair; probably obsolete */
+ picoos_int32 alphaHashTabSize; /* size of pair alphabet hash table */
+ picoos_int32 alphaHashTabPos; /* absolute address of the start of the pair alphabet */
+ picoos_int32 transTabEntrySize; /* size in bytes of each transition table entry */
+ picoos_int32 transTabPos; /* absolute address of the start of the transition table */
+ picoos_int32 inEpsStateTabPos; /* absolute address of the start of the input epsilon transition table */
+ picoos_int32 accStateTabPos; /* absolute address of the table of accepting states */
+} kfst_subobj_t;
+
+
+
+/* ************************************************************/
+/* primitives for reading from byte stream */
+/* ************************************************************/
+
+/* Converts 'nrBytes' bytes starting at position '*pos' in byte stream 'stream' into unsigned number 'num'.
+ '*pos' is modified to the position right after the number */
+static void FixedBytesToUnsignedNum (picoos_uint8 * stream, picoos_uint8 nrBytes, picoos_uint32 * pos, picoos_uint32 * num)
+{
+ picoos_int32 i;
+
+ (*num) = 0;
+ for (i = 0; i < nrBytes; i++) {
+ (*num) = ((*num) << 8) + (picoos_uint32)stream[*pos];
+ (*pos)++;
+ }
+}
+
+
+/* Converts 'nrBytes' bytes starting at position '*pos' in byte stream 'stream' into signed number 'num'.
+ '*pos' is modified to the position right after the number */
+static void FixedBytesToSignedNum (picoos_uint8 * stream, picoos_uint8 nrBytes, picoos_uint32 * pos, picoos_int32 * num)
+{
+ picoos_int32 i;
+ picoos_uint32 val;
+
+ val = 0;
+ for (i = 0; i < nrBytes; i++) {
+ val = (val << 8) + (picoos_uint32)stream[*pos];
+ (*pos)++;
+ }
+ if (val % 2 == 1) {
+ /* negative number */
+ (*num) = -((picoos_int32)((val - 1) / 2)) - 1;
+ } else {
+ /* positive number */
+ (*num) = val / 2;
+ }
+}
+
+
+/* Converts varying-sized sequence of bytes starting at position '*pos' in byte stream 'stream'
+ into (signed) number 'num'. '*pos' is modified to the position right after the number. */
+static void BytesToNum (picoos_uint8 * stream, picoos_uint32 * pos, picoos_int32 * num)
+{
+ picoos_uint32 val;
+ picoos_uint32 b;
+
+ val = 0;
+ b = (picoos_uint32)stream[*pos];
+ (*pos)++;
+ while (b < 128) {
+ val = (val << 7) + b;
+ b = (picoos_uint32)stream[*pos];
+ (*pos)++;
+ }
+ val = (val << 7) + (b - 128);
+ if (val % 2 == 1) {
+ /* negative number */
+ (*num) = -((picoos_int32)((val - 1) / 2)) - 1;
+ } else {
+ /* positive number */
+ (*num) = val / 2;
+ }
+}
+
+
+/* ************************************************************/
+/* setting up FST from byte stream */
+/* ************************************************************/
+
+static pico_status_t kfstInitialize(register picoknow_KnowledgeBase this,
+ picoos_Common common)
+{
+ picoos_uint32 curpos;
+ picoos_int32 offs;
+ kfst_subobj_t * kfst;
+
+ PICODBG_DEBUG(("kfstInitialize -- start\n"));
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING, NULL,
+ NULL);
+ }
+ kfst = (kfst_subobj_t *) this->subObj;
+
+ /* +CT+ */
+ kfst->fstStream = this->base;
+ PICODBG_TRACE(("base: %d\n",this->base));
+ kfst->hdrLen = FileHdrSize;
+ curpos = kfst->hdrLen;
+ BytesToNum(kfst->fstStream,& curpos,& kfst->transductionMode);
+ BytesToNum(kfst->fstStream,& curpos,& kfst->nrClasses);
+ BytesToNum(kfst->fstStream,& curpos,& kfst->nrStates);
+ BytesToNum(kfst->fstStream,& curpos,& kfst->termClass);
+ BytesToNum(kfst->fstStream,& curpos,& kfst->alphaHashTabSize);
+ BytesToNum(kfst->fstStream,& curpos,& offs);
+ kfst->alphaHashTabPos = kfst->hdrLen + offs;
+ BytesToNum(kfst->fstStream,& curpos,& kfst->transTabEntrySize);
+ BytesToNum(kfst->fstStream,& curpos,& offs);
+ kfst->transTabPos = kfst->hdrLen + offs;
+ BytesToNum(kfst->fstStream,& curpos,& offs);
+ kfst->inEpsStateTabPos = kfst->hdrLen + offs;
+ BytesToNum(kfst->fstStream,& curpos,& offs);
+ kfst->accStateTabPos = kfst->hdrLen + offs;
+ /* -CT- */
+
+ return PICO_OK;
+}
+
+
+static pico_status_t kfstSubObjDeallocate(register picoknow_KnowledgeBase this,
+ picoos_MemoryManager mm)
+{
+ if (NULL != this) {
+ picoos_deallocate(mm, (void *) &this->subObj);
+ }
+ return PICO_OK;
+}
+
+
+/* calculates a small number of data (e.g. addresses) from kb for fast access.
+ * This data is encapsulated in a picokfst_FST that can later be retrieved
+ * with picokfst_getFST. */
+pico_status_t picokfst_specializeFSTKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common)
+{
+ pico_status_t status;
+
+ if (NULL == this) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING, NULL, NULL);
+ }
+ if (0 < this->size) {
+ /* not a dummy kb */
+ this->subDeallocate = kfstSubObjDeallocate;
+
+ this->subObj = picoos_allocate(common->mm, sizeof(kfst_subobj_t));
+
+ if (NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM, NULL, NULL);
+ }
+ status = kfstInitialize(this, common);
+ if (PICO_OK != status) {
+ picoos_deallocate(common->mm,(void **)&this->subObj);
+ }
+ }
+ return PICO_OK;
+}
+
+
+/* ************************************************************/
+/* FST type and getFST function */
+/* ************************************************************/
+
+
+
+/* return kb FST for usage in PU */
+picokfst_FST picokfst_getFST(picoknow_KnowledgeBase this)
+{
+ if (NULL == this) {
+ return NULL;
+ } else {
+ return (picokfst_FST) this->subObj;
+ }
+}
+
+
+
+/* ************************************************************/
+/* FST access methods */
+/* ************************************************************/
+
+
+/* see description in header file */
+extern picoos_uint8 picokfst_kfstGetTransductionMode(picokfst_FST this)
+{
+ kfst_SubObj fst = (kfst_SubObj) this;
+ if (fst != NULL) {
+ return fst->transductionMode;
+ } else {
+ return 0;
+ }
+}
+
+
+/* see description in header file */
+extern void picokfst_kfstGetFSTSizes (picokfst_FST this, picoos_int32 *nrStates, picoos_int32 *nrClasses)
+{
+ kfst_SubObj fst = (kfst_SubObj) this;
+ if (fst != NULL) {
+ *nrStates = fst->nrStates;
+ *nrClasses = fst->nrClasses;
+ } else {
+ *nrStates = 0;
+ *nrClasses = 0;
+ }
+}
+
+/* see description in header file */
+extern void picokfst_kfstStartPairSearch (picokfst_FST this, picokfst_symid_t inSym,
+ picoos_bool * inSymFound, picoos_int32 * searchState)
+{
+ picoos_uint32 pos;
+ picoos_int32 offs;
+ picoos_int32 h;
+ picoos_int32 inSymCellPos;
+ picoos_int32 inSymX;
+ picoos_int32 nextSameHashInSymOffs;
+
+ kfst_SubObj fst = (kfst_SubObj) this;
+ (*searchState) = -1;
+ (*inSymFound) = 0;
+ h = inSym % fst->alphaHashTabSize;
+ pos = fst->alphaHashTabPos + (h * 4);
+ FixedBytesToSignedNum(fst->fstStream,4,& pos,& offs);
+ if (offs > 0) {
+ inSymCellPos = fst->alphaHashTabPos + offs;
+ pos = inSymCellPos;
+ BytesToNum(fst->fstStream,& pos,& inSymX);
+ BytesToNum(fst->fstStream,& pos,& nextSameHashInSymOffs);
+ while ((inSymX != inSym) && (nextSameHashInSymOffs > 0)) {
+ inSymCellPos = inSymCellPos + nextSameHashInSymOffs;
+ pos = inSymCellPos;
+ BytesToNum(fst->fstStream,& pos,& inSymX);
+ BytesToNum(fst->fstStream,& pos,& nextSameHashInSymOffs);
+ }
+ if (inSymX == inSym) {
+ /* input symbol found; state is set to position after symbol cell */
+ (*searchState) = pos;
+ (*inSymFound) = 1;
+ }
+ }
+}
+
+
+/* see description in header file */
+extern void picokfst_kfstGetNextPair (picokfst_FST this, picoos_int32 * searchState,
+ picoos_bool * pairFound,
+ picokfst_symid_t * outSym, picokfst_class_t * pairClass)
+{
+ picoos_uint32 pos;
+ picoos_int32 val;
+
+ kfst_SubObj fst = (kfst_SubObj) this;
+ if ((*searchState) < 0) {
+ (*pairFound) = 0;
+ (*outSym) = PICOKFST_SYMID_ILLEG;
+ (*pairClass) = -1;
+ } else {
+ pos = (*searchState);
+ BytesToNum(fst->fstStream,& pos,& val);
+ *outSym = (picokfst_symid_t)val;
+ if ((*outSym) != PICOKFST_SYMID_ILLEG) {
+ BytesToNum(fst->fstStream,& pos,& val);
+ *pairClass = (picokfst_class_t)val;
+ (*pairFound) = 1;
+ (*searchState) = pos;
+ } else {
+ (*pairFound) = 0;
+ (*outSym) = PICOKFST_SYMID_ILLEG;
+ (*pairClass) = -1;
+ (*searchState) = -1;
+ }
+ }
+}
+
+
+
+/* see description in header file */
+extern void picokfst_kfstGetTrans (picokfst_FST this, picokfst_state_t startState, picokfst_class_t transClass,
+ picokfst_state_t * endState)
+{
+
+ picoos_uint32 pos;
+ picoos_int32 index;
+ picoos_uint32 endStateX;
+
+ kfst_SubObj fst = (kfst_SubObj) this;
+ if ((startState < 1) || (startState > fst->nrStates) || (transClass < 1) || (transClass > fst->nrClasses)) {
+ (*endState) = 0;
+ } else {
+ index = (startState - 1) * fst->nrClasses + transClass - 1;
+ pos = fst->transTabPos + (index * fst->transTabEntrySize);
+ FixedBytesToUnsignedNum(fst->fstStream,fst->transTabEntrySize,& pos,& endStateX);
+ (*endState) = endStateX;
+ }
+}
+
+
+/* see description in header file */
+extern void picokfst_kfstStartInEpsTransSearch (picokfst_FST this, picokfst_state_t startState,
+ picoos_bool * inEpsTransFound, picoos_int32 * searchState)
+{
+
+ picoos_int32 offs;
+ picoos_uint32 pos;
+
+ kfst_SubObj fst = (kfst_SubObj) this;
+ (*searchState) = -1;
+ (*inEpsTransFound) = 0;
+ if ((startState > 0) && (startState <= fst->nrStates)) {
+ pos = fst->inEpsStateTabPos + (startState - 1) * 4;
+ FixedBytesToSignedNum(fst->fstStream,4,& pos,& offs);
+ if (offs > 0) {
+ (*searchState) = fst->inEpsStateTabPos + offs;
+ (*inEpsTransFound) = 1;
+ }
+ }
+}
+
+
+
+/* see description in header file */
+extern void picokfst_kfstGetNextInEpsTrans (picokfst_FST this, picoos_int32 * searchState,
+ picoos_bool * inEpsTransFound,
+ picokfst_symid_t * outSym, picokfst_state_t * endState)
+{
+ picoos_uint32 pos;
+ picoos_int32 val;
+
+ kfst_SubObj fst = (kfst_SubObj) this;
+ if ((*searchState) < 0) {
+ (*inEpsTransFound) = 0;
+ (*outSym) = PICOKFST_SYMID_ILLEG;
+ (*endState) = 0;
+ } else {
+ pos = (*searchState);
+ BytesToNum(fst->fstStream,& pos,& val);
+ *outSym = (picokfst_symid_t)val;
+ if ((*outSym) != PICOKFST_SYMID_ILLEG) {
+ BytesToNum(fst->fstStream,& pos,& val);
+ *endState = (picokfst_state_t)val;
+ (*inEpsTransFound) = 1;
+ (*searchState) = pos;
+ } else {
+ (*inEpsTransFound) = 0;
+ (*outSym) = PICOKFST_SYMID_ILLEG;
+ (*endState) = 0;
+ (*searchState) = -1;
+ }
+ }
+}
+
+
+/* see description in header file */
+extern picoos_bool picokfst_kfstIsAcceptingState (picokfst_FST this, picokfst_state_t state)
+{
+
+ picoos_uint32 pos;
+ picoos_uint32 val;
+
+ kfst_SubObj fst = (kfst_SubObj) this;
+ if ((state > 0) && (state <= fst->nrStates)) {
+ pos = fst->accStateTabPos + (state - 1);
+ FixedBytesToUnsignedNum(fst->fstStream,1,& pos,& val);
+ return (val == 1);
+ } else {
+ return 0;
+ }
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+/* End picofst.c */
diff --git a/lib/picokfst.h b/lib/picokfst.h
new file mode 100644
index 0000000..b391013
--- /dev/null
+++ b/lib/picokfst.h
@@ -0,0 +1,169 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picokfst.h
+ *
+ * FST knowledge loading and access
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+#ifndef PICOKFST_H_
+#define PICOKFST_H_
+
+#include "picodefs.h"
+#include "picodbg.h"
+#include "picoos.h"
+#include "picoknow.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+typedef picoos_int16 picokfst_symid_t; /* type of symbol identifiers */
+typedef picoos_int16 picokfst_class_t; /* type of symbol pair classes */
+typedef picoos_int16 picokfst_state_t; /* type of states */
+
+#define PICOKFST_SYMID_EPS (picokfst_symid_t) 0 /* epsilon symbol id */
+#define PICOKFST_SYMID_ILLEG (picokfst_symid_t) -1 /* illegal symbol id */
+
+/**
+ * @addtogroup picokfst
+ *
+ * Mapping of values to FST symbol id (relevant for compiling the FST) \n
+ * Value FST symbol id \n
+ * -------------------------------------- \n
+ * phoneme_id -> phoneme_id + 256 * PICOKFST_PLANE_PHONEMES \n
+ * accentlevel_id -> accentlevel_id + 256 * PICOKFST_PLANE_ACCENTS \n
+ * POS_id -> POS_id + 256 * PICOKFST_PLANE_POS \n
+ * pb_strength_id -> pb_strength_id + 256 * PICOKFST_PLANE_PB_STRENGTHS \n
+ * phon_term_id -> phon_term_id + 256 * PICOKFST_PLANE_INTERN \n
+*/
+enum picokfst_symbol_plane {
+ PICOKFST_PLANE_PHONEMES = 0, /* phoneme plane */
+ PICOKFST_PLANE_ASCII = 1, /* "ascii" plane (values > 127 may be used internally) */
+ PICOKFST_PLANE_XSAMPA = 2, /* x-sampa primitives plane (pico-specific table) */
+ PICOKFST_PLANE_ACCENTS = 4, /* accent plane */
+ PICOKFST_PLANE_POS = 5, /* part of speech plane */
+ PICOKFST_PLANE_PB_STRENGTHS = 6, /* phrase boundary strength plane */
+ PICOKFST_PLANE_INTERN = 7 /* internal plane, e.g. phonStartId, phonTermId */
+};
+
+/* to be used as bit set, e.g.
+ * picoos_uint8 transductionMode = PICOKFST_TRANSMODE_NEWSYMS | PICOKFST_TRANSMODE_POSUSED;
+ */
+enum picofst_transduction_mode {
+ PICOKFST_TRANSMODE_NEWSYMS = 1, /* e.g. {#WB},{#PB-S},{#PB-W},{#ACC0},{#ACC1},{#ACC2},{#ACC3}, */
+ PICOKFST_TRANSMODE_POSUSED = 2 /* FST contains Part Of Speech symbols */
+
+};
+
+
+/* ************************************************************/
+/* function to create specialized kb, */
+/* to be used by knowledge layer (picorsrc) only */
+/* ************************************************************/
+
+/* calculates a small number of data (e.g. addresses) from kb for fast access.
+ * This data is encapsulated in a picokfst_FST that can later be retrieved
+ * with picokfst_getFST. */
+pico_status_t picokfst_specializeFSTKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common);
+
+
+/* ************************************************************/
+/* FST type and getFST function */
+/* ************************************************************/
+
+/* FST type */
+typedef struct picokfst_fst * picokfst_FST;
+
+/* return kb FST for usage in PU */
+picokfst_FST picokfst_getFST(picoknow_KnowledgeBase this);
+
+
+/* ************************************************************/
+/* FST access methods */
+/* ************************************************************/
+
+/* returns transduction mode specified with rule sources;
+ result to be interpreted as set of picofst_transduction_mode */
+picoos_uint8 picokfst_kfstGetTransductionMode(picokfst_FST this);
+
+/* returns number of states and number of pair classes in FST;
+ legal states are 1..nrStates, legal classes are 1..nrClasses */
+void picokfst_kfstGetFSTSizes (picokfst_FST this, picoos_int32 *nrStates, picoos_int32 *nrClasses);
+
+/* starts search for all pairs with input symbol 'inSym'; '*inSymFound' returns whether
+ such pairs exist at all; '*searchState' returns a search state to be used in
+ subsequent calls to function 'picokfst_kfstGetNextPair', which must be used
+ to get the symbol pairs */
+void picokfst_kfstStartPairSearch (picokfst_FST this, picokfst_symid_t inSym,
+ picoos_bool * inSymFound, picoos_int32 * searchState);
+
+/* gets next pair for input symbol specified with preceding call to 'picokfst_kfstStartPairSearch';
+ '*searchState' maintains the search state, 'pairFound' returns whether any more pair was found,
+ '*outSym' returns the output symbol of the found pair, and '*pairClass' returns the
+ transition class of the found symbol pair */
+void picokfst_kfstGetNextPair (picokfst_FST this, picoos_int32 * searchState,
+ picoos_bool * pairFound,
+ picokfst_symid_t * outSym, picokfst_class_t * pairClass);
+
+/* attempts to do FST transition from state 'startState' with pair class 'transClass';
+ if such a transition exists, 'endState' returns the end state of the transition (> 0),
+ otherwise 'endState' returns <= 0 */
+void picokfst_kfstGetTrans (picokfst_FST this, picokfst_state_t startState, picokfst_class_t transClass,
+ picokfst_state_t * endState);
+
+/* starts search for all pairs with input epsilon symbol and all correponding
+ FST transitions starting in state 'startState'; to be used for fast
+ computation of epsilon closures;
+ '*inEpsTransFound' returns whether any such transition was found at all;
+ if so, '*searchState' returns a search state to be used in subsequent calls
+ to 'picokfst_kfstGetNextInEpsTrans' */
+void picokfst_kfstStartInEpsTransSearch (picokfst_FST this, picokfst_state_t startState,
+ picoos_bool * inEpsTransFound, picoos_int32 * searchState);
+
+/* gets next FST transition with a pair with empty input symbol starting from a state
+ previoulsy specified in 'picokfst_kfstStartInEpsTransSearch';
+ '*searchState' maintains the search state, '*inEpsTransFound' returns
+ whether a new transition with input epsilon was found, '*outSym 'returns
+ the output symbol of the found pair, and '*endState' returns the end state
+ of the found transition with that pair */
+void picokfst_kfstGetNextInEpsTrans (picokfst_FST this, picoos_int32 * searchState,
+ picoos_bool * inEpsTransFound,
+ picokfst_symid_t * outSym, picokfst_state_t * endState);
+
+/* returns whether 'state' is an accepting state of FST; originally, only
+ state 1 was an accepting state; however, in order to remove the need to
+ always do a last transition with a termination symbol pair, this function
+ defines a state as an accepting state if there is transition to state 1
+ with the terminator symbol pair */
+picoos_bool picokfst_kfstIsAcceptingState (picokfst_FST this, picokfst_state_t state);
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif /*PICOKFST_H_*/
diff --git a/lib/picoklex.c b/lib/picoklex.c
new file mode 100644
index 0000000..7ff0a33
--- /dev/null
+++ b/lib/picoklex.c
@@ -0,0 +1,572 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picoklex.c
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+#include "picoos.h"
+#include "picodbg.h"
+#include "picodata.h"
+#include "picoknow.h"
+#include "picoklex.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+/* ************************************************************/
+/* lexicon */
+/* ************************************************************/
+
+/**
+ * @addtogroup picolex
+ *
+ overview:
+ - lex consists of optional searchindex and a non-empty list of lexblocks
+ - lexblocks are fixed size, at the start of a block there is also the
+ start of an entry
+ - using the searchindex a unambiguous lexblock can be determined which
+ contains the entry (or there is no entry)
+ - one lex entry has POS GRAPH PHON, all mandatory, but
+ - PHON can be empty string -> no pronunciation in the resulting TTS output
+ - PHON can be :G2P -> use G2P later to add pronunciation
+ - (POS,GRAPH) is a uniq key (only one entry allowed)
+ - (GRAPH) is almost a uniq key (2-4 entries with the same GRAPH, and
+ differing POS and differing PHON possible)
+ - for one graph we can have two or three solutions from the lex
+ which all need to be passed on the the next PU
+ - in this case GRAPH, POS, and PHON all must be available in lex
+
+ sizing:
+ - 3 bytes entry index -> 16MB addressable
+ - 2 bytes searchindex nr -> 64K blocks possible
+ - 5 bytes per searchindex entry
+ - 3 bytes for graph-prefix
+ - 2 bytes blockadr in searchindex -> 64K blocks possible
+ - lexblock size 512B:
+ - 32M possible
+ - with ~20 bytes per entry
+ -> max. average of ~26 entries to be searched per lookup
+ - overhead of ~10 bytes per block to sync with
+ block boundaries
+ - examples:
+ - 500KB lex -> 1000 blocks,
+ 1000 entries in searchindex, ~25.6K lex-entries,
+ - ~5KB searchindex
+ ~10KB overhead for block sync
+ - 100KB lex -> 200 blocks,
+ 200 entries in searchindex, ~5.1K lex-entries,
+ - ~1KB searchindex
+ ~2KB overhead for block sync
+
+ pil-file: lexicon knowledge base in binary form
+
+ lex-kb = content
+
+ content = searchindex {lexblock}1:NRBLOCKS2
+
+ lexblock = {lexentry}1: (lexblock size is fixed 512Bytes)
+
+ searchindex = NRBLOCKS2 {GRAPH1 GRAPH1 GRAPH1 LEXBLOCKIND2}=NRBLOCKS2
+
+ lexentry = LENGRAPH1 {GRAPH1}=LENGRAPH1-1
+ LENPOSPHON1 POS1 {PHON1}=LENPOSPHON1-2
+
+ - special cases:
+ - PHON is empty string (no pronunciation in the resulting TTS output):
+ lexentry = LENGRAPH1 {GRAPH1}=LENGRAPH1-1 2 POS1
+ - PHON can be :G2P -> use G2P later to add pronunciation:
+ lexentry = LENGRAPH1 {GRAPH1}=LENGRAPH1-1 3 POS1 <reserved-phon-val=5>
+ - multi-byte values always little endian
+*/
+
+
+/* ************************************************************/
+/* lexicon data defines */
+/* may not be changed with current implementation */
+/* ************************************************************/
+
+/* nr bytes of nrblocks info */
+#define PICOKLEX_LEX_NRBLOCKS_SIZE 2
+
+/* search index entry: - nr graphs
+ - nr bytes of block index
+ - nr bytes per entry, NRGRAPHS*INDSIZE */
+#define PICOKLEX_LEX_SIE_NRGRAPHS 3
+#define PICOKLEX_LEX_SIE_INDSIZE 2
+#define PICOKLEX_LEX_SIE_SIZE 5
+
+/* nr of bytes per lexblock */
+#define PICOKLEX_LEXBLOCK_SIZE 512
+
+
+/* reserved values in klex to indicate :G2P needed for a lexentry */
+#define PICOKLEX_NEEDS_G2P 5
+
+
+/* ************************************************************/
+/* lexicon type and loading */
+/* ************************************************************/
+
+/** object : LexKnowledgeBase
+ * shortcut : klex
+ * derived from : picoknow_KnowledgeBase
+ */
+
+typedef struct klex_subobj *klex_SubObj;
+
+typedef struct klex_subobj
+{
+ picoos_uint16 nrblocks; /* nr lexblocks = nr eles in searchind */
+ picoos_uint8 *searchind;
+ picoos_uint8 *lexblocks;
+} klex_subobj_t;
+
+
+static pico_status_t klexInitialize(register picoknow_KnowledgeBase this,
+ picoos_Common common)
+{
+ picoos_uint32 curpos = 0;
+ klex_subobj_t *klex;
+
+ PICODBG_DEBUG(("start"));
+
+ /* check whether (this->size != 0) done before calling this function */
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ klex = (klex_subobj_t *) this->subObj;
+
+ if (PICO_OK == picoos_read_mem_pi_uint16(this->base, &curpos,
+ &(klex->nrblocks))) {
+ if (klex->nrblocks > 0) {
+ PICODBG_DEBUG(("nr blocks: %i, curpos: %i", klex->nrblocks,curpos));
+ klex->searchind = this->base + curpos;
+ } else {
+ klex->searchind = NULL;
+ }
+ klex->lexblocks = this->base + PICOKLEX_LEX_NRBLOCKS_SIZE +
+ (klex->nrblocks * (PICOKLEX_LEX_SIE_SIZE));
+ return PICO_OK;
+ } else {
+ return picoos_emRaiseException(common->em, PICO_EXC_FILE_CORRUPT,
+ NULL, NULL);
+ }
+}
+
+
+static pico_status_t klexSubObjDeallocate(register picoknow_KnowledgeBase this,
+ picoos_MemoryManager mm)
+{
+ if (NULL != this) {
+ picoos_deallocate(mm, (void *) &this->subObj);
+ }
+ return PICO_OK;
+}
+
+
+/* we don't offer a specialized constructor for a LexKnowledgeBase but
+ * instead a "specializer" of an allready existing generic
+ * picoknow_KnowledgeBase */
+
+pico_status_t picoklex_specializeLexKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common)
+{
+ if (NULL == this) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ if (this->size > 0) {
+ this->subDeallocate = klexSubObjDeallocate;
+ this->subObj = picoos_allocate(common->mm, sizeof(klex_subobj_t));
+ if (NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM,
+ NULL, NULL);
+ }
+ return klexInitialize(this, common);
+ } else {
+ /* some dummy klex */
+ return PICO_OK;
+ }
+}
+
+/* for now we don't need to do anything special for the main lex */
+/*
+pico_status_t picoklex_specializeMainLexKnowledgeBase(
+ picoknow_KnowledgeBase this,
+ picoos_Common common)
+{
+ return picoklex_specializeLexKnowledgeBase(this,common);
+}
+*/
+
+
+/* ************************************************************/
+/* lexicon getLex */
+/* ************************************************************/
+
+picoklex_Lex picoklex_getLex(picoknow_KnowledgeBase this)
+{
+ if (NULL == this) {
+ return NULL;
+ } else {
+ return (picoklex_Lex) this->subObj;
+ }
+}
+
+
+/* ************************************************************/
+/* functions on searchindex */
+/* ************************************************************/
+
+
+static picoos_uint32 klex_getSearchIndexVal(const klex_SubObj this,
+ picoos_uint16 index)
+{
+ picoos_uint32 pos, val;
+ pos = index * PICOKLEX_LEX_SIE_SIZE;
+ val = this->searchind[pos];
+ val = (val << 8) + this->searchind[pos + 1];
+ val = (val << 8) + this->searchind[pos + 2];
+ return val;
+}
+
+
+/* Determine first lexblock containing entries for specified
+ grapheme. */
+
+static picoos_uint16 klex_getLexblockNr(const klex_SubObj this,
+ const picoos_uint8 *graphsi) {
+ /* graphsi is of len PICOKLEX_LEX_SI_NGRAPHS */
+ picoos_int32 low, mid, high;
+ picoos_uint32 searchval, indval;
+
+ /* PICOKLEX_LEX_SIE_NRGRAPHS */
+
+ /* convert graph-prefix to number with 'lexicographic' ordering */
+ searchval = graphsi[0];
+ searchval = (searchval << 8) + graphsi[1];
+ searchval = (searchval << 8) + graphsi[2];
+
+ low = 0;
+ high = this->nrblocks;
+
+ /* do binary search */
+ while (low < high) {
+ mid = (low + high) / 2;
+ indval = klex_getSearchIndexVal(this, mid);
+ if (indval < searchval) {
+ low = mid + 1;
+ } else {
+ high = mid;
+ }
+ }
+ PICODBG_ASSERT(high == low);
+ /* low points to the first entry greater than or equal to searchval */
+
+ if (low < this->nrblocks) {
+ indval = klex_getSearchIndexVal(this, low);
+ if (indval > searchval) {
+ low--;
+ /* if there are identical elements in the search index we have
+ to move to the first one */
+ if (low > 0) {
+ indval = klex_getSearchIndexVal(this, low);
+ while (indval == klex_getSearchIndexVal(this, low-1)) {
+ low--;
+ }
+ }
+ }
+ } else {
+ low = this->nrblocks - 1;
+ }
+
+#if defined(PICO_DEBUG)
+ {
+ picoos_uint32 pos = low * PICOKLEX_LEX_SIE_SIZE;
+ PICODBG_DEBUG(("binary search result is %c%c%c (%d)",
+ this->searchind[pos], this->searchind[pos + 1],
+ this->searchind[pos + 2], low));
+ }
+#endif
+
+ return (picoos_uint16) low;
+}
+
+
+/* Determine number of adjacent lexblocks containing entries for
+ the same grapheme search prefix (identified by search index). */
+
+static picoos_uint16 klex_getLexblockRange(const klex_SubObj this,
+ picoos_uint16 index)
+{
+ picoos_uint16 count;
+ picoos_uint32 sval1, sval2;
+
+ sval1 = klex_getSearchIndexVal(this, index);
+
+#if defined(PICO_DEBUG)
+ /* 'index' must point to first lexblock of its kind */
+ if (index > 0) {
+ sval2 = klex_getSearchIndexVal(this, index - 1);
+ PICODBG_ASSERT(sval1 != sval2);
+ }
+#endif
+
+ index++;
+ sval2 = klex_getSearchIndexVal(this, index);
+
+ count = 1;
+ while (sval1 == sval2) {
+ count++;
+ index++;
+ sval2 = klex_getSearchIndexVal(this, index);
+ }
+
+ return count;
+}
+
+
+/* ************************************************************/
+/* functions on single lexblock */
+/* ************************************************************/
+
+static picoos_int8 klex_lexMatch(picoos_uint8 *lexentry,
+ const picoos_uint8 *graph,
+ const picoos_uint16 graphlen) {
+ picoos_uint8 i;
+ picoos_uint8 lexlen;
+ picoos_uint8 *lexgraph;
+
+ lexlen = lexentry[0] - 1;
+ lexgraph = &(lexentry[1]);
+ for (i=0; (i<graphlen) && (i<lexlen); i++) {
+ PICODBG_TRACE(("%d|%d graph|lex: %c|%c", graphlen, lexlen,
+ graph[i], lexgraph[i]));
+ if (lexgraph[i] < graph[i]) {
+ return -1;
+ } else if (lexgraph[i] > graph[i]) {
+ return 1;
+ }
+ }
+ if (graphlen == lexlen) {
+ return 0;
+ } else if (lexlen < graphlen) {
+ return -1;
+ } else {
+ return 1;
+ }
+}
+
+
+static void klex_setLexResult(const picoos_uint8 *lexentry,
+ const picoos_uint32 lexpos,
+ picoklex_lexl_result_t *lexres) {
+ picoos_uint8 i;
+
+ /* check if :G2P */
+ if ((lexentry[lexentry[0] + 2]) == PICOKLEX_NEEDS_G2P) {
+ /* set pos */
+ lexres->posind[0] = lexentry[lexentry[0] + 1];
+ /* set rest */
+ lexres->phonfound = FALSE;
+ lexres->posindlen = 1;
+ lexres->nrres = 1;
+ PICODBG_DEBUG(("result %d :G2P", lexres->nrres));
+ } else {
+ i = lexres->nrres * (PICOKLEX_POSIND_SIZE);
+ lexres->posindlen += PICOKLEX_POSIND_SIZE;
+ lexres->phonfound = TRUE;
+ /* set pos */
+ lexres->posind[i++] = lexentry[lexentry[0] + 1];
+ /* set ind, PICOKLEX_IND_SIZE */
+ lexres->posind[i++] = 0x000000ff & (lexpos);
+ lexres->posind[i++] = 0x000000ff & (lexpos >> 8);
+ lexres->posind[i] = 0x000000ff & (lexpos >> 16);
+ lexres->nrres++;
+ PICODBG_DEBUG(("result %d", lexres->nrres));
+ }
+}
+
+
+static void klex_lexblockLookup(klex_SubObj this,
+ const picoos_uint32 lexposStart,
+ const picoos_uint32 lexposEnd,
+ const picoos_uint8 *graph,
+ const picoos_uint16 graphlen,
+ picoklex_lexl_result_t *lexres) {
+ picoos_uint32 lexpos;
+ picoos_int8 rv;
+
+ lexres->nrres = 0;
+
+ lexpos = lexposStart;
+ rv = -1;
+ while ((rv < 0) && (lexpos < lexposEnd)) {
+
+ rv = klex_lexMatch(&(this->lexblocks[lexpos]), graph, graphlen);
+
+ if (rv == 0) { /* found */
+ klex_setLexResult(&(this->lexblocks[lexpos]), lexpos, lexres);
+ if (lexres->phonfound) {
+ /* look for more results, up to MAX_NRRES, don't even
+ check if more results would be available */
+ while ((lexres->nrres < PICOKLEX_MAX_NRRES) &&
+ (lexpos < lexposEnd)) {
+ lexpos += this->lexblocks[lexpos];
+ lexpos += this->lexblocks[lexpos];
+ /* if there are no more entries in this block, advance
+ to next block by skipping all zeros */
+ while ((this->lexblocks[lexpos] == 0) &&
+ (lexpos < lexposEnd)) {
+ lexpos++;
+ }
+ if (lexpos < lexposEnd) {
+ if (klex_lexMatch(&(this->lexblocks[lexpos]), graph,
+ graphlen) == 0) {
+ klex_setLexResult(&(this->lexblocks[lexpos]),
+ lexpos, lexres);
+ } else {
+ /* no more results, quit loop */
+ lexpos = lexposEnd;
+ }
+ }
+ }
+ } else {
+ /* :G2P mark */
+ }
+ } else if (rv < 0) {
+ /* not found, goto next entry */
+ lexpos += this->lexblocks[lexpos];
+ lexpos += this->lexblocks[lexpos];
+ /* if there are no more entries in this block, advance
+ to next block by skipping all zeros */
+ while ((this->lexblocks[lexpos] == 0) && (lexpos < lexposEnd)) {
+ lexpos++;
+ }
+ } else {
+ /* rv > 0, not found, won't show up later in block */
+ }
+ }
+}
+
+
+/* ************************************************************/
+/* lexicon lookup functions */
+/* ************************************************************/
+
+picoos_uint8 picoklex_lexLookup(const picoklex_Lex this,
+ const picoos_uint8 *graph,
+ const picoos_uint16 graphlen,
+ picoklex_lexl_result_t *lexres) {
+ picoos_uint16 lbnr, lbc;
+ picoos_uint32 lexposStart, lexposEnd;
+ picoos_uint8 i;
+ picoos_uint8 tgraph[PICOKLEX_LEX_SIE_NRGRAPHS];
+ klex_SubObj klex = (klex_SubObj) this;
+
+ if (NULL == klex) {
+ PICODBG_ERROR(("no lexicon loaded"));
+ /* no exception here needed, already checked at initialization */
+ return FALSE;
+ }
+
+ lexres->nrres = 0;
+ lexres->posindlen = 0;
+ lexres->phonfound = FALSE;
+
+ for (i = 0; i<PICOKLEX_LEX_SIE_NRGRAPHS; i++) {
+ if (i < graphlen) {
+ tgraph[i] = graph[i];
+ } else {
+ tgraph[i] = '\0';
+ }
+ }
+ PICODBG_DEBUG(("tgraph: %c%c%c", tgraph[0],tgraph[1],tgraph[2]));
+
+ if ((klex->nrblocks) == 0) {
+ /* no searchindex, no lexblock */
+ PICODBG_WARN(("no searchindex, no lexblock"));
+ return FALSE;
+ } else {
+ lbnr = klex_getLexblockNr(klex, tgraph);
+ PICODBG_ASSERT(lbnr < klex->nrblocks);
+ lbc = klex_getLexblockRange(klex, lbnr);
+ PICODBG_ASSERT((lbc >= 1) && (lbc <= klex->nrblocks));
+ }
+ PICODBG_DEBUG(("lexblock nr: %d (#%d)", lbnr, lbc));
+
+ lexposStart = lbnr * PICOKLEX_LEXBLOCK_SIZE;
+ lexposEnd = lexposStart + lbc * PICOKLEX_LEXBLOCK_SIZE;
+
+ PICODBG_DEBUG(("lookup start, lexpos range %d..%d", lexposStart,lexposEnd));
+ klex_lexblockLookup(klex, lexposStart, lexposEnd, graph, graphlen, lexres);
+ PICODBG_DEBUG(("lookup done, %d found", lexres->nrres));
+
+ return (lexres->nrres > 0);
+}
+
+
+picoos_uint8 picoklex_lexIndLookup(const picoklex_Lex this,
+ const picoos_uint8 *ind,
+ const picoos_uint8 indlen,
+ picoos_uint8 *pos,
+ picoos_uint8 **phon,
+ picoos_uint8 *phonlen) {
+ picoos_uint32 pentry;
+ klex_SubObj klex = (klex_SubObj) this;
+
+ /* check indlen */
+ if (indlen != PICOKLEX_IND_SIZE) {
+ return FALSE;
+ }
+
+ /* PICOKLEX_IND_SIZE */
+ pentry = 0x000000ff & (ind[0]);
+ pentry |= ((picoos_uint32)(ind[1]) << 8);
+ pentry |= ((picoos_uint32)(ind[2]) << 16);
+
+ /* check ind if it is within lexblocks byte stream, if not, return FALSE */
+ if (pentry >= ((picoos_uint32)klex->nrblocks * PICOKLEX_LEXBLOCK_SIZE)) {
+ return FALSE;
+ }
+
+ pentry += (klex->lexblocks[pentry]);
+ *phonlen = (klex->lexblocks[pentry++]) - 2;
+ *pos = klex->lexblocks[pentry++];
+ *phon = &(klex->lexblocks[pentry]);
+
+ PICODBG_DEBUG(("pentry: %d, phonlen: %d", pentry, *phonlen));
+ return TRUE;
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+
+/* end */
diff --git a/lib/picoklex.h b/lib/picoklex.h
new file mode 100644
index 0000000..e186393
--- /dev/null
+++ b/lib/picoklex.h
@@ -0,0 +1,129 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picoklex.h
+ *
+ * knowledge base: lexicon
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#ifndef PICOKLEX_H_
+#define PICOKLEX_H_
+
+#include "picoos.h"
+#include "picoknow.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* ************************************************************/
+/* function to create specialized kb, */
+/* to be used by picorsrc only */
+/* ************************************************************/
+
+pico_status_t picoklex_specializeLexKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common);
+
+
+/* ************************************************************/
+/* lexicon type and getLex function */
+/* ************************************************************/
+
+/* lexicon type */
+typedef struct picoklex_lex * picoklex_Lex;
+
+/* return kb lex for usage in PU */
+picoklex_Lex picoklex_getLex(picoknow_KnowledgeBase this);
+
+
+/* ************************************************************/
+/* lexicon lookup result type */
+/* ************************************************************/
+
+/* max nr of results */
+#define PICOKLEX_MAX_NRRES 4
+
+/* nr of bytes used for pos and index, needs to fit in uint32, ie. max 4 */
+#define PICOKLEX_POSIND_SIZE 4
+/* nr of bytes used for index, needs to fit in uint32, ie. max 4 */
+#define PICOKLEX_IND_SIZE 3
+/* max len (in bytes) of ind, (PICOKLEX_MAX_NRRES * PICOKLEX_POSIND_SIZE) */
+#define PICOKLEX_POSIND_MAXLEN 16
+
+
+/* the lexicon lookup result(s) are stored in field posind, which
+ contains a sequence of
+ POS1-byte, IND1-bytes, POS2-byte, IND2-bytes, etc.
+
+ the IND-bytes are the byte position(s) in the lexblocks part of the
+ lexicon byte stream, starting at picoklex_lex_t.lexblocks.
+
+ for lexentries without phones only the POS (there can be only one)
+ is stored in posind, nrres equals one, and phonfound is FALSE.
+*/
+
+typedef struct {
+ picoos_uint8 nrres; /* number of results, 0 of no entry found */
+ picoos_uint8 posindlen; /* number of posind bytes */
+ picoos_uint8 phonfound; /* phones found flag, TRUE if found */
+ picoos_uint8 posind[PICOKLEX_POSIND_MAXLEN]; /* sequence of multi-ind,
+ one per result */
+} picoklex_lexl_result_t;
+
+
+/* ************************************************************/
+/* lexicon lookup functions */
+/* ************************************************************/
+
+/** lookup lex by graph; result(s) are in lexres, ie. the phones are
+ not returned directly (because they are used later and space can be
+ saved using indices first), lexres contains an index (or several)
+ to the entry for later fast lookup once the phones are needed.
+ PICOKLEX_IND_SIZE bytes are used for the index, these ind bytes are
+ saved in the WORDINDEX items. If at least one entry is found TRUE
+ is returned, FALSE otherwise */
+picoos_uint8 picoklex_lexLookup(const picoklex_Lex this,
+ const picoos_uint8 *graph,
+ const picoos_uint16 graphlen,
+ picoklex_lexl_result_t *lexres);
+
+/** lookup lex entry by index ind; ind is a sequence of bytes with
+ length indlen (must be equal PICOKLEX_IND_SIZE) that is the content
+ of a WORDINDEX item. Returns TRUE if okay, FALSE otherwise */
+picoos_uint8 picoklex_lexIndLookup(const picoklex_Lex this,
+ const picoos_uint8 *ind,
+ const picoos_uint8 indlen,
+ picoos_uint8 *pos,
+ picoos_uint8 **phon,
+ picoos_uint8 *phonlen);
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif /*PICOKLEX_H_*/
diff --git a/lib/picoknow.c b/lib/picoknow.c
new file mode 100644
index 0000000..70b471c
--- /dev/null
+++ b/lib/picoknow.c
@@ -0,0 +1,81 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picoknow.c
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#include "picodefs.h"
+#include "picoos.h"
+#include "picodbg.h"
+#include "picoknow.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/** class : KnowledgeBase
+ * shortcut : kb
+ *
+ */
+extern picoknow_KnowledgeBase picoknow_newKnowledgeBase(picoos_MemoryManager mm)
+{
+ picoknow_KnowledgeBase this;
+ PICODBG_TRACE(("start"));
+
+ this = picoos_allocate(mm,sizeof(*this));
+ if (NULL != this) {
+ PICODBG_TRACE(("allocated KnowledgeBase at address %i with size %i",(picoos_uint32)this,sizeof(*this)));
+ /* initialize */
+ this->next = NULL;
+ this->id = PICOKNOW_KBID_NULL;
+ this->base = NULL;
+ this->size = 0;
+ this->subObj = NULL;
+ this->subDeallocate = NULL;
+ }
+ return this;
+}
+
+extern void picoknow_disposeKnowledgeBase(picoos_MemoryManager mm, picoknow_KnowledgeBase * this)
+{
+ picoos_uint8 id;
+ if (NULL != (*this)) {
+ id = (*this)->id;
+ PICODBG_TRACE(("disposing KnowledgeBase id=%i",id));
+ /* terminate */
+ if ((*this)->subObj != NULL) {
+ (*this)->subDeallocate((*this),mm);
+ }
+ picoos_deallocate(mm,(void**)this);
+ }
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+/* End picoknow.c */
diff --git a/lib/picoknow.h b/lib/picoknow.h
new file mode 100644
index 0000000..1444856
--- /dev/null
+++ b/lib/picoknow.h
@@ -0,0 +1,190 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picoknow.h
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+/**
+ * @addtogroup picoknow
+
+ * <b> Pico knowledge base </b>\n
+ *
+*/
+
+#ifndef PICOKNOW_H_
+#define PICOKNOW_H_
+
+#include "picodefs.h"
+#include "picoos.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+typedef enum picoknow_kb_id {
+ PICOKNOW_KBID_NULL = 0,
+ /* base / tpp 1 - 7 */
+ PICOKNOW_KBID_TPP_MAIN = 1,
+ PICOKNOW_KBID_TAB_GRAPHS = 2,
+ PICOKNOW_KBID_TAB_PHONES = 3,
+ PICOKNOW_KBID_TAB_POS = 4,
+ PICOKNOW_KBID_FIXED_IDS = 7,
+ /* debug */
+ PICOKNOW_KBID_DBG = 8,
+
+ /* textana 9 - 32 */
+ PICOKNOW_KBID_LEX_MAIN = 9,
+ PICOKNOW_KBID_DT_POSP = 10,
+ PICOKNOW_KBID_DT_POSD = 11,
+ PICOKNOW_KBID_DT_G2P = 12,
+ PICOKNOW_KBID_FST_WPHO_1 = 13,
+ PICOKNOW_KBID_FST_WPHO_2 = 14,
+ PICOKNOW_KBID_FST_WPHO_3 = 15,
+ PICOKNOW_KBID_FST_WPHO_4 = 16,
+ PICOKNOW_KBID_FST_WPHO_5 = 17,
+ PICOKNOW_KBID_DT_PHR = 18,
+ PICOKNOW_KBID_DT_ACC = 19,
+ PICOKNOW_KBID_FST_SPHO_1 = 20,
+ PICOKNOW_KBID_FST_SPHO_2 = 21,
+ PICOKNOW_KBID_FST_SPHO_3 = 22,
+ PICOKNOW_KBID_FST_SPHO_4 = 23,
+ PICOKNOW_KBID_FST_SPHO_5 = 24,
+
+ PICOKNOW_KBID_FST_XSAMPA_PARSE = 25,
+ PICOKNOW_KBID_FST_SVOXPA_PARSE = 26,
+ PICOKNOW_KBID_FST_XSAMPA2SVOXPA = 27,
+
+ PICOKNOW_KBID_FST_SPHO_6 = 28,
+ PICOKNOW_KBID_FST_SPHO_7 = 29,
+ PICOKNOW_KBID_FST_SPHO_8 = 30,
+ PICOKNOW_KBID_FST_SPHO_9 = 31,
+ PICOKNOW_KBID_FST_SPHO_10 = 32,
+
+
+ /* siggen 33 - 48 */
+ PICOKNOW_KBID_DT_DUR = 34,
+ PICOKNOW_KBID_DT_LFZ1 = 35,
+ PICOKNOW_KBID_DT_LFZ2 = 36,
+ PICOKNOW_KBID_DT_LFZ3 = 37,
+ PICOKNOW_KBID_DT_LFZ4 = 38,
+ PICOKNOW_KBID_DT_LFZ5 = 39,
+ PICOKNOW_KBID_DT_MGC1 = 40,
+ PICOKNOW_KBID_DT_MGC2 = 41,
+ PICOKNOW_KBID_DT_MGC3 = 42,
+ PICOKNOW_KBID_DT_MGC4 = 43,
+ PICOKNOW_KBID_DT_MGC5 = 44,
+ PICOKNOW_KBID_PDF_DUR = 45,
+ PICOKNOW_KBID_PDF_LFZ = 46,
+ PICOKNOW_KBID_PDF_MGC = 47,
+ PICOKNOW_KBID_PDF_PHS = 48,
+
+ /* user tpp 49 - 56 */
+ PICOKNOW_KBID_TPP_USER_1 = 49,
+ PICOKNOW_KBID_TPP_USER_2 = 50,
+
+ /* user lex 57 - 63 */
+ PICOKNOW_KBID_LEX_USER_1 = 57,
+ PICOKNOW_KBID_LEX_USER_2 = 58,
+
+ PICOKNOW_KBID_DUMMY = 127
+
+} picoknow_kb_id_t;
+
+#define PICOKNOW_DEFAULT_RESOURCE_NAME (picoos_char *) "__PICO_DEF_RSRC"
+
+#define PICOKNOW_MAX_NUM_WPHO_FSTS 5
+#define PICOKNOW_MAX_NUM_SPHO_FSTS 10
+#define PICOKNOW_MAX_NUM_ULEX 2
+#define PICOKNOW_MAX_NUM_UTPP 2
+
+#define PICOKNOW_KBID_WPHO_ARRAY { \
+ PICOKNOW_KBID_FST_WPHO_1, \
+ PICOKNOW_KBID_FST_WPHO_2, \
+ PICOKNOW_KBID_FST_WPHO_3, \
+ PICOKNOW_KBID_FST_WPHO_4, \
+ PICOKNOW_KBID_FST_WPHO_5 \
+}
+
+#define PICOKNOW_KBID_SPHO_ARRAY { \
+ PICOKNOW_KBID_FST_SPHO_1, \
+ PICOKNOW_KBID_FST_SPHO_2, \
+ PICOKNOW_KBID_FST_SPHO_3, \
+ PICOKNOW_KBID_FST_SPHO_4, \
+ PICOKNOW_KBID_FST_SPHO_5, \
+ PICOKNOW_KBID_FST_SPHO_6, \
+ PICOKNOW_KBID_FST_SPHO_7, \
+ PICOKNOW_KBID_FST_SPHO_8, \
+ PICOKNOW_KBID_FST_SPHO_9, \
+ PICOKNOW_KBID_FST_SPHO_10 \
+}
+
+#define PICOKNOW_KBID_ULEX_ARRAY { \
+ PICOKNOW_KBID_LEX_USER_1, \
+ PICOKNOW_KBID_LEX_USER_2, \
+}
+
+#define PICOKNOW_KBID_UTPP_ARRAY { \
+ PICOKNOW_KBID_TPP_USER_1, \
+ PICOKNOW_KBID_TPP_USER_2, \
+}
+
+/* max size (including NULLC) of descriptive name corresponding to KBID */
+#define PICOKNOW_MAX_KB_NAME_SIZ 16
+
+/* maximum number of kbs in one resource */
+#define PICOKNOW_MAX_NUM_RESOURCE_KBS 64
+
+
+/** class : KnowledgeBase
+ * shortcut : kb
+ *
+ */
+typedef struct picoknow_knowledge_base * picoknow_KnowledgeBase;
+
+typedef pico_status_t (* picoknow_kbSubDeallocate) (register picoknow_KnowledgeBase this, picoos_MemoryManager mm);
+
+typedef struct picoknow_knowledge_base {
+ /* public */
+ picoknow_KnowledgeBase next;
+ picoknow_kb_id_t id;
+ picoos_uint8 * base; /* start address */
+ picoos_uint32 size; /* size */
+
+ /* protected */
+ picoknow_kbSubDeallocate subDeallocate;
+ void * subObj;
+} picoknow_knowledge_base_t;
+
+extern picoknow_KnowledgeBase picoknow_newKnowledgeBase(picoos_MemoryManager mm);
+
+extern void picoknow_disposeKnowledgeBase(picoos_MemoryManager mm, picoknow_KnowledgeBase * this);
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif /*PICOKNOW_H_*/
diff --git a/lib/picokpdf.c b/lib/picokpdf.c
new file mode 100644
index 0000000..bbdbec4
--- /dev/null
+++ b/lib/picokpdf.c
@@ -0,0 +1,381 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picokpdf.c
+ *
+ * knowledge handling for pdf
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#include "picoos.h"
+#include "picodbg.h"
+#include "picoknow.h"
+#include "picokpdf.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* ************************************************************/
+/* pdf */
+/* ************************************************************/
+
+/*
+ * @addtogroup picokpdf
+ *
+ overview: format of knowledge base pdf file
+
+ This is the format for the dur pdf file:
+ - Numframes: 1 uint16
+ - Vecsize: 1 uint8
+ - sampperframe: 1 uint8
+ - Phonquantlen: 1 uint8
+ - Phonquant: Phonquantlen uint8
+ - Statequantlen: 1 uint8
+ - Statequantlen: Statequantlen uint8
+ - And then numframes x vecsize uint8
+
+ This is the format for mul (mgc and lfz) pdf files:
+ - numframes: 1 uint16
+ - vecsize: 1 uint8
+ - numstates: 1 uint8
+ - numframesperstate: numstates uint16
+ - ceporder: 1 uint8
+ - numvuv 1 uint8
+ - numdeltas: 1 uint8
+ - scmeanpow: 1 uint8
+ - maxbigpow: 1 uint8
+ - scmeanpowum KPDF_NUMSTREAMS * ceporder uint8
+ - scivarpow KPDF_NUMSTREAMS * ceporder uint8
+
+ And then numframes x vecsize uint8
+
+*/
+
+
+/* ************************************************************/
+/* pdf data defines */
+/* may not be changed with current implementation */
+/* ************************************************************/
+
+
+#define KPDF_NUMSTREAMS 3 /* coeff, delta, deltadelta */
+
+
+/* ************************************************************/
+/* pdf loading */
+/* ************************************************************/
+
+static pico_status_t kpdfDURInitialize(register picoknow_KnowledgeBase this,
+ picoos_Common common) {
+ picokpdf_pdfdur_t *pdfdur;
+ picoos_uint16 pos;
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ pdfdur = (picokpdf_pdfdur_t *)this->subObj;
+
+ pos = 0;
+
+ pdfdur->numframes = ((picoos_uint16)(this->base[pos+1])) << 8 |
+ this->base[pos];
+ pos += 2;
+ pdfdur->vecsize = this->base[pos++];
+ pdfdur->sampperframe = this->base[pos++];
+ pdfdur->phonquantlen = this->base[pos++];
+ pdfdur->phonquant = &(this->base[pos]);
+ pos += pdfdur->phonquantlen;
+ pdfdur->statequantlen = this->base[pos++];
+ pdfdur->statequant = &(this->base[pos]);
+ pos += pdfdur->statequantlen;
+ pdfdur->content = &(this->base[pos]);
+ PICODBG_DEBUG(("numframes %d, vecsize %d, phonquantlen %d, "
+ "statequantlen %d", pdfdur->numframes, pdfdur->vecsize,
+ pdfdur->phonquantlen, pdfdur->statequantlen));
+ if ((picoos_uint32)(pos + (pdfdur->numframes * pdfdur->vecsize)) != this->size) {
+ PICODBG_DEBUG(("header-spec size %d, kb-size %d",
+ pos + (pdfdur->numframes * pdfdur->vecsize),
+ this->size));
+ return picoos_emRaiseException(common->em, PICO_EXC_FILE_CORRUPT,
+ NULL, NULL);
+ }
+ PICODBG_DEBUG(("dur pdf initialized"));
+ return PICO_OK;
+}
+
+static picoos_uint8 convScaleFactorToBig(picoos_uint8 pow, picoos_uint8 bigpow)
+{
+ if (pow > 0x0F) {
+ pow = bigpow + (0xFF - pow + 1); /* take 2's complement of negative pow */
+ } else if (bigpow >= pow) {
+ pow = bigpow - pow;
+ } else {
+ /* error: bigpow is smaller than input pow */
+ return 0;
+ }
+ return pow;
+}
+
+static pico_status_t kpdfMULInitialize(register picoknow_KnowledgeBase this,
+ picoos_Common common) {
+ picokpdf_pdfmul_t *pdfmul;
+ picoos_uint16 pos;
+ picoos_uint8 scmeanpow, maxbigpow, nummean;
+ picoos_uint8 i;
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ pdfmul = (picokpdf_pdfmul_t *)this->subObj;
+
+ pos = 0;
+
+ pdfmul->numframes = ((picoos_uint16)(this->base[pos+1])) << 8 |
+ this->base[pos];
+ pos += 2;
+ pdfmul->vecsize = this->base[pos++];
+ pdfmul->numstates = this->base[pos++];
+ {
+ pdfmul->stateoffset[0] = (picoos_uint16) 0;
+ for (i=1; i<pdfmul->numstates; i++) {
+ pdfmul->stateoffset[i] = pdfmul->stateoffset[i-1] + (this->base[pos] | ((picoos_uint16) this->base[pos+1] << 8));
+ pos += 2;
+ }
+ pos += 2; /* we don't need the last number if we only need the offset (i.e. how to get to the vector start) */
+ }
+
+ pdfmul->ceporder = this->base[pos++];
+ pdfmul->numvuv = this->base[pos++];
+ pdfmul->numdeltas = this->base[pos++];
+ scmeanpow = this->base[pos++];
+ maxbigpow = this->base[pos++];
+ if (maxbigpow < PICOKPDF_BIG_POW) {
+ PICODBG_ERROR(("bigpow %i is larger than maxbigpow %i defined in pdf lingware", PICOKPDF_BIG_POW, maxbigpow));
+ return picoos_emRaiseException(common->em, PICO_EXC_MAX_NUM_EXCEED,NULL,NULL);
+ }
+ pdfmul->bigpow = PICOKPDF_BIG_POW; /* what we have to use is the smaller number! */
+
+ pdfmul->amplif = this->base[pos++];
+
+ /* bigpow corrected by scmeanpow, multiply means by 2^meanpow to obtain fixed point representation */
+ pdfmul->meanpow = convScaleFactorToBig(scmeanpow, pdfmul->bigpow);
+ if (0 == pdfmul->meanpow) {
+ PICODBG_ERROR(("error in convScaleFactorToBig"));
+ return picoos_emRaiseException(common->em, PICO_EXC_MAX_NUM_EXCEED,NULL,NULL);
+ }
+ nummean = 3*pdfmul->ceporder;
+
+ pdfmul->meanpowUm = picoos_allocate(common->mm,nummean*sizeof(picoos_uint8));
+ pdfmul->ivarpow = picoos_allocate(common->mm,nummean*sizeof(picoos_uint8));
+ if ((NULL == pdfmul->meanpowUm) || (NULL == pdfmul->ivarpow)) {
+ picoos_deallocate(common->mm,(void *) &(pdfmul->meanpowUm));
+ picoos_deallocate(common->mm,(void *) &(pdfmul->ivarpow));
+ return picoos_emRaiseException(common->em,PICO_EXC_OUT_OF_MEM,NULL,NULL);
+ }
+
+ /* read meanpowUm and convert on the fly */
+ /* meaning of meanpowUm becomes: multiply means from pdf stream by 2^meanpowUm
+ * to achieve fixed point scaling by big
+ */
+ for (i=0; i<nummean; i++) {
+ pdfmul->meanpowUm[i] = convScaleFactorToBig(this->base[pos++], pdfmul->bigpow);
+ }
+
+ /*read ivarpow and convert on the fly */
+ for (i=0; i<nummean; i++) {
+ pdfmul->ivarpow[i] = convScaleFactorToBig(this->base[pos++], pdfmul->bigpow);
+ }
+
+ /* check numdeltas */
+ if ((pdfmul->numdeltas == 0xFF) && (pdfmul->vecsize != (pdfmul->numvuv + pdfmul->ceporder * 3 * (2+1)))) {
+ PICODBG_ERROR(("header has inconsistent values for vecsize, ceporder, numvuv, and numdeltas"));
+ return picoos_emRaiseException(common->em,PICO_EXC_FILE_CORRUPT,NULL,NULL);
+ }
+
+/* vecsize: 1 uint8 for vuv
+ + ceporder short for static means
+ + numdeltas uint8 and short for sparse delta means
+ + ceporder*3 uint8 for static and delta inverse variances
+*/
+ if ((pdfmul->numdeltas != 0xFF) && (pdfmul->vecsize != pdfmul->numvuv+pdfmul->ceporder*2+pdfmul->numdeltas*3+pdfmul->ceporder*3)) {
+ PICODBG_ERROR(("header has inconsistent values for vecsize, ceporder, numvuv, and numdeltas\n"
+ "vecsize = %i while numvuv+ceporder*2 + numdeltas*3 + ceporder*3 = %i",
+ pdfmul->vecsize, pdfmul->numvuv + pdfmul->ceporder*2 + pdfmul->numdeltas * 3 + pdfmul->ceporder * 3));
+ return picoos_emRaiseException(common->em,PICO_EXC_FILE_CORRUPT,NULL,NULL);
+ }
+ pdfmul->content = &(this->base[pos]);
+ PICODBG_DEBUG(("numframes %d, vecsize %d, numstates %d, ceporder %d, "
+ "numvuv %d, numdeltas %d, meanpow %d, bigpow %d",
+ pdfmul->numframes, pdfmul->vecsize, pdfmul->numstates,
+ pdfmul->ceporder, pdfmul->numvuv, pdfmul->numdeltas,
+ pdfmul->meanpow, pdfmul->bigpow));
+ if ((picoos_uint32)(pos + (pdfmul->numframes * pdfmul->vecsize)) != this->size) {
+ PICODBG_DEBUG(("header-spec size %d, kb-size %d",
+ pos + (pdfmul->numframes * pdfmul->vecsize),
+ this->size));
+ return picoos_emRaiseException(common->em, PICO_EXC_FILE_CORRUPT,
+ NULL, NULL);
+ }
+ PICODBG_DEBUG(("mul pdf initialized"));
+ return PICO_OK;
+}
+
+static pico_status_t kpdfPHSInitialize(register picoknow_KnowledgeBase this,
+ picoos_Common common) {
+ picokpdf_pdfphs_t *pdfphs;
+ picoos_uint16 pos;
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ pdfphs = (picokpdf_pdfphs_t *)this->subObj;
+
+ pos = 0;
+
+ pdfphs->numvectors = ((picoos_uint16)(this->base[pos+1])) << 8 |
+ this->base[pos];
+ pos += 2;
+ pdfphs->indexBase = &(this->base[pos]);
+ pdfphs->contentBase = pdfphs->indexBase + pdfphs->numvectors * sizeof(picoos_uint32);
+ PICODBG_DEBUG(("phs pdf initialized"));
+ return PICO_OK;
+}
+
+
+
+static pico_status_t kpdfMULSubObjDeallocate(register picoknow_KnowledgeBase this,
+ picoos_MemoryManager mm) {
+
+
+ picokpdf_pdfmul_t *pdfmul;
+
+ if ((NULL != this) && (NULL != this->subObj)) {
+ pdfmul = (picokpdf_pdfmul_t *)this->subObj;
+ picoos_deallocate(mm,(void *) &(pdfmul->meanpowUm));
+ picoos_deallocate(mm,(void *) &(pdfmul->ivarpow));
+ picoos_deallocate(mm, (void *) &(this->subObj));
+ }
+ return PICO_OK;
+}
+
+static pico_status_t kpdfDURSubObjDeallocate(register picoknow_KnowledgeBase this,
+ picoos_MemoryManager mm) {
+ if (NULL != this) {
+ picoos_deallocate(mm, (void *) &this->subObj);
+ }
+ return PICO_OK;
+}
+
+static pico_status_t kpdfPHSSubObjDeallocate(register picoknow_KnowledgeBase this,
+ picoos_MemoryManager mm) {
+ if (NULL != this) {
+ picoos_deallocate(mm, (void *) &this->subObj);
+ }
+ return PICO_OK;
+}
+
+/* we don't offer a specialized constructor for a *KnowledgeBase but
+ * instead a "specializer" of an allready existing generic
+ * picoknow_KnowledgeBase */
+
+pico_status_t picokpdf_specializePdfKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common,
+ const picokpdf_kpdftype_t kpdftype) {
+ pico_status_t status;
+
+ if (NULL == this) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ switch (kpdftype) {
+ case PICOKPDF_KPDFTYPE_DUR:
+ this->subDeallocate = kpdfDURSubObjDeallocate;
+ this->subObj = picoos_allocate(common->mm,sizeof(picokpdf_pdfdur_t));
+ if (NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM,
+ NULL, NULL);
+ }
+ status = kpdfDURInitialize(this, common);
+ break;
+ case PICOKPDF_KPDFTYPE_MUL:
+ this->subDeallocate = kpdfMULSubObjDeallocate;
+ this->subObj = picoos_allocate(common->mm,sizeof(picokpdf_pdfmul_t));
+ if (NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM,
+ NULL, NULL);
+ }
+ status = kpdfMULInitialize(this, common);
+ break;
+ case PICOKPDF_KPDFTYPE_PHS:
+ this->subDeallocate = kpdfPHSSubObjDeallocate;
+ this->subObj = picoos_allocate(common->mm,sizeof(picokpdf_pdfphs_t));
+ if (NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM,
+ NULL, NULL);
+ }
+ status = kpdfPHSInitialize(this, common);
+ break;
+
+ default:
+ return picoos_emRaiseException(common->em, PICO_ERR_OTHER,
+ NULL, NULL);
+ }
+
+ if (status != PICO_OK) {
+ picoos_deallocate(common->mm, (void *) &this->subObj);
+ return picoos_emRaiseException(common->em, status, NULL, NULL);
+ }
+ return PICO_OK;
+}
+
+
+/* ************************************************************/
+/* pdf getPdf* */
+/* ************************************************************/
+
+picokpdf_PdfDUR picokpdf_getPdfDUR(picoknow_KnowledgeBase this) {
+ return ((NULL == this) ? NULL : ((picokpdf_PdfDUR) this->subObj));
+}
+
+picokpdf_PdfMUL picokpdf_getPdfMUL(picoknow_KnowledgeBase this) {
+ return ((NULL == this) ? NULL : ((picokpdf_PdfMUL) this->subObj));
+}
+
+picokpdf_PdfPHS picokpdf_getPdfPHS(picoknow_KnowledgeBase this) {
+ return ((NULL == this) ? NULL : ((picokpdf_PdfPHS) this->subObj));
+}
+
+
+#ifdef __cplusplus
+}
+#endif
+
+
+/* end */
diff --git a/lib/picokpdf.h b/lib/picokpdf.h
new file mode 100644
index 0000000..f4508a1
--- /dev/null
+++ b/lib/picokpdf.h
@@ -0,0 +1,160 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picokpdf.h
+ *
+ * knowledge handling for pdf
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#ifndef PICOKPDF_H_
+#define PICOKPDF_H_
+
+#include "picoos.h"
+#include "picoknow.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* ************************************************************/
+/**
+ * @addtogroup picokpdf
+ *
+ Two specialized pdf kb types are provided by this knowledge
+ handling module:
+
+ - pdf dur: ...kpdf_DUR (for dur)
+ - pdf mul: ...kpdf_MUL (for lfz and mgc)
+ - pdf phs: ...kpdf_PHS (for phase)
+
+*/
+/* ************************************************************/
+
+
+/* ************************************************************/
+/* defines and functions to create specialized kb, */
+/* to be used by picorsrc only */
+/* ************************************************************/
+
+#define PICOKPDF_MAX_NUM_STATES 10
+
+#define PICOKPDF_MAX_MUL_LFZ_CEPORDER 1
+#define PICOKPDF_MAX_MUL_MGC_CEPORDER 25
+
+/* trade accuracy against computation: more long multiplications.
+ * Maximum is 15 when invdiag0=(1<<(2*bigpow))/diag0 used
+ * currently observing instability in mlpg when bigpow >= 14, this needs to be investigated */
+
+#define PICOKPDF_BIG_POW 12
+
+typedef enum {
+ PICOKPDF_KPDFTYPE_DUR,
+ PICOKPDF_KPDFTYPE_MUL,
+ PICOKPDF_KPDFTYPE_PHS
+} picokpdf_kpdftype_t;
+
+pico_status_t picokpdf_specializePdfKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common,
+ const picokpdf_kpdftype_t type);
+
+
+/* ************************************************************/
+/* pdf types and get Pdf functions */
+/* ************************************************************/
+
+/** object : PdfDur, PdfMUL
+ * shortcut : kpdf*
+ * derived from : picoknow_KnowledgeBase
+ */
+
+typedef struct picokpdf_pdfdur *picokpdf_PdfDUR;
+typedef struct picokpdf_pdfmul *picokpdf_PdfMUL;
+typedef struct picokpdf_pdfphs *picokpdf_PdfPHS;
+
+/* subobj specific for pdf dur type */
+typedef struct picokpdf_pdfdur {
+ picoos_uint16 numframes;
+ picoos_uint8 vecsize;
+ picoos_uint8 sampperframe;
+ picoos_uint8 phonquantlen;
+ picoos_uint8 *phonquant;
+ picoos_uint8 statequantlen;
+ picoos_uint8 *statequant;
+ picoos_uint8 *content;
+} picokpdf_pdfdur_t;
+
+/* subobj specific for pdf mul type */
+typedef struct picokpdf_pdfmul {
+ picoos_uint16 numframes;
+ picoos_uint8 vecsize;
+ picoos_uint8 numstates;
+ picoos_uint16 stateoffset[PICOKPDF_MAX_NUM_STATES]; /* offset within a phone to find the state ? */
+ picoos_uint8 ceporder;
+ picoos_uint8 numvuv;
+ picoos_uint8 numdeltas;
+ picoos_uint8 meanpow;
+ picoos_uint8 bigpow;
+ picoos_uint8 amplif;
+ picoos_uint8 *meanpowUm; /* KPDF_NUMSTREAMS x ceporder values */
+ picoos_uint8 *ivarpow; /* KPDF_NUMSTREAMS x ceporder values */
+ picoos_uint8 *content;
+} picokpdf_pdfmul_t;
+
+/* subobj specific for pdf phs type */
+typedef struct picokpdf_pdfphs {
+ picoos_uint16 numvectors;
+ picoos_uint8 *indexBase;
+ picoos_uint8 *contentBase;
+} picokpdf_pdfphs_t;
+
+/* return kb pdf for usage in PU */
+picokpdf_PdfDUR picokpdf_getPdfDUR(picoknow_KnowledgeBase this);
+picokpdf_PdfMUL picokpdf_getPdfMUL(picoknow_KnowledgeBase this);
+picokpdf_PdfPHS picokpdf_getPdfPHS(picoknow_KnowledgeBase this);
+
+
+/* ************************************************************/
+/* PDF DUR functions */
+/* ************************************************************/
+
+/* e.g. */
+/*picoos_uint8 picokpdf_pdfDURgetEle(const picokpdf_PdfDUR this,
+ const picoos_uint16 row,
+ const picoos_uint16 col,
+ picoos_uint16 *val);
+*/
+
+/* ************************************************************/
+/* PDF MUL functions */
+/* ************************************************************/
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif /*PICOKPDF_H_*/
diff --git a/lib/picokpr.c b/lib/picokpr.c
new file mode 100644
index 0000000..956ae39
--- /dev/null
+++ b/lib/picokpr.c
@@ -0,0 +1,652 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picokpr.c
+ *
+ * knowledge handling for text preprocessing
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+#include "picoos.h"
+#include "picodbg.h"
+#include "picodata.h"
+#include "picoknow.h"
+#include "picokpr.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* ************************************************************/
+/* preproc */
+/* ************************************************************/
+
+/*
+ overview:
+*/
+
+
+/* ************************************************************/
+/* preproc data defines */
+/* ************************************************************/
+
+#define KPR_STR_SIZE 1
+#define KPR_LEXCAT_SIZE 2
+#define KPR_ATTRVAL_SIZE 4
+#define KPR_OUTITEM_SIZE 7
+#define KPR_TOK_SIZE 16
+#define KPR_PROD_SIZE 12
+#define KPR_CTX_SIZE 12
+
+#define KPR_NETNAME_OFFSET 0
+#define KPR_STRARRLEN_OFFSET 4
+#define KPR_LEXCATARRLEN_OFFSET 8
+#define KPR_ATTRVALARRLEN_OFFSET 12
+#define KPR_OUTITEMARRLEN_OFFSET 16
+#define KPR_TOKARRLEN_OFFSET 20
+#define KPR_PRODARRLEN_OFFSET 24
+#define KPR_CTXARRLEN_OFFSET 28
+
+#define KPR_ARRAY_START 32
+
+#define KPR_MAX_INT32 2147483647
+
+#define KPR_STR_OFS 0
+
+#define KPR_LEXCAT_OFS 0
+
+#define KPR_ATTRVAL_INT_OFS 0
+#define KPR_ATTRVAL_STROFS_OFS 0
+#define KPR_ATTRVAL_PRODOFS_OFS 0
+#define KPR_ATTRVAL_OUTITMOFS_OFS 0
+#define KPR_ATTRVAL_LEXCATOFS_OFS 0
+
+#define KPR_OUTITEM_NEXTOFS_OFS 0
+#define KPR_OUTITEM_TYPE_OFS 2
+#define KPR_OUTITEM_STROFS_OFS 3
+#define KPR_OUTITEM_VAL_OFS 3
+#define KPR_OUTITEM_ARGOFS_OFS 3
+
+#define KPR_TOK_SETWP_OFS 0
+#define KPR_TOK_SETNP_OFS 4
+#define KPR_TOK_NEXTOFS_OFS 8
+#define KPR_TOK_ALTLOFS_OFS 10
+#define KPR_TOK_ALTROFS_OFS 12
+#define KPR_TOK_ATTRIBOFS_OFS 14
+
+#define KPR_PROD_PRODPREFCOST_OFS 0
+#define KPR_PROD_PRODNAMEOFS_OFS 4
+#define KPR_PROD_ATOKOFS_OFS 8
+#define KPR_PROD_ETOKOFS_OFS 10
+
+#define KPR_CTX_CTXNAMEOFS_OFS 0
+#define KPR_CTX_NETNAMEOFS_OFS 4
+#define KPR_CTX_PRODNAMEOFS_OFS 8
+
+/* ************************************************************/
+/* preproc type and loading */
+/* ************************************************************/
+
+/* variable array element types */
+typedef picoos_uint8 picokpr_Str[KPR_STR_SIZE];
+typedef picoos_uint16 picokpr_LexCat2;
+typedef picoos_uint8 picokpr_AttrVal[KPR_ATTRVAL_SIZE];
+typedef picoos_uint8 picokpr_OutItem[KPR_OUTITEM_SIZE];
+typedef picoos_uint8 picokpr_Tok[KPR_TOK_SIZE];
+typedef picoos_uint8 picokpr_Prod[KPR_PROD_SIZE];
+typedef picoos_uint8 picokpr_Ctx[KPR_CTX_SIZE];
+
+/* variable array types */
+typedef picokpr_Str * picokpr_VarStrArr;
+typedef picokpr_LexCat2 * picokpr_VarLexCatArr;
+typedef picokpr_AttrVal * picokpr_VarAttrValArr;
+typedef picokpr_OutItem * picokpr_VarOutItemArr;
+typedef picokpr_Tok * picokpr_VarTokArr;
+typedef picokpr_Prod * picokpr_VarProdArr;
+typedef picokpr_Ctx * picokpr_VarCtxArr;
+
+/* ************************************************************/
+/* preproc type and loading */
+/* ************************************************************/
+
+/** object : PreprocKnowledgeBase
+ * shortcut : kpr
+ * derived from : picoknow_KnowledgeBase
+ */
+
+typedef struct kpr_subobj * kpr_SubObj;
+
+typedef struct kpr_subobj
+{
+ picoos_uchar * rNetName;
+
+ picoos_int32 rStrArrLen;
+ picoos_int32 rLexCatArrLen;
+ picoos_int32 rAttrValArrLen;
+ picoos_int32 rOutItemArrLen;
+ picoos_int32 rTokArrLen;
+ picoos_int32 rProdArrLen;
+ picoos_int32 rCtxArrLen;
+
+ picoos_uint8 * rStrArr;
+ picokpr_LexCat2 * rLexCatArr;
+ picokpr_AttrVal * rAttrValArr;
+ picokpr_OutItem * rOutItemArr;
+ picokpr_Tok * rTokArr;
+ picokpr_Prod * rProdArr;
+ picokpr_Ctx * rCtxArr;
+} kpr_subobj_t;
+
+
+static picoos_uint32 kpr_getUInt32(picoos_uint8 * p)
+{
+ return p[0] + 256*p[1] + 256*256*p[2] + 256*256*256*p[3];
+}
+
+
+static pico_status_t kprInitialize(register picoknow_KnowledgeBase this,
+ picoos_Common common)
+{
+ picoos_uint32 offset = 0;
+ kpr_subobj_t * kpr;
+
+ PICODBG_DEBUG(("start"));
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ kpr = (kpr_subobj_t *) this->subObj;
+
+ kpr->rStrArrLen = kpr_getUInt32(&(this->base[KPR_STRARRLEN_OFFSET]));
+ kpr->rLexCatArrLen = kpr_getUInt32(&(this->base[KPR_LEXCATARRLEN_OFFSET]));
+ kpr->rAttrValArrLen = kpr_getUInt32(&(this->base[KPR_ATTRVALARRLEN_OFFSET]));
+ kpr->rOutItemArrLen = kpr_getUInt32(&(this->base[KPR_OUTITEMARRLEN_OFFSET]));
+ kpr->rTokArrLen = kpr_getUInt32(&(this->base[KPR_TOKARRLEN_OFFSET]));
+ kpr->rProdArrLen = kpr_getUInt32(&(this->base[KPR_PRODARRLEN_OFFSET]));
+ kpr->rCtxArrLen = kpr_getUInt32(&(this->base[KPR_CTXARRLEN_OFFSET]));
+
+ offset = KPR_ARRAY_START;
+ kpr->rStrArr = &(this->base[offset]);
+ PICODBG_DEBUG(("rStrArr : cs: %i, ss: %i, offset: %i", sizeof(picokpr_Str), KPR_STR_SIZE, offset));
+ offset = offset + kpr->rStrArrLen * 1;
+
+ kpr->rLexCatArr = (picokpr_LexCat2 *)&(this->base[offset]);
+ PICODBG_DEBUG(("rLexCatArr : cs: %i, ss: %i, offset: %i", KPR_LEXCAT_SIZE, sizeof(picokpr_LexCat2), offset));
+ offset = offset + kpr->rLexCatArrLen * KPR_LEXCAT_SIZE;
+
+ kpr->rAttrValArr = (picokpr_AttrVal *)&(this->base[offset]);
+ PICODBG_DEBUG(("rAttrValArr : cs: %i, ss: %i, offset: %i", KPR_ATTRVAL_SIZE, sizeof(picokpr_AttrVal), offset));
+ offset = offset + kpr->rAttrValArrLen * KPR_ATTRVAL_SIZE;
+
+ kpr->rOutItemArr = (picokpr_OutItem *)&(this->base[offset]);
+ PICODBG_DEBUG(("rOutItemArr : cs: %i, ss: %i, offset: %i", KPR_OUTITEM_SIZE, sizeof(picokpr_OutItem), offset));
+ offset = offset + kpr->rOutItemArrLen * KPR_OUTITEM_SIZE;
+
+ kpr->rTokArr = (picokpr_Tok *)&(this->base[offset]);
+ PICODBG_DEBUG(("rTokArr : cs: %i, ss: %i, offset: %i", KPR_TOK_SIZE, sizeof(picokpr_Tok), offset));
+ offset = offset + kpr->rTokArrLen * KPR_TOK_SIZE;
+
+ kpr->rProdArr = (picokpr_Prod *)&(this->base[offset]);
+ PICODBG_DEBUG(("rProdArr : cs: %i, ss: %i, offset: %i", KPR_PROD_SIZE, sizeof(picokpr_Prod), offset));
+ offset = offset + kpr->rProdArrLen * KPR_PROD_SIZE;
+
+ kpr->rCtxArr = (picokpr_Ctx *)&(this->base[offset]);
+ PICODBG_DEBUG(("rCtxArr : cs: %i, ss: %i, offset: %i", KPR_CTX_SIZE, sizeof(picokpr_Ctx), offset));
+ offset = offset + kpr->rCtxArrLen * KPR_CTX_SIZE;
+
+ kpr->rNetName = &(kpr->rStrArr[kpr_getUInt32(&(this->base[KPR_NETNAME_OFFSET]))]);
+
+ return PICO_OK;
+}
+
+
+static pico_status_t kprSubObjDeallocate(register picoknow_KnowledgeBase this,
+ picoos_MemoryManager mm)
+{
+ if (NULL != this) {
+ picoos_deallocate(mm, (void *) &this->subObj);
+ }
+ return PICO_OK;
+}
+
+
+/* we don't offer a specialized constructor for a PreprocKnowledgeBase but
+ * instead a "specializer" of an allready existing generic
+ * picoknow_KnowledgeBase */
+
+pico_status_t picokpr_specializePreprocKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common)
+{
+ if (NULL == this) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ this->subDeallocate = kprSubObjDeallocate;
+ this->subObj = picoos_allocate(common->mm, sizeof(kpr_subobj_t));
+ if (NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM,
+ NULL, NULL);
+ }
+ return kprInitialize(this, common);
+}
+
+/* ************************************************************/
+/* preproc getPreproc */
+/* ************************************************************/
+
+picokpr_Preproc picokpr_getPreproc(picoknow_KnowledgeBase this)
+{
+ if (NULL == this) {
+ return NULL;
+ } else {
+ return (picokpr_Preproc) this->subObj;
+ }
+}
+
+
+/* *****************************************************************************/
+/* knowledge base access routines for strings in StrArr */
+/* *****************************************************************************/
+
+extern picokpr_VarStrPtr picokpr_getVarStrPtr(picokpr_Preproc preproc, picokpr_StrArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rStrArr[ofs]);
+
+ return p;
+}
+
+/* *****************************************************************************/
+
+extern picoos_bool picokpr_isEqual (picokpr_Preproc preproc, picoos_uchar str[], picoos_int32 len__9, picokpr_StrArrOffset str2)
+{
+ picokpr_VarStrPtr lstrp;
+ len__9 = len__9; /*PP 13.10.08 : fix warning "var not used in this function"*/
+ lstrp = (picokpr_VarStrPtr)&((kpr_SubObj)preproc)->rStrArr[str2];
+ return picoos_strcmp((picoos_char*)lstrp,(picoos_char*)str) == 0;
+}
+
+
+
+extern picoos_bool picokpr_isEqualHead (picokpr_Preproc preproc, picoos_uchar str[], picoos_int32 len__10, picokpr_StrArrOffset head)
+{
+ picokpr_VarStrPtr lstrp;
+ len__10 = len__10; /*PP 13.10.08 : fix warning "var not used in this function"*/
+ lstrp = (picokpr_VarStrPtr)&((kpr_SubObj)preproc)->rStrArr[head];
+ return (picoos_strstr((picoos_char*)str, (picoos_char*)lstrp) == (picoos_char*)str);
+}
+
+
+
+extern picoos_bool picokpr_isEqualMid (picokpr_Preproc preproc, picoos_uchar str[], picoos_int32 len__11, picokpr_StrArrOffset mid)
+{
+ picokpr_VarStrPtr lstrp;
+ len__11 = len__11; /*PP 13.10.08 : fix warning "var not used in this function"*/
+ lstrp = (picokpr_VarStrPtr)(void *) &((kpr_SubObj)preproc)->rStrArr[mid];
+ return (picoos_strstr((picoos_char*)str, (picoos_char*)lstrp) != NULL);
+}
+
+
+
+extern picoos_bool picokpr_isEqualTail (picokpr_Preproc preproc, picoos_uchar str[], picoos_int32 len__12, picokpr_StrArrOffset tail)
+{
+ picoos_int32 lstart;
+ picokpr_VarStrPtr lstrp;
+ len__12 = len__12; /* avoid warning "var not used in this function"*/
+ lstrp = (picokpr_VarStrPtr)&((kpr_SubObj)preproc)->rStrArr[tail];
+ lstart = picoos_strlen((picoos_char*)str) - picoos_strlen((picoos_char*)lstrp);
+ if (lstart >= 0) {
+ return (picoos_strstr((picoos_char*)&(str[lstart]), (picoos_char*)lstrp) != NULL);
+ } else {
+ return FALSE;
+ }
+}
+
+/* *****************************************************************************/
+/* knowledge base access routines for lexical categories in LexCatArr */
+/* *****************************************************************************/
+
+extern picokpr_LexCat picokpr_getLexCat(picokpr_Preproc preproc, picokpr_LexCatArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rLexCatArr[ofs]);
+
+ return p[0] + 256*p[1];
+}
+
+/* *****************************************************************************/
+/* knowledge base access routines for AttrVal fields in AttrValArr */
+/* *****************************************************************************/
+
+extern picoos_int32 picokpr_getAttrValArrInt32(picokpr_Preproc preproc, picokpr_AttrValArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rAttrValArr[ofs]);
+ picoos_uint32 c = p[KPR_ATTRVAL_INT_OFS] +
+ 256*p[KPR_ATTRVAL_INT_OFS+1] +
+ 256*256*p[KPR_ATTRVAL_INT_OFS+2] +
+ 256*256*256*p[KPR_ATTRVAL_INT_OFS+3];
+
+ if (c > KPR_MAX_INT32) {
+ return (c - KPR_MAX_INT32) - 1;
+ } else {
+ return (((int)c + (int) -(KPR_MAX_INT32)) - 1);
+ }
+}
+
+/* *****************************************************************************/
+/* knowledge base access routines for AttrVal fields in AttrValArr */
+/* *****************************************************************************/
+
+extern picokpr_OutItemArrOffset picokpr_getOutItemNextOfs(picokpr_Preproc preproc, picokpr_OutItemArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rOutItemArr[ofs]);
+
+ return p[KPR_OUTITEM_NEXTOFS_OFS+0] + 256*p[KPR_OUTITEM_NEXTOFS_OFS+1];
+}
+
+
+extern picoos_int32 picokpr_getOutItemType(picokpr_Preproc preproc, picokpr_OutItemArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rOutItemArr[ofs]);
+
+ return p[KPR_OUTITEM_TYPE_OFS+0];
+}
+
+
+extern picokpr_StrArrOffset picokpr_getOutItemStrOfs(picokpr_Preproc preproc, picokpr_OutItemArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rOutItemArr[ofs]);
+
+ return p[KPR_OUTITEM_STROFS_OFS+0] +
+ 256*p[KPR_OUTITEM_STROFS_OFS+1] +
+ 256*256*p[KPR_OUTITEM_STROFS_OFS+2] +
+ 256*256*256*p[KPR_OUTITEM_STROFS_OFS+3];
+}
+
+
+extern picokpr_VarStrPtr picokpr_getOutItemStr(picokpr_Preproc preproc, picokpr_OutItemArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rOutItemArr[ofs]);
+ picoos_uint32 c = p[KPR_OUTITEM_STROFS_OFS+0] +
+ 256*p[KPR_OUTITEM_STROFS_OFS+1] +
+ 256*256*p[KPR_OUTITEM_STROFS_OFS+2] +
+ 256*256*256*p[KPR_OUTITEM_STROFS_OFS+3];
+
+ return (picoos_uint8 *)&(((kpr_SubObj)preproc)->rStrArr[c]);
+}
+
+extern picoos_int32 picokpr_getOutItemVal(picokpr_Preproc preproc, picokpr_OutItemArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rOutItemArr[ofs]);
+ picoos_uint32 c = p[KPR_OUTITEM_VAL_OFS+0] +
+ 256*p[KPR_OUTITEM_VAL_OFS+1] +
+ 256*256*p[KPR_OUTITEM_VAL_OFS+2] +
+ 256*256*256*p[KPR_OUTITEM_VAL_OFS+3];
+
+ if (c > KPR_MAX_INT32) {
+ return (c - KPR_MAX_INT32) - 1;
+ } else {
+ return (((int)c + (int) -(KPR_MAX_INT32)) - 1);
+ }
+}
+
+
+extern picokpr_OutItemArrOffset picokpr_getOutItemArgOfs(picokpr_Preproc preproc, picokpr_OutItemArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rOutItemArr[ofs]);
+
+ return p[KPR_OUTITEM_ARGOFS_OFS+0] +
+ 256*p[KPR_OUTITEM_ARGOFS_OFS+1] +
+ 256*256*p[KPR_OUTITEM_ARGOFS_OFS+2] +
+ 256*256*256*p[KPR_OUTITEM_ARGOFS_OFS+3];
+}
+
+
+/* *****************************************************************************/
+/* knowledge base access routines for tokens in TokArr */
+/* *****************************************************************************/
+
+extern picokpr_TokSetNP picokpr_getTokSetNP(picokpr_Preproc preproc, picokpr_TokArrOffset ofs)
+{
+ picoos_uint32 c/*, b*/;
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rTokArr[ofs]) + KPR_TOK_SETNP_OFS;
+ /*picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rTokArr[ofs]);*/
+ picoos_uint32 p0, p1, p2, p3;
+
+ p0 = *(p++);
+ p1 = *(p++);
+ p2 = *(p++);
+ p3 = *p;
+
+ c = p0 + (p1<<8) + (p2<<16) + (p3<<24);
+
+ return c;
+
+ /*
+ c = p[KPR_TOK_SETNP_OFS+0] +
+ 256*p[KPR_TOK_SETNP_OFS+1] +
+ 256*256*p[KPR_TOK_SETNP_OFS+2] +
+ 256*256*256*p[KPR_TOK_SETNP_OFS+3];
+
+ b = 0;
+ i = 0;
+ while ((i <= 31) && (c > 0)) {
+ if (c % 2 == 1) {
+ b |= (1<<i);
+ }
+ c = c / 2;
+ i++;
+ }
+
+ return b;
+ */
+}
+
+
+extern picokpr_TokSetWP picokpr_getTokSetWP(picokpr_Preproc preproc, picokpr_TokArrOffset ofs)
+{
+ picoos_uint32 c/*, b*/;
+ /* picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rTokArr[ofs]);*/
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rTokArr[ofs]) + KPR_TOK_SETWP_OFS;
+ picoos_uint32 p0, p1, p2, p3;
+
+ p0 = *(p++);
+ p1 = *(p++);
+ p2 = *(p++);
+ p3 = *p;
+
+ c = p0 + (p1<<8) + (p2<<16) + (p3<<24);
+ return c;
+
+ /*
+ c = p[KPR_TOK_SETWP_OFS+0] +
+ 256*p[KPR_TOK_SETWP_OFS+1] +
+ 256*256*p[KPR_TOK_SETWP_OFS+2] +
+ 256*256*256*p[KPR_TOK_SETWP_OFS+3];
+
+ b = 0;
+ i = 0;
+ while ((i <= 31) && (c > 0)) {
+ if (c % 2 == 1) {
+ b |= (1<<i);
+ }
+ c = c / 2;
+ i++;
+ }
+
+ return b;
+ */
+}
+
+
+extern picokpr_TokArrOffset picokpr_getTokNextOfs(picokpr_Preproc preproc, picokpr_TokArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rTokArr[ofs]);
+
+ return p[KPR_TOK_NEXTOFS_OFS+0] + 256*p[KPR_TOK_NEXTOFS_OFS+1];
+}
+
+
+extern picokpr_TokArrOffset picokpr_getTokAltLOfs(picokpr_Preproc preproc, picokpr_TokArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rTokArr[ofs]) + KPR_TOK_ALTLOFS_OFS;
+ picokpr_TokArrOffset c = *p++;
+ return c + 256**p;
+
+ /*picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rTokArr[ofs]);
+
+ return p[KPR_TOK_ALTLOFS_OFS+0] + 256*p[KPR_TOK_ALTLOFS_OFS+1];
+ */
+}
+
+
+extern picokpr_TokArrOffset picokpr_getTokAltROfs(picokpr_Preproc preproc, picokpr_TokArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rTokArr[ofs]) + KPR_TOK_ALTROFS_OFS;
+ picokpr_TokArrOffset c = *p++;
+ return c + 256**p;
+
+ /*picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rTokArr[ofs]);
+
+ return p[KPR_TOK_ALTROFS_OFS+0] + 256*p[KPR_TOK_ALTROFS_OFS+1];*/
+}
+
+
+extern picokpr_AttrValArrOffset picokpr_getTokAttribOfs(picokpr_Preproc preproc, picokpr_TokArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rTokArr[ofs]);
+
+ return p[KPR_TOK_ATTRIBOFS_OFS+0] + 256*p[KPR_TOK_ATTRIBOFS_OFS+1];
+}
+
+/* *****************************************************************************/
+/* knowledge base access routines for productions in ProdArr */
+/* *****************************************************************************/
+
+extern picoos_int32 picokpr_getProdArrLen(picokpr_Preproc preproc)
+{
+ return ((kpr_SubObj)preproc)->rProdArrLen;
+}
+
+extern picoos_int32 picokpr_getProdPrefCost(picokpr_Preproc preproc, picokpr_ProdArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rProdArr[ofs]);
+ picoos_uint32 c = p[KPR_PROD_PRODPREFCOST_OFS+0] +
+ 256*p[KPR_PROD_PRODPREFCOST_OFS+1] +
+ 256*256*p[KPR_PROD_PRODPREFCOST_OFS+2] +
+ 256*256*256*p[KPR_PROD_PRODPREFCOST_OFS+3];
+
+
+ if (c > KPR_MAX_INT32) {
+ return (c - KPR_MAX_INT32) - 1;
+ } else {
+ return (((int)c + (int) -(KPR_MAX_INT32)) - 1);
+ }
+}
+
+
+extern picokpr_StrArrOffset picokpr_getProdNameOfs(picokpr_Preproc preproc, picokpr_ProdArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rProdArr[ofs]);
+
+ return p[KPR_PROD_PRODNAMEOFS_OFS+0] +
+ 256*p[KPR_PROD_PRODNAMEOFS_OFS+1] +
+ 256*256*p[KPR_PROD_PRODNAMEOFS_OFS+2] +
+ 256*256*256*p[KPR_PROD_PRODNAMEOFS_OFS+3];
+
+}
+
+
+extern picokpr_TokArrOffset picokpr_getProdATokOfs(picokpr_Preproc preproc, picokpr_ProdArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rProdArr[ofs]);
+
+ return p[KPR_PROD_ATOKOFS_OFS+0] + 256*p[KPR_PROD_ATOKOFS_OFS+1];
+}
+
+
+extern picokpr_TokArrOffset picokpr_getProdETokOfs(picokpr_Preproc preproc, picokpr_ProdArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rProdArr[ofs]);
+
+ return p[KPR_PROD_ETOKOFS_OFS+0] + 256*p[KPR_PROD_ETOKOFS_OFS+1];
+}
+
+/* *****************************************************************************/
+/* knowledge base access routines for contexts in CtxArr */
+/* *****************************************************************************/
+
+extern picoos_int32 picokpr_getCtxArrLen(picokpr_Preproc preproc)
+{
+ return ((kpr_SubObj)preproc)->rCtxArrLen;
+}
+
+extern picokpr_StrArrOffset picokpr_getCtxCtxNameOfs(picokpr_Preproc preproc, picokpr_CtxArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rCtxArr[ofs]);
+
+ return p[KPR_CTX_CTXNAMEOFS_OFS+0] +
+ 256*p[KPR_CTX_CTXNAMEOFS_OFS+1] +
+ 256*256*p[KPR_CTX_CTXNAMEOFS_OFS+2] +
+ 256*256*256*p[KPR_CTX_CTXNAMEOFS_OFS+3];
+}
+
+
+extern picokpr_StrArrOffset picokpr_getCtxNetNameOfs(picokpr_Preproc preproc, picokpr_CtxArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rCtxArr[ofs]);
+
+ return p[KPR_CTX_NETNAMEOFS_OFS+0] +
+ 256*p[KPR_CTX_NETNAMEOFS_OFS+1] +
+ 256*256*p[KPR_CTX_NETNAMEOFS_OFS+2] +
+ 256*256*256*p[KPR_CTX_NETNAMEOFS_OFS+3];
+}
+
+
+extern picokpr_StrArrOffset picokpr_getCtxProdNameOfs(picokpr_Preproc preproc, picokpr_CtxArrOffset ofs)
+{
+ picoos_uint8 * p = (picoos_uint8 *)&(((kpr_SubObj)preproc)->rCtxArr[ofs]);
+
+ return p[KPR_CTX_PRODNAMEOFS_OFS+0] +
+ 256*p[KPR_CTX_PRODNAMEOFS_OFS+1] +
+ 256*256*p[KPR_CTX_PRODNAMEOFS_OFS+2] +
+ 256*256*256*p[KPR_CTX_PRODNAMEOFS_OFS+3];
+}
+
+/* *****************************************************************************/
+/* knowledge base access routines for networks */
+/* *****************************************************************************/
+
+extern picokpr_VarStrPtr picokpr_getPreprocNetName(picokpr_Preproc preproc)
+{
+ return ((kpr_SubObj)preproc)->rNetName;
+}
+
+
+#ifdef __cplusplus
+}
+#endif
+
+
+/* end */
diff --git a/lib/picokpr.h b/lib/picokpr.h
new file mode 100644
index 0000000..0532b49
--- /dev/null
+++ b/lib/picokpr.h
@@ -0,0 +1,147 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picokpr.h
+ *
+ * knowledge handling for text preprocessing
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+/**
+ * @addtogroup picokpr
+
+ * <b> Knowledge handling for text preprocessing </b>\n
+ *
+*/
+
+#ifndef PICOKPR_H_
+#define PICOKPR_H_
+
+#include "picoos.h"
+#include "picoknow.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* ************************************************************/
+/* function to create specialized kb, */
+/* to be used by picorsrc only */
+/* ************************************************************/
+
+pico_status_t picokpr_specializePreprocKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common);
+
+
+/* ************************************************************/
+/* preproc type and getPreproc function */
+/* ************************************************************/
+
+/* maximal array length in preproc knowledge */
+#define NPPMaxStrArrLen 100000000
+#define NPPMaxLexCatArrLen 65536
+#define NPPMaxAttrValLen 65536
+#define NPPMaxOutItemArrLen 65536
+#define NPPMaxTokArrLen 65536
+#define NPPMaxProdArrLen 65536
+#define NPPMaxCtxArrLen 65536
+
+/* array offset types */
+typedef picoos_uint32 picokpr_StrArrOffset;
+typedef picoos_uint16 picokpr_LexCatArrOffset;
+typedef picoos_uint16 picokpr_AttrValArrOffset;
+typedef picoos_uint16 picokpr_OutItemArrOffset;
+typedef picoos_uint16 picokpr_TokArrOffset;
+typedef picoos_uint16 picokpr_ProdArrOffset;
+typedef picoos_uint16 picokpr_CtxArrOffset;
+
+typedef picoos_uchar * picokpr_VarStrPtr;
+
+typedef picoos_int16 picokpr_LexCat;
+typedef picoos_uint32 picokpr_TokSetNP;
+typedef picoos_uint32 picokpr_TokSetWP;
+
+/* preproc types */
+typedef struct picokpr_preproc * picokpr_Preproc;
+
+/* return kb preproc for usage in PU */
+picokpr_Preproc picokpr_getPreproc(picoknow_KnowledgeBase this);
+
+
+/* *****************************************************************************/
+/* access routines */
+/* *****************************************************************************/
+
+/* knowledge base access routines for strings in StrArr */
+extern picokpr_VarStrPtr picokpr_getVarStrPtr(picokpr_Preproc preproc, picokpr_StrArrOffset ofs);
+extern picoos_bool picokpr_isEqual (picokpr_Preproc preproc, picoos_uchar str[], picoos_int32 len__9, picokpr_StrArrOffset str2);
+extern picoos_bool picokpr_isEqualHead (picokpr_Preproc preproc, picoos_uchar str[], picoos_int32 len__10, picokpr_StrArrOffset head);
+extern picoos_bool picokpr_isEqualMid (picokpr_Preproc preproc, picoos_uchar str[], picoos_int32 len__11, picokpr_StrArrOffset mid);
+extern picoos_bool picokpr_isEqualTail (picokpr_Preproc preproc, picoos_uchar str[], picoos_int32 len__12, picokpr_StrArrOffset tail);
+
+/* knowledge base access routines for lexical categories in LexCatArr */
+extern picokpr_LexCat picokpr_getLexCat(picokpr_Preproc preproc, picokpr_LexCatArrOffset ofs);
+
+/* knowledge base access routines for AttrVal fields in AttrValArr */
+extern picoos_int32 picokpr_getAttrValArrInt32(picokpr_Preproc preproc, picokpr_AttrValArrOffset ofs);
+
+/* knowledge base access routines for out items fields in OutItemArr */
+extern picokpr_StrArrOffset picokpr_getOutItemStrOfs(picokpr_Preproc preproc, picokpr_OutItemArrOffset ofs);
+extern picokpr_VarStrPtr picokpr_getOutItemStr(picokpr_Preproc preproc, picokpr_OutItemArrOffset ofs);
+extern picoos_int32 picokpr_getOutItemType(picokpr_Preproc preproc, picokpr_OutItemArrOffset ofs);
+extern picoos_int32 picokpr_getOutItemVal(picokpr_Preproc preproc, picokpr_OutItemArrOffset ofs);
+extern picokpr_OutItemArrOffset picokpr_getOutItemArgOfs(picokpr_Preproc preproc, picokpr_OutItemArrOffset ofs);
+extern picokpr_OutItemArrOffset picokpr_getOutItemNextOfs(picokpr_Preproc preproc, picokpr_OutItemArrOffset ofs);
+
+/* knowledge base access routines for tokens in TokArr */
+extern picokpr_TokSetNP picokpr_getTokSetNP(picokpr_Preproc preproc, picokpr_TokArrOffset ofs);
+extern picokpr_TokSetWP picokpr_getTokSetWP(picokpr_Preproc preproc, picokpr_TokArrOffset ofs);
+extern picokpr_TokArrOffset picokpr_getTokNextOfs(picokpr_Preproc preproc, picokpr_TokArrOffset ofs);
+extern picokpr_TokArrOffset picokpr_getTokAltLOfs(picokpr_Preproc preproc, picokpr_TokArrOffset ofs);
+extern picokpr_TokArrOffset picokpr_getTokAltROfs(picokpr_Preproc preproc, picokpr_TokArrOffset ofs);
+extern picokpr_AttrValArrOffset picokpr_getTokAttribOfs(picokpr_Preproc preproc, picokpr_TokArrOffset ofs);
+
+/* knowledge base access routines for productions in ProdArr */
+extern picoos_int32 picokpr_getProdArrLen(picokpr_Preproc preproc);
+extern picoos_int32 picokpr_getProdPrefCost(picokpr_Preproc preproc, picokpr_ProdArrOffset ofs);
+extern picokpr_StrArrOffset picokpr_getProdNameOfs(picokpr_Preproc preproc, picokpr_ProdArrOffset ofs);
+extern picokpr_TokArrOffset picokpr_getProdATokOfs(picokpr_Preproc preproc, picokpr_ProdArrOffset ofs);
+extern picokpr_TokArrOffset picokpr_getProdETokOfs(picokpr_Preproc preproc, picokpr_ProdArrOffset ofs);
+
+/* knowledge base access routines for contexts in CtxArr */
+extern picoos_int32 picokpr_getCtxArrLen(picokpr_Preproc preproc);
+extern picokpr_StrArrOffset picokpr_getCtxCtxNameOfs(picokpr_Preproc preproc, picokpr_CtxArrOffset ofs);
+extern picokpr_StrArrOffset picokpr_getCtxNetNameOfs(picokpr_Preproc preproc, picokpr_CtxArrOffset ofs);
+extern picokpr_StrArrOffset picokpr_getCtxProdNameOfs(picokpr_Preproc preproc, picokpr_CtxArrOffset ofs);
+
+/* knowledge base access routines for preprocs */
+extern picokpr_VarStrPtr picokpr_getPreprocNetName(picokpr_Preproc preproc);
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif /*PICOKPR_H_*/
diff --git a/lib/picoktab.c b/lib/picoktab.c
new file mode 100644
index 0000000..ca8b470
--- /dev/null
+++ b/lib/picoktab.c
@@ -0,0 +1,1118 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picoktab.c
+ *
+ * symbol tables needed at runtime
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#include "picoos.h"
+#include "picodbg.h"
+#include "picoknow.h"
+#include "picobase.h"
+#include "picoktab.h"
+#include "picodata.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/** @todo : the following would be better part of a knowledge base.
+ * Make sure it is consistent with the phoneme symbol table used in the lingware */
+
+/* PLANE_PHONEMES */
+
+/* PLANE_POS */
+
+/* PLANE_PB_STRENGTHS */
+
+/* PLANE_ACCENTS */
+
+/* PLANE_INTERN */
+#define PICOKTAB_TMPID_PHONSTART '\x26' /* 38 '&' */
+#define PICOKTAB_TMPID_PHONTERM '\x23' /* 35 '#' */
+
+
+/* ************************************************************/
+/* fixed ids */
+/* ************************************************************/
+
+
+static pico_status_t ktabIdsInitialize(register picoknow_KnowledgeBase this,
+ picoos_Common common)
+{
+ picoktab_FixedIds ids;
+
+ PICODBG_DEBUG(("start"));
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ ids = (picoktab_FixedIds) this->subObj;
+
+ ids->phonStartId = PICOKTAB_TMPID_PHONSTART;
+ ids->phonTermId = PICOKTAB_TMPID_PHONTERM;
+ return PICO_OK;
+}
+
+
+static pico_status_t ktabIdsSubObjDeallocate(register picoknow_KnowledgeBase this,
+ picoos_MemoryManager mm)
+{
+ if (NULL != this) {
+ picoos_deallocate(mm, (void *) &this->subObj);
+ }
+ return PICO_OK;
+}
+
+pico_status_t picoktab_specializeIdsKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common)
+{
+ if (NULL == this) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ this->subDeallocate = ktabIdsSubObjDeallocate;
+ this->subObj = picoos_allocate(common->mm, sizeof(picoktab_fixed_ids_t));
+ if (NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM,
+ NULL, NULL);
+ }
+ return ktabIdsInitialize(this, common);
+}
+
+picoktab_FixedIds picoktab_getFixedIds(picoknow_KnowledgeBase this)
+{
+ return ((NULL == this) ? NULL : ((picoktab_FixedIds) this->subObj));
+}
+
+
+picoktab_FixedIds picoktab_newFixedIds(picoos_MemoryManager mm)
+{
+ picoktab_FixedIds this = (picoktab_FixedIds) picoos_allocate(mm,sizeof(*this));
+ if (NULL != this) {
+ /* initialize */
+ }
+ return this;
+}
+
+
+void picoktab_disposeFixedIds(picoos_MemoryManager mm, picoktab_FixedIds * this)
+{
+ if (NULL != (*this)) {
+ /* terminate */
+ picoos_deallocate(mm,(void *)this);
+ }
+}
+
+
+
+/* ************************************************************/
+/* Graphs */
+/* ************************************************************/
+
+/* overview binary file format for graphs kb:
+
+ graphs-kb = NROFSENTRIES SIZEOFSENTRY ofstable graphs
+
+ NROFSENTRIES : 2 bytes, number of entries in offset table
+ SIZEOFSENTRY : 1 byte, size of one entry in offset table
+
+ ofstable = {OFFSET}=NROFSENTRIES (contains NROFSENTRIES entries of OFFSET)
+
+ OFFSET: SIZEOFSENTRY bytes, offset to baseaddress of graphs-kb to entry in graphs
+
+ graphs = {graph}=NROFSENTRIES (contains NROFSENTRIES entries of graph)
+
+ graph = PROPSET FROM TO [TOKENTYPE] [TOKENSUBTYPE] [VALUE] [LOWERCASE] [GRAPHSUBS1] [GRAPHSUBS2]
+
+ FROM : 1..4 unsigned bytes, UTF8 character without terminating 0
+ TO : 1..4 unsigned bytes, UTF8 character without terminating 0
+ PROPSET : 1 unsigned byte, least significant bit : has TO field
+ next bit : has TOKENTYPE
+ next bit : has TOKENSUBTYPE
+ next bit : has VALUE
+ next bit : has LOWERCASE
+ next bit : has GRAPHSUBS1
+ next bit : has GRAPHSUBS2
+ next bit : has PUNC
+
+ TOKENTYPE : 1 unsigned byte
+ TOKENSUBTYPE : 1 unsigned byte
+ VALUE : 1 unsigned byte
+ LOWERCASE : 1..4 unsigned bytes, UTF8 character without terminating 0
+ GRAPHSUBS1 : 1..4 unsigned bytes, UTF8 character without terminating 0
+ GRAPHSUBS2 : 1..4 unsigned bytes, UTF8 character without terminating 0
+ PUNC : 1 unsigned byte
+*/
+
+static picoos_uint32 ktab_propOffset (const picoktab_Graphs this, picoos_uint32 graphsOffset, picoos_uint32 prop);
+
+#define KTAB_START_GRAPHS_NR_OFFSET 0
+#define KTAB_START_GRAPHS_SIZE_OFFSET 2
+#define KTAB_START_GRAPHS_OFFSET_TABLE 3
+#define KTAB_START_GRAPHS_GRAPH_TABLE 0
+
+/* bitmasks to extract the grapheme properties info from the property set */
+#define KTAB_GRAPH_PROPSET_TO ((picoos_uint8)'\x01')
+#define KTAB_GRAPH_PROPSET_TOKENTYPE ((picoos_uint8)'\x02')
+#define KTAB_GRAPH_PROPSET_TOKENSUBTYPE ((picoos_uint8)'\x04')
+#define KTAB_GRAPH_PROPSET_VALUE ((picoos_uint8)'\x08')
+#define KTAB_GRAPH_PROPSET_LOWERCASE ((picoos_uint8)'\x010')
+#define KTAB_GRAPH_PROPSET_GRAPHSUBS1 ((picoos_uint8)'\x020')
+#define KTAB_GRAPH_PROPSET_GRAPHSUBS2 ((picoos_uint8)'\x040')
+#define KTAB_GRAPH_PROPSET_PUNCT ((picoos_uint8)'\x080')
+
+
+typedef struct ktabgraphs_subobj *ktabgraphs_SubObj;
+
+typedef struct ktabgraphs_subobj {
+ picoos_uint16 nrOffset;
+ picoos_uint16 sizeOffset;
+
+ picoos_uint8 * offsetTable;
+ picoos_uint8 * graphTable;
+} ktabgraphs_subobj_t;
+
+
+
+static pico_status_t ktabGraphsInitialize(register picoknow_KnowledgeBase this,
+ picoos_Common common) {
+ ktabgraphs_subobj_t * ktabgraphs;
+
+ PICODBG_DEBUG(("start"));
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ ktabgraphs = (ktabgraphs_subobj_t *) this->subObj;
+ ktabgraphs->nrOffset = ((int)(this->base[KTAB_START_GRAPHS_NR_OFFSET])) + 256*(int)(this->base[KTAB_START_GRAPHS_NR_OFFSET+1]);
+ ktabgraphs->sizeOffset = (int)(this->base[KTAB_START_GRAPHS_SIZE_OFFSET]);
+ ktabgraphs->offsetTable = &(this->base[KTAB_START_GRAPHS_OFFSET_TABLE]);
+ ktabgraphs->graphTable = &(this->base[KTAB_START_GRAPHS_GRAPH_TABLE]);
+ return PICO_OK;
+}
+
+static pico_status_t ktabGraphsSubObjDeallocate(register picoknow_KnowledgeBase this,
+ picoos_MemoryManager mm) {
+ if (NULL != this) {
+ picoos_deallocate(mm, (void *) &this->subObj);
+ }
+ return PICO_OK;
+}
+
+
+pico_status_t picoktab_specializeGraphsKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common) {
+ if (NULL == this) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ this->subDeallocate = ktabGraphsSubObjDeallocate;
+ this->subObj = picoos_allocate(common->mm, sizeof(ktabgraphs_subobj_t));
+ if (NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM,
+ NULL, NULL);
+ }
+ return ktabGraphsInitialize(this, common);
+}
+
+
+picoktab_Graphs picoktab_getGraphs(picoknow_KnowledgeBase this) {
+ if (NULL == this) {
+ return NULL;
+ } else {
+ return (picoktab_Graphs) this->subObj;
+ }
+}
+
+
+/* Graphs methods */
+
+picoos_uint8 picoktab_hasVowellikeProp(const picoktab_Graphs this,
+ const picoos_uint8 *graph,
+ const picoos_uint8 graphlenmax) {
+
+ picoos_uint8 ui8App;
+ picoos_uint32 graphsOffset;
+ ktabgraphs_subobj_t * g = (ktabgraphs_SubObj)this;
+
+ ui8App = graphlenmax; /* avoid warning "var not used in this function"*/
+
+ graphsOffset = picoktab_graphOffset (this, (picoos_uchar *)graph);
+ return g->graphTable[graphsOffset + ktab_propOffset (this, graphsOffset, KTAB_GRAPH_PROPSET_TOKENTYPE)] == PICODATA_ITEMINFO1_TOKTYPE_LETTERV;
+}
+
+
+static void ktab_getStrProp (const picoktab_Graphs this, picoos_uint32 graphsOffset, picoos_uint32 propOffset, picoos_uchar * str)
+{
+ ktabgraphs_subobj_t * g = (ktabgraphs_SubObj)this;
+ picoos_uint32 i, l;
+
+ i = 0;
+ l = picobase_det_utf8_length(g->graphTable[graphsOffset+propOffset]);
+ while (i<l) {
+ str[i] = g->graphTable[graphsOffset+propOffset+i];
+ i++;
+ }
+ str[l] = 0;
+}
+
+
+static picoos_uint32 ktab_propOffset(const picoktab_Graphs this,
+ picoos_uint32 graphsOffset, picoos_uint32 prop)
+/* Returns offset of property 'prop' inside the graph with offset 'graphsOffset' in graphs table;
+ If the property is found, a value > 0 is returned otherwise 0 */
+{
+ picoos_uint32 n = 0;
+ ktabgraphs_subobj_t * g = (ktabgraphs_SubObj) this;
+
+ if ((g->graphTable[graphsOffset] & prop) == prop) {
+ n = n + 1; /* overread PROPSET field */
+ n = n + picobase_det_utf8_length(g->graphTable[graphsOffset+n]); /* overread FROM field */
+ if (prop > KTAB_GRAPH_PROPSET_TO) {
+ if ((g->graphTable[graphsOffset] & KTAB_GRAPH_PROPSET_TO)
+ == KTAB_GRAPH_PROPSET_TO) {
+ n = n + picobase_det_utf8_length(g->graphTable[graphsOffset+n]); /* overread TO field */
+ }
+ } else {
+ return n;
+ }
+ if (prop > KTAB_GRAPH_PROPSET_TOKENTYPE) {
+ if ((g->graphTable[graphsOffset] & KTAB_GRAPH_PROPSET_TOKENTYPE)
+ == KTAB_GRAPH_PROPSET_TOKENTYPE) {
+ n = n + 1; /* overread TOKENTYPE field */
+ }
+ } else {
+ return n;
+ }
+ if (prop > KTAB_GRAPH_PROPSET_TOKENSUBTYPE) {
+ if ((g->graphTable[graphsOffset] & KTAB_GRAPH_PROPSET_TOKENSUBTYPE)
+ == KTAB_GRAPH_PROPSET_TOKENSUBTYPE) {
+ n = n + 1; /* overread stokentype field */
+ }
+ } else {
+ return n;
+ }
+ if (prop > KTAB_GRAPH_PROPSET_VALUE) {
+ if ((g->graphTable[graphsOffset] & KTAB_GRAPH_PROPSET_VALUE)
+ == KTAB_GRAPH_PROPSET_VALUE) {
+ n = n + 1; /* overread value field */
+ }
+ } else {
+ return n;
+ }
+ if (prop > KTAB_GRAPH_PROPSET_LOWERCASE) {
+ if ((g->graphTable[graphsOffset] & KTAB_GRAPH_PROPSET_LOWERCASE)
+ == KTAB_GRAPH_PROPSET_LOWERCASE) {
+ n = n + picobase_det_utf8_length(g->graphTable[graphsOffset+n]); /* overread lowercase field */
+ }
+ } else {
+ return n;
+ }
+ if (prop > KTAB_GRAPH_PROPSET_GRAPHSUBS1) {
+ if ((g->graphTable[graphsOffset] & KTAB_GRAPH_PROPSET_GRAPHSUBS1)
+ == KTAB_GRAPH_PROPSET_GRAPHSUBS1) {
+ n = n + picobase_det_utf8_length(g->graphTable[graphsOffset+n]); /* overread graphsubs1 field */
+ }
+ } else {
+ return n;
+ }
+ if (prop > KTAB_GRAPH_PROPSET_GRAPHSUBS2) {
+ if ((g->graphTable[graphsOffset] & KTAB_GRAPH_PROPSET_GRAPHSUBS2)
+ == KTAB_GRAPH_PROPSET_GRAPHSUBS2) {
+ n = n + picobase_det_utf8_length(g->graphTable[graphsOffset+n]); /* overread graphsubs2 field */
+ }
+ } else {
+ return n;
+ }
+ if (prop > KTAB_GRAPH_PROPSET_PUNCT) {
+ if ((g->graphTable[graphsOffset] & KTAB_GRAPH_PROPSET_PUNCT)
+ == KTAB_GRAPH_PROPSET_PUNCT) {
+ n = n + 1; /* overread value field */
+ }
+ } else {
+ return n;
+ }
+ }
+
+ return n;
+}
+
+
+picoos_uint32 picoktab_graphOffset (const picoktab_Graphs this, picoos_uchar * utf8graph)
+{ ktabgraphs_subobj_t * g = (ktabgraphs_SubObj)this;
+ picoos_int32 a, b, m;
+ picoos_uint32 graphsOffset;
+ picoos_uint32 propOffset;
+ picobase_utf8char from;
+ picobase_utf8char to;
+ picoos_bool utfGEfrom;
+ picoos_bool utfLEto;
+
+ if (g->nrOffset > 0) {
+ a = 0;
+ b = g->nrOffset-1;
+ do {
+ m = (a+b) / 2;
+
+ /* get offset to graph[m] */
+ if (g->sizeOffset == 1) {
+ graphsOffset = g->offsetTable[g->sizeOffset*m];
+ }
+ else {
+ graphsOffset = g->offsetTable[g->sizeOffset*m ] +
+ 256*g->offsetTable[g->sizeOffset*m + 1];
+ /* PICODBG_DEBUG(("picoktab_graphOffset: %i %i %i %i", m, g->offsetTable[g->sizeOffset*m], g->offsetTable[g->sizeOffset*m + 1], graphsOffset));
+ */
+ }
+
+ /* get FROM and TO field of graph[m] */
+ ktab_getStrProp(this, graphsOffset, 1, from);
+ propOffset = ktab_propOffset(this, graphsOffset, KTAB_GRAPH_PROPSET_TO);
+ if (propOffset > 0) {
+ ktab_getStrProp(this, graphsOffset, propOffset, to);
+ }
+ else {
+ picoos_strcpy((picoos_char *)to, (picoos_char *)from);
+ }
+
+ /* PICODBG_DEBUG(("picoktab_graphOffset: %i %i %i '%s' '%s' '%s'", a, m, b, from, utf8graph, to));
+ */
+ utfGEfrom = picoos_strcmp((picoos_char *)utf8graph, (picoos_char *)from) >= 0;
+ utfLEto = picoos_strcmp((picoos_char *)utf8graph, (picoos_char *)to) <= 0;
+
+ if (utfGEfrom && utfLEto) {
+ /* PICODBG_DEBUG(("picoktab_graphOffset: utf char '%s' found", utf8graph));
+ */
+ return graphsOffset;
+ }
+ if (!utfGEfrom) {
+ b = m-1;
+ }
+ else if (!utfLEto) {
+ a = m+1;
+ }
+ } while (a<=b);
+ }
+ PICODBG_DEBUG(("picoktab_graphOffset: utf char '%s' not found", utf8graph));
+ return 0;
+}
+
+
+
+
+picoos_bool picoktab_getIntPropTokenType (const picoktab_Graphs this, picoos_uint32 graphsOffset, picoos_uint8 * stokenType)
+{
+ picoos_uint32 propOffset;
+ ktabgraphs_subobj_t * g = (ktabgraphs_SubObj)this;
+
+ propOffset = ktab_propOffset(this, graphsOffset, KTAB_GRAPH_PROPSET_TOKENTYPE);
+ if (propOffset > 0) {
+ *stokenType = (picoos_uint8)(g->graphTable[graphsOffset+propOffset]);
+ return TRUE;
+ }
+ else {
+ return FALSE;
+ }
+}
+
+
+picoos_bool picoktab_getIntPropTokenSubType (const picoktab_Graphs this, picoos_uint32 graphsOffset, picoos_int8 * stokenSubType)
+{
+ picoos_uint32 propOffset;
+ ktabgraphs_subobj_t * g = (ktabgraphs_SubObj)this;
+
+ propOffset = ktab_propOffset(this, graphsOffset, KTAB_GRAPH_PROPSET_TOKENSUBTYPE);
+ if (propOffset > 0) {
+ *stokenSubType = (picoos_int8)(g->graphTable[graphsOffset+propOffset]);
+ return TRUE;
+ }
+ else {
+ return FALSE;
+ }
+}
+
+picoos_bool picoktab_getIntPropValue (const picoktab_Graphs this, picoos_uint32 graphsOffset, picoos_uint32 * value)
+{
+ picoos_uint32 propOffset;
+ ktabgraphs_subobj_t * g = (ktabgraphs_SubObj)this;
+
+ propOffset = ktab_propOffset(this, graphsOffset, KTAB_GRAPH_PROPSET_VALUE);
+ if (propOffset > 0) {
+ *value = (picoos_uint32)(g->graphTable[graphsOffset+propOffset]);
+ return TRUE;
+ }
+ else {
+ return FALSE;
+ }
+}
+
+
+picoos_bool picoktab_getIntPropPunct (const picoktab_Graphs this, picoos_uint32 graphsOffset, picoos_uint8 * info1, picoos_uint8 * info2)
+{
+ picoos_uint32 propOffset;
+ ktabgraphs_subobj_t * g = (ktabgraphs_SubObj)this;
+
+ propOffset = ktab_propOffset(this, graphsOffset, KTAB_GRAPH_PROPSET_PUNCT);
+ if (propOffset > 0) {
+ if (g->graphTable[graphsOffset+propOffset] == 2) {
+ *info1 = PICODATA_ITEMINFO1_PUNC_SENTEND;
+ }
+ else {
+ *info1 = PICODATA_ITEMINFO1_PUNC_PHRASEEND;
+ }
+ if (g->graphTable[graphsOffset+1] == '.') {
+ *info2 = PICODATA_ITEMINFO2_PUNC_SENT_T;
+ }
+ else if (g->graphTable[graphsOffset+1] == '?') {
+ *info2 = PICODATA_ITEMINFO2_PUNC_SENT_Q;
+ }
+ else if (g->graphTable[graphsOffset+1] == '!') {
+ *info2 = PICODATA_ITEMINFO2_PUNC_SENT_E;
+ }
+ else {
+ *info2 = PICODATA_ITEMINFO2_PUNC_PHRASE;
+ }
+ return TRUE;
+ }
+ else {
+ return FALSE;
+ }
+}
+
+
+picoos_bool picoktab_getStrPropLowercase (const picoktab_Graphs this, picoos_uint32 graphsOffset, picoos_uchar * lowercase)
+{
+ picoos_uint32 propOffset;
+
+ propOffset = ktab_propOffset(this, graphsOffset, KTAB_GRAPH_PROPSET_LOWERCASE);
+ if (propOffset > 0) {
+ ktab_getStrProp(this, graphsOffset, propOffset, lowercase);
+ return TRUE;
+ }
+ else {
+ return FALSE;
+ }
+}
+
+
+picoos_bool picoktab_getStrPropGraphsubs1 (const picoktab_Graphs this, picoos_uint32 graphsOffset, picoos_uchar * graphsubs1)
+{
+ picoos_uint32 propOffset;
+
+ propOffset = ktab_propOffset(this, graphsOffset, KTAB_GRAPH_PROPSET_GRAPHSUBS1);
+ if (propOffset > 0) {
+ ktab_getStrProp(this, graphsOffset, propOffset, graphsubs1);
+ return TRUE;
+ }
+ else {
+ return FALSE;
+ }
+}
+
+
+picoos_bool picoktab_getStrPropGraphsubs2 (const picoktab_Graphs this, picoos_uint32 graphsOffset, picoos_uchar * graphsubs2)
+{
+ picoos_uint32 propOffset;
+
+ propOffset = ktab_propOffset(this, graphsOffset, KTAB_GRAPH_PROPSET_GRAPHSUBS2);
+ if (propOffset > 0) {
+ ktab_getStrProp(this, graphsOffset, propOffset, graphsubs2);
+ return TRUE;
+ }
+ else {
+ return FALSE;
+ }
+}
+/* *****************************************************************/
+/* used for tools */
+
+static void ktab_getUtf8 (picoos_uchar ** pos, picoos_uchar * to)
+{
+ picoos_uint32 l;
+ l = picobase_det_utf8_length(**pos);
+ while (l>0) {
+ *(to++) = *((*pos)++);
+ l--;
+ }
+ *to = 0;
+}
+
+picoos_uint16 picoktab_graphsGetNumEntries(const picoktab_Graphs this)
+{
+ ktabgraphs_subobj_t * g = (ktabgraphs_SubObj) this;
+ return g->nrOffset;
+}
+
+void picoktab_graphsGetGraphInfo(const picoktab_Graphs this,
+ picoos_uint16 graphIndex, picoos_uchar * from, picoos_uchar * to,
+ picoos_uint8 * propset,
+ picoos_uint8 * stokenType, picoos_uint8 * stokenSubType,
+ picoos_uint8 * value, picoos_uchar * lowercase,
+ picoos_uchar * graphsubs1, picoos_uchar * graphsubs2,
+ picoos_uint8 * punct) {
+ ktabgraphs_subobj_t * g = (ktabgraphs_SubObj) this;
+ picoos_uint32 graphsOffset;
+ picoos_uint8 * pos;
+
+ /* calculate offset of graph[graphIndex] */
+ if (g->sizeOffset == 1) {
+ graphsOffset = g->offsetTable[graphIndex];
+ } else {
+ graphsOffset = g->offsetTable[2 * graphIndex]
+ + (g->offsetTable[2 * graphIndex + 1] << 8);
+ }
+ pos = &(g->graphTable[graphsOffset]);
+ *propset = *pos;
+
+ pos++; /* advance to FROM */
+ ktab_getUtf8(&pos, from); /* get FROM and advance */
+ if ((*propset) & KTAB_GRAPH_PROPSET_TO) {
+ ktab_getUtf8(&pos, to); /* get TO and advance */
+ } else {
+ picoos_strcpy((picoos_char *)to, (picoos_char *)from);
+ }
+ if ((*propset) & KTAB_GRAPH_PROPSET_TOKENTYPE) {
+ (*stokenType) = *(pos++); /* get TOKENTYPE and advance */
+ } else {
+ (*stokenType) = -1;
+ }
+ if ((*propset) & KTAB_GRAPH_PROPSET_TOKENSUBTYPE) {
+ (*stokenSubType) = *(pos++); /* get TOKENSUBTYPE and advance */
+ } else {
+ (*stokenSubType) = -1;
+ }
+ if ((*propset) & KTAB_GRAPH_PROPSET_VALUE) {
+ (*value) = *(pos++); /* get VALUE and advance */
+ } else {
+ (*value) = -1;
+ }
+ if ((*propset) & KTAB_GRAPH_PROPSET_LOWERCASE) {
+ ktab_getUtf8(&pos, lowercase); /* get LOWERCASE and advance */
+ } else {
+ lowercase[0] = NULLC;
+ }
+ if ((*propset) & KTAB_GRAPH_PROPSET_GRAPHSUBS1) {
+ ktab_getUtf8(&pos, graphsubs1); /* get GRAPHSUBS1 and advance */
+ } else {
+ graphsubs1[0] = NULLC;
+ }
+ if ((*propset) & KTAB_GRAPH_PROPSET_GRAPHSUBS2) {
+ ktab_getUtf8(&pos, graphsubs2); /* get GRAPHSUBS2 and advance */
+ } else {
+ graphsubs2[0] = NULLC;
+ }
+ if ((*propset) & KTAB_GRAPH_PROPSET_PUNCT) {
+ (*punct) = *(pos++); /* get PUNCT and advance */
+ } else {
+ (*punct) = -1;
+ }
+}
+
+/* ************************************************************/
+/* Phones */
+/* ************************************************************/
+
+/* overview binary file format for phones kb:
+
+ phones-kb = specids propertytable
+
+ specids = PRIMSTRESSID1 SECSTRESSID1 SYLLBOUNDID1 PAUSEID1 WORDBOUNDID1
+ RESERVE1 RESERVE1 RESERVE1
+
+ propertytable = {PHONEPROP2}=256
+
+ PRIMSTRESSID1: one byte, ID of primary stress
+ SECSTRESSID1: one byte, ID of secondary stress
+ SYLLBOUNDID1: one byte, ID of syllable boundary
+ PAUSEID1: one byte, ID of pause
+ RESERVE1: reserved for future use
+
+ PHONEPROP2: one byte, max. of 256 phones directly access this table
+ to check a property for a phone; binary properties
+ encoded (1 bit per prop)
+ least significant bit: vowel
+ next bit: diphth
+ next bit: glott
+ next bit: nonsyllvowel
+ next bit: syllcons
+ 3 bits spare
+ */
+
+#define KTAB_START_SPECIDS 0
+#define KTAB_IND_PRIMSTRESS 0
+#define KTAB_IND_SECSTRESS 1
+#define KTAB_IND_SYLLBOUND 2
+#define KTAB_IND_PAUSE 3
+#define KTAB_IND_WORDBOUND 4
+
+#define KTAB_START_PROPS 8
+
+
+typedef struct ktabphones_subobj *ktabphones_SubObj;
+
+typedef struct ktabphones_subobj {
+ picoos_uint8 *specids;
+ picoos_uint8 *props;
+} ktabphones_subobj_t;
+
+
+/* bitmasks to extract the property info from props */
+#define KTAB_PPROP_VOWEL '\x01'
+#define KTAB_PPROP_DIPHTH '\x02'
+#define KTAB_PPROP_GLOTT '\x04'
+#define KTAB_PPROP_NONSYLLVOWEL '\x08'
+#define KTAB_PPROP_SYLLCONS '\x10'
+
+
+static pico_status_t ktabPhonesInitialize(register picoknow_KnowledgeBase this,
+ picoos_Common common) {
+ ktabphones_subobj_t * ktabphones;
+
+ PICODBG_DEBUG(("start"));
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ ktabphones = (ktabphones_subobj_t *) this->subObj;
+ ktabphones->specids = &(this->base[KTAB_START_SPECIDS]);
+ ktabphones->props = &(this->base[KTAB_START_PROPS]);
+ return PICO_OK;
+}
+
+static pico_status_t ktabPhonesSubObjDeallocate(register picoknow_KnowledgeBase this,
+ picoos_MemoryManager mm) {
+ if (NULL != this) {
+ picoos_deallocate(mm, (void *) &this->subObj);
+ }
+ return PICO_OK;
+}
+
+pico_status_t picoktab_specializePhonesKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common) {
+ if (NULL == this) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ this->subDeallocate = ktabPhonesSubObjDeallocate;
+ this->subObj = picoos_allocate(common->mm, sizeof(ktabphones_subobj_t));
+ if (NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM,
+ NULL, NULL);
+ }
+ return ktabPhonesInitialize(this, common);
+}
+
+picoktab_Phones picoktab_getPhones(picoknow_KnowledgeBase this) {
+ if (NULL == this) {
+ return NULL;
+ } else {
+ return (picoktab_Phones) this->subObj;
+ }
+}
+
+
+/* Phones methods */
+
+picoos_uint8 picoktab_hasVowelProp(const picoktab_Phones this,
+ const picoos_uint8 ch) {
+ return (KTAB_PPROP_VOWEL & ((ktabphones_SubObj)this)->props[ch]);
+}
+picoos_uint8 picoktab_hasDiphthProp(const picoktab_Phones this,
+ const picoos_uint8 ch) {
+ return (KTAB_PPROP_DIPHTH & ((ktabphones_SubObj)this)->props[ch]);
+}
+picoos_uint8 picoktab_hasGlottProp(const picoktab_Phones this,
+ const picoos_uint8 ch) {
+ return (KTAB_PPROP_GLOTT & ((ktabphones_SubObj)this)->props[ch]);
+}
+picoos_uint8 picoktab_hasNonsyllvowelProp(const picoktab_Phones this,
+ const picoos_uint8 ch) {
+ return (KTAB_PPROP_NONSYLLVOWEL & ((ktabphones_SubObj)this)->props[ch]);
+}
+picoos_uint8 picoktab_hasSyllconsProp(const picoktab_Phones this,
+ const picoos_uint8 ch) {
+ return (KTAB_PPROP_SYLLCONS & ((ktabphones_SubObj)this)->props[ch]);
+}
+
+picoos_bool picoktab_isSyllCarrier(const picoktab_Phones this,
+ const picoos_uint8 ch) {
+ picoos_uint8 props;
+ props = ((ktabphones_SubObj)this)->props[ch];
+ return (((KTAB_PPROP_VOWEL & props) &&
+ !(KTAB_PPROP_NONSYLLVOWEL & props))
+ || (KTAB_PPROP_SYLLCONS & props));
+}
+
+picoos_bool picoktab_isPrimstress(const picoktab_Phones this,
+ const picoos_uint8 ch) {
+ return (ch == ((ktabphones_SubObj)this)->specids[KTAB_IND_PRIMSTRESS]);
+}
+picoos_bool picoktab_isSecstress(const picoktab_Phones this,
+ const picoos_uint8 ch) {
+ return (ch == ((ktabphones_SubObj)this)->specids[KTAB_IND_SECSTRESS]);
+}
+picoos_bool picoktab_isSyllbound(const picoktab_Phones this,
+ const picoos_uint8 ch) {
+ return (ch == ((ktabphones_SubObj)this)->specids[KTAB_IND_SYLLBOUND]);
+}
+picoos_bool picoktab_isWordbound(const picoktab_Phones this,
+ const picoos_uint8 ch) {
+ return (ch == ((ktabphones_SubObj)this)->specids[KTAB_IND_WORDBOUND]);
+}
+picoos_bool picoktab_isPause(const picoktab_Phones this,
+ const picoos_uint8 ch) {
+ return (ch == ((ktabphones_SubObj)this)->specids[KTAB_IND_PAUSE]);
+}
+
+picoos_uint8 picoktab_getPrimstressID(const picoktab_Phones this) {
+ return ((ktabphones_SubObj)this)->specids[KTAB_IND_PRIMSTRESS];
+}
+picoos_uint8 picoktab_getSecstressID(const picoktab_Phones this) {
+ return ((ktabphones_SubObj)this)->specids[KTAB_IND_SECSTRESS];
+}
+picoos_uint8 picoktab_getSyllboundID(const picoktab_Phones this) {
+ return ((ktabphones_SubObj)this)->specids[KTAB_IND_SYLLBOUND];
+}
+picoos_uint8 picoktab_getWordboundID(const picoktab_Phones this) {
+ return ((ktabphones_SubObj)this)->specids[KTAB_IND_WORDBOUND];
+}
+picoos_uint8 picoktab_getPauseID(const picoktab_Phones this) {
+ return ((ktabphones_SubObj)this)->specids[KTAB_IND_PAUSE];
+}
+
+/* ************************************************************/
+/* Pos */
+/* ************************************************************/
+
+/* overview binary file format for pos kb:
+
+ pos-kb = header posids
+ header = {COUNT2 OFFS2}=8
+ posids = {POSID1 {PARTID1}0:8}1:
+
+ where POSID1 is the value of the (combined) part-of-speech symbol,
+ and {PARTID1} are the symbol values of its components (empty if it
+ is not a combined symbol). The {PARTID1} list is sorted.
+ Part-of-speech symbols with equal number of components are grouped
+ together.
+
+ The header contains information about these groups:
+
+ COUNT2 specifies the number of elements in the group, and OFFS2
+ specifies the offset (relative to the beginning of the kb) where
+ the group data starts, i.e.:
+
+ 25 32 -> 25 not-combined elements, starting at offset 32
+ 44 57 -> 44 elements composed of 2 symbols, starting at offset 57
+ 23 189 -> 23 elements composed of 3 symbols, starting at offset 189
+ ...
+
+ Currently, each symbol may be composed of up to 8 other symbols.
+ Therefore, the header has 8 entries, too. The header starts with
+ the unique POS list, and then in increasing order, 2 symbols, 3
+ symbols,...
+
+Zur Anschauung die ge-printf-te Version:
+
+ 25 32
+ 44 57
+ 23 189
+ 12 281
+ 4 341
+ 1 365
+ 0 0
+ 0 0
+ 33 |
+ 34 |
+ 35 |
+ 60 |
+ etc.
+ 36 | 35 60
+ 50 | 35 95
+ 51 | 35 97
+ 58 | 35 120
+ 59 | 35 131
+ 61 | 60 75
+ 63 | 60 95
+ 64 | 60 97
+ etc.
+ 42 | 35 60 117
+ 44 | 35 60 131
+ 45 | 35 73 97
+ 48 | 35 84 97
+ 54 | 35 97 131
+ 56 | 35 113 120
+ 57 | 35 117 120
+ 62 | 60 84 122
+ etc.
+ */
+
+typedef struct ktabpos_subobj *ktabpos_SubObj;
+
+typedef struct ktabpos_subobj {
+ picoos_uint16 nrcomb[PICOKTAB_MAXNRPOS_IN_COMB];
+ picoos_uint8 *nrcombstart[PICOKTAB_MAXNRPOS_IN_COMB];
+} ktabpos_subobj_t;
+
+
+static pico_status_t ktabPosInitialize(register picoknow_KnowledgeBase this,
+ picoos_Common common) {
+ ktabpos_subobj_t *ktabpos;
+ picoos_uint16 osprev;
+ picoos_uint16 os, pos;
+ picoos_uint8 i;
+
+ PICODBG_DEBUG(("start"));
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ ktabpos = (ktabpos_subobj_t *)this->subObj;
+
+ os = 0;
+ for (i = 0, pos = 0; i < PICOKTAB_MAXNRPOS_IN_COMB; i++, pos += 4) {
+ ktabpos->nrcomb[i] = ((picoos_uint16)(this->base[pos+1])) << 8 |
+ this->base[pos];
+ if (ktabpos->nrcomb[i] > 0) {
+ osprev = os;
+ os = ((picoos_uint16)(this->base[pos+3])) << 8 | this->base[pos+2];
+ ktabpos->nrcombstart[i] = &(this->base[os]);
+ PICODBG_TRACE(("i %d, pos %d, nr %d, osprev %d, os %d", i, pos,
+ ktabpos->nrcomb[i], osprev, os));
+ if (osprev >= os) {
+ /* cannot be, in a valid kb */
+ return picoos_emRaiseException(common->em,
+ PICO_EXC_FILE_CORRUPT,
+ NULL, NULL);
+ }
+ } else {
+ if (i == 0) {
+ /* cannot be, in a valid kb */
+ return picoos_emRaiseException(common->em,
+ PICO_EXC_FILE_CORRUPT,
+ NULL, NULL);
+ }
+ ktabpos->nrcombstart[i] = NULL;
+ }
+ }
+ return PICO_OK;
+}
+
+static pico_status_t ktabPosSubObjDeallocate(register picoknow_KnowledgeBase this,
+ picoos_MemoryManager mm) {
+ if (NULL != this) {
+ picoos_deallocate(mm, (void *) &this->subObj);
+ }
+ return PICO_OK;
+}
+
+pico_status_t picoktab_specializePosKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common) {
+ if (NULL == this) {
+ return picoos_emRaiseException(common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ this->subDeallocate = ktabPosSubObjDeallocate;
+ this->subObj = picoos_allocate(common->mm, sizeof(ktabpos_subobj_t));
+ if (NULL == this->subObj) {
+ return picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM,
+ NULL, NULL);
+ }
+ return ktabPosInitialize(this, common);
+}
+
+picoktab_Pos picoktab_getPos(picoknow_KnowledgeBase this) {
+ if (NULL == this) {
+ return NULL;
+ } else {
+ return (picoktab_Pos) this->subObj;
+ }
+}
+
+
+/* Pos methods */
+
+static picoos_int16 ktab_isEqualPosGroup(const picoos_uint8 *grp1,
+ const picoos_uint8 *grp2,
+ picoos_uint8 len)
+{
+ /* if both, grp1 and grp2 would be sorted in ascending order
+ we could implement a function picoktab_comparePosGroup in
+ a similar manner as strcmp */
+
+ picoos_uint16 i, j, equal;
+
+ equal = 1;
+
+ i = 0;
+ while (equal && (i < len)) {
+ /* search grp1[i] in grp2 */
+ j = 0;
+ while ((j < len) && (grp1[i] != grp2[j])) {
+ j++;
+ }
+ equal = (j < len);
+ i++;
+ }
+
+ return equal;
+}
+
+
+picoos_bool picoktab_isUniquePos(const picoktab_Pos this,
+ const picoos_uint8 pos) {
+ ktabpos_subobj_t *ktabpos;
+ picoos_uint16 i;
+
+ /* speed-up possible with e.g. binary search */
+
+ ktabpos = (ktabpos_subobj_t *)this;
+ PICODBG_TRACE(("pos %d, nrcombinations %d", pos, ktabpos->nrcomb[0]));
+ i = 0;
+ while ((i < ktabpos->nrcomb[0]) && (pos > ktabpos->nrcombstart[0][i])) {
+ PICODBG_TRACE(("compare with pos %d at position %d",
+ ktabpos->nrcombstart[0][i], pos, i));
+ i++;
+ }
+ return ((i < ktabpos->nrcomb[0]) && (pos == ktabpos->nrcombstart[0][i]));
+}
+
+
+picoos_bool picoktab_isPartOfPosGroup(const picoktab_Pos this,
+ const picoos_uint8 pos,
+ const picoos_uint8 posgroup)
+{
+ ktabpos_subobj_t *ktabpos;
+ picoos_uint8 *grp;
+ picoos_uint16 i, j, n, s, grplen;
+ picoos_uint8 *e;
+ picoos_uint8 found;
+
+ ktabpos = (ktabpos_subobj_t *) this;
+
+ grp = NULL;
+ found = FALSE;
+ grplen = 0;
+
+ /* currently, a linear search is required to find 'posgroup'; the
+ knowledge base should be extended to allow for a faster search */
+
+ /* treat case i==0, grplen==0, ie. pos == posgroup */
+ if (pos == posgroup) {
+ found = TRUE;
+ }
+
+ i = 1;
+ while ((grp == NULL) && (i < PICOKTAB_MAXNRPOS_IN_COMB)) {
+ n = ktabpos->nrcomb[i]; /* number of entries */
+ e = ktabpos->nrcombstart[i]; /* ptr to first entry */
+ s = i + 2; /* size of an entry in bytes */
+ /* was with while starting at 0:
+ s = i > 0 ? i + 2 : 1;
+ */
+ j = 0;
+ while ((grp == NULL) && (j < n)) {
+ if (posgroup == e[0]) {
+ grp = e + 1;
+ grplen = s - 1;
+ }
+ e += s;
+ j++;
+ }
+ i++;
+ }
+
+ /* test if 'pos' is contained in the components of 'posgroup' */
+ if (grp != NULL) {
+ for (i = 0; !found && (i < grplen); i++) {
+ if (pos == grp[i]) {
+ found = TRUE;
+ }
+ }
+
+ /* just a way to test picoktab_getPosGroup */
+ /*
+ PICODBG_ASSERT(picoktab_getPosGroup(this, grp, grplen) == posgroup);
+ */
+ }
+
+ return found;
+}
+
+
+picoos_uint8 picoktab_getPosGroup(const picoktab_Pos this,
+ const picoos_uint8 *poslist,
+ const picoos_uint8 poslistlen)
+{
+ picoos_uint8 poscomb;
+ ktabpos_subobj_t *ktabpos;
+ picoos_uint16 i, j, n, s;
+ picoos_uint8 *e;
+
+ ktabpos = (ktabpos_subobj_t *) this;
+ poscomb = 0;
+
+ if ((poslistlen > 0) && (poslistlen <= PICOKTAB_MAXNRPOS_IN_COMB)) {
+ i = poslistlen - 1;
+ if (i > 0) {
+ n = ktabpos->nrcomb[i]; /* number of entries */
+ e = ktabpos->nrcombstart[i]; /* ptr to first entry */
+ s = i + 2; /* size of an entry in bytes */
+ j = 0;
+ while (!poscomb && (j < n)) {
+ if (ktab_isEqualPosGroup(poslist, e + 1, poslistlen)) {
+ poscomb = *e;
+ }
+ e += s;
+ j++;
+ }
+ if (!poscomb) {
+ /* combination not found; shouldn't occur if lingware OK! */
+ /* contingency solution: take first */
+ PICODBG_WARN(("dynamically created POS combination not found in table; taking first (%i)",poslist[0]));
+ poscomb = poslist[0];
+ }
+ } else { /* not a composed POS */
+ poscomb = poslist[0];
+ }
+ }
+
+ return poscomb;
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+
+/* end */
diff --git a/lib/picoktab.h b/lib/picoktab.h
new file mode 100644
index 0000000..71ba2ab
--- /dev/null
+++ b/lib/picoktab.h
@@ -0,0 +1,224 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picoktab.h
+ *
+ * symbol tables needed at runtime
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+/**
+ * @addtogroup picoktab
+
+ * <b> Symbol tables needed at runtime </b>\n
+ *
+*/
+
+#ifndef PICOKTAB_H_
+#define PICOKTAB_H_
+
+#include "picoos.h"
+#include "picoknow.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* ************************************************************/
+/* fixed IDs type and functions */
+/* ************************************************************/
+
+/** object : FixedIds
+ * shortcut : ids
+ */
+typedef struct picoktab_fixed_ids * picoktab_FixedIds;
+
+typedef struct picoktab_fixed_ids {
+ picoos_uint8 phonStartId;
+ picoos_uint8 phonTermId;
+} picoktab_fixed_ids_t;
+
+/* to be used by picorsrc only */
+pico_status_t picoktab_specializeIdsKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common);
+
+picoktab_FixedIds picoktab_getFixedIds(picoknow_KnowledgeBase this);
+
+
+/* ************************************************************/
+/* Graphs type and functions */
+/* ************************************************************/
+
+typedef struct picoktab_graphs *picoktab_Graphs;
+
+/* to be used by picorsrc only */
+pico_status_t picoktab_specializeGraphsKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common);
+
+/* return kb graphs for usage in PU */
+picoktab_Graphs picoktab_getGraphs(picoknow_KnowledgeBase this);
+
+/* graph access routine: if the desired graph 'utf8graph' exists in
+ the graph table a graph offset > 0 is returned, which then can be
+ used to access the properties */
+picoos_uint32 picoktab_graphOffset(const picoktab_Graphs this,
+ picoos_uchar * utf8graph);
+
+
+/* check if UTF8 char 'graph' has property vowellike, return non-zero
+ if 'ch' has the property, 0 otherwise */
+picoos_uint8 picoktab_hasVowellikeProp(const picoktab_Graphs this,
+ const picoos_uint8 *graph,
+ const picoos_uint8 graphlenmax);
+
+/* graph properties access routines: if graph with offset 'graphsOffset' has the
+ desired property, returns TRUE if 'ch' has the property, FALSE otherwise */
+picoos_bool picoktab_getIntPropTokenType(const picoktab_Graphs this,
+ picoos_uint32 graphsOffset,
+ picoos_uint8 *stokenType);
+picoos_bool picoktab_getIntPropTokenSubType(const picoktab_Graphs this,
+ picoos_uint32 graphsOffset,
+ picoos_int8 *stokenSubType);
+picoos_bool picoktab_getIntPropValue(const picoktab_Graphs this,
+ picoos_uint32 graphsOffset,
+ picoos_uint32 *value);
+picoos_bool picoktab_getStrPropLowercase(const picoktab_Graphs this,
+ picoos_uint32 graphsOffset,
+ picoos_uchar *lowercase);
+picoos_bool picoktab_getStrPropGraphsubs1(const picoktab_Graphs this,
+ picoos_uint32 graphsOffset,
+ picoos_uchar *graphsubs1);
+picoos_bool picoktab_getStrPropGraphsubs2(const picoktab_Graphs this,
+ picoos_uint32 graphsOffset,
+ picoos_uchar *graphsubs2);
+picoos_bool picoktab_getIntPropPunct(const picoktab_Graphs this,
+ picoos_uint32 graphsOffset,
+ picoos_uint8 *info1,
+ picoos_uint8 *info2);
+
+picoos_uint16 picoktab_graphsGetNumEntries(const picoktab_Graphs this);
+void picoktab_graphsGetGraphInfo(const picoktab_Graphs this,
+ picoos_uint16 graphIndex, picoos_uchar * from, picoos_uchar * to,
+ picoos_uint8 * propset,
+ picoos_uint8 * stokenType, picoos_uint8 * stokenSubType,
+ picoos_uint8 * value, picoos_uchar * lowercase,
+ picoos_uchar * graphsubs1, picoos_uchar * graphsubs2,
+ picoos_uint8 * punct);
+
+
+/* ************************************************************/
+/* Phones type and functions */
+/* ************************************************************/
+
+/* to be used by picorsrc only */
+pico_status_t picoktab_specializePhonesKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common);
+
+typedef struct picoktab_phones *picoktab_Phones;
+
+/* return kb Phones for usage in PU */
+picoktab_Phones picoktab_getPhones(picoknow_KnowledgeBase this);
+
+/* check if 'ch' has a property, return non-zero if 'ch' has the
+ property, 0 otherwise */
+picoos_uint8 picoktab_hasVowelProp(const picoktab_Phones this,
+ const picoos_uint8 ch);
+picoos_uint8 picoktab_hasDiphthProp(const picoktab_Phones this,
+ const picoos_uint8 ch);
+picoos_uint8 picoktab_hasGlottProp(const picoktab_Phones this,
+ const picoos_uint8 ch);
+picoos_uint8 picoktab_hasNonsyllvowelProp(const picoktab_Phones this,
+ const picoos_uint8 ch);
+picoos_uint8 picoktab_hasSyllconsProp(const picoktab_Phones this,
+ const picoos_uint8 ch);
+
+/* to speed up processing for often used combinations of properties
+ the following functions are provided, which check if the property
+ combination is true for 'ch' */
+picoos_bool picoktab_isSyllCarrier(const picoktab_Phones this,
+ const picoos_uint8 ch);
+
+/* some properties can be assigned to a single sym only, check if 'ch'
+ is a special sym, return TRUE if it is the special sym, FALSE
+ otherwise */
+picoos_bool picoktab_isPrimstress(const picoktab_Phones this,
+ const picoos_uint8 ch);
+picoos_bool picoktab_isSecstress(const picoktab_Phones this,
+ const picoos_uint8 ch);
+picoos_bool picoktab_isSyllbound(const picoktab_Phones this,
+ const picoos_uint8 ch);
+picoos_bool picoktab_isWordbound(const picoktab_Phones this,
+ const picoos_uint8 ch);
+picoos_bool picoktab_isPause(const picoktab_Phones this,
+ const picoos_uint8 ch);
+
+/* get specific sym values */
+picoos_uint8 picoktab_getPrimstressID(const picoktab_Phones this);
+picoos_uint8 picoktab_getSecstressID(const picoktab_Phones this);
+picoos_uint8 picoktab_getSyllboundID(const picoktab_Phones this);
+picoos_uint8 picoktab_getWordboundID(const picoktab_Phones this);
+picoos_uint8 picoktab_getPauseID(const picoktab_Phones this);
+
+/* ************************************************************/
+/* Pos type and functions */
+/* ************************************************************/
+
+/* to be used by picorsrc only */
+pico_status_t picoktab_specializePosKnowledgeBase(picoknow_KnowledgeBase this,
+ picoos_Common common);
+
+typedef struct picoktab_pos *picoktab_Pos;
+
+#define PICOKTAB_MAXNRPOS_IN_COMB 8
+
+/* return kb Pos for usage in PU */
+picoktab_Pos picoktab_getPos(picoknow_KnowledgeBase this);
+
+/* returns TRUE if 'pos' is the ID of a unique (ie. non-combined) POS,
+ returns FALSE otherwise */
+picoos_bool picoktab_isUniquePos(const picoktab_Pos this,
+ const picoos_uint8 pos);
+
+/* returns TRUE if the non-combined 'pos' is one of the POSes in the
+ combined POS group 'posgroup, returns FALSE otherwise. Note: if
+ 'posgroup' is itself non-combined, this function returns TRUE if it
+ matches with 'pos', and FALSE otherwise */
+picoos_bool picoktab_isPartOfPosGroup(const picoktab_Pos this,
+ const picoos_uint8 pos,
+ const picoos_uint8 posgroup);
+
+/* return the combined POS group ID that is a representative ID for
+ all the 'poslistlen' POSes (which can be combined themselves) in
+ poslist. Returns '0' in case of error. */
+picoos_uint8 picoktab_getPosGroup(const picoktab_Pos this,
+ const picoos_uint8 *poslist,
+ const picoos_uint8 poslistlen);
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif /*PICOKTAB_H_*/
diff --git a/lib/picoos.c b/lib/picoos.c
new file mode 100644
index 0000000..4e2b439
--- /dev/null
+++ b/lib/picoos.c
@@ -0,0 +1,2309 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picoos.c
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#include <stdarg.h>
+#include "picodefs.h"
+#include "picopal.h"
+#include "picoos.h"
+#include "picodbg.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+#define picoos_SVOXFileHeader (picoos_char *)" (C) SVOX AG "
+
+/* **********************************************
+ * default error and warning messages
+ * **********************************************/
+
+
+#define PICOOS_MSG_EXC_NUMBER_FORMAT (picoos_char *) "wrong number format"
+#define PICOOS_MSG_EXC_MAX_NUM_EXCEED (picoos_char *) "number exceeded"
+#define PICOOS_MSG_EXC_NAME_CONFLICT (picoos_char *) "name conflict"
+#define PICOOS_MSG_EXC_NAME_UNDEFINED (picoos_char *) "name undefined"
+#define PICOOS_MSG_EXC_NAME_ILLEGAL (picoos_char *) "illegal name"
+
+/* buffer interaction */
+#define PICOOS_MSG_EXC_BUF_OVERFLOW (picoos_char *) "buffer overflow"
+#define PICOOS_MSG_EXC_BUF_UNDERFLOW (picoos_char *) "buffer underflow"
+#define PICOOS_MSG_EXC_BUF_IGNORE (picoos_char *) "buffer error"
+
+/* memory allocation */
+#define PICOOS_MSG_EXC_OUT_OF_MEM (picoos_char *) "out of memory"
+
+/* files */
+#define PICOOS_MSG_EXC_CANT_OPEN_FILE (picoos_char *) "cannot open file"
+#define PICOOS_MSG_EXC_UNEXPECTED_FILE_TYPE (picoos_char *) "unexpected file type"
+#define PICOOS_MSG_EXC_FILE_CORRUPT (picoos_char *) "corrupt file"
+#define PICOOS_MSG_EXC_FILE_NOT_FOUND (picoos_char *) "file not found"
+
+/* resources */
+#define PICOOS_MSG_EXC_RESOURCE_BUSY (picoos_char *) "resource is busy"
+#define PICOOS_MSG_EXC_RESOURCE_MISSING (picoos_char *) "cannot find resource"
+
+/* knowledge bases */
+#define PICOOS_MSG_EXC_KB_MISSING (picoos_char *) "knowledge base missing"
+
+/* runtime exceptions (programming problems, usually a bug. E.g. trying to access null pointer) */
+#define PICOOS_MSG_ERR_NULLPTR_ACCESS (picoos_char *) "access violation"
+#define PICOOS_MSG_ERR_INVALID_HANDLE (picoos_char *) "invalid handle value"
+#define PICOOS_MSG_ERR_INVALID_ARGUMENT (picoos_char *) "invalid argument supplied"
+#define PICOOS_MSG_ERR_INDEX_OUT_OF_RANGE (picoos_char *) "index out of range"
+
+
+/* errors ("external" errors, e.g. hardware failure. Usually cannot be dealt with from inside pico.) */
+#define PICOOS_MSG_ERR_OTHER (picoos_char *) "other error"
+
+#define PICOOS_MSG_ERR_PU (picoos_char *) "error in processing unit"
+
+/* WARNINGS */
+
+/* general */
+#define PICOOS_MSG_WARN_INCOMPLETE (picoos_char *) "incomplete output"
+#define PICOOS_MSG_WARN_FALLBACK (picoos_char *) "using fall-back"
+#define PICOOS_MSG_WARN_OTHER (picoos_char *) "other warning"
+
+/* resources */
+#define PICOOS_MSG_WARN_KB_OVERWRITE (picoos_char *) "overwriting knowledge base"
+#define PICOOS_MSG_WARN_RESOURCE_DOUBLE_LOAD (picoos_char *) "resource already loaded"
+
+/* decision trees */
+#define PICOOS_MSG_WARN_INVECTOR (picoos_char *) "input vector not constructed"
+#define PICOOS_MSG_WARN_CLASSIFICATION (picoos_char *) "output not classified"
+#define PICOOS_MSG_WARN_OUTVECTOR (picoos_char *) "output vector not decomposed"
+
+/* processing units */
+#define PICOOS_MSG_WARN_PU_IRREG_ITEM (picoos_char *) "irregular item in processing unit"
+#define PICOOS_MSG_WARN_PU_DISCARD_BUF (picoos_char *) "discarding processing unit buffer"
+
+
+/* **********************************************
+ * wrappers for picopal functions
+ * **********************************************/
+
+picoos_int32 picoos_atoi(const picoos_char *s)
+{
+ return (picoos_int32)picopal_atoi((const picoos_char *)s);
+}
+
+
+picoos_int8 picoos_strcmp(const picoos_char *a, const picoos_char *b)
+{
+ picopal_int32 res = picopal_strcmp((const picopal_char *)a,
+ (const picopal_char *)b);
+ return (picoos_int8) ((res < 0) ? -1 : (res > 0) ? 1 : 0);
+}
+picoos_int8 picoos_strncmp(const picoos_char *a, const picoos_char *b, picoos_objsize_t siz)
+{
+ picopal_int32 res = picopal_strncmp((const picopal_char *)a,
+ (const picopal_char *)b, siz);
+ return (picoos_int8) ((res < 0) ? -1 : (res > 0) ? 1 : 0);
+}
+
+picoos_uint32 picoos_strlen(const picoos_char *s)
+{
+ return (picoos_uint32)picopal_strlen((const picopal_char *)s);
+}
+
+picoos_char *picoos_strchr(const picoos_char *s, picoos_char c)
+{
+ return (picoos_char *)picopal_strchr((const picopal_char *)s,
+ (picopal_char)c);
+}
+
+picoos_char *picoos_strstr(const picoos_char *s, const picoos_char * substr)
+{
+ return (picoos_char *)picopal_strstr((const picopal_char *)s,
+ (const picopal_char *)substr);
+}
+
+picoos_int16 picoos_slprintf(picoos_char * b, picoos_uint32 bsize, const picoos_char *f, ...)
+{
+ picopal_int16 i;
+ va_list args;
+
+ va_start(args, (char *)f);
+ i = (picoos_int16)picopal_vslprintf((picoos_char *) b, bsize, (const picoos_char *)f, args);
+ va_end(args);
+ return i;
+}
+
+picoos_char *picoos_strcpy(picoos_char *d, const picoos_char *s)
+{
+ return (picoos_char *)picopal_strcpy((picopal_char *)d,
+ (const picopal_char *)s);
+}
+
+picoos_char *picoos_strcat(picoos_char *d, const picoos_char *s)
+{
+ return (picoos_char *)picopal_strcat((picopal_char *)d,
+ (const picopal_char *)s);
+}
+
+picoos_objsize_t picoos_strlcpy(picoos_char *dst, const picoos_char *src, picoos_objsize_t siz)
+{
+ return (picoos_objsize_t) picopal_strlcpy((picopal_char *) dst, (const picopal_char *) src, (picopal_objsize_t) siz);
+}
+
+/* copies 'length' bytes from 'src' to 'dest'. (regions may be overlapping) no error checks! */
+void * picoos_mem_copy(const void * src, void * dst, picoos_objsize_t length)
+{
+ return picopal_mem_copy(src,dst,(picopal_objsize_t) length);
+}
+
+/* sets 'length' bytes starting at dest[0] to 'byte_val' */
+void * picoos_mem_set(void * dest, picoos_uint8 byte_val, picoos_objsize_t length) {
+ return picopal_mem_set(dest,(picopal_uint8)byte_val, (picopal_objsize_t)length);
+}
+
+
+picoos_double picoos_cos (const picoos_double cos_arg)
+{
+ return (picoos_double) picopal_cos ((picopal_double) cos_arg);
+}
+
+
+picoos_double picoos_sin (const picoos_double sin_arg)
+{
+ return (picoos_double) picopal_sin((picopal_double) sin_arg);
+}
+picoos_double picoos_fabs (const picoos_double fabs_arg)
+{
+ return (picoos_double) picopal_fabs((picopal_double) fabs_arg);
+}
+
+picoos_double picoos_quick_exp(const picoos_double y) {
+ return (picoos_double) picopal_quick_exp ((picopal_double)y);
+}
+
+
+/* *****************************************************************/
+/* "Common" */
+/* *****************************************************************/
+/* picoos_common is a collection of basic functionalities that must be globally
+ * accessible from every "big" function. It includes pointers to the MemoryManasger,
+ * ExceptionManager and a system-wide list of open files. */
+
+picoos_Common picoos_newCommon(picoos_MemoryManager mm)
+{
+ picoos_Common this = (picoos_Common) picoos_allocate(mm,sizeof(*this));
+ if (NULL != this) {
+ /* initialize */
+ this->em = NULL;
+ this->mm = NULL;
+ this->fileList = NULL;
+ }
+ return this;
+}
+
+void picoos_disposeCommon(picoos_MemoryManager mm, picoos_Common * this)
+{
+ if (NULL != (*this)) {
+ /* terminate */
+ picoos_deallocate(mm,(void *)this);
+ }
+}
+
+
+/* *****************************************************************/
+/* Memory Management */
+/* *****************************************************************/
+
+typedef struct mem_block_hdr * MemBlockHdr;
+typedef struct mem_block_hdr
+{
+ MemBlockHdr next;
+ byte_ptr_t data;
+ picoos_objsize_t size;
+} mem_block_hdr_t;
+
+typedef struct mem_cell_hdr * MemCellHdr;
+typedef struct mem_cell_hdr
+{
+ /* size may be <0 if used */
+ picoos_ptrdiff_t size;
+ MemCellHdr leftCell;
+ MemCellHdr prevFree, nextFree;
+} mem_cell_hdr_t;
+
+typedef struct memory_manager
+{
+ MemBlockHdr firstBlock, lastBlock; /* memory blockList */
+ MemCellHdr freeCells, lastFree; /* free memory cells (first/last sentinel */
+ /* "constants" */
+ picoos_objsize_t fullCellHdrSize; /* aligned size of full cell header, including free-links */
+ picoos_objsize_t usedCellHdrSize; /* aligned size of header part without free-links */
+ picoos_objsize_t minContSize; /* minimum requestable application content size for allocation;
+ must hold free-list info; = fullCellHdrSize-usedCellHdrSize */
+ picoos_objsize_t minCellSize; /* minimum remaining cell size when a free cell is split */
+ picoos_bool protMem; /* true if memory protection is enabled */
+ picoos_ptrdiff_t usedSize;
+ picoos_ptrdiff_t prevUsedSize;
+ picoos_ptrdiff_t maxUsedSize;
+} memory_manager_t;
+
+/** allocates 'alloc_size' bytes at start of raw memory block ('raw_mem',raw_mem_size)
+ * and returns pointer to allocated region. Returns remaining (correctly aligned) raw memory block
+ * in ('rest_mem','rest_mem_size').
+ * The allocated memory is not subject to memory management, so that it can never be freed again!
+ *
+ */
+void * picoos_raw_malloc(byte_ptr_t raw_mem,
+ picoos_objsize_t raw_mem_size, picoos_objsize_t alloc_size,
+ byte_ptr_t * rest_mem, picoos_objsize_t * rest_mem_size)
+{
+ picoos_ptrdiff_t rest;
+ if (raw_mem == NULL) {
+ return NULL;
+ } else {
+ if (alloc_size < 1) {
+ alloc_size = 1;
+ }
+ alloc_size = ((alloc_size + PICOOS_ALIGN_SIZE - 1) / PICOOS_ALIGN_SIZE)
+ * PICOOS_ALIGN_SIZE;
+
+ rest = raw_mem_size - alloc_size;
+ if (rest < 0) {
+ return NULL;
+ } else {
+ *rest_mem_size = rest;
+ *rest_mem = raw_mem + alloc_size;
+ return (void *) raw_mem;
+ }
+ }
+}
+
+/** initializes the last block of mm */
+static int os_init_mem_block(picoos_MemoryManager this)
+{
+ int isFirstBlock;
+ void * newBlockAddr;
+ picoos_objsize_t size;
+ MemCellHdr cbeg, cmid, cend;
+
+ isFirstBlock = (this->freeCells == NULL);
+ newBlockAddr = (void *) this->lastBlock->data;
+ size = this->lastBlock->size;
+ cbeg = (MemCellHdr) newBlockAddr;
+ cmid = (MemCellHdr)((picoos_objsize_t)newBlockAddr + this->fullCellHdrSize);
+ cend = (MemCellHdr)((picoos_objsize_t)newBlockAddr + size
+ - this->fullCellHdrSize);
+ cbeg->size = 0;
+
+ cbeg->leftCell = NULL;
+ cmid->size = size - 2 * this->fullCellHdrSize;
+ cmid->leftCell = cbeg;
+ cend->size = 0;
+ cend->leftCell = cmid;
+ if (isFirstBlock) {
+ cbeg->nextFree = cmid;
+ cbeg->prevFree = NULL;
+ cmid->nextFree = cend;
+ cmid->prevFree = cbeg;
+ cend->nextFree = NULL;
+ cend->prevFree = cmid;
+ this->freeCells = cbeg;
+ this->lastFree = cend;
+ } else {
+ /* add cmid to free cell list */
+ cbeg->nextFree = NULL;
+ cbeg->prevFree = NULL;
+ cmid->nextFree = this->freeCells->nextFree;
+ cmid->prevFree = this->freeCells;
+ cmid->nextFree->prevFree = cmid;
+ cmid->prevFree->nextFree = cmid;
+ cend->nextFree = NULL;
+ cbeg->prevFree = NULL;
+ }
+ return PICO_OK;
+}
+
+
+picoos_MemoryManager picoos_newMemoryManager(
+ void *raw_memory,
+ picoos_objsize_t size,
+ picoos_bool enableMemProt)
+{
+ byte_ptr_t rest_mem;
+ picoos_objsize_t rest_mem_size;
+ picoos_MemoryManager this;
+ picoos_objsize_t size2;
+ mem_cell_hdr_t test_cell;
+
+ this = picoos_raw_malloc(raw_memory, size, sizeof(memory_manager_t),
+ &rest_mem, &rest_mem_size);
+ if (this == NULL) {
+ return NULL;
+ }
+
+ /* test if memory protection functionality is available on the current
+ platform (if not, picopal_mpr_alloc() always returns NULL) */
+ if (enableMemProt) {
+ void *addr = picopal_mpr_alloc(100);
+ if (addr == NULL) {
+ enableMemProt = FALSE;
+ } else {
+ picopal_mpr_free(&addr);
+ }
+ }
+
+ this->firstBlock = NULL;
+ this->lastBlock = NULL;
+ this->freeCells = NULL;
+ this->lastFree = NULL;
+
+ this->protMem = enableMemProt;
+ this->usedSize = 0;
+ this->prevUsedSize = 0;
+ this->maxUsedSize = 0;
+
+ /* get aligned full header size */
+ this->fullCellHdrSize = ((sizeof(mem_cell_hdr_t) + PICOOS_ALIGN_SIZE - 1)
+ / PICOOS_ALIGN_SIZE) * PICOOS_ALIGN_SIZE;
+ /* get aligned size of header without free-list fields; the result may be compiler-dependent;
+ the size is therefore computed by inspecting the end addresses of the fields 'size' and 'leftCell';
+ the higher of the ending addresses is used to get the (aligned) starting address
+ of the application contents */
+ this->usedCellHdrSize = (picoos_objsize_t) &test_cell.size
+ - (picoos_objsize_t) &test_cell + sizeof(picoos_objsize_t);
+ size2 = (picoos_objsize_t) &test_cell.leftCell - (picoos_objsize_t)
+ &test_cell + sizeof(MemCellHdr);
+ if (size2 > this->usedCellHdrSize) {
+ this->usedCellHdrSize = size2;
+ }
+ /* get minimum application-usable size; must be large enough to hold remainder of
+ cell header (free-list links) when in free-list */
+ this->minContSize = this->fullCellHdrSize - this->usedCellHdrSize;
+ /* get minimum required size of a cell remaining after a cell split */
+ this->minCellSize = this->fullCellHdrSize + PICOOS_ALIGN_SIZE;
+
+ /* install remainder of raw memory block as first block */
+ raw_memory = rest_mem;
+ size = rest_mem_size;
+ this->firstBlock = this->lastBlock = picoos_raw_malloc(raw_memory, size,
+ sizeof(mem_block_hdr_t), &rest_mem, &rest_mem_size);
+ if (this->lastBlock == NULL) {
+ return NULL;
+ }
+ this->lastBlock->next = NULL;
+ this->lastBlock->data = rest_mem;
+ this->lastBlock->size = rest_mem_size;
+
+ os_init_mem_block(this);
+
+ return this;
+}
+
+void picoos_disposeMemoryManager(picoos_MemoryManager * mm)
+{
+ *mm = NULL;
+}
+
+
+/* the following memory manager routines are for testing and
+ debugging purposes */
+
+
+void *picoos_allocProtMem(picoos_MemoryManager mm, picoos_objsize_t byteSize)
+{
+ if (mm->protMem) {
+ return picopal_mpr_alloc(byteSize);
+ } else {
+ return picoos_allocate(mm, byteSize);
+ }
+}
+
+
+void picoos_deallocProtMem(picoos_MemoryManager mm, void **addr)
+{
+ if (mm->protMem) {
+ picopal_mpr_free(addr);
+ } else {
+ picoos_deallocate(mm, addr);
+ }
+}
+
+
+void picoos_protectMem(
+ picoos_MemoryManager mm,
+ void *addr,
+ picoos_objsize_t len,
+ picoos_bool enable)
+{
+ if (mm->protMem) {
+ int prot = PICOPAL_PROT_READ;
+ if (!enable) {
+ prot |= PICOPAL_PROT_WRITE;
+ }
+ picopal_mpr_protect(addr, len, prot);
+ } else {
+ /* memory protection disabled; nothing to do */
+ }
+}
+
+#define PICOPAL_PROT_NONE 0 /* the memory cannot be accessed at all */
+#define PICOPAL_PROT_READ 1 /* the memory can be read */
+#define PICOPAL_PROT_WRITE 2 /* the memory can be written to */
+
+void picoos_getMemUsage(
+ picoos_MemoryManager this,
+ picoos_bool resetIncremental,
+ picoos_int32 *usedBytes,
+ picoos_int32 *incrUsedBytes,
+ picoos_int32 *maxUsedBytes)
+{
+ *usedBytes = (picoos_int32) this->usedSize;
+ *incrUsedBytes = (picoos_int32) (this->usedSize - this->prevUsedSize);
+ *maxUsedBytes = (picoos_int32) this->maxUsedSize;
+ if (resetIncremental) {
+ this->prevUsedSize = this->usedSize;
+ }
+}
+
+
+void picoos_showMemUsage(picoos_MemoryManager this, picoos_bool incremental,
+ picoos_bool resetIncremental)
+{
+ picoos_int32 usedBytes, incrUsedBytes, maxUsedBytes;
+
+ picoos_getMemUsage(this, resetIncremental, &usedBytes, &incrUsedBytes,
+ &maxUsedBytes);
+ if (incremental) {
+ PICODBG_DEBUG(("additional memory used: %d", incrUsedBytes));
+ } else {
+ PICODBG_DEBUG(("memory used: %d, maximally used: %d", usedBytes, maxUsedBytes));
+ }
+}
+
+
+void * picoos_allocate(picoos_MemoryManager this,
+ picoos_objsize_t byteSize)
+{
+
+ picoos_objsize_t cellSize;
+ MemCellHdr c, c2, c2r;
+ void * adr;
+
+ if (byteSize < this->minContSize) {
+ byteSize = this->minContSize;
+ }
+ byteSize = ((byteSize + PICOOS_ALIGN_SIZE - 1) / PICOOS_ALIGN_SIZE)
+ * PICOOS_ALIGN_SIZE;
+
+ cellSize = byteSize + this->usedCellHdrSize;
+ /*PICODBG_TRACE(("allocating %d", cellSize));*/
+ c = this->freeCells->nextFree;
+ while (
+ (c != NULL) &&
+ (c->size != (picoos_ptrdiff_t) cellSize) &&
+ (c->size < (picoos_ptrdiff_t)(cellSize+ this->minCellSize))) {
+ c = c->nextFree;
+ }
+ if (c == NULL) {
+ return NULL;
+ }
+ if ((c->size == (picoos_ptrdiff_t) cellSize)) {
+ c->prevFree->nextFree = c->nextFree;
+ c->nextFree->prevFree = c->prevFree;
+ } else {
+ c2 = (MemCellHdr)((picoos_objsize_t)c + cellSize);
+ c2->size = c->size - cellSize;
+ c->size = cellSize;
+ c2->leftCell = c;
+ c2r = (MemCellHdr)((picoos_objsize_t)c2 + c2->size);
+ c2r->leftCell = c2;
+ c2->nextFree = c->nextFree;
+ c2->nextFree->prevFree = c2;
+ c2->prevFree = c->prevFree;
+ c2->prevFree->nextFree = c2;
+ }
+
+ /* statistics */
+ this->usedSize += cellSize;
+ if (this->usedSize > this->maxUsedSize) {
+ this->maxUsedSize = this->usedSize;
+ }
+
+ c->size = -(c->size);
+ adr = (void *)((picoos_objsize_t)c + this->usedCellHdrSize);
+ return adr;
+}
+
+void picoos_deallocate(picoos_MemoryManager this, void * * adr)
+{
+ MemCellHdr c;
+ MemCellHdr cr;
+ MemCellHdr cl;
+ MemCellHdr crr;
+
+
+ if ((*adr) != NULL) {
+ c = (MemCellHdr)((picoos_objsize_t)(*adr) - this->usedCellHdrSize);
+ c->size = -(c->size);
+
+ /*PICODBG_TRACE(("deallocating %d", c->size));*/
+ /* statistics */
+ this->usedSize -= c->size;
+
+ cr = (MemCellHdr)((picoos_objsize_t)c + c->size);
+ cl = c->leftCell;
+ if (cl->size > 0) {
+ if (cr->size > 0) {
+ crr = (MemCellHdr)((picoos_objsize_t)cr + cr->size);
+ crr->leftCell = cl;
+ cl->size = ((cl->size + c->size) + cr->size);
+ cr->nextFree->prevFree = cr->prevFree;
+ cr->prevFree->nextFree = cr->nextFree;
+ } else {
+ cl->size = (cl->size + c->size);
+ cr->leftCell = cl;
+ }
+ } else {
+ if ((cr->size > 0)) {
+ crr = (MemCellHdr)((picoos_objsize_t)cr + cr->size);
+ crr->leftCell = c;
+ c->size = (c->size + cr->size);
+ c->nextFree = cr->nextFree;
+ c->prevFree = cr->prevFree;
+ c->nextFree->prevFree = c;
+ c->prevFree->nextFree = c;
+ } else {
+ c->nextFree = this->freeCells->nextFree;
+ c->prevFree = this->freeCells;
+ c->nextFree->prevFree = c;
+ c->prevFree->nextFree = c;
+ }
+ }
+ }
+ *adr = NULL;
+}
+
+/* *****************************************************************/
+/* Exception Management */
+/* *****************************************************************/
+/** object : exceptionManager
+ * shortcut : em
+ *
+ */
+
+typedef picoos_char picoos_exc_msg[PICOOS_MAX_EXC_MSG_LEN];
+typedef picoos_char picoos_warn_msg[PICOOS_MAX_WARN_MSG_LEN];
+
+typedef struct picoos_exception_manager
+{
+ picoos_int32 curExceptionCode;
+ picoos_exc_msg curExceptionMessage;
+
+ picoos_uint8 curNumWarnings;
+ picoos_int32 curWarningCode[PICOOS_MAX_NUM_WARNINGS];
+ picoos_warn_msg curWarningMessage[PICOOS_MAX_NUM_WARNINGS];
+
+} picoos_exception_manager_t;
+
+void picoos_emReset(picoos_ExceptionManager this)
+{
+ this->curExceptionCode = PICO_OK;
+ this->curExceptionMessage[0] = '\0';
+ this->curNumWarnings = 0;
+}
+
+picoos_ExceptionManager picoos_newExceptionManager(picoos_MemoryManager mm)
+{
+ picoos_ExceptionManager this = (picoos_ExceptionManager) picoos_allocate(
+ mm, sizeof(*this));
+ if (NULL != this) {
+ /* initialize */
+ picoos_emReset(this);
+ }
+ return this;
+}
+
+void picoos_disposeExceptionManager(picoos_MemoryManager mm,
+ picoos_ExceptionManager * this)
+{
+ if (NULL != (*this)) {
+ /* terminate */
+ picoos_deallocate(mm, (void *)this);
+ }
+}
+
+static void picoos_vSetErrorMsg(picoos_char * dst, picoos_objsize_t siz,
+ picoos_int16 code, picoos_char * base, const picoos_char *fmt, va_list args)
+{
+ picoos_uint16 bsize;
+
+ if (NULL == base) {
+ switch (code) {
+ case PICO_EXC_NUMBER_FORMAT:
+ base = PICOOS_MSG_EXC_NUMBER_FORMAT;
+ break;
+ case PICO_EXC_MAX_NUM_EXCEED:
+ base = PICOOS_MSG_EXC_MAX_NUM_EXCEED;
+ break;
+ case PICO_EXC_NAME_CONFLICT:
+ base = PICOOS_MSG_EXC_NAME_CONFLICT;
+ break;
+ case PICO_EXC_NAME_UNDEFINED:
+ base = PICOOS_MSG_EXC_NAME_UNDEFINED;
+ break;
+ case PICO_EXC_NAME_ILLEGAL:
+ base = PICOOS_MSG_EXC_NAME_ILLEGAL;
+ break;
+
+ /* buffer interaction */
+ case PICO_EXC_BUF_OVERFLOW:
+ base = PICOOS_MSG_EXC_BUF_OVERFLOW;
+ break;
+ case PICO_EXC_BUF_UNDERFLOW:
+ base = PICOOS_MSG_EXC_BUF_UNDERFLOW;
+ break;
+ case PICO_EXC_BUF_IGNORE:
+ base = PICOOS_MSG_EXC_BUF_IGNORE;
+ break;
+
+ /* memory allocation */
+ case PICO_EXC_OUT_OF_MEM:
+ base = PICOOS_MSG_EXC_OUT_OF_MEM;
+ break;
+
+ /* files */
+ case PICO_EXC_CANT_OPEN_FILE:
+ base = PICOOS_MSG_EXC_CANT_OPEN_FILE;
+ break;
+ case PICO_EXC_UNEXPECTED_FILE_TYPE:
+ base = PICOOS_MSG_EXC_UNEXPECTED_FILE_TYPE;
+ break;
+ case PICO_EXC_FILE_CORRUPT:
+ base = PICOOS_MSG_EXC_FILE_CORRUPT;
+ break;
+
+ case PICO_EXC_FILE_NOT_FOUND:
+ base = PICOOS_MSG_EXC_FILE_NOT_FOUND;
+ break;
+
+ /* resources */
+ case PICO_EXC_RESOURCE_BUSY:
+ base = PICOOS_MSG_EXC_RESOURCE_BUSY;
+ break;
+ case PICO_EXC_RESOURCE_MISSING:
+ base = PICOOS_MSG_EXC_RESOURCE_MISSING;
+ break;
+
+ /* knowledge bases */
+ case PICO_EXC_KB_MISSING:
+ fmt = PICOOS_MSG_EXC_KB_MISSING;
+ break;
+
+ /* runtime exceptions (programming problems, usually a bug. E.g. trying to access null pointer) */
+ case PICO_ERR_NULLPTR_ACCESS:
+ base = PICOOS_MSG_ERR_NULLPTR_ACCESS;
+ break;
+ case PICO_ERR_INVALID_HANDLE:
+ base = PICOOS_MSG_ERR_INVALID_HANDLE;
+ break;
+ case PICO_ERR_INVALID_ARGUMENT:
+ base = PICOOS_MSG_ERR_INVALID_ARGUMENT;
+ break;
+ case PICO_ERR_INDEX_OUT_OF_RANGE:
+ base = PICOOS_MSG_ERR_INDEX_OUT_OF_RANGE;
+ break;
+
+ /* errors ("external" errors, e.g. hardware failure. Usually cannot be dealt with from inside pico.) */
+ case PICO_ERR_OTHER:
+ base = PICOOS_MSG_ERR_OTHER;
+ break;
+
+ /* other error inside pu */
+ case PICO_STEP_ERROR:
+ base = PICOOS_MSG_ERR_PU;
+ break;
+
+ /* WARNINGS */
+
+ /* general */
+ case PICO_WARN_INCOMPLETE:
+ base = PICOOS_MSG_WARN_INCOMPLETE;
+ break;
+ case PICO_WARN_FALLBACK:
+ base = PICOOS_MSG_WARN_FALLBACK;
+ break;
+
+ case PICO_WARN_OTHER:
+ base = PICOOS_MSG_WARN_OTHER;
+ break;
+
+ /* resources */
+ case PICO_WARN_KB_OVERWRITE:
+ base = PICOOS_MSG_WARN_KB_OVERWRITE;
+ break;
+ case PICO_WARN_RESOURCE_DOUBLE_LOAD:
+ base = PICOOS_MSG_WARN_RESOURCE_DOUBLE_LOAD;
+ break;
+
+ /* decision trees */
+ case PICO_WARN_INVECTOR:
+ base = PICOOS_MSG_WARN_INVECTOR;
+ break;
+ case PICO_WARN_CLASSIFICATION:
+ base = PICOOS_MSG_WARN_CLASSIFICATION;
+ break;
+ case PICO_WARN_OUTVECTOR:
+ base = PICOOS_MSG_WARN_OUTVECTOR;
+ break;
+
+ /* processing units */
+ case PICO_WARN_PU_IRREG_ITEM:
+ base = PICOOS_MSG_WARN_PU_IRREG_ITEM;
+ break;
+ case PICO_WARN_PU_DISCARD_BUF:
+ base = PICOOS_MSG_WARN_PU_DISCARD_BUF;
+ break;
+
+ default:
+ base = (picoos_char *) "unknown error";
+ break;
+ }
+ }
+ bsize = picoos_strlcpy(dst,base,siz);
+ if ((NULL != fmt) && (bsize < siz)) { /* there is something to add and more space to add it */
+ if (bsize > 0) {
+ dst += bsize;
+ siz -= bsize;
+ bsize = picoos_strlcpy(dst,(picoos_char *)": ",siz);
+ }
+ if (bsize < siz) {
+ picopal_vslprintf((picopal_char *) dst + bsize, siz - bsize, (picopal_char *)fmt, args);
+ }
+ }
+}
+
+void picoos_setErrorMsg(picoos_char * dst, picoos_objsize_t siz,
+ picoos_int16 code, picoos_char * base, const picoos_char *fmt, ...)
+{
+ va_list args;
+ va_start(args, (char *)fmt);
+ picoos_vSetErrorMsg(dst,siz, code, base, fmt,args);
+ va_end(args);
+}
+
+/* For convenience, this function returns the resulting current exception code. The return value therefore is NOT the status of raising
+ * the error! */
+pico_status_t picoos_emRaiseException(picoos_ExceptionManager this,
+ pico_status_t exceptionCode, picoos_char * baseMessage, picoos_char * fmt, ...)
+{
+ va_list args;
+
+
+ if (PICO_OK == this->curExceptionCode && PICO_OK != exceptionCode) {
+ this->curExceptionCode = exceptionCode;
+ va_start(args, (char *)fmt);
+ picoos_vSetErrorMsg(this->curExceptionMessage,PICOOS_MAX_EXC_MSG_LEN, exceptionCode, baseMessage, fmt,args);
+ PICODBG_DEBUG((
+ "exit with exception code=%i, exception message='%s'",
+ this->curExceptionCode, this->curExceptionMessage));
+
+ va_end(args);
+
+ }
+ return this->curExceptionCode;
+}
+
+pico_status_t picoos_emGetExceptionCode(picoos_ExceptionManager this)
+{
+ return this->curExceptionCode;
+}
+
+void picoos_emGetExceptionMessage(picoos_ExceptionManager this, picoos_char * msg, picoos_uint16 maxsize)
+{
+ picoos_strlcpy(msg,this->curExceptionMessage,maxsize);
+}
+
+void picoos_emRaiseWarning(picoos_ExceptionManager this,
+ pico_status_t warningCode, picoos_char * baseMessage, picoos_char * fmt, ...)
+{
+ va_list args;
+ if ((this->curNumWarnings < PICOOS_MAX_NUM_WARNINGS) && (PICO_OK != warningCode)) {
+ if (PICOOS_MAX_NUM_WARNINGS-1 == this->curNumWarnings) {
+ this->curWarningCode[this->curNumWarnings] = PICO_EXC_MAX_NUM_EXCEED;
+ picoos_strlcpy(this->curWarningMessage[this->curNumWarnings],(picoos_char *) "too many warnings",PICOOS_MAX_WARN_MSG_LEN);
+ } else {
+ this->curWarningCode[this->curNumWarnings] = warningCode;
+ va_start(args, (char *)fmt);
+ picoos_vSetErrorMsg(this->curWarningMessage[this->curNumWarnings],PICOOS_MAX_WARN_MSG_LEN, warningCode, baseMessage, fmt,args);
+ va_end(args);
+ }
+ this->curNumWarnings++;
+ }
+ PICODBG_DEBUG((
+ "exit with code=%i and message='%s', resulting in #warnings=%i",
+ this->curWarningCode[this->curNumWarnings-1],
+ this->curWarningMessage[this->curNumWarnings-1],
+ this->curNumWarnings));
+}
+
+picoos_uint8 picoos_emGetNumOfWarnings(picoos_ExceptionManager this)
+{
+ return this->curNumWarnings;
+}
+
+pico_status_t picoos_emGetWarningCode(picoos_ExceptionManager this, picoos_uint8 index)
+{
+ if (index < this->curNumWarnings) {
+ return this->curWarningCode[index];
+ } else {
+ return PICO_OK;
+ }
+}
+
+void picoos_emGetWarningMessage(picoos_ExceptionManager this, picoos_uint8 index, picoos_char * msg, picoos_uint16 maxsize)
+{
+ if (index < this->curNumWarnings) {
+ picoos_strlcpy(msg,this->curWarningMessage[index],maxsize);
+ } else {
+ msg[0] = NULLC;
+ }
+}
+
+
+
+
+/* *****************************************************************/
+/* File Access */
+/* *****************************************************************/
+
+#define picoos_MagicNumber 192837465
+#define picoos_MaxBufSize 1000000
+#define picoos_MaxNrOfBuffers 1000000
+#define picoos_MaxBufLen 8192
+#define picoos_HashFuncId0 0
+#define picoos_HashTableSize0 101
+#define picoos_HashTableSize1 1731
+#define picoos_MaxHashTableSize HashTableSize1
+
+#define cardinal_ptr_t picoos_uint32 *
+
+typedef struct picoos_buffer
+{
+ picoos_char * data;
+ int start; /* denotes the file position of the buffer beginning */
+ int len; /* denotes the length of the buffer; -1 means invalid buffer */
+ int pos; /* denotes the position in the buffer */
+} picoos_buffer_t;
+
+typedef picoos_buffer_t picoos_buffer_array_t[picoos_MaxNrOfBuffers];
+typedef picoos_int32 picoos_buffer_index_array_t[picoos_MaxNrOfBuffers];
+
+/** object : File
+ * shortcut : f
+ *
+ */
+typedef struct picoos_file
+{
+ picoos_FileName name;
+ picoos_uint8 binary;
+ picoos_uint8 write;
+
+ picopal_File nf;
+
+ picoos_uint32 lFileLen;
+ picoos_uint32 lPos;
+
+ picoos_File next;
+ picoos_File prev;
+
+} picoos_file_t;
+
+picoos_File picoos_newFile(picoos_MemoryManager mm)
+{
+ picoos_File this = (picoos_File) picoos_allocate(mm, sizeof(*this));
+ if (NULL != this) {
+ /* initialize */
+ }
+ return this;
+}
+
+void picoos_disposeFile(picoos_MemoryManager mm, picoos_File * this)
+{
+ if (NULL != (*this)) {
+ /* terminate */
+ picoos_deallocate(mm, (void *)this);
+ }
+}
+
+
+/* ************************************************************
+ * low-level file operations
+ **************************************************************/
+
+static picoos_int32 os_min(const picoos_int32 x, const picoos_int32 y)
+{
+ return (x < y) ? x : y;
+}
+
+/*
+ static picoos_uint8 LReadChar (picoos_File f, picoos_char * ch);
+
+
+ static picoos_uint8 LSetPos (picoos_File f, unsigned int pos);
+
+
+ static picoos_uint8 LGetPos (picoos_File f, picoos_uint32 * pos);
+
+
+ static picoos_uint8 LEof (picoos_File f);
+ */
+
+static picoos_bool LOpen(picoos_Common g, picoos_File * f,
+ picoos_char fileName[], picopal_access_mode mode)
+{
+ picoos_bool done = TRUE;
+
+ *f = picoos_newFile(g->mm);
+ picopal_strcpy((*f)->name, fileName);
+ (*f)->write = ((mode == PICOPAL_TEXT_WRITE) || (mode
+ == PICOPAL_BINARY_WRITE));
+ (*f)->binary = (mode
+ == PICOPAL_BINARY_WRITE);
+ (*f)->next = NULL;
+ (*f)->prev = NULL;
+ (*f)->nf = picopal_get_fnil();
+ (*f)->lFileLen = 0;
+ (*f)->lPos = 0;
+ if (picopal_strlen((*f)->name)) {
+ (*f)->nf = picopal_fopen((*f)->name, mode);
+ done = !(picopal_is_fnil((*f)->nf));
+ if (done) {
+ (*f)->lFileLen = picopal_flength((*f)->nf);
+ }
+ }
+ if (done) {
+ (*f)->next = g->fileList;
+ if (g->fileList != NULL) {
+ g->fileList->prev = (*f);
+ }
+ g->fileList = (*f);
+ } else {
+ picoos_disposeFile(g->mm, f);
+ (*f) = NULL;
+ }
+ return done;
+}
+
+static picoos_bool LClose(picoos_Common g, picoos_File * f)
+{
+
+ picoos_bool done;
+
+ if (((*f) != NULL)) {
+ done = (PICO_OK == picopal_fclose((*f)->nf));
+ if (((*f)->next != NULL)) {
+ (*f)->next->prev = (*f)->prev;
+ }
+ if (((*f)->prev != NULL)) {
+ (*f)->prev->next = (*f)->next;
+ } else {
+ g->fileList = (*f)->next;
+ }
+ picoos_disposeFile(g->mm, f);
+
+ done = TRUE;
+ } else {
+ done = FALSE;
+ }
+ return done;
+
+}
+
+/* caller must ensure that bytes[] has at least len allocated bytes */
+static picoos_bool LReadBytes(picoos_File f, picoos_uint8 bytes[],
+ picoos_uint32 * len)
+{
+ picoos_bool done;
+ picoos_int32 res;
+
+ PICODBG_TRACE(("trying to read %i bytes",*len));
+ if ((f != NULL)) {
+ res = picopal_fread_bytes(f->nf, (void *) &bytes[(0)], 1, (*len));
+ PICODBG_TRACE(("res = %i",res));
+ if (res < 0) { /* non-ansi */
+ (*len) = 0;
+ done = FALSE;
+ } else if (((picoos_uint32)res != (*len))) {
+ (*len) = res;
+ done = FALSE;
+ } else {
+ done = TRUE;
+ }
+ f->lPos = (f->lPos + (*len));
+ } else {
+ (*len) = 0;
+ done = FALSE;
+ }
+ return done;
+}
+
+ static picoos_bool LWriteBytes(picoos_File f, const picoos_char bytes[], int * len) {
+ picoos_bool done;
+ int res;
+ /*int n;
+ void * bptr; */
+
+ if (f != NULL) {
+ res = picopal_fwrite_bytes(f->nf, (void *) bytes, 1, *len);
+ if ((res < 0)) {
+ (*len) = 0;
+ done = FALSE;
+ } else if ((res != (*len))) {
+ (*len) = res;
+ done = FALSE;
+ } else {
+ done = TRUE;
+ }
+ f->lPos = (f->lPos + (unsigned int) (*len));
+ if ((f->lPos > f->lFileLen)) {
+ f->lFileLen = f->lPos;
+ }
+ } else {
+ (*len) = 0;
+ done = FALSE;
+ }
+ return done;
+}
+
+
+static picoos_bool LSetPos(picoos_File f, unsigned int pos)
+{
+
+ picoos_bool done;
+
+ if ((f != NULL)) {
+ if ((pos == f->lPos)) {
+ done = TRUE;
+ } else {
+ done = (PICO_OK == picopal_fseek(f->nf, pos, PICOPAL_SEEK_SET));
+ if (done) {
+ f->lPos = pos;
+ }
+ }
+ } else {
+ done = FALSE;
+ }
+ return done;
+}
+
+static picoos_bool LGetPos(picoos_File f, picoos_uint32 * pos)
+{
+ picoos_bool done = TRUE;
+ if ((f != NULL)) {
+ (*pos) = f->lPos;
+ } else {
+ done = FALSE;
+ (*pos) = 0;
+ }
+ return done;
+
+}
+
+static picoos_bool LEof(picoos_File f)
+{
+ picoos_bool isEof;
+
+ if ((f != NULL)) {
+ isEof = picopal_feof(f->nf);
+ } else {
+ isEof = TRUE;
+ }
+ return isEof;
+
+}
+
+/* **************************************************************************************/
+
+
+
+/* read a given string 'str' from file. If no match was found, the read position is advanced until and including the first
+ * non-matching character */
+static picoos_bool picoos_StrRead (picoos_File f, picoos_char str[])
+{
+ picoos_uint32 i = 0;
+ picoos_bool done = TRUE;
+ picoos_char b;
+
+ while (done && (str[i] != NULLC)) {
+ done = done && picoos_ReadByte(f,(picoos_char *)&b);
+ done = done && (b == str[i]);
+ i++;
+ }
+ return done;
+}
+
+/* write 'str' to file */
+static picoos_bool picoos_WriteStr (picoos_File f, picoos_char str[])
+{
+ picoos_uint32 i = 0;
+ picoos_bool done = TRUE;
+
+ while (done && (str[i] != NULLC)) {
+ done = done && picoos_WriteByte(f,str[i]);
+ i++;
+ }
+ return done;
+}
+
+
+
+/* **** Sequential binary file access ******/
+
+/* Remark: 'ReadByte', 'ReadBytes' and 'ReadVar' may be mixed;
+ 'WriteByte', 'WriteBytes' and 'WriteVar' may be mixed. */
+
+/* Open existing binary file for read access. Reading is buffered
+ * with 'nrOfBufs' buffers of size 'bufSize'. If 'nrOfBufs' or
+ * 'bufSize' is 0 reading is not buffered.
+ * If 'key' is not empty, the file is decrypted with 'key'.
+ * If the opened file is in an encrypted archive file, it
+ */
+picoos_uint8 picoos_OpenBinary(picoos_Common g, picoos_File * f,
+ picoos_char fileName[])
+{
+ return LOpen(g, f, fileName, PICOPAL_BINARY_READ);
+}
+
+
+/* Read next byte from file 'f'. */
+picoos_bool picoos_ReadByte(picoos_File f, picoos_uint8 * by)
+{
+ picoos_uint32 n = 1;
+
+ return picoos_ReadBytes(f, by, &n) && (n == 1);
+
+}
+
+/* Read next 'len' bytes from 'f' into 'bytes'; 'len' returns the
+ number of bytes actually read (may be smaller than requested
+ length if at end of file). bytes[] must be big enough to hold at least len bytes.
+*/
+picoos_bool picoos_ReadBytes(picoos_File f, picoos_uint8 bytes[],
+ picoos_uint32 * len)
+{
+ picoos_bool done = TRUE;
+ /* unsigned int origPos; */
+
+ if ((f != NULL)) {
+ done = LReadBytes(f, bytes, len);
+ /*if ((f->keyLen > 0)) {
+ DecryptBytes(f->key,picoos_MaxKeyLen,f->keyLen,origPos,bytes,(*len));
+ }*/
+ }
+
+ return done;
+}
+
+
+/* Create new binary file.
+ If 'key' is not empty, the file is encrypted with 'key'. */
+picoos_bool picoos_CreateBinary(picoos_Common g, picoos_File * f,
+ picoos_char fileName[])
+{
+ return LOpen(g, f, fileName, PICOPAL_BINARY_WRITE);
+
+}
+
+
+picoos_uint8 picoos_WriteByte(picoos_File f, picoos_char by)
+{
+ int n = 1;
+
+ return picoos_WriteBytes(f, (picoos_char *) &by, &n);
+}
+
+
+/* Writes 'len' bytes from 'bytes' onto file 'f'; 'len' returns
+ the number of bytes actually written. */
+picoos_bool picoos_WriteBytes(picoos_File f, const picoos_char bytes[], picoos_int32 * len) {
+ picoos_bool done = FALSE;
+
+ if (f != NULL) {
+ done = LWriteBytes(f, bytes, len);
+ }
+
+ return done;
+}
+
+
+
+/* Close previously opened binary file. */
+picoos_uint8 picoos_CloseBinary(picoos_Common g, picoos_File * f)
+{
+ return LClose(g, f);
+
+}
+
+/* **************************************************************************************/
+/* *** general routines *****/
+
+
+/* Returns whether end of file was encountered in previous
+ read operation. */
+picoos_bool picoos_Eof(picoos_File f)
+{
+ if ((NULL != f)) {
+ return LEof(f);
+ } else {
+ return TRUE;
+ }
+
+}
+
+/* sets the file pointer to
+ 'pos' bytes from beginning (first byte = byte 0). This
+ routine should only be used for binary files. */
+picoos_bool picoos_SetPos(picoos_File f, picoos_int32 pos)
+{
+ picoos_bool done = TRUE;
+ if ((NULL != f)) {
+ done = LSetPos(f, pos);
+ } else {
+ done = FALSE;
+ }
+ return done;
+
+}
+
+/* Get position from file 'f'. */
+picoos_bool picoos_GetPos(picoos_File f, picoos_uint32 * pos)
+{
+ if (NULL != f) {
+ /* if (f->bFile) {
+ (*pos) = BGetPos(f);
+ } else { */
+ (*pos) = LGetPos(f, pos);
+ /* } */
+ return TRUE;
+ } else {
+ (*pos) = 0;
+ return FALSE;
+ }
+}
+
+/* Returns the length of the file in bytes. */
+picoos_bool picoos_FileLength(picoos_File f, picoos_uint32 * len)
+{
+
+ if (NULL != f) {
+ *len = f->lFileLen;
+ return TRUE;
+ } else {
+ *len = 0;
+ return FALSE;
+ }
+}
+
+/* Return full name of file 'f'. maxsize is the size of 'name[]' in bytes */
+picoos_bool picoos_Name(picoos_File f, picoos_char name[], picoos_uint32 maxsize)
+{
+ picoos_bool done = TRUE;
+
+ if (NULL != f) {
+ done = (picoos_strlcpy(name, f->name,maxsize) < maxsize);
+ } else {
+ name[0] = (picoos_char)NULLC;
+ done = FALSE;
+ }
+
+ return done;
+}
+
+/* Returns whether file 'name' exists or not. */
+picoos_bool picoos_FileExists(picoos_Common g, picoos_char name[])
+{
+ picoos_File f;
+
+ if (picoos_OpenBinary(g, & f,name)) {
+ picoos_CloseBinary(g, & f);
+ return TRUE;
+ } else {
+ return FALSE;
+ }
+}
+
+
+/* ******************************************************************/
+/* Array conversion operations: all procedures convert 'nrElems' values from
+ 'src' starting with index 'srcStartInd' into corresponding (possibly
+ rounded) values in 'dst' starting at 'dstStartInd'. */
+
+/* taking pi to be le, these are just the array versions of read_mem_pi_*int16 */
+typedef picoos_uint8 two_byte_t[2];
+
+static void arr_conv_le_int16 (picoos_uint8 src[], picoos_uint32 srcShortStartInd, picoos_uint32 nrElems, picoos_int16 dst[], picoos_uint32 dstStartInd)
+{
+ two_byte_t * src_p = (two_byte_t *) (src + (srcShortStartInd * 2));
+ picoos_int16 * dst_p = dst + dstStartInd;
+ picoos_uint32 i;
+
+ for (i=0; i<nrElems; i++) {
+ *(dst_p++) = (*src_p)[0] + (((*src_p)[1] & 0x7F) << 8) - (((*src_p)[1] & 0x80) ? 0x8000 : 0);
+ src_p++;
+ }
+}
+
+
+
+/* convert array of int16 into little-endian format */
+static void arr_conv_int16_le (picoos_int16 src[], picoos_uint32 srcStartInd, picoos_uint32 nrElems, picoos_uint8 dst[], picoos_uint32 dstShortStartInd)
+{
+ two_byte_t * dst_p = (two_byte_t *) (dst + (dstShortStartInd * 2));
+ picoos_int16 * src_p = src + srcStartInd;
+ picoos_uint32 i;
+ picoos_uint16 val;
+
+ for (i=0; i<nrElems; i++) {
+ val = (picoos_uint16) *(src_p++);
+ (*dst_p)[0] = (picoos_uint8)(val & 0x00FF);
+ (*dst_p)[1] = (picoos_uint8)((val & 0xFF00)>>8);
+ dst_p++;
+ }
+}
+
+/* *****************************************************************/
+/* Sampled Data Files */
+/* *****************************************************************/
+
+#define PICOOS_SDF_BUF_LEN 1024
+
+#define PICOOS_INT16_MIN -32768
+#define PICOOS_INT16_MAX 32767
+#define PICOOS_UINT16_MAX 0xffff
+#define PICOOS_INT32_MIN -2147483648
+#define PICOOS_INT32_MAX 2147483647
+#define PICOOS_UINT32_MAX 0xffffffff
+
+/** object : SDFile
+ * shortcut : sdf
+ *
+ */
+/* typedef struct picoos_sd_file * picoos_SDFile */
+typedef struct picoos_sd_file
+{
+ picoos_uint32 sf;
+ wave_file_type_t fileType; /* (acoustic) wav, au, raw, other */
+ picoos_uint32 hdrSize;
+ picoos_encoding_t enc;
+ picoos_File file;
+ picoos_uint32 nrFileSamples;
+ picoos_int16 buf[PICOOS_SDF_BUF_LEN];
+ picoos_int32 bufPos;
+ picoos_uint8 bBuf[2*PICOOS_SDF_BUF_LEN];
+ picoos_bool aborted;
+} picoos_sd_file_t;
+
+
+/* Tries to read wav header at beginning of file 'f';
+ returns sampling rate 'sf', encoding type 'enc',
+ nr of samples in file 'nrSamples', header size 'hdrSize',
+ and byte order 'bOrder'; returns whether a supported
+ wav header and format was found. */
+static picoos_bool picoos_readWavHeader(picoos_File f, picoos_uint32 * sf,
+ picoos_encoding_t * enc, picoos_uint32 * nrSamples,
+ picoos_uint32 * hdrSize) {
+ picoos_uint16 n16;
+ picoos_uint32 n32;
+ picoos_uint16 formatTag;
+ picoos_uint32 sampleRate;
+ picoos_uint32 bytesPerSec;
+ picoos_uint16 blockAlign;
+ picoos_uint16 sampleSize;
+ picoos_uint32 dataLength;
+ picoos_uint32 fileLen;
+ picoos_uint32 nrFileSamples;
+ picoos_bool done;
+
+
+ picoos_SetPos(f, 0);
+ picoos_FileLength(f, &fileLen);
+ done = picoos_StrRead(f, (picoos_char *) "RIFF");
+ done = done && (PICO_OK == picoos_read_le_uint32(f, &n32)); /* length of riff chunk, unused */
+ done = done && picoos_StrRead(f, (picoos_char *) "WAVE");
+ done = done && picoos_StrRead(f, (picoos_char *) "fmt ");
+ done = done && (PICO_OK == picoos_read_le_uint32(f, &n32)); /* length of fmt chunk in bytes; must be 16 */
+ done = done && (n32 == 16);
+ done = done && (PICO_OK == picoos_read_le_uint16(f, &formatTag));
+ done = done && (PICO_OK == picoos_read_le_uint16(f, &n16)); /* number of channels; must be mono */
+ done = done && (n16 == 1);
+ done = done && (PICO_OK == picoos_read_le_uint32(f, &sampleRate));
+ done = done && (PICO_OK == picoos_read_le_uint32(f, &bytesPerSec));
+ done = done && (PICO_OK == picoos_read_le_uint16(f, &blockAlign));
+ done = done && (PICO_OK == picoos_read_le_uint16(f, &sampleSize));
+ done = done && picoos_StrRead(f, (picoos_char *) "data");
+ done = done && (PICO_OK == picoos_read_le_uint32(f, &dataLength)); /* length of data chunk in bytes */
+ (*hdrSize) = 44;
+ if (done) {
+ (*sf) = sampleRate;
+ (*nrSamples) = 0;
+ switch (formatTag) {
+ case FORMAT_TAG_LIN:
+ (*enc) = PICOOS_ENC_LIN;
+ done = ((blockAlign == 2) && (sampleSize == 16));
+ (*nrSamples) = (dataLength / 2);
+ nrFileSamples = ((fileLen - (*hdrSize)) / 2);
+ break;
+ case FORMAT_TAG_ULAW:
+ (*enc) = PICOOS_ENC_ULAW;
+ done = ((blockAlign == 1) && (sampleSize == 8));
+ (*nrSamples) = dataLength;
+ nrFileSamples = (fileLen - (*hdrSize));
+ break;
+ case FORMAT_TAG_ALAW:
+ (*enc) = PICOOS_ENC_ALAW;
+ done = ((blockAlign == 1) && (sampleSize == 8));
+ (*nrSamples) = dataLength;
+ nrFileSamples = (fileLen - (*hdrSize));
+ break;
+ default:
+ done = FALSE;
+ break;
+ }
+ if (!done) {
+ /* communicate "unsupported format" */
+ PICODBG_WARN(("unsupported wav format"));
+ } else {
+ if (nrFileSamples != (*nrSamples)) {
+ /* warn "inconsistent number of samples" */
+ PICODBG_WARN(("inconsistent number of samples in wav file: %d vs. %d",nrFileSamples,(*nrSamples)));
+ (*nrSamples) = nrFileSamples;
+ }
+ }
+ }
+ return done;
+}
+
+
+
+extern picoos_bool picoos_sdfOpenIn(picoos_Common g, picoos_SDFile * sdFile,
+ picoos_char fileName[], picoos_uint32 * sf, picoos_encoding_t * enc,
+ picoos_uint32 * numSamples)
+{
+ picoos_bool done = FALSE;
+ picoos_sd_file_t * sdf = NULL;
+ wave_file_type_t fileType = FILE_TYPE_OTHER;
+
+ (*sf) = 0;
+ (*numSamples) = 0;
+ (*enc) = PICOOS_ENC_LIN;
+ (*sdFile) = NULL;
+
+ sdf = picoos_allocate(g->mm,sizeof(picoos_sd_file_t));
+ if (NULL == sdf) {
+ picoos_emRaiseWarning(g->em,PICO_EXC_OUT_OF_MEM,NULL,NULL);
+ return FALSE;
+ }
+
+ /* buffered access not supported, yet */
+ if (picoos_OpenBinary(g,&(sdf->file),fileName)) {
+ if (picoos_has_extension(fileName,(picoos_char *) ".wav")) {
+ fileType = FILE_TYPE_WAV;
+ done = picoos_readWavHeader(sdf->file,&(sdf->sf),&(sdf->enc),&(sdf->nrFileSamples),&(sdf->hdrSize));
+ } else {
+ /* we prefer not to treat other formats, rather than treat it as raw */
+ /* fileType = FILE_TYPE_RAW; */
+ fileType = FILE_TYPE_OTHER;
+ done = FALSE;
+ }
+
+ if (FILE_TYPE_OTHER == fileType) {
+ picoos_emRaiseWarning(g->em,PICO_EXC_UNEXPECTED_FILE_TYPE,(picoos_char *)"unsupported filename suffix",NULL);
+ } else if (!done) {
+ picoos_emRaiseWarning(g->em,PICO_EXC_UNEXPECTED_FILE_TYPE,(picoos_char *)"non-conforming header",NULL);
+ } else {
+ (*numSamples) = sdf->nrFileSamples;
+ (*sf) = sdf->sf;
+ (*enc) = sdf->enc;
+ /* check whether sd file properties are supported */
+ if (PICOOS_ENC_LIN != sdf->enc) {
+ done = FALSE;
+ picoos_emRaiseWarning(g->em,PICO_EXC_UNEXPECTED_FILE_TYPE,NULL,(picoos_char *)"encoding not supported");
+ }
+ if (SAMPLE_FREQ_16KHZ != sdf->sf) {
+ done = FALSE;
+ picoos_emRaiseWarning(g->em,PICO_EXC_UNEXPECTED_FILE_TYPE,NULL,(picoos_char *)"sample frequency not supported");
+ }
+ (*sdFile) = sdf;
+ }
+ if (!done){
+ picoos_CloseBinary(g,&(sdf->file));
+ }
+ } else {
+ picoos_emRaiseException(g->em,PICO_EXC_CANT_OPEN_FILE,NULL,NULL);
+ }
+ if (!done) {
+ picoos_deallocate(g->mm,(void *)&sdf);
+ (*sdFile) = NULL;
+ }
+ return done;
+}
+
+
+static void picoos_sdfLoadSamples(picoos_SDFile sdFile,
+ picoos_uint32 * nrSamples) {
+ picoos_uint32 len;
+ picoos_sd_file_t * sdf = sdFile;
+
+ switch (sdFile->enc) {
+ case PICOOS_ENC_LIN:
+ if ((*nrSamples) > PICOOS_SDF_BUF_LEN) {
+ (*nrSamples) = PICOOS_SDF_BUF_LEN;
+ }
+ len = 2 * (*nrSamples);
+ picoos_ReadBytes(sdf->file, sdf->bBuf, &len);
+ (*nrSamples) = len / 2;
+ arr_conv_le_int16(sdf->bBuf, 0, (*nrSamples), sdf->buf, 0);
+ break;
+ /* @todo : may be useful */
+ case PICOOS_ENC_ULAW:
+ case PICOOS_ENC_ALAW:
+ default:
+ (*nrSamples) = 0;
+ }
+
+}
+
+extern picoos_bool picoos_sdfGetSamples (
+ picoos_SDFile sdFile,
+ picoos_uint32 start,
+ picoos_uint32 * nrSamples,
+ picoos_int16 samples[])
+{
+ picoos_uint32 b;
+ picoos_uint32 rem;
+ picoos_uint32 n;
+ picoos_uint32 i;
+ picoos_uint32 j;
+ picoos_bool done = FALSE;
+
+ if (NULL == sdFile) {
+ (*nrSamples) = 0;
+ } else {
+ if (start >= sdFile->nrFileSamples) {
+ if (start > sdFile->nrFileSamples) {
+ PICODBG_WARN(("start has to be <= sdFile->nrFileSamples"));
+ }
+ (*nrSamples) = 0;
+ } else {
+ if (((start + (*nrSamples)) > sdFile->nrFileSamples)) {
+ (*nrSamples) = (sdFile->nrFileSamples - start);
+ }
+ if ((sdFile->enc == PICOOS_ENC_LIN)) {
+ b = 2;
+ } else {
+ b = 1;
+ }
+ picoos_SetPos(sdFile->file,(sdFile->hdrSize + (b * start)));
+ j = 0;
+ rem = (*nrSamples);
+ n = rem;
+ while ((rem > 0) && (n > 0)) {
+ /* set n=min(rem,buffer_length) and try loading next n samples */
+ n = (rem < PICOOS_SDF_BUF_LEN) ? rem : PICOOS_SDF_BUF_LEN;
+ picoos_sdfLoadSamples(sdFile, &n);
+ /* n may be smaller now */
+ for (i = 0; i < n; i++) {
+ samples[j] = sdFile->buf[i];
+ j++;
+ }
+ rem -= n;
+ start += n;
+ }
+ (*nrSamples) = j;
+ done = ((*nrSamples) > 0);
+ }
+ }
+ return done;
+}
+
+
+extern picoos_bool picoos_sdfCloseIn (picoos_Common g, picoos_SDFile * sdFile)
+{
+ if (NULL != (*sdFile)) {
+ picoos_CloseBinary(g,&((*sdFile)->file));
+ picoos_deallocate(g->mm,(void *)sdFile);
+ }
+ return TRUE;
+}
+
+
+static picoos_bool picoos_writeWavHeader(picoos_File f, picoos_uint32 sf,
+ picoos_encoding_t enc, picoos_uint32 nrSamples,
+ picoos_uint32 * hdrSize) {
+ picoos_uint16 formatTag = FORMAT_TAG_LIN;
+ picoos_uint32 sampleRate;
+ picoos_uint32 bytesPerSec;
+ picoos_uint32 bytesPerSample = 2;
+ picoos_uint16 blockAlign;
+ picoos_uint16 sampleSize = 16;
+ picoos_uint32 dataLength;
+ picoos_bool done = TRUE;
+
+ picoos_SetPos(f, 0);
+
+ switch (enc) {
+ case PICOOS_ENC_LIN:
+ formatTag = FORMAT_TAG_LIN;
+ bytesPerSample = 2;
+ sampleSize = 16;
+ break;
+ case PICOOS_ENC_ULAW:
+ formatTag = FORMAT_TAG_ULAW;
+ bytesPerSample = 1;
+ sampleSize = 8;
+ break;
+ case PICOOS_ENC_ALAW:
+ formatTag = FORMAT_TAG_ALAW;
+ bytesPerSample = 1;
+ sampleSize = 8;
+ break;
+ default:
+ done = FALSE;
+ break;
+ }
+
+ bytesPerSec = (sf * bytesPerSample);
+ blockAlign = bytesPerSample;
+ sampleRate = sf;
+ dataLength = (bytesPerSample * nrSamples);
+ done = done && picoos_WriteStr(f,(picoos_char *)"RIFF");
+ done = done && picoos_write_le_uint32(f,dataLength + 36);
+ done = done && picoos_WriteStr(f,(picoos_char *)"WAVE");
+ done = done && picoos_WriteStr(f,(picoos_char *)"fmt ");
+ done = done && picoos_write_le_uint32(f,16);
+ done = done && picoos_write_le_uint16(f,formatTag);
+ done = done && picoos_write_le_uint16(f,1);
+ done = done && picoos_write_le_uint32(f,sampleRate);
+ done = done && picoos_write_le_uint32(f,bytesPerSec);
+ done = done && picoos_write_le_uint16(f,blockAlign);
+ done = done && picoos_write_le_uint16(f,sampleSize);
+ done = done && picoos_WriteStr(f,(picoos_char *)"data");
+ done = done && picoos_write_le_uint32(f,dataLength);
+ (*hdrSize) = 44;
+ return done;
+}
+
+
+#define DummyLen 100000000
+
+extern picoos_bool picoos_sdfOpenOut(picoos_Common g, picoos_SDFile * sdFile,
+ picoos_char fileName[], int sf, picoos_encoding_t enc)
+{
+ picoos_bool done = TRUE;
+ picoos_sd_file_t * sdf = NULL;
+
+ (*sdFile) = NULL;
+ sdf = picoos_allocate(g->mm, sizeof(picoos_sd_file_t));
+ if (NULL == sdf) {
+ picoos_emRaiseWarning(g->em, PICO_EXC_OUT_OF_MEM, NULL, NULL);
+ return FALSE;
+ }
+ sdf->sf = sf;
+ sdf->enc = enc;
+ /* check whether sd file properties are supported */
+ if (PICOOS_ENC_LIN != sdf->enc) {
+ done = FALSE;
+ picoos_emRaiseWarning(g->em, PICO_EXC_UNEXPECTED_FILE_TYPE, NULL,
+ (picoos_char *) "encoding not supported");
+ }
+ if (SAMPLE_FREQ_16KHZ != sdf->sf) {
+ done = FALSE;
+ picoos_emRaiseWarning(g->em, PICO_EXC_UNEXPECTED_FILE_TYPE, NULL,
+ (picoos_char *) "sample frequency not supported");
+ }
+ if (done) {
+ sdf->nrFileSamples = 0;
+ sdf->bufPos = 0;
+ sdf->aborted = FALSE;
+ if (picoos_CreateBinary(g, &(sdf->file), fileName)) {
+ if (picoos_has_extension(fileName, (picoos_char *) ".wav")) {
+ sdf->fileType = FILE_TYPE_WAV;
+ done = picoos_writeWavHeader(sdf->file, sdf->sf, sdf->enc,
+ DummyLen, &(sdf->hdrSize));
+ } else {
+ /* we prefer not to treat other formats, rather than treat it as raw */
+ /* fileType = FILE_TYPE_RAW; */
+ sdf->fileType = FILE_TYPE_OTHER;
+ done = FALSE;
+ }
+
+ if (FILE_TYPE_OTHER == sdf->fileType) {
+ picoos_emRaiseWarning(g->em, PICO_EXC_UNEXPECTED_FILE_TYPE,
+ (picoos_char *) "unsupported filename suffix", NULL);
+ } else if (!done) {
+ picoos_emRaiseWarning(g->em, PICO_EXC_UNEXPECTED_FILE_TYPE,
+ (picoos_char *) "non-conforming header", NULL);
+ } else {
+ (*sdFile) = sdf;
+ }
+ if (!done) {
+ picoos_CloseBinary(g, &(sdf->file));
+ }
+ } else {
+ picoos_emRaiseException(g->em, PICO_EXC_CANT_OPEN_FILE, NULL, NULL);
+ }
+ }
+ if (!done) {
+ picoos_deallocate(g->mm, (void *) &sdf);
+ (*sdFile) = NULL;
+ }
+ return done;
+}
+
+static picoos_bool picoos_sdfFlushOutBuf(picoos_SDFile sdFile)
+{
+ picoos_bool done = FALSE;
+ picoos_int32 len;
+ picoos_int32 nrSamples;
+
+ if (!(sdFile->aborted)) {
+ nrSamples = sdFile->bufPos;
+ switch (sdFile->enc) {
+ case PICOOS_ENC_LIN:
+ arr_conv_int16_le(sdFile->buf, 0, nrSamples, sdFile->bBuf, 0);
+ len = (nrSamples * 2);
+ done = picoos_WriteBytes(sdFile->file, sdFile->bBuf, &len)
+ && ((nrSamples * 2) == len);
+ break;
+ case PICOOS_ENC_ULAW:
+ case PICOOS_ENC_ALAW:
+ default:
+ nrSamples = 0;
+ break;
+ }
+ sdFile->nrFileSamples = (sdFile->nrFileSamples + nrSamples);
+ }
+
+ sdFile->bufPos = 0;
+ return done;
+}
+
+extern picoos_bool picoos_sdfFlushOutput(picoos_SDFile sdFile)
+{
+ if ((sdFile != NULL) && !(sdFile->aborted) && (sdFile->bufPos > 0)) {
+ return picoos_sdfFlushOutBuf(sdFile);
+ }
+ return TRUE;
+}
+
+
+
+extern picoos_bool picoos_sdfPutSamples (picoos_SDFile sdFile, picoos_uint32 nrSamples, picoos_int16 samples[])
+{
+ picoos_uint32 i;
+ picoos_int32 s;
+ picoos_bool done = FALSE;
+
+ if ((sdFile != NULL) && !(sdFile->aborted)) {
+ done = TRUE;
+ for (i = 0; i < nrSamples; i++) {
+ s = samples[i];
+ if ((s > PICOOS_INT16_MAX)) {
+ s = PICOOS_INT16_MAX;
+ } else if (s < PICOOS_INT16_MIN) {
+ s = PICOOS_INT16_MIN;
+ }
+ sdFile->buf[sdFile->bufPos++] = s;
+ if (sdFile->bufPos >= PICOOS_SDF_BUF_LEN) {
+ done = picoos_sdfFlushOutBuf(sdFile);
+ }
+ }
+ } else {
+ done = FALSE;
+ }
+ return done;
+}
+
+
+extern picoos_bool picoos_sdfCloseOut (picoos_Common g, picoos_SDFile * sdFile)
+{
+
+ picoos_bool done = TRUE;
+ picoos_uint32 hdrSize;
+
+ if (NULL != (*sdFile)) {
+ if (!((*sdFile)->aborted) && ((*sdFile)->bufPos > 0)) {
+ done = picoos_sdfFlushOutBuf(*sdFile);
+ }
+ if (FILE_TYPE_WAV == (*sdFile)->fileType) {
+ done = picoos_writeWavHeader((*sdFile)->file, (*sdFile)->sf,
+ (*sdFile)->enc, (*sdFile)->nrFileSamples, &hdrSize);
+ }
+ done = picoos_CloseBinary(g, &((*sdFile)->file));
+ picoos_deallocate(g->mm, (void *) sdFile);
+ }
+ return done;
+}
+
+
+/* *****************************************************************/
+/* FileHeader */
+/* *****************************************************************/
+
+
+
+pico_status_t picoos_clearHeader(picoos_FileHeader header)
+{
+ picoos_uint8 i;
+ for (i=0; i < PICOOS_MAX_NUM_HEADER_FIELDS; i++) {
+ header->field[i].key[0] = NULLC;
+ header->field[i].value[0] = NULLC;
+ header->field[i].op = PICOOS_FIELD_IGNORE;
+ }
+ header->numFields = 0;
+ return PICO_OK;
+}
+
+pico_status_t picoos_setHeaderField(picoos_FileHeader header,
+ picoos_uint8 index, picoos_char * key, picoos_char * value,
+ picoos_compare_op_t op)
+{
+ if (index >= header->numFields) {
+ return PICO_ERR_INDEX_OUT_OF_RANGE;
+ }
+ header->field[index].op = op;
+ if ((picoos_strlcpy(header->field[index].key, key,
+ PICOOS_MAX_FIELD_STRING_LEN) < PICOOS_MAX_FIELD_STRING_LEN)
+ && (picoos_strlcpy(header->field[index].value, value,
+ PICOOS_MAX_FIELD_STRING_LEN)) < PICOOS_MAX_FIELD_STRING_LEN) {
+ return PICO_OK;
+ } else {
+ return PICO_ERR_INDEX_OUT_OF_RANGE;
+ }
+}
+
+
+/* caller has to make sure allocated space at key and value are large enough to hold a picoos_field_string */
+pico_status_t picoos_getHeaderField(picoos_FileHeader header, picoos_uint8 index, picoos_field_string_t key, picoos_field_string_t value, picoos_compare_op_t * op)
+{
+ if (index >= header->numFields) {
+ return PICO_ERR_INDEX_OUT_OF_RANGE;
+ }
+ *op = header->field[index].op;
+ if ((picoos_strlcpy(key,header->field[index].key,
+ PICOOS_MAX_FIELD_STRING_LEN) < PICOOS_MAX_FIELD_STRING_LEN)
+ && (picoos_strlcpy(value,header->field[index].value,
+ PICOOS_MAX_FIELD_STRING_LEN)) < PICOOS_MAX_FIELD_STRING_LEN) {
+ return PICO_OK;
+ } else {
+ return PICO_ERR_INDEX_OUT_OF_RANGE;
+ }
+ return PICO_OK;
+}
+
+
+/* check whether 'str' of length strlen matches contents in circular buffer buf, located in the first strlen bytes and ending
+ * position bufpos. */
+static picoos_uint8 os_matched( picoos_char * str, picoos_uint32 strlen, picoos_char * buf, picoos_int32 bufpos) {
+ picoos_int32 i = strlen-1;
+ while (i >= 0 && buf[bufpos] == str[i]) {
+ i--;
+ bufpos--;
+ if (bufpos < 0) {
+ bufpos = strlen-1;
+ }
+ }
+ return (i<0);
+}
+
+pico_status_t picoos_getSVOXHeaderString(picoos_char * str, picoos_uint8 * len, picoos_uint32 maxlen)
+{
+ picoos_char * ch;
+ *len = picoos_strlcpy(str,picoos_SVOXFileHeader,maxlen);
+ if (*len < maxlen) {
+ ch = str;
+ /* SVOX header is made less readable */
+ while (*ch) {
+ *ch -= ' ';
+ ch++;
+ }
+ return PICO_OK;
+ } else {
+ return PICO_ERR_OTHER;
+ }
+}
+
+pico_status_t picoos_readPicoHeader(picoos_File f, picoos_uint32 * headerlen)
+{
+ picoos_char str[32];
+ picoos_char buf[32];
+ picoos_uint8 strlen, bufpos;
+ picoos_uint32 n;
+ picoos_uint8 done;
+
+ picoos_getSVOXHeaderString(str,&strlen,32);
+ /* search for svox header somewhere near the file start. This allows for initial
+ * non-svox-header bytes for a customer-specific header and/or filling bytes for alignment */
+ *headerlen = 0;
+ /* read in initial chunk of length strlen */
+ n = strlen;
+ done = picoos_ReadBytes(f,(picoos_uint8 *)buf,&n) && (n == strlen);
+ if (done) {
+ *headerlen = n;
+ bufpos = strlen-1; /* last legal buf position */
+ done = os_matched(str,strlen,buf,bufpos);
+ while (!done && *headerlen < PICO_MAX_FOREIGN_HEADER_LEN) {
+ n = 1;
+ bufpos = (bufpos + 1) % strlen;
+ done = picoos_ReadBytes(f,(picoos_uint8 *)buf+bufpos,&n) && 1 == n;
+ done = done && os_matched(str,strlen,buf,bufpos);
+ headerlen++;
+ }
+ }
+ if (done) {
+ return PICO_OK;
+ } else {
+ return PICO_EXC_UNEXPECTED_FILE_TYPE;
+ }
+}
+
+picoos_uint8 picoos_get_str (picoos_char * fromStr, picoos_uint32 * pos, picoos_char * toStr, picoos_objsize_t maxsize)
+{
+ picoos_uint8 i = 0;
+ /* skip non-printables */
+
+ while ((fromStr[*pos] != NULLC) && (fromStr[*pos] <= ' ')) {
+ (*pos)++;
+ }
+ /* copy printable portion */
+ while ((fromStr[*pos] != NULLC) && (fromStr[*pos] > ' ') && (i < maxsize-1)) {
+ toStr[i++] = fromStr[(*pos)++];
+ }
+ toStr[i] = NULLC;
+ return (i > 0) && (fromStr[*pos] <= ' ');
+}
+
+pico_status_t picoos_hdrParseHeader(picoos_FileHeader header, picoos_header_string_t str)
+{
+ picoos_uint32 curpos = 0;
+ picoos_uint8 i, numFields;
+
+
+ /* read number of fields */
+ numFields = str[curpos++];
+ numFields = os_min(numFields,PICOOS_MAX_NUM_HEADER_FIELDS);
+ /* read in all field pairs */
+ PICODBG_DEBUG(("number of fields = %i", numFields));
+ for (i = 0; i < numFields; i++) {
+ picoos_get_str(str,&curpos,header->field[i].key,PICOOS_MAX_FIELD_STRING_LEN);
+ picoos_get_str(str,&curpos,header->field[i].value,PICOOS_MAX_FIELD_STRING_LEN);
+ }
+ return PICO_OK;
+}
+
+
+
+
+
+/* **************************************************************************/
+/* Read little-endian / platform-independent integers from file or memory */
+/* **************************************************************************/
+
+/* read little-endian */
+pico_status_t picoos_read_le_uint16 (picoos_File file, picoos_uint16 * val)
+{
+ picoos_uint8 by[2];
+ picoos_uint32 n = 2;
+ if (picoos_ReadBytes(file, by, &n) && 2 == n) {
+ /* little-endian */
+ *val = (picoos_uint16) by[1] << 8 | (picoos_uint16) by[0];
+ return PICO_OK;
+ } else {
+ *val = 0;
+ return PICO_ERR_OTHER;
+ }
+}
+
+pico_status_t picoos_read_le_int16 (picoos_File file, picoos_int16 * val)
+{
+ return picoos_read_le_uint16(file, (picoos_uint16 *)val);
+}
+
+pico_status_t picoos_read_le_uint32 (picoos_File file, picoos_uint32 * val)
+{
+ picoos_uint8 by[4];
+ picoos_uint32 n = 4;
+ if (picoos_ReadBytes(file, by, &n) && (4 == n)) {
+ /* little-endian */
+ PICODBG_TRACE(("reading uint 32: %i %i %i %i",
+ by[0], by[1], by[2], by[3]));
+ *val = (((picoos_uint32) by[3] << 8 | (picoos_uint32) by[2]) << 8 | (picoos_uint32) by[1]) << 8 | (picoos_uint32) by[0];
+ PICODBG_TRACE(("uint 32: %i %i %i %i corresponds %i",
+ by[0], by[1], by[2], by[3], *val));
+ return PICO_OK;
+ } else {
+ *val = 0;
+ return PICO_ERR_OTHER;
+ }
+}
+
+/* platform-independent */
+/* our convention is that pi is little-endian. */
+
+/** @todo : direct implementation if too slow */
+
+pico_status_t picoos_read_pi_uint16 (picoos_File file, picoos_uint16 * val)
+{
+ return picoos_read_le_uint16(file,val);
+}
+
+pico_status_t picoos_read_pi_uint32 (picoos_File file, picoos_uint32 * val)
+{
+ return picoos_read_le_uint32(file, val);
+}
+
+pico_status_t picoos_read_pi_int32 (picoos_File file, picoos_int32 * val)
+{
+ return picoos_read_le_uint32(file, (picoos_uint32 *)val);
+}
+
+/* read pi from memory */
+
+pico_status_t picoos_read_mem_pi_uint16 (picoos_uint8 * data, picoos_uint32 * pos, picoos_uint16 * val)
+{
+ picoos_uint8 * by = data + *pos;
+
+ /* little-endian */
+ *val = (picoos_uint16) by[1] << 8 | (picoos_uint16) by[0];
+ (*pos) += 2;
+ return PICO_OK;
+}
+
+pico_status_t picoos_read_mem_pi_uint32 (picoos_uint8 * data, picoos_uint32 * pos, picoos_uint32 * val)
+{
+ picoos_uint8 * by = data + *pos;
+
+ /* little-endian */
+ *val = (((picoos_uint32) by[3] << 8 | (picoos_uint32) by[2]) << 8 | (picoos_uint32) by[1]) << 8 | (picoos_uint32) by[0];
+ (*pos) += 4;
+ return PICO_OK;
+}
+
+/* **************************************************************************/
+/* Write little-endian / platform-independent integers into file or memory */
+/* **************************************************************************/
+/* write little-endian */
+pico_status_t picoos_write_le_uint16 (picoos_File file, picoos_uint16 val)
+{
+ picoos_int32 len = 2;
+ picoos_uint8 by[2];
+
+ by[0] = (picoos_uint8)((val) & 0x00FF);
+ by[1] = (picoos_uint8)(((val) & 0xFF00)>>8);
+ return (picoos_WriteBytes(file,by,&len) && (2 == len));
+}
+pico_status_t picoos_write_le_uint32 (picoos_File file, picoos_uint32 val)
+{
+ picoos_int32 len = 4;
+ picoos_uint8 by[4];
+
+ by[0] = (picoos_uint8)(val & 0x000000FF);
+ by[1] = (picoos_uint8)((val & 0x0000FF00)>>8);
+ by[2] = (picoos_uint8)((val & 0x00FF0000)>>16);
+ by[3] = (picoos_uint8)((val & 0xFF000000)>>24);
+ return (picoos_WriteBytes(file,by,&len) && (4 == len));
+}
+
+/* write pi to mem */
+pico_status_t picoos_write_mem_pi_uint16 (picoos_uint8 * data, picoos_uint32 * pos, picoos_uint16 val)
+{
+ picoos_uint8 * by = data + *pos;
+ /* little-endian */
+ by[0] = (picoos_uint8)((val) & 0x00FF);
+ by[1] = (picoos_uint8)(((val) & 0xFF00)>>8);
+ (*pos) += 2;
+ return PICO_OK;
+}
+
+/* *****************************************************************/
+/* String search and compare operations */
+/* *****************************************************************/
+
+/* this function is case-sensitive */
+picoos_uint8 picoos_has_extension(const picoos_char *str, const picoos_char *suf)
+{
+ picoos_int32 istr = picoos_strlen(str)-1;
+ picoos_int32 isuf = picoos_strlen(suf)-1;
+ while ((istr >= 0) && (isuf >=0) && (str[istr] == suf[isuf])) {
+ istr--;
+ isuf--;
+ }
+ return (isuf < 0);
+}
+
+
+/* *****************************************************************/
+/* String/Number Conversions (may be moved to picopal) */
+/* *****************************************************************/
+
+pico_status_t picoos_string_to_int32(picoos_char str[],
+ picoos_int32 * res)
+{
+ /* syntax: [+|-] dig {dig} */
+
+ int i;
+ int neg;
+ int val;
+ int err;
+
+ err = 0;
+ i = 0;
+ while ((str[i] <= ' ') && (str[i] != '\0')) {
+ i++;
+ }
+ neg = 0;
+ if (str[i] == '-') {
+ neg = 1;
+ i++;
+ } else if (str[i] == '+') {
+ i++;
+ }
+ val = 0;
+ if ((str[i] < '0') || (str[i]> '9')) {
+ err = 1;
+ }
+ while ((str[i] >= '0') && (str[i] <= '9')) {
+ val = val * 10 + (str[i] - '0');
+ i++;
+ }
+ while ((str[i] <= ' ') && (str[i] != '\0')) {
+ i++;
+ }
+ if (neg == 1) {
+ val = -val;
+ }
+ if ((err == 0) && (str[i] == '\0')) {
+ (*res) = val;
+ return PICO_OK;
+ } else {
+ (*res) = 0;
+ return PICO_EXC_NUMBER_FORMAT;
+ }
+}
+
+pico_status_t picoos_string_to_uint32(picoos_char str[],
+ picoos_uint32 * res)
+{
+ /* syntax: [+] dig {dig} */
+
+ int i;
+ int val;
+ int err;
+
+ err = 0;
+ i = 0;
+ while ((str[i] <= ' ') && (str[i] != '\0')) {
+ i++;
+ }
+ if (str[i] == '+') {
+ i++;
+ }
+ val = 0;
+ if ((str[i] < '0') || (str[i]> '9')) {
+ err = 1;
+ }
+ while ((str[i] >= '0') && (str[i] <= '9')) {
+ val = val * 10 + (str[i] - '0');
+ i++;
+ }
+ while ((str[i] <= ' ') && (str[i] != '\0')) {
+ i++;
+ }
+ if ((err == 0) && (str[i] == '\0')) {
+ (*res) = val;
+ return PICO_OK;
+ } else {
+ (*res) = 0;
+ return PICO_EXC_NUMBER_FORMAT;
+ }
+}
+
+/* 'stringlen' is the part of input string to be considered,
+ * possibly not containing NULLC (e.g. result of strlen).
+ * 'maxsize' is the maximal size of 'part' including a byte
+ * for the terminating NULLC! */
+void picoos_get_sep_part_str(picoos_char string[],
+ picoos_int32 stringlen, picoos_int32 * ind, picoos_char sepCh,
+ picoos_char part[], picoos_int32 maxsize, picoos_uint8 * done)
+{
+
+ picoos_int32 j;
+ picoos_uint8 done1;
+
+ if (((*ind) >= stringlen)) {
+ (*done) = 0;
+ part[0] = (picoos_char) NULLC;
+ } else {
+ done1 = 1;
+ j = 0;
+ while ((((*ind) < stringlen) && (string[(*ind)] != sepCh)) && (string[((*ind))] != (picoos_char)NULLC)) {
+ if ((j < maxsize-1)) {
+ part[(j)] = string[(*ind)];
+ j++;
+ } else {
+ done1 = 0;
+ }
+ (*ind)++;
+ }
+ part[j] = (picoos_char)NULLC;
+ if ((*ind) < stringlen) {
+ if ((string[(*ind)] == sepCh)) {
+ (*ind)++; /* skip separator character */
+ } else if (string[(*ind)] == (picoos_char)NULLC) {
+ /* reached end of input; set ind to stringlen so that no
+ more (empty) partial strings will be found */
+ (*ind) = stringlen;
+ }
+ }
+ (*done) = done1;
+ }
+}
+
+/* *****************************************************************/
+/* timer function */
+/* *****************************************************************/
+
+extern void picoos_get_timer(picopal_uint32 * sec, picopal_uint32 * usec)
+{
+ picopal_get_timer(sec, usec);
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+
+/* end */
diff --git a/lib/picoos.h b/lib/picoos.h
new file mode 100644
index 0000000..859e176
--- /dev/null
+++ b/lib/picoos.h
@@ -0,0 +1,550 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picoos.h
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+/**
+ * @addtogroup picoos
+
+ * <b> Operating system generalization module </b>\n
+ *
+*/
+
+#ifndef PICOOS_H_
+#define PICOOS_H_
+
+#include "picodefs.h"
+#include "picopal.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* some "switch" used in picopal and in picocep ... */
+#define PICOOS_DIV_USE_INV PICOPAL_DIV_USE_INV
+
+/* *************************************************/
+/* types */
+/* *************************************************/
+
+typedef picopal_uint8 picoos_uint8;
+typedef picopal_uint16 picoos_uint16;
+typedef picopal_uint32 picoos_uint32;
+
+typedef picopal_int8 picoos_int8;
+typedef picopal_int16 picoos_int16;
+typedef picopal_int32 picoos_int32;
+
+typedef picopal_double picoos_double;
+typedef picopal_single picoos_single;
+
+typedef picopal_char picoos_char;
+typedef picopal_uchar picoos_uchar;
+
+typedef picopal_uint8 picoos_bool;
+
+typedef picopal_objsize_t picoos_objsize_t;
+typedef picopal_ptrdiff_t picoos_ptrdiff_t;
+
+/* *************************************************/
+/* functions */
+/* *************************************************/
+
+
+picoos_int32 picoos_atoi(const picoos_char *);
+picoos_int8 picoos_strcmp(const picoos_char *, const picoos_char *);
+picoos_int8 picoos_strncmp(const picoos_char *a, const picoos_char *b, picoos_objsize_t siz);
+picoos_uint32 picoos_strlen(const picoos_char *);
+picoos_char * picoos_strchr(const picoos_char *, picoos_char);
+picoos_char *picoos_strstr(const picoos_char *s, const picoos_char *substr);
+picoos_int16 picoos_slprintf(picoos_char * b, picoos_uint32 bsize, const picoos_char *f, ...);
+picoos_char * picoos_strcpy(picoos_char *, const picoos_char *);
+picoos_char * picoos_strcat(picoos_char *, const picoos_char *);
+
+/* copies 'length' bytes from 'src' to 'dest'. (regions may be overlapping) no error checks! */
+void * picoos_mem_copy(const void * src, void * dst, picoos_objsize_t length);
+
+/* safe versions */
+picoos_objsize_t picoos_strlcpy(picoos_char *dst, const picoos_char *src, picoos_objsize_t siz);
+void * picoos_mem_set(void * dest, picoos_uint8 byte_val, picoos_objsize_t length);
+
+picoos_double picoos_cos(const picoos_double cos_arg);
+picoos_double picoos_sin(const picoos_double sin_arg);
+picoos_double picoos_fabs(const picoos_double fabs_arg);
+
+picoos_double picoos_quick_exp(const picoos_double y);
+
+
+void picoos_get_sep_part_str (picoos_char string[], picoos_int32 stringlen, picoos_int32 * ind, picoos_char sepCh, picoos_char part[], picoos_int32 maxsize, picoos_uint8 * done);
+pico_status_t picoos_string_to_uint32 (picoos_char str[], picoos_uint32 * res);
+pico_status_t picoos_string_to_int32 (picoos_char str[], picoos_int32 * res);
+
+/* *****************************************************************/
+/* "Common" */
+/* *****************************************************************/
+/* picoos_common is a collection of basic functionalities that must be globally accesible from every "big" function.
+ * It includes pointers to the MemoryManasger, ExceptionManager and a list of open files. */
+
+typedef struct memory_manager * picoos_MemoryManager;
+typedef struct picoos_exception_manager * picoos_ExceptionManager;
+typedef struct picoos_file * picoos_File;
+
+
+/** object : Common
+ * shortcut : common
+ *
+ */
+typedef struct picoos_common * picoos_Common;
+
+/* the picoos_common structure itself is exported so no access functions are needed. Handle with care! (might be changed later) */
+typedef struct picoos_common {
+ picoos_ExceptionManager em;
+ picoos_MemoryManager mm;
+ picoos_File fileList;
+} picoos_common_t;
+
+picoos_Common picoos_newCommon(picoos_MemoryManager mm);
+
+void picoos_disposeCommon(picoos_MemoryManager mm, picoos_Common * this);
+
+
+/* *****************************************************************/
+/* Memory Management */
+/* *****************************************************************/
+
+typedef picoos_char * byte_ptr_t;
+
+#define PICOOS_ALIGN_SIZE 8
+
+
+
+void * picoos_raw_malloc(byte_ptr_t raw_mem,
+ picoos_objsize_t raw_mem_size, picoos_objsize_t alloc_size,
+ byte_ptr_t * rest_mem, picoos_objsize_t * rest_mem_size);
+
+/**
+ * Creates a new memory manager object for the specified raw memory
+ * block. 'enableProtMem' enables or disables memory protection
+ * functionality; if disabled, picoos_protectMem() has no effect.
+ */
+picoos_MemoryManager picoos_newMemoryManager(
+ void *raw_memory,
+ picoos_objsize_t size,
+ picoos_bool enableMemProt);
+
+
+
+void picoos_disposeMemoryManager(picoos_MemoryManager * mm);
+
+
+void * picoos_allocate(picoos_MemoryManager this, picoos_objsize_t byteSize);
+void picoos_deallocate(picoos_MemoryManager this, void * * adr);
+
+/* the following memory manager routines are for testing and
+ debugging purposes */
+
+/**
+ * Same as picoos_allocate, but write access to the memory block may be
+ * prohibited by a subsequent call to picoos_protectMem().
+ */
+void *picoos_allocProtMem(picoos_MemoryManager mm, picoos_objsize_t byteSize);
+
+/**
+ * Releases a memory block previously allocated by picoos_allocProtMem().
+ */
+void picoos_deallocProtMem(picoos_MemoryManager mm, void **addr);
+
+/**
+ * Enables or disables write protection of a memory block previously
+ * allocated by picoos_allocProtMem(). If write protection is enabled,
+ * any subsequent write access will cause a segmentation fault.
+ */
+void picoos_protectMem(
+ picoos_MemoryManager mm,
+ void *addr,
+ picoos_objsize_t len,
+ picoos_bool enable);
+
+void picoos_getMemUsage(
+ picoos_MemoryManager this,
+ picoos_bool resetIncremental,
+ picoos_int32 *usedBytes,
+ picoos_int32 *incrUsedBytes,
+ picoos_int32 *maxUsedBytes);
+
+void picoos_showMemUsage(
+ picoos_MemoryManager this,
+ picoos_bool incremental,
+ picoos_bool resetIncremental);
+
+/* *****************************************************************/
+/* Exception Management */
+/* *****************************************************************/
+/** object : ExceptionManager
+ * shortcut : em
+ *
+ */
+
+
+#define PICOOS_MAX_EXC_MSG_LEN 512
+#define PICOOS_MAX_WARN_MSG_LEN 64
+#define PICOOS_MAX_NUM_WARNINGS 8
+
+void picoos_setErrorMsg(picoos_char * dst, picoos_objsize_t siz,
+ picoos_int16 code, picoos_char * base, const picoos_char *fmt, ...);
+
+
+picoos_ExceptionManager picoos_newExceptionManager(picoos_MemoryManager mm);
+
+void picoos_disposeExceptionManager(picoos_MemoryManager mm,
+ picoos_ExceptionManager * this);
+
+
+void picoos_emReset(picoos_ExceptionManager this);
+
+/* For convenience, this function returns the resulting exception code of 'this'
+ * (as would be returned by emGetExceptionCode).
+ * The return value therefore is NOT the status of raising
+ * the error! */
+pico_status_t picoos_emRaiseException(picoos_ExceptionManager this,
+ pico_status_t exceptionCode, picoos_char * baseMessage, picoos_char * fmt, ...);
+
+pico_status_t picoos_emGetExceptionCode(picoos_ExceptionManager this);
+
+void picoos_emGetExceptionMessage(picoos_ExceptionManager this, picoos_char * msg, picoos_uint16 maxsize);
+
+void picoos_emRaiseWarning(picoos_ExceptionManager this,
+ pico_status_t warningCode, picoos_char * baseMessage, picoos_char * fmt, ...);
+
+picoos_uint8 picoos_emGetNumOfWarnings(picoos_ExceptionManager this);
+
+pico_status_t picoos_emGetWarningCode(picoos_ExceptionManager this, picoos_uint8 warnNum);
+
+void picoos_emGetWarningMessage(picoos_ExceptionManager this, picoos_uint8 warnNum, picoos_char * msg, picoos_uint16 maxsize);
+
+
+
+
+/* *****************************************************************/
+/* File Access */
+/* *****************************************************************/
+
+#define picoos_MaxFileNameLen 512
+#define picoos_MaxKeyLen 512
+#define picoos_MaxPathLen 512
+#define picoos_MaxPathListLen 2048
+
+typedef picoos_char picoos_Key[picoos_MaxKeyLen];
+typedef picoos_char picoos_FileName[picoos_MaxFileNameLen];
+typedef picoos_char picoos_Path[picoos_MaxPathLen];
+typedef picoos_char picoos_PathList[picoos_MaxPathListLen];
+
+
+/* ***** Sequential binary file access ******/
+
+/* Remark: 'ReadByte', 'ReadBytes' and 'ReadVar' may be mixed;
+ 'WriteByte', 'WriteBytes' and 'WriteVar' may be mixed. */
+
+/* Open existing binary file for read access. */
+picoos_uint8 picoos_OpenBinary(picoos_Common g, picoos_File * f, picoos_char name[]);
+
+
+/* Read next byte from file 'f'. */
+picoos_uint8 picoos_ReadByte(picoos_File f, picoos_uint8 * by);
+
+/* Read next 'len' bytes from 'f' into 'bytes'; 'len' returns the
+ number of bytes actually read (may be smaller than requested
+ length if 'bytes' is too small to hold all bytes or at end of file).
+ Remark: 'bytes' is compabtible with any variable of any size. */
+picoos_uint8 picoos_ReadBytes(picoos_File f, picoos_uint8 bytes[],
+ picoos_uint32 * len);
+
+
+/* Create new binary file.
+ If 'key' is not empty, the file is encrypted with 'key'. */
+picoos_uint8 picoos_CreateBinary(picoos_Common g, picoos_File * f, picoos_char name[]);
+
+picoos_uint8 picoos_WriteByte(picoos_File f, picoos_char by);
+
+/* Writes 'len' bytes from 'bytes' onto file 'f'; 'len' returns
+ the number of bytes actually written. */
+picoos_uint8 picoos_WriteBytes(picoos_File f, const picoos_char bytes[],
+ picoos_int32 * len);
+
+
+/* Close previously opened binary file. */
+picoos_uint8 picoos_CloseBinary(picoos_Common g, picoos_File * f);
+
+
+
+
+
+
+pico_status_t picoos_read_le_int16 (picoos_File file, picoos_int16 * val);
+pico_status_t picoos_read_le_uint16 (picoos_File file, picoos_uint16 * val);
+pico_status_t picoos_read_le_uint32 (picoos_File file, picoos_uint32 * val);
+
+
+pico_status_t picoos_read_pi_uint16 (picoos_File file, picoos_uint16 * val);
+pico_status_t picoos_read_pi_uint32 (picoos_File file, picoos_uint32 * val);
+
+pico_status_t picoos_write_le_uint16 (picoos_File file, picoos_uint16 val);
+pico_status_t picoos_write_le_uint32 (picoos_File file, picoos_uint32 val);
+
+/*
+pico_status_t picoos_write_pi_uint32 (picoos_File file, const picoos_uint32 val);
+
+pico_status_t picoos_write_pi_uint16 (picoos_File file, const picoos_uint16 val);
+*/
+
+
+/* **************************************************************************************/
+/* *** general file routines *****/
+
+/* Returns whether end of file was encountered in previous
+ read operation. */
+picoos_bool picoos_Eof(picoos_File f);
+
+/* sets the file pointer to
+ 'pos' bytes from beginning (first byte = byte 0). This
+ routine should only be used for binary files. */
+picoos_bool picoos_SetPos(picoos_File f, picoos_int32 pos);
+
+/* Get position from file 'f'. */
+picoos_bool picoos_GetPos(picoos_File f, picoos_uint32 * pos);
+
+/* Returns the length of the file in bytes. */
+picoos_bool picoos_FileLength(picoos_File f, picoos_uint32 * len);
+
+/* Return full name of file 'f'. */
+picoos_bool picoos_Name(picoos_File f, picoos_char name[], picoos_uint32 maxsize);
+
+/* Returns whether file 'name' exists or not. */
+picoos_bool picoos_FileExists(picoos_Common g, picoos_char name[] /*, picoos_char ckey[] */);
+
+/* Delete a file. */
+picoos_bool picoos_Delete(picoos_char name[]);
+
+/* Rename a file. */
+picoos_bool picoos_Rename(picoos_char oldName[], picoos_char newName[]);
+
+
+/* *****************************************************************/
+/* Sampled Data Files */
+/* *****************************************************************/
+
+#define SAMPLE_FREQ_16KHZ (picoos_uint32) 16000
+
+typedef enum {
+ FILE_TYPE_WAV,
+ FILE_TYPE_AU,
+ FILE_TYPE_RAW,
+ FILE_TYPE_OTHER
+} wave_file_type_t;
+
+typedef enum {
+ FORMAT_TAG_LIN = 1, /**< linear 16-bit encoding */
+ FORMAT_TAG_ALAW = 6, /**< a-law encoding, 8 bit */
+ FORMAT_TAG_ULAW = 7 /**< u-law encoding, 8 bit */
+ /* there are many more */
+} wave_format_tag_t;
+
+
+typedef enum {
+ /* values corresponding RIFF wFormatTag */
+ PICOOS_ENC_LIN = FORMAT_TAG_LIN, /**< linear 16-bit encoding; standard */
+ PICOOS_ENC_ALAW = FORMAT_TAG_ALAW, /**< a-law encoding, 8 bit */
+ PICOOS_ENC_ULAW = FORMAT_TAG_ULAW, /**< u-law encoding, 8 bit */
+ /* values outside RIFF wFormatTag values (above 4100) */
+ PICOOS_ENC_OTHER = 5000 /**< other; (treated as raw) */
+ } picoos_encoding_t;
+
+typedef struct picoos_sd_file * picoos_SDFile;
+
+/* SDFile input functions */
+
+/* orig. comment from SDInOut.def
+ Opens sampled data file 'fileName' for input and returns
+ the encoding 'enc' of the file, sampling rate 'sf',
+ nr of samples 'nrSamples', and a handle to the opened file
+ in 'sdFile'.
+
+ If 'fileName' is empty, the input is taken from the direct
+ acoustic input using the sampling rate specified by
+ "SetRawDefaults". In this case, 'encoding' returns 'EncLin',
+ and 'nrSamples' returns 0.
+
+ The file format is taken from the file name extension:
+ ".wav" --> wav file
+ ".au" --> au file
+ other extensions --> headerless files
+
+ For wav and au files, the sampling rate and encoding are taken
+ from the file header and returned in 'sf' and 'enc'. For
+ headerless files, 'sf' and 'enc' are taken from the
+ most recently set default values (procedure SetRawDefaults).
+
+ 'done' returns whether the sampled data file was successfully
+ opened. */
+extern picoos_bool picoos_sdfOpenIn (picoos_Common g, picoos_SDFile * sdFile, picoos_char fileName[], picoos_uint32 * sf, picoos_encoding_t * enc, picoos_uint32 * nrSamples);
+
+
+extern picoos_bool picoos_sdfGetSamples (picoos_SDFile sdFile, picoos_uint32 start, picoos_uint32 * nrSamples, picoos_int16 samples[]);
+
+
+extern picoos_bool picoos_sdfCloseIn (picoos_Common g, picoos_SDFile * sdFile);
+
+
+/* SDFile output functions*/
+
+extern picoos_bool picoos_sdfOpenOut (picoos_Common g, picoos_SDFile * sdFile, picoos_char fileName[], int sf, picoos_encoding_t enc);
+
+
+extern picoos_bool picoos_sdfPutSamples (picoos_SDFile sdFile, picoos_uint32 nrSamples, picoos_int16 samples[]);
+
+/*
+extern picoos_bool picoos_AbortOutput (picoos_SDFile sdFile);
+
+
+extern picoos_bool picoos_ResumeOutput (picoos_SDFile sdFile);
+
+
+extern picoos_bool picoos_FlushOutput (picoos_SDFile sdFile);
+*/
+
+extern picoos_bool picoos_sdfCloseOut (picoos_Common g, picoos_SDFile * sdFile);
+
+
+/* *****************************************************************/
+/* File Headers */
+/* *****************************************************************/
+
+#define PICOOS_MAX_FIELD_STRING_LEN 32 /* including terminating char */
+
+#define PICOOS_MAX_NUM_HEADER_FIELDS 10
+#define PICOOS_NUM_BASIC_HEADER_FIELDS 5
+
+#define PICOOS_HEADER_NAME 0
+#define PICOOS_HEADER_VERSION 1
+#define PICOOS_HEADER_DATE 2
+#define PICOOS_HEADER_TIME 3
+#define PICOOS_HEADER_CONTENT_TYPE 4
+
+#define PICOOS_MAX_HEADER_STRING_LEN (PICOOS_MAX_NUM_HEADER_FIELDS * (2 * PICOOS_MAX_FIELD_STRING_LEN))
+
+typedef picoos_char picoos_field_string_t[PICOOS_MAX_FIELD_STRING_LEN];
+
+typedef picoos_char picoos_header_string_t[PICOOS_MAX_HEADER_STRING_LEN];
+
+typedef enum {PICOOS_FIELD_IGNORE, PICOOS_FIELD_EQUAL, PICOOS_FIELD_COMPAT} picoos_compare_op_t;
+
+/* private */
+typedef struct picoos_file_header_field {
+ picoos_field_string_t key;
+ picoos_field_string_t value;
+ picoos_compare_op_t op;
+} picoos_file_header_field_t;
+
+/* public */
+typedef struct picoos_file_header * picoos_FileHeader;
+typedef struct picoos_file_header {
+ picoos_uint8 numFields;
+ picoos_file_header_field_t field[PICOOS_MAX_NUM_HEADER_FIELDS];
+} picoos_file_header_t;
+
+
+pico_status_t picoos_clearHeader(picoos_FileHeader header);
+
+pico_status_t picoos_setHeaderField(picoos_FileHeader header, picoos_uint8 index, picoos_char * key, picoos_char * value, picoos_compare_op_t op);
+
+/* caller has to make sure allocated space at key and value are large enough to hold a picoos_field_string */
+pico_status_t picoos_getHeaderField(picoos_FileHeader header, picoos_uint8 index, picoos_field_string_t key, picoos_field_string_t value, picoos_compare_op_t * op);
+
+/* caller has to make sure allocated space at str is large enough to hold the header in question */
+/*
+pico_status_t picoos_hdrToString(picoos_FileHeader header, picoos_header_string_t str);
+*/
+
+pico_status_t picoos_hdrParseHeader(picoos_FileHeader header, picoos_header_string_t str);
+
+pico_status_t picoos_getSVOXHeaderString(picoos_char * str, picoos_uint8 * len, picoos_uint32 maxlen);
+
+pico_status_t picoos_readPicoHeader(picoos_File f, picoos_uint32 * headerlen);
+
+
+
+/* *****************************************************************/
+/* String search and compare operations */
+/* *****************************************************************/
+
+
+picoos_uint8 picoos_has_extension(const picoos_char *str, const picoos_char *suf);
+
+/* *****************************************************************/
+/* String/Number Manipulations (might be moved to picopal) */
+/* *****************************************************************/
+
+pico_status_t picoos_string_to_int32(picoos_char str[],
+ picoos_int32 * res);
+
+pico_status_t picoos_string_to_uint32(picoos_char str[],
+ picoos_uint32 * res);
+
+/* 'stringlen' is the part of input string to be considered, possibly not containing NULLC (e.g. result of strlen).
+ * 'maxsize' is the maximal size of 'part' including a byte for the terminating NULLC! */
+void picoos_get_sep_part_str(picoos_char string[],
+ picoos_int32 stringlen, picoos_int32 * ind, picoos_char sepCh,
+ picoos_char part[], picoos_int32 maxsize, picoos_uint8 * done);
+
+/* searches for the first contiguous string of printable characters (> ' ') inside fromStr, possibly skipping
+ * chars <= ' ') and returns it in toStr.
+ * fromStr is assumed to be NULLC terminated and toStr is forced to be NULLC terminated within maxsize.
+ * The search is started at *pos inside fromStr and at return, *pos is the position within fromStr after the
+ * found string, or the position after the end of fromStr, if no string was found.
+ * the function returns TRUE if a string was found and fitted toStr, or FALSE otherwise. */
+picoos_uint8 picoos_get_str (picoos_char * fromStr, picoos_uint32 * pos, picoos_char * toStr, picoos_objsize_t maxsize);
+
+
+pico_status_t picoos_read_mem_pi_uint16 (picoos_uint8 * data, picoos_uint32 * pos, picoos_uint16 * val);
+
+pico_status_t picoos_read_mem_pi_uint32 (picoos_uint8 * data, picoos_uint32 * pos, picoos_uint32 * val);
+
+pico_status_t picoos_write_mem_pi_uint16 (picoos_uint8 * data, picoos_uint32 * pos, picoos_uint16 val);
+
+
+/* *****************************************************************/
+/* timer function */
+/* *****************************************************************/
+
+void picoos_get_timer(picopal_uint32 * sec, picopal_uint32 * usec);
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif /*PICOOS_H_*/
diff --git a/lib/picopal.c b/lib/picopal.c
new file mode 100644
index 0000000..db05e2b
--- /dev/null
+++ b/lib/picopal.c
@@ -0,0 +1,554 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picopal.c
+ *
+ * pico platform abstraction layer
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+/* GCC does not supporte #pragma message */
+#if !defined(__GNUC__)
+#define PRAGMA_MESSAGE
+#endif
+
+#if defined(PRAGMA_MESSAGE) && defined(PICO_PLATFORM)
+#pragma message("PICO_PLATFORM : is defined externally")
+#endif
+
+#if defined(PRAGMA_MESSAGE) && defined(_WIN32)
+#pragma message("_WIN32 : is defined")
+#endif
+
+#if defined(PRAGMA_MESSAGE) && defined(__APPLE__)
+#pragma message("__APPLE__ : is defined")
+#endif
+
+#if defined(PRAGMA_MESSAGE) && defined(__MACH__)
+#pragma message("__MACH__ : is defined")
+#endif
+
+#if defined(PRAGMA_MESSAGE) && defined(macintosh)
+#pragma message("macintosh : is defined")
+#endif
+
+#if defined(PRAGMA_MESSAGE) && defined(linux)
+#pragma message("linux : is defined")
+#endif
+#if defined(PRAGMA_MESSAGE) && defined(__linux__)
+#pragma message("__linux__ : is defined")
+#endif
+#if defined(PRAGMA_MESSAGE) && defined(__linux)
+#pragma message("__linux : is defined")
+#endif
+
+
+#include <stdlib.h>
+#include <string.h>
+#include <stdio.h>
+#include <stdarg.h>
+
+#include "picodefs.h"
+#include "picopal.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+#if PICO_PLATFORM == PICO_Windows
+/* use high-resolution timer functions on Windows platform */
+#define USE_CLOCK 0
+#else
+/* use clock() function instead of high-resolution timer on
+ non-Windows platforms */
+#define USE_CLOCK 1
+#endif
+
+#include <time.h>
+#if PICO_PLATFORM == PICO_Windows
+#include <windows.h>
+#endif
+
+#if defined(PRAGMA_MESSAGE)
+#pragma message("PICO_PLATFORM : " PICO_PLATFORM_STRING)
+#endif
+
+
+picopal_int32 picopal_atoi(const picopal_char *s) {
+ return (picopal_int32)atoi((const char *)s);
+}
+
+picopal_int32 picopal_strcmp(const picopal_char *a, const picopal_char *b) {
+ return (picopal_int32)strcmp((const char *)a, (const char *)b);
+}
+
+picopal_int32 picopal_strncmp(const picopal_char *a, const picopal_char *b, picopal_objsize_t siz) {
+ return (picopal_int32)strncmp((const char *)a, (const char *)b, (size_t) siz);
+}
+
+picopal_objsize_t picopal_strlen(const picopal_char *s) {
+ return (picopal_objsize_t)strlen((const char *)s);
+}
+
+picopal_char *picopal_strchr(const picopal_char *s, picopal_char c) {
+ return (picopal_char *)strchr((const char *)s, (int)c);
+}
+
+picopal_char *picopal_strstr(const picopal_char *s, const picopal_char *substr) {
+ return (picopal_char *)strstr((const char *)s, (const char *)substr);
+}
+
+picopal_char *picopal_strcpy(picopal_char *d, const picopal_char *s) {
+ return (picopal_char *)strcpy((char *)d, (const char *)s);
+}
+
+picopal_char *picopal_strcat(picopal_char *dest, const picopal_char *src) {
+ return (picopal_char *)strcat((char *)dest, (const char *)src);
+}
+
+
+/* copy src into dst, but make sure that dst is not accessed beyond its size 'siz' and is allways NULLC-terminated.
+ * 'siz' is the number of bytes of the destination, including one byte for NULLC!
+ * returns the full length of the input string, not including termination NULLC (strlen(src)).
+ * the copy is successfull without truncation if picopal_strlcpy(dst,src,siz) < siz */
+picopal_objsize_t picopal_strlcpy(picopal_char *dst, const picopal_char *src, picopal_objsize_t siz)
+{
+ picopal_char *d = dst;
+ const picopal_char *s = src;
+ picopal_objsize_t n = siz;
+
+ /* Copy as many bytes as will fit */
+ if (n != 0) {
+ while (--n != 0) {
+ if ((*(d++) = *(s++)) == NULLC) {
+ break;
+ }
+ }
+ }
+
+ /* Not enough room in dst, add NULLC and traverse rest of src */
+ if (n == 0) {
+ if (siz != 0) {
+ *d = NULLC; /* NULLC-terminate dst */
+ }
+ while (*(s++)) {}
+ ;
+ }
+
+ return(s - src - 1); /* count does not include NULLC */
+}
+
+
+picopal_int16 picopal_sprintf(picopal_char * dst, const picopal_char *fmt, ...)
+{
+ picopal_int16 i;
+ va_list args;
+
+ va_start(args, (char *)fmt);
+ i = (picopal_int16)vsprintf((char *) dst, (const char *)fmt, args);
+ va_end(args);
+ return i;
+}
+
+
+picopal_objsize_t picopal_vslprintf(picopal_char * dst, picopal_objsize_t siz, const picopal_char *fmt, va_list args) {
+ picopal_char buf[21];
+ picopal_char *d = dst;
+ const picopal_char *f = fmt;
+ picopal_char * b;
+ picopal_objsize_t len, nnew, n = siz;
+ picopal_int32 ival;
+ picopal_char cval;
+ picopal_objsize_t i = 0;
+
+ if (!f) {
+ f = (picopal_char *) "";
+ }
+ while (*f) {
+ if (*f == '%') {
+ switch (*(++f)) {
+ case 'i':
+ f++;
+ ival = va_arg(args,int);
+ picopal_sprintf(buf,(picopal_char *)"%i",ival);
+ b = buf;
+ break;
+ case 'c':
+ f++;
+ cval = va_arg(args,int);
+ picopal_sprintf(buf,(picopal_char *)"%c",cval);
+ b = buf;
+ break;
+ case 's':
+ f++;
+ b = (picopal_char *) va_arg(args, char*);
+ break;
+ default:
+ if (n > 0) {
+ (*d++) = '%';
+ n--;
+ }
+ i++;
+ b = NULL;
+ break;
+ }
+ if (b) {
+ len = picopal_strlcpy(d,b,n); /* n1 is the actual length of sval */
+ i += len;
+ nnew = (n > len) ? n-len : 0; /* nnew is the new value of n */
+ d += (n - nnew);
+ n = nnew;
+ }
+ } else {
+ if (n) {
+ (*d++) = (*f);
+ n--;
+ }
+ i++;
+ f++;
+ }
+ }
+
+ return i;
+}
+
+picopal_objsize_t picopal_slprintf(picopal_char * dst, picopal_objsize_t siz, const picopal_char *fmt, /*args*/ ...) {
+ picopal_objsize_t i;
+ va_list args;
+
+ va_start(args, (char *)fmt);
+ i = picopal_vslprintf(dst, siz, fmt, args);
+ va_end(args);
+ return i;
+
+}
+
+
+/* copies 'length' bytes from 'src' to 'dest'. (regions may be overlapping) no error checks! */
+void * picopal_mem_copy(const void * src, void * dst, picopal_objsize_t length)
+{
+ return memmove(dst, src, (size_t) length);
+}
+
+/* sets 'length' bytes starting at dest[0] to 'byte_val' */
+void * picopal_mem_set(void * dest, picopal_uint8 byte_val, picopal_objsize_t length)
+{
+ return memset(dest, (int) byte_val, (size_t) length);
+}
+
+/* *************************************************/
+/* fixed-point math */
+/* *************************************************/
+
+/* *************************************************/
+/* transcendent math */
+/* *************************************************/
+
+
+picopal_double picopal_cos(const picopal_double cos_arg)
+{
+ return (picopal_double) cos((double)cos_arg);
+}
+picopal_double picopal_sin(const picopal_double sin_arg)
+{
+ return (picopal_double) sin((double) sin_arg);
+}
+picopal_double picopal_fabs(const picopal_double fabs_arg)
+{
+ return (picopal_double) fabs((double) fabs_arg);
+}
+
+
+/* *************************************************/
+/* file access */
+/* *************************************************/
+#define PICOPAL_EOL '\n'
+
+picopal_char picopal_eol(void)
+{
+ return PICOPAL_EOL;
+}
+
+/* 'fopen' opens the file with name 'filename'. Depending on
+ 'mode' :
+ 'PICOPAL_TEXT_READ' : Opens an existing text file for reading.
+ The file is positioned at the beginning of the file.
+ 'PICOPAL_TEXT_WRITE' : Opens and truncates an existing file or creates a new
+ text file for writing. The file is positioned at the
+ beginning.
+ 'PICOPAL_BIN_READ' : Opens an existing binary file for reading.
+ The file is positioned at the beginning of the file.
+ 'PICOPAL_BIN_WRITE' : Opens and truncates an existing file or creates a new
+ binary file for writing. The file is positioned at the
+ beginning.
+ If the opening of the file is successful a file pointer is given
+ back. Otherwise a NIL-File is given back.
+*/
+picopal_File picopal_fopen (picopal_char filename[], picopal_access_mode mode)
+{
+ picopal_File res;
+
+ switch (mode) {
+ case PICOPAL_TEXT_READ :
+ res = (picopal_File) fopen((char *)filename, (char *)"r");
+ break;
+ case PICOPAL_TEXT_WRITE :
+ res = (picopal_File) fopen((char *)filename, (char *)"w");
+ break;
+ case PICOPAL_BINARY_READ :
+ res = (picopal_File) fopen((char *)filename, (char *)"rb");
+ break;
+ case PICOPAL_BINARY_WRITE :
+ res = (picopal_File) fopen((char *)filename, (char *)"wb");
+ break;
+ default :
+ res = (picopal_File) NULL;
+ }
+ return res;
+
+}
+
+
+picopal_File picopal_get_fnil (void)
+{
+ return (picopal_File) NULL;
+}
+
+
+picopal_int8 picopal_is_fnil (picopal_File f)
+{
+ return (NULL == f);
+}
+
+pico_status_t picopal_fflush (picopal_File f)
+{
+ return (0 == fflush((FILE *)f)) ? PICO_OK : PICO_EOF;
+}
+
+
+pico_status_t picopal_fclose (picopal_File f)
+{
+ return (0 == fclose((FILE *)f)) ? PICO_OK : PICO_EOF;
+}
+
+
+
+picopal_uint32 picopal_flength (picopal_File stream)
+{
+ fpos_t fpos;
+ picopal_int32 len;
+
+ fgetpos((FILE *)stream,&fpos);
+ picopal_fseek(stream,0,SEEK_END);
+ len = ftell((FILE *)stream);
+ fsetpos((FILE *)stream,&fpos);
+ clearerr((FILE *)stream);
+ return len;
+
+}
+
+picopal_uint8 picopal_feof (picopal_File stream)
+{
+ return (0 != feof((FILE *)stream));
+
+}
+
+pico_status_t picopal_fseek (picopal_File f, picopal_uint32 offset, picopal_int8 seekmode)
+{
+ return (0 == fseek((FILE *)f, offset, seekmode)) ? PICO_OK : PICO_EOF;
+}
+
+pico_status_t picopal_fget_char (picopal_File f, picopal_char * ch)
+{
+ picopal_int16 res;
+
+ res = fgetc((FILE *)f);
+ if (res >= 0) {
+ *ch = (picopal_char) res;
+ }
+ else {
+ *ch = '\0';
+ }
+ return (res >= 0) ? PICO_OK : PICO_EOF;
+}
+
+picopal_objsize_t picopal_fread_bytes (picopal_File f, void * ptr, picopal_objsize_t objsize, picopal_uint32 nobj)
+{
+ return (picopal_objsize_t) fread(ptr, objsize, nobj, (FILE *)f);
+}
+
+picopal_objsize_t picopal_fwrite_bytes (picopal_File f, void * ptr, picopal_objsize_t objsize, picopal_uint32 nobj){ return (picopal_objsize_t) fwrite(ptr, objsize, nobj, (FILE *)f);}
+/* *************************************************/
+/* functions for debugging/testing purposes only */
+/* *************************************************/
+
+void *picopal_mpr_alloc(picopal_objsize_t size)
+{
+#if PICO_PLATFORM == PICO_Windows
+ return VirtualAlloc(NULL, size, MEM_COMMIT, PAGE_READWRITE);
+#else
+ /* not yet implemented for other platforms; corresponding
+ function on UNIX systems is pvalloc */
+ return NULL;
+#endif
+ size = size; /* avoid warning "var not used in this function"*/
+
+}
+
+void picopal_mpr_free(void **p)
+{
+#if PICO_PLATFORM == PICO_Windows
+ VirtualFree(*p, 0, MEM_RELEASE);
+#else
+ /* not yet implemented for other platforms */
+#endif
+ *p = NULL;
+}
+
+pico_status_t picopal_mpr_protect(void *addr, picopal_objsize_t len, picopal_int16 prot)
+{
+ pico_status_t status = PICO_OK;
+
+#if PICO_PLATFORM == PICO_Windows
+ DWORD dwNewProtect, dwOldProtect;
+ dwNewProtect = PICOPAL_PROT_NONE;
+ if (prot & PICOPAL_PROT_READ) {
+ if (prot & PICOPAL_PROT_WRITE) {
+ dwNewProtect = PAGE_READWRITE;
+ } else {
+ dwNewProtect = PAGE_READONLY;
+ }
+ } else if (prot & PICOPAL_PROT_WRITE) {
+ /* under Windows write-only is not possible */
+ dwNewProtect = PAGE_READWRITE;
+ }
+ if (!VirtualProtect(addr, len, dwNewProtect, &dwOldProtect)) {
+ status = PICO_ERR_OTHER;
+ }
+#else
+ /* not yet implemented for other platforms */
+ addr = addr; /* avoid warning "var not used in this function"*/
+ len = len; /* avoid warning "var not used in this function"*/
+ prot = prot; /* avoid warning "var not used in this function"*/
+
+#endif
+ return status;
+}
+/*
+ * Reference:
+ * A Fast, Compact Approximation of the Exponential Function by Nicol N. Schraudolph in Neural Computation, 11,853-862 (1999)
+ * See also: http://www-h.eng.cam.ac.uk/help/tpl/programs/Matlab/mex.html
+ *
+ */
+picopal_double picopal_quick_exp(const picopal_double y) {
+ union {
+ picopal_double d;
+ struct {
+ #if PICO_ENDIANNESS == ENDIANNESS_LITTLE /* little endian */
+ picopal_int32 j,i;
+ #else
+ picopal_int32 i,j;
+ #endif
+ } n;
+
+ } _eco;
+ _eco.n.i = (picopal_int32)(1512775.3951951856938297995605697f * y) + 1072632447;
+ return _eco.d;
+}
+
+
+/* *************************************************/
+/* timer */
+/* *************************************************/
+
+#if IMPLEMENT_TIMER
+
+#define USEC_PER_SEC 1000000
+
+typedef clock_t picopal_clock_t;
+
+
+#if USE_CLOCK
+picopal_clock_t startTime;
+#else
+int timerInit = 0;
+LARGE_INTEGER startTime;
+LARGE_INTEGER timerFreq;
+#endif
+
+
+picopal_clock_t picopal_clock(void)
+{
+ return (picopal_clock_t)clock();
+}
+
+#endif /* IMPLEMENT_TIMER */
+
+void picopal_get_timer(picopal_uint32 * sec, picopal_uint32 * usec)
+{
+#if IMPLEMENT_TIMER
+#if USE_CLOCK
+ picopal_clock_t dt;
+ dt = picopal_clock() - startTime;
+ *sec = dt / CLOCKS_PER_SEC;
+ *usec = USEC_PER_SEC * (dt % CLOCKS_PER_SEC) / CLOCKS_PER_SEC;
+#else
+ LARGE_INTEGER now;
+ if (!timerInit) {
+ QueryPerformanceFrequency(&timerFreq);
+ timerInit = 1;
+ }
+ if (QueryPerformanceCounter(&now) && 0) {
+/*
+ LONGLONG dt, tf;
+ dt = now.QuadPart - GLOB(startTime).QuadPart;
+ tf = GLOB(timerFreq).QuadPart;
+ *sec = (unsigned int) (dt / tf);
+ *usec = (unsigned int) (USEC_PER_SEC * (dt % tf) / tf);
+*/
+ double dt, tf;
+ dt = (double)(now.QuadPart - startTime.QuadPart);
+ tf = (double)(timerFreq.QuadPart);
+ *sec = (unsigned int) (dt /tf);
+ *usec = (unsigned int) ((double)USEC_PER_SEC * (dt / tf)) % USEC_PER_SEC;
+ } else {
+ /* high freq counter not supported by system */
+ DWORD dt;
+ dt = GetTickCount() - startTime.LowPart;
+ *sec = dt / 1000;
+ *usec = 1000 * (dt % 1000);
+ }
+#endif /* USE_CLOCK */
+#else
+ *sec = 0;
+ *usec = 0;
+#endif /* IMPLEMENT_TIMER */
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+/* End picopal.c */
diff --git a/lib/picopal.h b/lib/picopal.h
new file mode 100644
index 0000000..acfc8e0
--- /dev/null
+++ b/lib/picopal.h
@@ -0,0 +1,288 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picopal.h
+ *
+ * pico plattform abstraction layer
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+/**
+ * @addtogroup picoos
+
+ * <b> Operating system Platform Specific implementation module </b>\n
+ *
+*/
+#ifndef PICOPAL_H_
+#define PICOPAL_H_
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <stdarg.h>
+#include <math.h>
+#include <stddef.h>
+#include "picopltf.h"
+#include "picodefs.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* *********************************************************/
+/* general defines and typedefs (used to be in picodefs.h) */
+/* *********************************************************/
+
+#define TRUE 1
+#define FALSE 0
+
+#ifndef NULL
+#define NULL 0
+#endif
+
+#define NULLC '\000'
+
+
+/* "strange" defines to switch variants... */
+#define PICOPAL_DIV_USE_INV 0
+
+
+/*---------------Externals-----------------------*/
+/* used by picocep*/
+#if defined(PICO_DEBUG)
+ extern int numlongmult, numshortmult;
+#endif
+
+
+typedef signed int pico_status_t;
+
+
+/* unfortunately, ANSI-C uses eof for results and exceptional results .. */
+/* in the context of reading from a CharBuffer, eof means "no more
+ input available FOR NOW" */
+
+#define PICO_EOF (pico_status_t) -1
+
+
+/* *************************************************/
+/* constants */
+/* *************************************************/
+
+
+ /* operating system identifications */
+#define PICOPAL_OS_NIL 0 /* just an unchangeable first value */
+#define PICOPAL_OS_WINDOWS 1
+/* ... */
+#define PICOPAL_OS_GENERIC 99 /* must always be the last value */
+
+/* *************************************************/
+/* types */
+/* *************************************************/
+
+typedef unsigned char picopal_uint8;
+typedef unsigned short picopal_uint16;
+typedef unsigned int picopal_uint32;
+
+typedef signed char picopal_int8;
+typedef signed short picopal_int16;
+typedef signed int picopal_int32;
+
+typedef float picopal_single;
+typedef double picopal_double;
+
+typedef unsigned char picopal_char;
+
+typedef unsigned char picopal_uchar;
+
+typedef size_t picopal_objsize_t;
+typedef ptrdiff_t picopal_ptrdiff_t;
+
+/* *************************************************/
+/* functions */
+/* *************************************************/
+
+picopal_int32 picopal_atoi(const picopal_char *);
+
+picopal_int32 picopal_strcmp(const picopal_char *, const picopal_char *);
+picopal_int32 picopal_strncmp(const picopal_char *a, const picopal_char *b, picopal_objsize_t siz);
+picopal_objsize_t picopal_strlen(const picopal_char *);
+picopal_char * picopal_strchr(const picopal_char *, picopal_char);
+picopal_char * picopal_strcpy(picopal_char *d, const picopal_char *s);
+picopal_char *picopal_strstr(const picopal_char *s, const picopal_char *substr);
+picopal_char *picopal_strcat(picopal_char *dest, const picopal_char *src);
+picopal_int16 picopal_sprintf(picopal_char * dst, const picopal_char *fmt, ...);
+
+/* copies 'length' bytes from 'src' to 'dest'. (regions may be overlapping) no error checks! */
+void * picopal_mem_copy(const void * src, void * dst, picopal_objsize_t length);
+
+/* sets 'length' bytes starting at dest[0] to 'byte_val' */
+void * picopal_mem_set(void * dest, picopal_uint8 byte_val, picopal_objsize_t length);
+
+/* safe versions */
+picopal_objsize_t picopal_vslprintf(picopal_char * dst, picopal_objsize_t siz, const picopal_char *fmt, va_list args);
+picopal_objsize_t picopal_slprintf(picopal_char * dst, picopal_objsize_t siz, const picopal_char *fmt, /*args*/ ...);
+picopal_objsize_t picopal_strlcpy(picopal_char *dst, const picopal_char *src, picopal_objsize_t siz);
+
+/*Fixed point computation*/
+/*
+picopal_int32 picopal_fixptdiv(picopal_int32 a, picopal_int32 b, picopal_uint8 bigpow);
+picopal_int32 picopal_fixptmult(picopal_int32 x, picopal_int32 y, picopal_uint8 bigpow);
+picopal_int32 picopal_fixptdivORinv(picopal_int32 a, picopal_int32 b, picopal_int32 invb, picopal_uint8 bigpow);
+picopal_int32 picopal_fixptmultdouble(picopal_int32 x, picopal_int32 y, picopal_uint8 bigpow);
+picopal_uint8 picopal_highestBit(picopal_int32 x);
+*/
+
+/* *************************************************/
+/* math */
+/* *************************************************/
+
+picopal_double picopal_cos (const picopal_double cos_arg);
+picopal_double picopal_sin (const picopal_double sin_arg);
+picopal_double picopal_fabs (const picopal_double fabs_arg);
+
+
+
+
+/* *************************************************/
+/* file access */
+/* *************************************************/
+
+extern picopal_char picopal_eol(void);
+
+#define picopal_FILE FILE
+
+
+/* seek modes to be used with the 'FSeek' procedure */
+#define PICOPAL_SEEK_SET 0 /* absolut seek position */
+#define PICOPAL_SEEK_CUR 1 /* relative to current */
+#define PICOPAL_SEEK_END 2 /* relative to the end of the file */
+
+
+typedef enum {PICOPAL_BINARY_READ, PICOPAL_BINARY_WRITE, PICOPAL_TEXT_READ, PICOPAL_TEXT_WRITE} picopal_access_mode;
+
+typedef picopal_FILE * picopal_File;
+
+extern picopal_File picopal_fopen (picopal_char fileName[], picopal_access_mode mode);
+/* 'FOpen' opens the file with name 'filename'. Depending on
+ 'mode' :
+ 'TextRead' : Opens an existing text file for reading.
+ The file is positioned at the beginning of the file.
+ 'TextWrite' : Opens and truncates an existing file or creates a new
+ text file for writing. The file is positioned at the
+ beginning.
+ 'BinaryRead' : Opens an existing binary file for reading.
+ The file is positioned at the beginning of the file.
+ 'BinaryWrite' : Opens and truncates an existing file or creates a new
+ binary file for writing. The file is positioned at the
+ beginning.
+ If the opening of the file is successful a file pointer is given
+ back. Otherwise a NIL-File is given back.
+*/
+
+
+extern picopal_File picopal_get_fnil (void);
+
+
+extern picopal_int8 picopal_is_fnil (picopal_File f);
+
+
+extern pico_status_t picopal_fclose (picopal_File f);
+
+
+extern picopal_uint32 picopal_flength (picopal_File f);
+
+
+extern picopal_uint8 picopal_feof (picopal_File f);
+
+
+extern pico_status_t picopal_fseek (picopal_File f, picopal_uint32 offset, picopal_int8 seekmode);
+
+
+extern pico_status_t picopal_fget_char (picopal_File f, picopal_char * ch);
+
+
+extern picopal_objsize_t picopal_fread_bytes (picopal_File f, void * ptr, picopal_objsize_t objsize, picopal_uint32 nobj);
+
+extern picopal_objsize_t picopal_fwrite_bytes (picopal_File f, void * ptr, picopal_objsize_t objsize, picopal_uint32 nobj);
+
+
+extern pico_status_t picopal_fflush (picopal_File f);
+
+/*
+extern pico_status_t picopal_fput_char (picopal_File f, picopal_char ch);
+*/
+
+
+/*
+extern pico_status_t picopal_remove (picopal_char filename[]);
+
+
+extern pico_status_t picopal_rename (picopal_char oldname[], picopal_char newname[]);
+
+*/
+
+/* *************************************************/
+/* functions for debugging/testing purposes only */
+/* *************************************************/
+
+/**
+ * Returns a pointer to a newly allocated chunk of 'size' bytes, aligned
+ * to the system page size.
+ * Memory allocated by this routine may be protected by calling function
+ * picopal_mrp_protect().
+ */
+void *picopal_mpr_alloc(picopal_objsize_t size);
+
+/**
+ * Releases the chunk of memory pointed to by '*p'. 'p' must be previously
+ * allocated by a call to picopal_mpr_alloc().
+ */
+void picopal_mpr_free(void **p);
+
+#define PICOPAL_PROT_NONE 0 /* the memory cannot be accessed at all */
+#define PICOPAL_PROT_READ 1 /* the memory can be read */
+#define PICOPAL_PROT_WRITE 2 /* the memory can be written to */
+
+/**
+ * Specifies the desired protection 'prot' for the memory page(s) containing
+ * part or all of the interval [addr, addr+len-1]. If an access is disallowed
+ * by the protection given it, the program receives a SIGSEGV.
+ */
+pico_status_t picopal_mpr_protect(void *addr, picopal_objsize_t len, picopal_int16 prot);
+
+/* Fast, Compact Approximation of the Exponential Function */
+picopal_double picopal_quick_exp(const picopal_double y);
+
+/* *************************************************/
+/* types functions for time measurement */
+/* *************************************************/
+
+extern void picopal_get_timer(picopal_uint32 * sec, picopal_uint32 * usec);
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif /*PICOPAL_H_*/
diff --git a/lib/picopam.c b/lib/picopam.c
new file mode 100644
index 0000000..02e5cc2
--- /dev/null
+++ b/lib/picopam.c
@@ -0,0 +1,4803 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picopam.c
+ *
+ * Phonetic to Acoustic Mapping PU - Implementation
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#include "picodefs.h"
+#include "picoos.h"
+#include "picodbg.h"
+#include "picodata.h"
+#include "picopam.h"
+#include "picokdt.h"
+#include "picokpdf.h"
+#include "picoktab.h"
+#include "picokdbg.h"
+#include "picodsp.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+#define PICOPAM_IN_BUFF_SIZE PICODATA_BUFSIZE_PAM /*input buffer size for PAM */
+#define PICOPAM_OUT_PAM_SIZE PICODATA_BUFSIZE_PAM /*output buffer size for PAM*/
+#define PICOPAM_DT_NRLFZ 5 /* nr of lfz decision trees per phoneme */
+#define PICOPAM_DT_NRMGC 5 /* nr of mgc decision trees per phoneme */
+#define PICOPAM_NRSTPF 5 /* nr of states per phone */
+
+#define PICOPAM_COLLECT 0
+#define PICOPAM_SCHEDULE 1
+#define PICOPAM_IMMEDIATE 2
+#define PICOPAM_FORWARD 3
+#define PICOPAM_FORWARD_FORCE_TERM 4
+#define PICOPAM_PROCESS 5
+#define PICOPAM_PLAY 6
+#define PICOPAM_FEED 7
+
+#define PICOPAM_CONTINUE 100
+#define PICOPAM_GOTO_SCHEDULE 1
+#define PICOPAM_FLUSH_RECEIVED 6
+#define PICOPAM_GOTO_FEED 7
+#define PICOPAM_PRE_SYLL_ENDED 10
+
+#define PICOPAM_BREAK_ADD_SIZE 4 /*syllable feature vector increment dued to BREAK and SILENCE*/
+#define PICOPAM_VECT_SIZE 64+PICOPAM_BREAK_ADD_SIZE /*syllable feature vector size (bytes)*/
+#define PICOPAM_INVEC_SIZE 60 /*phone feature vector size */
+#define PICOPAM_MAX_SYLL_PER_SENT 100 /*maximum number of syllables per sentece*/
+#define PICOPAM_MAX_PH_PER_SENT 400 /*maximum number of phonemes per sentece*/
+#define PICOPAM_MAX_ITEM_PER_SENT 255 /*maximum number of attached items per sentence*/
+#define PICOPAM_MAX_ITEM_SIZE_PER_SENT 4096 /*maximum size of attached items per sentence*/
+
+#define PICOPAM_READY 20 /*PAM could start backward processing*/
+#define PICOPAM_MORE 21 /*PAM has still to collect */
+#define PICOPAM_NA 22 /*PAM has not to deal with this item*/
+#define PICOPAM_ERR 23 /*input item is not a valid item*/
+
+/*sentence types:cfr pam_map_sentence_type*/
+#define PICOPAM_DECLARATIVE 0
+#define PICOPAM_INTERROGATIVE 1
+#define PICOPAM_EXCLAMATIVE 2
+
+#define PICOPAM_T 0
+#define PICOPAM_P 1
+#define PICOPAM_p 2
+#define PICOPAM_Y 3
+
+#if 1
+#define PAM_PHR2_WITH_PR1 1 /*deal with PHR2 boundaries as with PHR1*/
+#else
+#define PAM_PHR2_WITH_PR3 1 /*deal with PHR2 boundaries as with PHR3*/
+#endif
+
+#define PICOPAM_DONT_CARE_VALUE 250 /*don't care value for tree printout */
+#define PICOPAM_DONT_CARE_VAL 10 /*don't care value for tree feeding */
+#define PICOPAM_PH_DONT_CARE_VAL 7 /*don't care value for tree feeding (phonetic)*/
+
+#define PICOPAM_MAX_STATES_PER_PHONE 5 /*number of states per phone */
+#define PICOPAM_STATE_SIZE_IN_ITEM 6 /*size of a state in frame item */
+#define PICOPAM_FRAME_ITEM_SIZE 4+PICOPAM_MAX_STATES_PER_PHONE*PICOPAM_STATE_SIZE_IN_ITEM
+
+#define PICOPAM_DIR_FORW 0 /*forward adapter processing*/
+#define PICOPAM_DIR_BACK 1 /*backward adapter processing*/
+#define PICOPAM_DIR_SIL 2 /*final silence attributes*/
+
+#define PICOPAM_SYLL_PAUSE 0 /*syllable but containing a pause phone*/
+#define PICOPAM_SYLL_SYLL 1 /*a real syllable with phonemes*/
+
+#define PICOPAM_EVENT_P_BOUND 0 /*primary boundary*/
+#define PICOPAM_EVENT_S_BOUND 1 /*secondary boundary*/
+#define PICOPAM_EVENT_W_BOUND 3 /*word boundary*/
+#define PICOPAM_EVENT_SYLL 4 /*syllable*/
+
+/* ----- CONSTANTS FOR BREAK COMMAND SUPPORT ----- */
+#define PICOPAM_PWIDX_SBEG 0
+#define PICOPAM_PWIDX_PHR1 1
+#define PICOPAM_PWIDX_PHR2 2
+#define PICOPAM_PWIDX_SEND 3
+#define PICOPAM_PWIDX_DEFA 4
+#define PICOPAM_PWIDX_SIZE 5
+
+/*----------------------------------------------------------------*/
+/*structure related to the feature vectors for feeding the trees */
+/*NOTE : the same data structure is used to manage the syllables */
+/* Using the first 8 fields for marking the boundaries */
+/* and using the last 4 bytes as follows */
+/* byte 61 : 1st attached non PAM item id(0=no item attached) */
+/* in the "sSyllItemOffs" data structure */
+/* byte 62 : last attached non PAM item id(0=no item attached)*/
+/* in the "sSyllItemOffs" data structure */
+/* byte 63..64 : offset of the start of the syllable in */
+/* the "sPhIds" data structure */
+typedef struct
+{
+ picopal_uint8 phoneV[PICOPAM_VECT_SIZE];
+} sFtVect, *pSftVect;
+
+/*----------------------------------------------------------
+ Name : pam_subobj
+ Function: subobject definition for the pam processing
+ Shortcut: pam
+ ---------------------------------------------------------*/
+typedef struct pam_subobj
+{
+ /*----------------------PU voice management------------------------------*/
+ /* picorsrc_Voice voice; */
+ /*----------------------PU state management------------------------------*/
+ picoos_uint8 procState; /* where to take up work at next processing step */
+ picoos_uint8 retState; /* where to go back from feed state at next p.s. */
+ picoos_uint8 needMoreInput; /* more data necessary to start processing */
+ /*----------------------PU input management------------------------------*/
+ picoos_uint8 inBuf[PICOPAM_IN_BUFF_SIZE]; /* internal input buffer */
+ picoos_uint16 inBufSize; /* actually allocated size */
+ picoos_uint16 inReadPos, inWritePos; /* next pos to read/write from/to inBuf*/
+ /*----------------------PU output management-----------------------------*/
+ picoos_uint8 outBuf[PICOPAM_OUT_PAM_SIZE]; /* internal output buffer */
+ picoos_uint16 outBufSize; /* actually allocated size */
+ picoos_uint16 outReadPos, outWritePos; /* next pos to read/write from/to outBuf*/
+ /*---------------------- adapter working buffers --------------------*/
+ picoos_uint8 *sPhFeats; /*feature vector for a single phone */
+ sFtVect *sSyllFeats; /*Syllable feature vector set for the
+ full sentence */
+ picoos_uint8 *sPhIds; /*phone ids for the full sentence */
+ picoos_uint8 *sSyllItems; /*items attached to the syllable */
+ picoos_int16 *sSyllItemOffs;/*offset of items attached to the syllable*/
+ /*---------------------- adapter general variables ---------------------*/
+ picoos_int16 nTotalPhonemes; /*number of phonemes in the sentence*/
+ picoos_int16 nCurrPhoneme; /*current phoneme in the sentence */
+ picoos_int16 nSyllPhoneme; /*current phoneme in the syllable */
+ picoos_int16 nCurrSyllable; /*current syllable in the sentence */
+ picoos_int16 nTotalSyllables; /*number of syllables in the sentence -> J1*/
+ picoos_uint8 nLastAttachedItemId;/*last attached item id*/
+ picoos_uint8 nCurrAttachedItem; /*current attached item*/
+ picoos_int16 nAttachedItemsSize; /*total size of the attached items*/
+ picoos_uint8 sType; /*Sentence type*/
+ picoos_uint8 pType; /*Phrase type*/
+ picoos_single pMod; /*pitch modifier*/
+ picoos_single dMod; /*Duration modifier*/
+ picoos_single dRest; /*Duration modifier rest*/
+ /*---------------------- adapter specific component variables ----------*/
+ picoos_uint8 a3_overall_syllable; /* A3 */
+ picoos_uint8 a3_primary_phrase_syllable;
+ picoos_uint8 b4_b5_syllable; /* B4,B5 */
+ picoos_uint8 b6_b7_syllable; /* B6,B7 */
+ picoos_uint8 b6_b7_state;
+ picoos_uint8 b8_b9_stressed_syllable; /* B8,B9 */
+ picoos_uint8 b10_b11_accented_syllable; /* B10,B11 */
+ picoos_uint8 b12_b13_syllable; /* B12,B13 */
+ picoos_uint8 b12_b13_state;
+ picoos_uint8 b14_b15_syllable; /* B14,B15 */
+ picoos_uint8 b14_b15_state;
+ picoos_uint8 b17_b19_syllable; /* B17,B19 */
+ picoos_uint8 b17_b19_state;
+ picoos_uint8 b18_b20_b21_syllable; /* B18,B20,B21 */
+ picoos_uint8 b18_b20_b21_state;
+ picoos_uint8 c3_overall_syllable; /* C3 */
+ picoos_uint8 c3_primary_phrase_syllable;
+ picoos_uint8 d2_syllable_in_word; /* D2 */
+ picoos_uint8 d2_prev_syllable_in_word;
+ picoos_uint8 d2_current_primary_phrase_word;
+ picoos_int8 e1_syllable_word_start; /* E1 */
+ picoos_int8 e1_syllable_word_end;
+ picoos_uint8 e1_content;
+ picoos_int8 e2_syllable_word_start; /* E2 */
+ picoos_int8 e2_syllable_word_end;
+ picoos_uint8 e3_e4_word; /* E3,E4 */
+ picoos_uint8 e3_e4_state;
+ picoos_uint8 e5_e6_content_word; /* E5,E6 */
+ picoos_uint8 e5_e6_content;
+ picoos_uint8 e7_e8_word; /* E7,E8 */
+ picoos_uint8 e7_e8_content;
+ picoos_uint8 e7_e8_state;
+ picoos_uint8 e9_e11_word; /* E9,E11 */
+ picoos_uint8 e9_e11_saw_word;
+ picoos_uint8 e9_e11_state;
+ picoos_uint8 e10_e12_e13_word; /* E10,E12,E13 */
+ picoos_uint8 e10_e12_e13_state;
+ picoos_uint8 e10_e12_e13_saw_word;
+ picoos_uint8 f2_overall_word; /* F2 */
+ picoos_uint8 f2_word_syllable;
+ picoos_uint8 f2_next_word_syllable;
+ picoos_uint8 f2_current_primary_phrase_word;
+ picoos_int8 g1_current_secondary_phrase_syllable; /*G1 */
+ picoos_int8 g1_current_syllable;
+ picoos_int8 g2_current_secondary_phrase_word; /*G2 */
+ picoos_int8 g2_current_word;
+ picoos_uint8 h1_current_secondary_phrase_syll; /*H1 */
+ picoos_uint8 h2_current_secondary_phrase_word; /*H2 */
+ picoos_uint8 h3_h4_current_secondary_phrase_word; /*H3,H4 */
+ picoos_uint8 h5_current_phrase_type; /*H5 */
+
+ picoos_uint8 h5_syllable; /* H5 */
+ picoos_uint8 h5_state;
+
+ picoos_uint8 i1_secondary_phrase_syllable; /*I1 */
+ picoos_uint8 i1_next_secondary_phrase_syllable;
+ picoos_uint8 i2_secondary_phrase_word; /*I2 */
+ picoos_uint8 i2_next_secondary_phrase_word;
+ picoos_uint8 j1_utterance_syllable; /*J1 */
+ picoos_uint8 j2_utterance_word; /*J2 */
+ picoos_uint8 j3_utterance_sec_phrases; /*J3 */
+ /*---------------------- constant data -------------------*/
+ picoos_uint16 sil_weights[PICOPAM_PWIDX_SIZE][PICOPAM_MAX_STATES_PER_PHONE];
+ /*---------------------- LINGWARE related data -------------------*/
+ picokdt_DtPAM dtdur; /* dtdur knowledge base */
+ picokdt_DtPAM dtlfz[PICOPAM_DT_NRLFZ]; /* dtlfz knowledge bases */
+ picokdt_DtPAM dtmgc[PICOPAM_DT_NRMGC]; /* dtmgc knowledge bases */
+ /*---------------------- Pdfs related data -------------------*/
+ picokpdf_PdfDUR pdfdur; /* pdfdur knowledge base */
+ picokpdf_PdfMUL pdflfz; /* pdflfz knowledge base */
+ /*---------------------- Tree traversal related data -------------------*/
+ picoos_uint16 durIndex;
+ picoos_uint8 numFramesState[PICOPAM_DT_NRLFZ];
+ picoos_uint16 lf0Index[PICOPAM_DT_NRLFZ];
+ picoos_uint16 mgcIndex[PICOPAM_DT_NRMGC];
+ /*---------------------- temps for updating the feature vector ---------*/
+ picoos_uint16 phonDur;
+ picoos_single phonF0[PICOPAM_DT_NRLFZ];
+ /*---------------------- Phones related data -------------------*/
+ picoktab_Phones tabphones;
+} pam_subobj_t;
+
+
+/* ----- CONSTANTS FOR FEATURE VECTOR BUILDING (NOT PREFIXED WITH "PICOPAM_" FOR BREVITY) ----- */
+#define P1 0 /*field 1 of the input vector*/
+#define P2 1
+#define P3 2
+#define P4 3
+#define P5 4
+#define P6 5
+#define P7 6
+#define bnd 6 /*boundary type item associated to the syllable = P7 */
+#define P8 7
+#define A3 8
+#define B1 9
+#define B2 10
+#define B3 11
+#define B4 12
+#define B5 13
+#define B6 14
+#define B7 15
+#define B8 16
+#define B9 17
+#define B10 18
+#define B11 19
+#define B12 20
+#define B13 21
+#define B14 22
+#define B15 23
+#define B16 24
+#define B17 25
+#define B18 26
+#define B19 27
+#define B20 28
+#define B21 29
+#define C3 30
+#define D2 31
+#define E1 32
+#define E2 33
+#define E3 34
+#define E4 35
+#define E5 36
+#define E6 37
+#define E7 38
+#define E8 39
+#define E9 40
+#define E10 41
+#define E11 42
+#define E12 43
+#define E13 44
+#define F2 45
+#define G1 46
+#define G2 47
+#define H1 48
+#define H2 49
+#define H3 50
+#define H4 51
+#define H5 52
+#define I1 53
+#define I2 54
+#define J1 55
+#define J2 56
+#define J3 57
+#define DUR 58 /*duration component*/
+#define F0 59 /*F0 component*/
+#define ITM 60 /*Item Offset into sSyllItems item list*/
+#define itm 61 /*second byte of the Item Offset */
+#define FID 62 /*Phoneme offset in the sPhIds phoneme list*/
+#define fid 63 /*second byte of the Phoneme offset */
+#define Min 64 /*offset to min syllable duration (uint 16,pauses)*/
+#define Max 66 /*offset to max syllable duration (uint 16,pauses)*/
+/* -------------------------------------------------------------------
+ PAM feature vector indices position changes,
+ ------------------------------------------------------------------- */
+#define T_B1 8
+#define T_B2 9
+#define T_B3 10
+#define T_B4 11
+#define T_B5 12
+#define T_B6 13
+#define T_B7 14
+#define T_B8 15
+#define T_B9 16
+#define T_B10 17
+#define T_B11 18
+#define T_B12 19
+#define T_B13 20
+#define T_B14 21
+#define T_B15 22
+#define T_B16 23
+#define T_B17 24
+#define T_B18 25
+#define T_B19 26
+#define T_B20 27
+#define T_B21 28
+#define T_E1 29
+#define T_E2 30
+#define T_E3 31
+#define T_E4 32
+#define T_E5 33
+#define T_E6 34
+#define T_E7 35
+#define T_E8 36
+#define T_E9 37
+#define T_E10 38
+#define T_E11 39
+#define T_E12 40
+#define T_E13 41
+#define T_A3 42
+#define T_C3 43
+#define T_D2 44
+#define T_F2 45
+#define T_G1 46
+#define T_I1 47
+#define T_G2 48
+#define T_I2 49
+#define T_H1 50
+#define T_H2 51
+#define T_H3 52
+#define T_H4 53
+#define T_H5 54
+
+/*------------------------------------------------------------------
+ Service routines :
+ ------------------------------------------------------------------*/
+static pico_status_t pam_initialize(register picodata_ProcessingUnit this);
+static pico_status_t pam_terminate(register picodata_ProcessingUnit this);
+static pico_status_t pam_allocate(picoos_MemoryManager mm, pam_subobj_t *pam);
+static void pam_deallocate(picoos_MemoryManager mm, pam_subobj_t *pam);
+static pico_status_t pam_subobj_deallocate(register picodata_ProcessingUnit this,
+ picoos_MemoryManager mm);
+/*------------------------------------------------------------------
+ Processing routines :
+ ------------------------------------------------------------------*/
+static picodata_step_result_t pam_step(register picodata_ProcessingUnit this,
+ picoos_int16 mode, picoos_uint16 * numBytesOutput);
+static pico_status_t pam_deal_with(const picoos_uint8 *item);
+/*Utility*/
+static picoos_uint8 pam_get_vowel_name(register picodata_ProcessingUnit this,
+ picoos_uint8 *item, picoos_uint8 *pos);
+static picoos_uint8 pam_get_pause_id(register picodata_ProcessingUnit this);
+
+static picoos_uint8 pam_map_sentence_type(picoos_uint8 iteminfo1,
+ picoos_uint8 iteminfo2);
+static picoos_uint8 pam_map_phrase_type(picoos_uint8 iteminfo1,
+ picoos_uint8 iteminfo2);
+
+/*Adapter*/
+static pico_status_t pam_reset_processors(register picodata_ProcessingUnit this);
+static pico_status_t pam_reset_processors_back(
+ register picodata_ProcessingUnit this);
+static pico_status_t pam_create_syllable(register picodata_ProcessingUnit this,
+ picoos_uint8 syllType, picoos_uint8 *sContent, picoos_uint8 sentType,
+ picoos_uint8 phType, picoos_uint8 uBoundType, picoos_uint16 uMin,
+ picoos_uint16 uMax);
+static pico_status_t pam_process_event_feature(
+ register picodata_ProcessingUnit this, picoos_uint8 nFeat,
+ picoos_uint8 event_type, picoos_uint8 direction);
+static pico_status_t pam_process_event(register picodata_ProcessingUnit this,
+ picoos_uint8 event_type, picoos_uint8 direction);
+static pico_status_t pam_adapter_forward_step(
+ register picodata_ProcessingUnit this, picoos_uint8 *itemBase);
+static pico_status_t pam_adapter_backward_step(
+ register picodata_ProcessingUnit this);
+static pico_status_t pam_do_pause(register picodata_ProcessingUnit this);
+static pico_status_t pam_adapter_do_pauses(register picodata_ProcessingUnit this);
+/*-------------- tree traversal ---------------------------------------*/
+static pico_status_t pam_expand_vector(register picodata_ProcessingUnit this);
+static picoos_uint8 pam_do_tree(register picodata_ProcessingUnit this,
+ const picokdt_DtPAM dtpam, const picoos_uint8 *invec,
+ const picoos_uint8 inveclen, picokdt_classify_result_t *dtres);
+static pico_status_t pam_get_f0(register picodata_ProcessingUnit this,
+ picoos_uint16 *lf0Index, picoos_uint8 nState, picoos_single *phonF0);
+static pico_status_t pam_get_duration(register picodata_ProcessingUnit this,
+ picoos_uint16 durIndex, picoos_uint16 *phonDur,
+ picoos_uint8 *numFramesState);
+static pico_status_t pam_update_vector(register picodata_ProcessingUnit this);
+/*-------------- FINAL ITEM FEEDING -----------------------------------------*/
+static pico_status_t pam_put_item(register picodata_ProcessingUnit this,
+ picoos_uint8 *outBuff, picoos_uint16 outWritePos,
+ picoos_uint8 *bytesWr);
+
+static pico_status_t pam_put_term(picoos_uint8 *outBuff,
+ picoos_uint16 outWritePos, picoos_uint8 *bytesWr);
+
+static pico_status_t is_pam_command(const picoos_uint8 *qItem);
+
+static void get_default_boundary_limit(picoos_uint8 uBoundType,
+ picoos_uint16 *uMinDur, picoos_uint16 *uMaxDur);
+
+/* -------------------------------------------------------------
+ * Pico System functions
+ * -------------------------------------------------------------
+ */
+
+/**
+ * allocation for PAM memory on pam PU
+ * @param mm : handle to engine memory manager
+ * @param pam : handle to a pam struct
+ * @return PICO_OK : allocation successful
+ * @return PICO_ERR_OTHER : allocation errors
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_allocate(picoos_MemoryManager mm, pam_subobj_t *pam)
+{
+ picoos_uint8 *data;
+ picoos_int16 *dataI;
+
+ pam->sSyllFeats = NULL;
+ pam->sPhIds = NULL;
+ pam->sPhFeats = NULL;
+ pam->sSyllItems = NULL;
+ pam->sSyllItemOffs = NULL;
+
+ /*-----------------------------------------------------------------
+ * PAM Local buffers ALLOCATION
+ ------------------------------------------------------------------*/
+ /*PAM Local buffers*/
+ data = (picopal_uint8 *) picoos_allocate(mm, sizeof(sFtVect)
+ * PICOPAM_MAX_SYLL_PER_SENT);
+ if (data == NULL)
+ return PICO_ERR_OTHER;
+ pam->sSyllFeats = (sFtVect*) data;
+
+ data = (picopal_uint8 *) picoos_allocate(mm, sizeof(picopal_uint8)
+ * PICOPAM_MAX_PH_PER_SENT);
+ if (data == NULL) {
+ pam_deallocate(mm, pam);
+ return PICO_ERR_OTHER;
+ }
+ pam->sPhIds = (picopal_uint8*) data;
+
+ data = (picopal_uint8 *) picoos_allocate(mm, sizeof(picopal_uint8)
+ * PICOPAM_VECT_SIZE);
+ if (data == NULL) {
+ pam_deallocate(mm, pam);
+ return PICO_ERR_OTHER;
+ }
+ pam->sPhFeats = (picopal_uint8*) data;
+
+ data = (picopal_uint8 *) picoos_allocate(mm, sizeof(picopal_uint8)
+ * PICOPAM_MAX_ITEM_SIZE_PER_SENT);
+ if (data == NULL) {
+ pam_deallocate(mm, pam);
+ return PICO_ERR_OTHER;
+ }
+ pam->sSyllItems = (picopal_uint8*) data;
+
+ dataI = (picoos_int16 *) picoos_allocate(mm, sizeof(picoos_int16)
+ * PICOPAM_MAX_ITEM_PER_SENT);
+ if (data == NULL) {
+ pam_deallocate(mm, pam);
+ return PICO_ERR_OTHER;
+ }
+ pam->sSyllItemOffs = (picoos_int16*) dataI;
+
+ return PICO_OK;
+}/*pam_allocate*/
+
+/**
+ * frees allocation for DSP memory on PAM PU
+ * @param mm : memory manager
+ * @param pam : pam PU internal sub-object
+ * @return void
+ * @remarks modified and inserted in sub obj removal PP 15.09.08
+ * @callgraph
+ * @callergraph
+ */
+static void pam_deallocate(picoos_MemoryManager mm, pam_subobj_t *pam)
+{
+ /*-----------------------------------------------------------------
+ * Memory de-allocations
+ * ------------------------------------------------------------------*/
+ if (pam->sSyllFeats != NULL)
+ picoos_deallocate(mm, (void *) &pam->sSyllFeats);
+ if (pam->sPhIds != NULL)
+ picoos_deallocate(mm, (void *) &pam->sPhIds);
+ if (pam->sPhFeats != NULL)
+ picoos_deallocate(mm, (void *) &pam->sPhFeats);
+ if (pam->sSyllItems != NULL)
+ picoos_deallocate(mm, (void *) &pam->sSyllItems);
+ if (pam->sSyllItemOffs != NULL)
+ picoos_deallocate(mm, (void *) &pam->sSyllItemOffs);
+
+}/*pam_deallocate*/
+
+/**
+ * initialization of a pam PU
+ * @param this : handle to a PU struct
+ * @return PICO_OK : init OK
+ * @return PICO_ERR_OTHER : error on getting pkbs addresses
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_initialize(register picodata_ProcessingUnit this)
+{
+ pico_status_t nI, nJ;
+ pam_subobj_t *pam;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+ pam->inBufSize = PICOPAM_IN_BUFF_SIZE;
+ pam->outBufSize = PICOPAM_OUT_PAM_SIZE;
+ pam->inReadPos = 0;
+ pam->inWritePos = 0;
+ pam->outReadPos = 0;
+ pam->outWritePos = 0;
+ pam->needMoreInput = 0;
+ pam->procState = 0;
+
+ /*-----------------------------------------------------------------
+ * MANAGE INTERNAL INITIALIZATION
+ ------------------------------------------------------------------*/
+ /*init the syllable structure*/
+ for (nI = 0; nI < PICOPAM_MAX_SYLL_PER_SENT; nI++)
+ for (nJ = 0; nJ < PICOPAM_VECT_SIZE; nJ++)
+ pam->sSyllFeats[nI].phoneV[nJ] = 0;
+
+ for (nI = 0; nI < PICOPAM_MAX_PH_PER_SENT; nI++)
+ pam->sPhIds[nI] = 0;
+
+ for (nI = 0; nI < PICOPAM_VECT_SIZE; nI++)
+ pam->sPhFeats[nI] = 0;
+
+ for (nI = 0; nI < PICOPAM_MAX_ITEM_SIZE_PER_SENT; nI++)
+ pam->sSyllItems[nI] = 0;
+
+ for (nI = 0; nI < PICOPAM_MAX_ITEM_PER_SENT; nI++)
+ pam->sSyllItemOffs[nI] = 0;
+
+ /*Other variables*/
+ pam_reset_processors(this);
+ pam->nLastAttachedItemId = pam->nCurrAttachedItem = 0;
+ pam->nAttachedItemsSize = 0;
+
+ /*pitch and duration modifiers*/
+ pam->pMod = 1.0f;
+ pam->dMod = 1.0f;
+ pam->dRest = 0.0f;
+
+
+ /* constant tables */
+ {
+ picoos_uint8 i, j;
+ picoos_uint16 tmp_weights[PICOPAM_PWIDX_SIZE][PICOPAM_MAX_STATES_PER_PHONE] = {
+ {10, 10, 10, 10, 1 }, /*SBEG*/
+ { 1, 4, 8, 4, 1 }, /*PHR1*/
+ { 1, 4, 8, 4, 1 }, /*PHR2*/
+ { 1, 10, 10, 10, 10 },/*SEND*/
+ { 1, 1, 1, 1, 1 } /*DEFAULT*/
+ };
+ for (i = 0; i < PICOPAM_PWIDX_SIZE; i++) {
+ for (j = 0; j < PICOPAM_PWIDX_SIZE; j++) {
+ pam->sil_weights[j][j] = tmp_weights[i][j];
+ }
+ }
+ }
+/*-----------------------------------------------------------------
+ * MANAGE LINGWARE INITIALIZATION IF NEEDED
+ ------------------------------------------------------------------*/
+ /* kb dtdur */
+ pam->dtdur = picokdt_getDtPAM(this->voice->kbArray[PICOKNOW_KBID_DT_DUR]);
+ if (pam->dtdur == NULL) {
+ picoos_emRaiseException(this->common->em, PICO_EXC_KB_MISSING, NULL,
+ NULL);
+ return PICO_ERR_OTHER;
+ }PICODBG_DEBUG(("got dtdur"));
+
+ /* kb dtlfz* */
+ pam->dtlfz[0] = picokdt_getDtPAM(
+ this->voice->kbArray[PICOKNOW_KBID_DT_LFZ1]);
+ pam->dtlfz[1] = picokdt_getDtPAM(
+ this->voice->kbArray[PICOKNOW_KBID_DT_LFZ2]);
+ pam->dtlfz[2] = picokdt_getDtPAM(
+ this->voice->kbArray[PICOKNOW_KBID_DT_LFZ3]);
+ pam->dtlfz[3] = picokdt_getDtPAM(
+ this->voice->kbArray[PICOKNOW_KBID_DT_LFZ4]);
+ pam->dtlfz[4] = picokdt_getDtPAM(
+ this->voice->kbArray[PICOKNOW_KBID_DT_LFZ5]);
+ for (nI = 0; nI < PICOPAM_DT_NRLFZ; nI++) {
+ if (pam->dtlfz[nI] == NULL) {
+ picoos_emRaiseException(this->common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ return PICO_ERR_OTHER;
+ }PICODBG_DEBUG(("got dtlfz%d", nI+1));
+ }
+
+ /* kb dtmgc* */
+ pam->dtmgc[0] = picokdt_getDtPAM(
+ this->voice->kbArray[PICOKNOW_KBID_DT_MGC1]);
+ pam->dtmgc[1] = picokdt_getDtPAM(
+ this->voice->kbArray[PICOKNOW_KBID_DT_MGC2]);
+ pam->dtmgc[2] = picokdt_getDtPAM(
+ this->voice->kbArray[PICOKNOW_KBID_DT_MGC3]);
+ pam->dtmgc[3] = picokdt_getDtPAM(
+ this->voice->kbArray[PICOKNOW_KBID_DT_MGC4]);
+ pam->dtmgc[4] = picokdt_getDtPAM(
+ this->voice->kbArray[PICOKNOW_KBID_DT_MGC5]);
+ for (nI = 0; nI < PICOPAM_DT_NRMGC; nI++) {
+ if (pam->dtmgc[nI] == NULL) {
+ picoos_emRaiseException(this->common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ return PICO_ERR_OTHER;
+ }PICODBG_DEBUG(("got dtmgc%d", nI+1));
+ }
+
+ /* kb pdfdur* */
+ pam->pdfdur = picokpdf_getPdfDUR(
+ this->voice->kbArray[PICOKNOW_KBID_PDF_DUR]);
+ if (pam->pdfdur == NULL) {
+ picoos_emRaiseException(this->common->em, PICO_EXC_KB_MISSING, NULL,
+ NULL);
+ return PICO_ERR_OTHER;
+ }PICODBG_DEBUG(("got pdfdur"));
+
+ /* kb pdflfz* */
+ pam->pdflfz = picokpdf_getPdfMUL(
+ this->voice->kbArray[PICOKNOW_KBID_PDF_LFZ]);
+ if (pam->pdflfz == NULL) {
+ picoos_emRaiseException(this->common->em, PICO_EXC_KB_MISSING, NULL,
+ NULL);
+ return PICO_ERR_OTHER;
+ }PICODBG_DEBUG(("got pdflfz"));
+
+ /* kb tabphones */
+ pam->tabphones = picoktab_getPhones(
+ this->voice->kbArray[PICOKNOW_KBID_TAB_PHONES]);
+ if (pam->tabphones == NULL) {
+ picoos_emRaiseException(this->common->em, PICO_EXC_KB_MISSING, NULL,
+ NULL);
+ return PICO_ERR_OTHER;
+ }PICODBG_DEBUG(("got tabphones"));
+
+ return PICO_OK;
+}/*pam_initialize*/
+
+/**
+ * termination of a pam PU
+ * @param this : handle to a pam PU struct
+ * @return PICO_OK
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_terminate(register picodata_ProcessingUnit this)
+{
+
+ pam_subobj_t *pam;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+
+ return PICO_OK;
+}/*pam_terminate*/
+
+/**
+ * deallocaton of a pam PU
+ * @param this : handle to a pam PU struct
+ * @param mm : engine memory manager
+ * @return PICO_OK
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_subobj_deallocate(register picodata_ProcessingUnit this,
+ picoos_MemoryManager mm)
+{
+
+ pam_subobj_t* pam;
+
+ if (NULL != this) {
+ pam = (pam_subobj_t *) this->subObj;
+ mm = mm; /* avoid warning "var not used in this function"*/
+ /*-----------------------------------------------------------------
+ * Memory de-allocations
+ * ------------------------------------------------------------------*/
+ if (pam->sSyllFeats != NULL) {
+ picoos_deallocate(this->common->mm, (void *) &pam->sSyllFeats);
+ }
+ if (pam->sPhIds != NULL) {
+ picoos_deallocate(this->common->mm, (void *) &pam->sPhIds);
+ }
+ if (pam->sPhFeats != NULL) {
+ picoos_deallocate(this->common->mm, (void *) &pam->sPhFeats);
+ }
+ if (pam->sSyllItems != NULL) {
+ picoos_deallocate(this->common->mm, (void *) &pam->sSyllItems);
+ }
+ if (pam->sSyllItemOffs != NULL) {
+ picoos_deallocate(this->common->mm, (void *) &pam->sSyllItemOffs);
+ }
+ picoos_deallocate(this->common->mm, (void *) &this->subObj);
+ }
+
+ return PICO_OK;
+}/*pam_subobj_deallocate*/
+
+/**
+ * creates a new pam processing unit
+ * @param mm : engine memory manager
+ * @param common : engine common object pointer
+ * @param cbIn : pointer to input buffer
+ * @param cbOut : pointer to output buffer
+ * @param voice : pointer to voice structure
+ * @return this : pam PU handle if success
+ * @return NULL : if error
+ * @callgraph
+ * @callergraph
+ */
+picodata_ProcessingUnit picopam_newPamUnit(picoos_MemoryManager mm,
+ picoos_Common common, picodata_CharBuffer cbIn,
+ picodata_CharBuffer cbOut, picorsrc_Voice voice)
+{
+
+ register pam_subobj_t * pam;
+
+ picodata_ProcessingUnit this = picodata_newProcessingUnit(mm, common, cbIn,
+ cbOut, voice);
+ if (this == NULL) {
+ return NULL;
+ }
+ this->initialize = pam_initialize;
+
+ PICODBG_DEBUG(("picotok_newPamUnit -- set this->step to pam_step"));
+
+ this->step = pam_step;
+ this->terminate = pam_terminate;
+ this->subDeallocate = pam_subobj_deallocate;
+ this->subObj = picoos_allocate(mm, sizeof(pam_subobj_t));
+ if (this->subObj == NULL) {
+ PICODBG_ERROR(("Error in Pam Object allocation"));
+ picoos_deallocate(mm, (void*) &this);
+ return NULL;
+ };
+
+ /*-----------------------------------------------------------------
+ * Allocate internal memory for PAM (only at PU creation time)
+ * ------------------------------------------------------------------*/
+ pam = (pam_subobj_t *) this->subObj;
+ if (PICO_OK != pam_allocate(mm, pam)) {
+ PICODBG_ERROR(("Error in Pam buffers Allocation"));
+ picoos_deallocate(mm, (void *) &this->subObj);
+ picoos_deallocate(mm, (void *) &this);
+ return NULL;
+ }
+
+ /*-----------------------------------------------------------------
+ * Initialize memory for PAM (this may be re-used elsewhere, e.g.Reset)
+ * ------------------------------------------------------------------*/
+ if (PICO_OK != pam_initialize(this)) {
+ PICODBG_ERROR(("problem initializing the pam sub-object"));
+ }
+ return this;
+}/*picopam_newPamUnit*/
+
+/*-------------------------------------------------------------------------------
+ PROCESSING AND INTERNAL FUNCTIONS
+ --------------------------------------------------------------------------------*/
+
+/**
+ * initializes default duration limits for boundary items
+ * @param uBoundType : type of input boundary type
+ * @param *uMinDur, *uMaxDur : addresses of values to initialize
+ * @return void
+ * @remarks so far initializes to 0 both values; this will leave the values given by tree prediction
+ * @callgraph
+ * @callergraph
+ */
+static void get_default_boundary_limit(picoos_uint8 uBoundType,
+ picoos_uint16 *uMinDur, picoos_uint16 *uMaxDur)
+{
+ switch (uBoundType) {
+ case PICODATA_ITEMINFO1_BOUND_SBEG:
+ *uMinDur = 0;
+ *uMaxDur = 20;
+ break;
+ case PICODATA_ITEMINFO1_BOUND_SEND:
+ *uMinDur = 550;
+ *uMaxDur = 650;
+ break;
+ case PICODATA_ITEMINFO1_BOUND_TERM:
+ *uMinDur = 0;
+ *uMaxDur = 0;
+ break;
+ case PICODATA_ITEMINFO1_BOUND_PHR0:
+ *uMinDur = 0;
+ *uMaxDur = 0;
+ break;
+ case PICODATA_ITEMINFO1_BOUND_PHR1:
+ *uMinDur = 275;
+ *uMaxDur = 325;
+ break;
+ case PICODATA_ITEMINFO1_BOUND_PHR2:
+ *uMinDur = 4;
+ *uMaxDur = 60;
+ break;
+ case PICODATA_ITEMINFO1_BOUND_PHR3:
+ *uMinDur = 0;
+ *uMaxDur = 0;
+ break;
+ default:
+ break;
+ }
+
+}/*get_default_boundary_limit*/
+
+/**
+ * checks if "neededSize" is available on "nCurrPhoneme"
+ * @param pam : pam subobj
+ * @param neededSize : the requested size
+ * @return PICO_OK : size is available
+ * @return !=PICO_OK : size not available
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t check_phones_size(pam_subobj_t *pam,
+ picoos_int16 neededSize)
+{
+ if ((pam->nCurrPhoneme + neededSize) > PICOPAM_MAX_PH_PER_SENT - 1) {
+ return PICO_ERR_OTHER;
+ }
+ return PICO_OK;
+}/*check_phones_size*/
+
+/**
+ * checks if neededSize is available on "nCurrSyllable"
+ * @param pam : pam subobj
+ * @param neededSize : the requested size
+ * @return PICO_OK : size is available
+ * @return !=PICO_OK : size not available
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t check_syllables_size(pam_subobj_t *pam,
+ picoos_int16 neededSize)
+{
+ if ((pam->nCurrSyllable + neededSize) > PICOPAM_MAX_SYLL_PER_SENT - 1) {
+ return PICO_ERR_OTHER;
+ }
+ return PICO_OK;
+}/*check_syllables_size*/
+
+/**
+ * verifies that local storage has enough space to receive 1 item
+ * @param this : pointer to current PU struct
+ * @param item : pointer to current item head
+ * @return TRUE : resource limits would be reached during processing of input item
+ * @return FALSE : item could be processed normally
+ * @remarks item pointed to by *item should be already valid
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pamCheckResourceLimits(
+ register picodata_ProcessingUnit this, const picoos_uint8 *item)
+{
+ register pam_subobj_t * pam;
+ picodata_itemhead_t head;
+ pico_status_t sResult;
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+ sResult = TRUE; /*default : resource limits reached*/
+ head.type = item[0];
+ head.info1 = item[1];
+ head.info2 = item[2];
+ head.len = item[3];
+
+ switch (head.type) {
+ /*commands that generate syllables/phonemes*/
+ case PICODATA_ITEM_SYLLPHON:
+ if (pam->nCurrSyllable >= PICOPAM_MAX_SYLL_PER_SENT - 2) {
+ return sResult; /*no room for more syllables*/
+ }
+ if ((pam->nCurrPhoneme + head.len) >= PICOPAM_MAX_PH_PER_SENT - 2) {
+ return sResult; /*no room for more phoneme*/
+ }
+ break;
+ case PICODATA_ITEM_BOUND:
+ if ((head.info1 == PICODATA_ITEMINFO1_BOUND_SBEG) || (head.info1
+ == PICODATA_ITEMINFO1_BOUND_SEND) || (head.info1
+ == PICODATA_ITEMINFO1_BOUND_TERM) || (head.info1
+ == PICODATA_ITEMINFO1_BOUND_PHR1)
+#ifdef PAM_PHR2_WITH_PR1
+ || (head.info1 == PICODATA_ITEMINFO1_BOUND_PHR2)
+#endif
+ ) {
+
+ if (pam->nCurrSyllable >= PICOPAM_MAX_SYLL_PER_SENT - 2) {
+ return sResult; /*no room for more syllables*/
+ }
+ if ((pam->nCurrPhoneme + 1) >= PICOPAM_MAX_PH_PER_SENT - 2) {
+ return sResult; /*no room for more phoneme*/
+ }
+ }
+ break;
+
+ default:
+ /*all other commands has to be queued*/
+ if ((pam->nAttachedItemsSize + head.len)
+ >= PICOPAM_MAX_ITEM_SIZE_PER_SENT - 1) {
+ return sResult; /*no room for more items*/
+ }
+ break;
+ }
+ return FALSE; /*no resource limits apply to current item*/
+} /*pamCheckResourceLimits*/
+
+/**
+ * selects items to be sent to next PU immedately
+ * @param this : pointer to current PU struct
+ * @param item : pointer to current item head
+ * @return TRUE : item should be passed on next PU NOW
+ * @return FALSE : item should not be passed on next PU now but should be processed
+ * @remarks item pointed to by *item should be already valid
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_check_immediate(register picodata_ProcessingUnit this,
+ const picoos_uint8 *item)
+{
+ register pam_subobj_t * pam;
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+
+ if (pam->nCurrSyllable <= -1) {
+ if (item[0] == PICODATA_ITEM_SYLLPHON)
+ return FALSE;
+ if ((item[0] == PICODATA_ITEM_BOUND) && (item[1]
+ == PICODATA_ITEMINFO1_BOUND_SBEG))
+ return FALSE;
+ if (is_pam_command((picoos_uint8 *) item) == TRUE)
+ return FALSE;
+ return TRUE; /*no need to process data : send it*/
+ }
+ return FALSE; /*syllable struct not void : do standard processing*/
+
+} /*pam_check_immediate*/
+
+/**
+ * checks if the input item has to be queued in local storage for later resynch
+ * @param this : pointer to current PU struct
+ * @param item : pointer to current item head
+ * @return TRUE : item should be queued
+ * @return FALSE : item should not be queued
+ * @remarks item pointed to by *item should be already valid
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_hastobe_queued(register picodata_ProcessingUnit this,
+ const picoos_uint8 *item)
+{
+ register pam_subobj_t * pam;
+ picodata_itemhead_t head;
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+ head.type = item[0];
+ head.info1 = item[1];
+
+ switch (head.type) {
+ /*commands that generate syllables/phonemes*/
+ case PICODATA_ITEM_SYLLPHON:
+ return FALSE; /*no queue needed*/
+ break;
+ case PICODATA_ITEM_BOUND:
+ if ((head.info1 == PICODATA_ITEMINFO1_BOUND_PHR3)
+#ifdef PAM_PHR2_WITH_PR3
+ ||(head.info1==PICODATA_ITEMINFO1_BOUND_PHR2)
+#endif
+ || (head.info1 == PICODATA_ITEMINFO1_BOUND_PHR0)) {
+ return FALSE; /*no queue needed*/
+ }
+ break;
+
+ default:
+ /*all other items has to be queued*/
+ break;
+ }
+ return TRUE; /*item has to be queued*/
+} /*pam_hastobe_queued*/
+
+/**
+ * queue item in local storage for later resynch
+ * @param this : pointer to current PU struct
+ * @param item : pointer to current item head
+ * @return TRUE : item queued
+ * @return FALSE : item not queued because of errors
+ * @remarks item pointed to by *item should be already valid
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_queue(register picodata_ProcessingUnit this,
+ const picoos_uint8 *item)
+{
+ register pam_subobj_t * pam;
+ picodata_itemhead_t head;
+ picoos_uint8 nI;
+ pico_status_t sResult;
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+ sResult = TRUE; /*default : item queued*/
+ head.type = item[0];
+ head.info1 = item[1];
+ head.info2 = item[2];
+ head.len = item[3];
+
+ /*test condition on enough room to store current item in the "sSyllItems" area*/
+ if ((pam->nAttachedItemsSize + head.len + sizeof(picodata_itemhead_t))
+ >= PICOPAM_MAX_ITEM_SIZE_PER_SENT - 1) {
+ return FALSE; /*resource limit reached*/
+ }
+ /*store current offset*/
+ pam->sSyllItemOffs[pam->nLastAttachedItemId] = pam->nAttachedItemsSize;
+ /*store the item to the "sSyllItems" area*/
+ for (nI = 0; nI < (head.len + sizeof(picodata_itemhead_t)); nI++) {
+ pam->sSyllItems[pam->nAttachedItemsSize + nI] = item[nI];
+ }
+ /*increment the attached items area*/
+ pam->nAttachedItemsSize += nI;
+
+ /*increment id*/
+ pam->nLastAttachedItemId++;
+ /*set start(if not initialized) and end ids of queued items in sSyllFeats*/
+ if (pam->nCurrSyllable > -1) {
+ /*normal case : the item is attached to current syllable*/
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[ITM] == 0) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[ITM]
+ = pam->nLastAttachedItemId;
+ }
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[itm]
+ = pam->nLastAttachedItemId;
+ } else {
+ /*special case : an item is requested to be queued even if no
+ syllables has been assigned to the sentence structure :
+ -->> use syll 0*/
+ if (pam->sSyllFeats[0].phoneV[ITM] == 0) {
+ pam->sSyllFeats[0].phoneV[ITM] = pam->nLastAttachedItemId;
+ }
+ pam->sSyllFeats[0].phoneV[itm] = pam->nLastAttachedItemId;
+ }
+ return TRUE; /*item queued successfully*/
+} /*pam_queue*/
+
+/**
+ * selects items to be dealth with by the PU processing
+ * @param item : pointer to current item head
+ * @return TRUE : item should be processed
+ * @return FALSE : item should not be processed (maybe it ontains commands or items for other PUs)
+ * @remarks item pointed to by *item should be already valid
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_deal_with(const picoos_uint8 *item)
+{
+ picodata_itemhead_t head;
+ pico_status_t sResult;
+ sResult = FALSE;
+ head.type = item[0];
+ head.info1 = item[1];
+ head.info2 = item[2];
+ head.len = item[3];
+ switch (head.type) {
+ case PICODATA_ITEM_SYLLPHON:
+ case PICODATA_ITEM_BOUND:
+ sResult = TRUE;
+ break;
+ default:
+ break;
+ }
+ return sResult;
+} /*pam_deal_with*/
+
+/**
+ * returns true if more items has to be produced for current syllable
+ * @param this : Pam object pointer
+ * @return TRUE : item is to be produced
+ * @return FALSE : item is not to be produced
+ * @remarks item pointed to by *item should be already valid
+ * @callgraph
+ * @callergraph
+ */
+static picoos_uint8 pamHasToProcess(register picodata_ProcessingUnit this)
+{
+ register pam_subobj_t * pam;
+ picoos_uint8 nCond1, nCond2, nCond3;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+ /*conditions originating a "NOT to be processed" result */
+ nCond1 = pam->nCurrSyllable <= -1;
+ nCond2 = pam->nCurrSyllable >= pam->nTotalSyllables;
+ nCond3 = pam->nSyllPhoneme
+ >= pam->sSyllFeats[pam->nCurrSyllable].phoneV[B3];
+
+ if ((nCond1) || (nCond2) || (nCond3))
+ return FALSE;
+
+ return TRUE;
+} /*pamHasToProcess*/
+
+/**
+ * modifies the process flags in order to point to next valid syllable phone or item to be produced
+ * @param this : Pam object pointer
+ * @return TRUE : item has to be produced
+ * @return FALSE : item has not to be produced
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pamUpdateProcess(register picodata_ProcessingUnit this)
+{
+ register pam_subobj_t * pam;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+
+ if (pam->nCurrSyllable == -1) {
+ /*this to be able to manage sudden PU cleanup after FLUSH CMD*/
+ return PICO_OK;
+ }
+ /*check number of phonemes for current syllable*/
+ if (pam->nSyllPhoneme < pam->sSyllFeats[pam->nCurrSyllable].phoneV[B3] - 1) {
+ pam->nSyllPhoneme++;
+ return PICO_OK;
+ }
+ if (pam->nSyllPhoneme == pam->sSyllFeats[pam->nCurrSyllable].phoneV[B3] - 1) {
+ /*this helps in identifyng the end of syllable condition in PamHasToProcess*/
+ pam->nSyllPhoneme++;
+ }
+ /*previous syllable phonemes are complete: test if any items are tied to this syllable*/
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[ITM] > 0) {
+ /*there are items tied to this syllable*/
+ if (pam->nCurrAttachedItem == 0) {
+ /*if it is the first item to be regenerated initialize it*/
+ pam->nCurrAttachedItem
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[ITM];
+ return PICO_OK;
+ } else {
+ /*not the first item : check if more*/
+ if (pam->nCurrAttachedItem
+ < pam->sSyllFeats[pam->nCurrSyllable].phoneV[itm]) {
+ /*more tied items to be regenerated*/
+ pam->nCurrAttachedItem++;
+ return PICO_OK;
+ }
+ }
+ }
+ /*previous syllable phonemes and items are complete: switch to next syllable*/
+ if (pam->nCurrSyllable < pam->nTotalSyllables - 1) {
+ pam->nCurrSyllable++;
+ pam->nSyllPhoneme = 0;
+ pam->nCurrAttachedItem = 0;
+ return PICO_OK;
+ }
+ /*no more phonemes or items to be produced*/
+ pam->nCurrSyllable++;
+ pam->nSyllPhoneme = 0;
+ return PICO_ERR_OTHER;
+
+} /*pamUpdateProcess*/
+
+/**
+ * returns true if more items has to be popped for current syllable
+ * @param this : Pam object pointer
+ * @return TRUE : item has to be popped
+ * @return FALSE : item has not to be popped
+ * @callgraph
+ * @callergraph
+ */
+static picoos_uint8 pamHasToPop(register picodata_ProcessingUnit this)
+{
+ register pam_subobj_t * pam;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+
+ /*Preliminary condition : at least 1 syllable*/
+ if (pam->nCurrSyllable <= -1)
+ return FALSE;
+
+ /*Preliminary condition : not maximum number of syllables*/
+ if (pam->nCurrSyllable >= pam->nTotalSyllables)
+ return FALSE;
+
+ /*Preliminary condition : start and end offset in current item > 0 */
+ if ((pam->sSyllFeats[pam->nCurrSyllable].phoneV[ITM] <= 0)
+ || (pam->sSyllFeats[pam->nCurrSyllable].phoneV[itm] <= 0))
+ return FALSE;
+
+ /*Final condition : current popped item less or eq to maximum*/
+ if (pam->nCurrAttachedItem
+ > pam->sSyllFeats[pam->nCurrSyllable].phoneV[itm])
+ return FALSE;
+
+ return TRUE;
+} /*pamHasToPop*/
+
+/**
+ * returns the address of an item to be popped from the current syllable queue
+ * @param this : Pam object pointer
+ * @return pop_address : item address
+ * @return NULL : item not poppable
+ * @callgraph
+ * @callergraph
+ */
+static picoos_uint8 *pamPopItem(register picodata_ProcessingUnit this)
+{
+ register pam_subobj_t * pam;
+ picoos_uint8 nItem;
+ if (NULL == this || NULL == this->subObj) {
+ return NULL;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+
+ /*Preliminary condition : at least 1 syllable*/
+ if (pam->nCurrSyllable <= -1)
+ return NULL;
+
+ /*Preliminary condition : not maximum number of syllables*/
+ if (pam->nCurrSyllable >= pam->nTotalSyllables)
+ return NULL;
+
+ /*Preliminary condition : start and end offset in current item > 0 */
+ if ((pam->sSyllFeats[pam->nCurrSyllable].phoneV[ITM] <= 0)
+ || (pam->sSyllFeats[pam->nCurrSyllable].phoneV[itm] <= 0))
+ return NULL;
+
+ /*Final condition : current popped item less than maximum*/
+ if (pam->nCurrAttachedItem
+ > pam->sSyllFeats[pam->nCurrSyllable].phoneV[itm])
+ return NULL;
+
+ nItem = pam->nCurrAttachedItem;
+ /*please note : nItem-1 should match with actions performed in function "pam_queue(..)" */
+ return &(pam->sSyllItems[pam->sSyllItemOffs[nItem - 1]]);
+
+} /*pamPopItem*/
+
+/**
+ * returns the address of an item popped from the syllable 0 queue
+ * @param this : Pam object pointer
+ * @return pop_address : item address
+ * @return NULL : item not poppable
+ * @remarks the item is popped only if it has been inserted in the queue before the first
+ * @remarks item assigned to the syllable 0 i.e.
+ * @remarks AttachedItem<=pam->sSyllFeats[pam->nCurrSyllable].phoneV[itm]-1
+ * @callgraph
+ * @callergraph
+ */
+static picoos_uint8 *pamPopAttachedSy0(register picodata_ProcessingUnit this)
+{
+ register pam_subobj_t * pam;
+ picoos_uint8 nItem;
+ if (NULL == this || NULL == this->subObj) {
+ return NULL;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+
+ /*should be syllable 0*/
+ if (pam->nCurrSyllable != 0)
+ return NULL;
+
+ /*start and end offset in current item > 0 */
+ if ((pam->sSyllFeats[pam->nCurrSyllable].phoneV[ITM] <= 0)
+ || (pam->sSyllFeats[pam->nCurrSyllable].phoneV[itm] <= 0))
+ return NULL;
+
+ /*if current popped item is > 0 test end condition*/
+ if (pam->nCurrAttachedItem > 0) {
+ /*Other condition : current popped item less than maximum*/
+ if (pam->nCurrAttachedItem
+ > pam->sSyllFeats[pam->nCurrSyllable].phoneV[itm] - 1)
+ return NULL;
+ }
+ nItem = pam->nCurrAttachedItem;
+ return &(pam->sSyllItems[pam->sSyllItemOffs[nItem]]);
+
+} /*pamPopAttachedSy0*/
+
+/**
+ * pdf access for duration
+ * @param this : Pam object pointer
+ * @param durIndex : index of duration in the pdf
+ * @param phonDur : pointer to base of array where to store the duration values
+ * @param numFramesState : pointer to base of array where to store the number of frames per state
+ * @return PICO_OK : pdf retrieved
+ * @return PICO_ERR_OTHER : pdf not retrieved
+ * @remarks Modifies phonDur (the requested duration value)
+ * @remarks Modifies numFramesState (the requested number of frames per state (vector))
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_get_duration(register picodata_ProcessingUnit this,
+ picoos_uint16 durIndex, picoos_uint16 *phonDur,
+ picoos_uint8 *numFramesState)
+{
+ pam_subobj_t *pam;
+ picokpdf_PdfDUR pdf;
+ picoos_uint8 *durItem;
+ picoos_uint16 nFrameSize, nI;
+ picoos_single fValue;
+ pam = (pam_subobj_t *) this->subObj;
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ pdf = pam->pdfdur;
+ /*make the index 0 based*/
+ if (durIndex > 0)
+ durIndex--;
+
+ /* check */
+ if (durIndex > pdf->numframes - 1) {
+ PICODBG_ERROR(("PAM durPdf access error, index overflow -> index: %d , numframes: %d", durIndex, pdf->numframes));
+ return PICO_ERR_OTHER;
+ }
+ /* base pointer */
+ durItem = &(pdf->content[durIndex * pdf->vecsize]);
+ if (durItem == NULL) {
+ PICODBG_ERROR(("PAM durPdf access error , frame pointer = NULL"));
+ return PICO_ERR_OTHER;
+ }
+ nFrameSize = pdf->sampperframe / 16;
+ *phonDur = ((pdf->phonquant[((*durItem) & 0xF0) >> 4]) * nFrameSize);
+ numFramesState[0] = pdf->statequant[((*durItem) & 0x0F)];
+ durItem++;
+ numFramesState[1] = pdf->statequant[((*durItem) & 0xF0) >> 4];
+ numFramesState[2] = pdf->statequant[((*durItem) & 0x0F)];
+ durItem++;
+ numFramesState[3] = pdf->statequant[((*durItem) & 0xF0) >> 4];
+ numFramesState[4] = pdf->statequant[((*durItem) & 0x0F)];
+
+ /*modification of the duration information based on the duration modifier*/
+ *phonDur = (picoos_uint16) (((picoos_single) * phonDur) * pam->dMod);
+ for (nI = 0; nI < 5; nI++) {
+ fValue = pam->dRest + (picoos_single) numFramesState[nI] * pam->dMod;
+ numFramesState[nI] = (picoos_uint8) (fValue);
+ pam->dRest = fValue - (picoos_single) numFramesState[nI];
+ }
+ return PICO_OK;
+}/*pam_get_duration*/
+
+/**
+ * pdf access for pitch
+ * @param this : Pam object pointer
+ * @param lf0Index : pointer to variable to receive index of pitch in the pdf
+ * @param nI : number of the phone's state
+ * @param phonF0 : pointer to variable to receive the pitch value
+ * @return PICO_OK : pdf retrieved
+ * @return PICO_ERR_OTHER : pdf not retrieved
+ * @remarks Modifies phonDur (the requested duration value)
+ * @remarks Modifies phonF0 (the requested pitch value (scalar))
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_get_f0(register picodata_ProcessingUnit this,
+ picoos_uint16 *lf0Index, picoos_uint8 nI, picoos_single *phonF0)
+{
+ pam_subobj_t *pam;
+ picoos_uint8 *lfItem, numstreams;
+ picoos_uint16 lf0IndexOffset, sTemp;
+ picoos_single lfum, lfivar, lfz;
+
+ pam = (pam_subobj_t *) this->subObj;
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ lf0IndexOffset = lf0Index[nI];
+
+ /*make the index 0 based*/
+ if (lf0IndexOffset > 0)
+ lf0IndexOffset--;
+
+ lf0IndexOffset += pam->pdflfz->stateoffset[nI];
+ if (lf0IndexOffset > pam->pdflfz->numframes - 1) {
+ PICODBG_ERROR(("PAM flfzPdf access error, index overflow -> index: %d , numframes: %d", lf0Index, pam->pdflfz->numframes));
+ return PICO_ERR_OTHER;
+ }
+ /* base pointer */
+ lf0IndexOffset *= pam->pdflfz->vecsize;
+
+ lfItem = &(pam->pdflfz->content[lf0IndexOffset]);
+ sTemp = (picoos_uint16) (((lfItem[1] << 8)) | lfItem[0]);
+
+ lfum = (picoos_single) (sTemp << (pam->pdflfz->meanpowUm[0]));
+ numstreams = 3;
+ lfivar = (picoos_single) (((picoos_uint16) lfItem[numstreams * 2])
+ << pam->pdflfz->ivarpow[0]);
+ lfz = (picoos_single) lfum / (picoos_single) lfivar;
+ lfz = (picoos_single) exp((double) lfz);
+ phonF0[nI] = (picoos_single) lfz;
+
+ /*pitch modoification*/
+ phonF0[nI] *= pam->pMod;
+ return PICO_OK;
+}/*pam_get_f0*/
+
+/**
+ * elementary rounding function
+ * @param fIn : (real) input value
+ * @return the rounded value
+ * @callgraph
+ * @callergraph
+ */
+static picoos_single f_round(picoos_single fIn)
+{
+ picoos_int32 iVal;
+ picoos_single fVal;
+
+ iVal = (picoos_int32) fIn;
+ fVal = (picoos_single) iVal;
+
+ if (fIn > (picoos_single) 0.0f) {
+ if ((fIn - fVal) < (picoos_single) 0.5f)
+ return fVal;
+ else
+ return fVal + (picoos_single) 1.0f;
+ } else {
+ if ((fVal - fIn) < (picoos_single) 0.5f)
+ return fVal;
+ else
+ return fVal - (picoos_single) 1.0f;
+ }
+}/*f_round*/
+
+/**
+ * updates the input vector for PAM
+ * @param this : Pam object pointer
+ * @return PICO_OK : update successful
+ * @return PICO_ERR_OTHER : errors on retrieving the PU pointer
+ * @remarks Modifies pam->sPhFeats[]
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_update_vector(register picodata_ProcessingUnit this)
+{
+ pam_subobj_t *pam;
+ picoos_uint8 numstates, nI;
+ picoos_single fDur, f0avg, f0quant, minf0, maxf0, durquant1, durquant2,
+ mindur, maxdur1, maxdur2;
+
+ pam = (pam_subobj_t *) this->subObj;
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ /*default init*/
+ pam->sPhFeats[DUR] = 0;
+ pam->sPhFeats[F0] = 0;
+ /*
+ Hard coded parameters for quantization
+ */
+ numstates = PICOPAM_NRSTPF;
+ f0quant = 30.0f;
+ minf0 = 90.0f;
+ maxf0 = 360.0f;
+
+ durquant1 = 20.0f;
+ durquant2 = 100.0f;
+ mindur = 40.0f;
+ maxdur1 = 160.0f;
+ maxdur2 = 600.0f;
+ f0avg = 0.0f;
+ for (nI = 0; nI < numstates; nI++)
+ f0avg += pam->phonF0[nI];
+ f0avg /= (picoos_single) numstates;
+
+ f0avg = f_round(f0avg / f0quant) * f0quant;
+ if (f0avg < minf0)
+ f0avg = minf0;
+ if (f0avg > maxf0)
+ f0avg = maxf0;
+
+ /*make initial silence of sentence shorter (see also pam_put_item)*/
+ if ((pam->nCurrSyllable == 0) && (pam->nSyllPhoneme == 0)) {
+ pam->phonDur = 2 * 4;
+ }
+
+ fDur = (picoos_single) pam->phonDur;
+ fDur = f_round(fDur / durquant1) * durquant1;
+ if (fDur < mindur)
+ fDur = mindur;
+ if (fDur > maxdur1) {
+ fDur = f_round(fDur / durquant2) * durquant2;
+ if (fDur > maxdur2)
+ fDur = maxdur2;
+ }
+ pam->sPhFeats[DUR] = (picoos_uint8) (fDur / (picoos_single) 10.0f);
+ pam->sPhFeats[F0] = (picoos_uint8) (f0avg / (picoos_single) 10.0f);
+
+ return PICO_OK;
+}/*pam_update_vector*/
+
+/**
+ * compress a single feature in the range 0..9
+ * @param inVal : the value to be compressed
+ * @return compVal : the compressed value
+ * @callgraph
+ * @callergraph
+ */
+static picoos_uint8 pamCompressComponent(picoos_uint8 inVal)
+{
+ if (inVal <= 5)
+ return inVal;
+ if ((5 < inVal) && (inVal <= 10))
+ return 6;
+ if ((10 < inVal) && (inVal <= 20))
+ return 7;
+ if ((20 < inVal) && (inVal <= 30))
+ return 8;
+ return 9;
+}/*pamCompressComponent*/
+
+/**
+ * prepares the input vector for tree feeding
+ * @param this : Pam object pointer
+ * @return PICO_OK : vector expanded
+ * @return PICO_ERR_OTHER : errors on expansion or retrieving the PU pointer
+ * @remarks Modifies pam->sPhFeats[]
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_expand_vector(register picodata_ProcessingUnit this)
+{
+ pam_subobj_t *pam;
+ picoos_uint8 *inVect, *phonVect, *outVect, nI;
+ picoos_int16 nOffs, nOffs1, nLen;
+ pam = (pam_subobj_t *) this->subObj;
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ inVect = &(pam->sSyllFeats[pam->nCurrSyllable].phoneV[0]);
+ phonVect = &(pam->sPhIds[0]);
+ outVect = &(pam->sPhFeats[0]);
+ /*just copy back*/
+ for (nI = 0; nI < PICOPAM_INVEC_SIZE; nI++)
+ outVect[nI] = inVect[nI];
+ /*now fill missing fields*/
+ picoos_mem_copy((void*) &(inVect[FID]), &nOffs, sizeof(nOffs));
+ /*offset to first phone of current syllable*/
+ nOffs = nOffs + pam->nSyllPhoneme; /*offset to current phone of current syllable*/
+ nLen = inVect[B3]; /*len of current syllable*/
+ if (pam->nSyllPhoneme >= nLen) {
+ /*error on addressing current phone*/
+ return PICO_ERR_OTHER;
+ }
+ /*previous of the previous phone*/
+ nOffs1 = nOffs - 2;
+ if (nOffs1 >= 0)
+ outVect[P1] = phonVect[nOffs1];
+ else
+ outVect[P1] = PICOPAM_PH_DONT_CARE_VAL;
+ /*previous phone*/
+ nOffs1 = nOffs - 1;
+ if (nOffs1 >= 0)
+ outVect[P2] = phonVect[nOffs1];
+ else
+ outVect[P2] = PICOPAM_PH_DONT_CARE_VAL;
+ /*^current phone*/
+ outVect[P3] = phonVect[nOffs];
+
+ /*next phone*/
+ nOffs1 = nOffs + 1;
+ if (nOffs1 < pam->nTotalPhonemes)
+ outVect[P4] = phonVect[nOffs1];
+ else
+ outVect[P4] = PICOPAM_PH_DONT_CARE_VAL;
+ /*next of the next phone*/
+ nOffs1 = nOffs + 2;
+ if (nOffs1 < pam->nTotalPhonemes)
+ outVect[P5] = phonVect[nOffs1];
+ else
+ outVect[P5] = PICOPAM_PH_DONT_CARE_VAL;
+ /*pos of curr phone with respect to left syllable boundary*/
+ outVect[P6] = pam->nSyllPhoneme + 1;
+ /*pos of curr phone with respect to right syllable boundary*/
+ outVect[P7] = nLen - pam->nSyllPhoneme;
+ /*is current phone in consonant syllable boundary? (1:yes)*/
+ if (pam->nSyllPhoneme < inVect[P8])
+ outVect[P8] = 1;
+ else
+ outVect[P8] = 0;
+ return PICO_OK;
+}/*pam_expand_vector*/
+
+/**
+ * compresses the input vector for PAM
+ * @param this : Pam object pointer
+ * @return PICO_OK : compression successful
+ * @return PICO_ERR_OTHER : errors on retrieving the PU pointer
+ * @remarks Modifies pam->sPhFeats[]
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pamCompressVector(register picodata_ProcessingUnit this)
+{
+ pam_subobj_t *pam;
+ picoos_uint8 *outVect, nI;
+ pam = (pam_subobj_t *) this->subObj;
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ outVect = &(pam->sPhFeats[0]);
+ for (nI = 0; nI < PICOPAM_INVEC_SIZE; nI++) {
+ switch (nI) {
+ case P1:
+ case P2:
+ case P3:
+ case P4:
+ case P5:
+ case B1:
+ case B2:
+ case B16:
+ case E1:
+ case H5:
+ /*don't do any compression*/
+ break;
+ default:
+ /*do compression*/
+ if (outVect[nI] != PICOPAM_DONT_CARE_VALUE)
+ outVect[nI] = pamCompressComponent(outVect[nI]);
+ else
+ outVect[nI] = PICOPAM_DONT_CARE_VAL;
+ break;
+ }
+ }
+ return PICO_OK;
+}/*pamCompressVector*/
+
+/**
+ * reorganizes the input vector for PAM
+ * @param this : Pam object pointer
+ * @return PICO_OK : reorganization successful
+ * @return PICO_ERR_OTHER : errors on retrieving the PU pointer
+ * @remarks Modifies pam->sPhFeats[]
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pamReorgVector(register picodata_ProcessingUnit this)
+{
+ pam_subobj_t *pam;
+ picoos_uint8 *outVect, inVect[60], nI;
+ pam = (pam_subobj_t *) this->subObj;
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ outVect = &(pam->sPhFeats[0]);
+ for (nI = 0; nI < PICOPAM_INVEC_SIZE; nI++) inVect[nI] = outVect[nI];
+ /*reorganize*/
+ for (nI = T_B1; nI <= T_H5; nI++) {
+ switch (nI) {
+ case T_B1:
+ outVect[T_B1] = inVect[B1];
+ break;
+ case T_B2:
+ outVect[T_B2] = inVect[B2];
+ break;
+ case T_B3:
+ outVect[T_B3] = inVect[B3];
+ break;
+ case T_B4:
+ outVect[T_B4] = inVect[B4];
+ break;
+ case T_B5:
+ outVect[T_B5] = inVect[B5];
+ break;
+ case T_B6:
+ outVect[T_B6] = inVect[B6];
+ break;
+ case T_B7:
+ outVect[T_B7] = inVect[B7];
+ break;
+ case T_B8:
+ outVect[T_B8] = inVect[B8];
+ break;
+ case T_B9:
+ outVect[T_B9] = inVect[B9];
+ break;
+ case T_B10:
+ outVect[T_B10] = inVect[B10];
+ break;
+ case T_B11:
+ outVect[T_B11] = inVect[B11];
+ break;
+ case T_B12:
+ outVect[T_B12] = inVect[B12];
+ break;
+ case T_B13:
+ outVect[T_B13] = inVect[B13];
+ break;
+ case T_B14:
+ outVect[T_B14] = inVect[B14];
+ break;
+ case T_B15:
+ outVect[T_B15] = inVect[B15];
+ break;
+ case T_B16:
+ outVect[T_B16] = inVect[B16];
+ break;
+ case T_B17:
+ outVect[T_B17] = inVect[B17];
+ break;
+ case T_B18:
+ outVect[T_B18] = inVect[B18];
+ break;
+ case T_B19:
+ outVect[T_B19] = inVect[B19];
+ break;
+ case T_B20:
+ outVect[T_B20] = inVect[B20];
+ break;
+ case T_B21:
+ outVect[T_B21] = inVect[B21];
+ break;
+
+ case T_E1:
+ outVect[T_E1] = inVect[E1];
+ break;
+ case T_E2:
+ outVect[T_E2] = inVect[E2];
+ break;
+ case T_E3:
+ outVect[T_E3] = inVect[E3];
+ break;
+ case T_E4:
+ outVect[T_E4] = inVect[E4];
+ break;
+ case T_E5:
+ outVect[T_E5] = inVect[E5];
+ break;
+ case T_E6:
+ outVect[T_E6] = inVect[E6];
+ break;
+ case T_E7:
+ outVect[T_E7] = inVect[E7];
+ break;
+ case T_E8:
+ outVect[T_E8] = inVect[E8];
+ break;
+ case T_E9:
+ outVect[T_E9] = inVect[E9];
+ break;
+ case T_E10:
+ outVect[T_E10] = inVect[E10];
+ break;
+ case T_E11:
+ outVect[T_E11] = inVect[E11];
+ break;
+ case T_E12:
+ outVect[T_E12] = inVect[E12];
+ break;
+ case T_E13:
+ outVect[T_E13] = inVect[E13];
+ break;
+
+ case T_A3:
+ outVect[T_A3] = inVect[A3];
+ break;
+ case T_C3:
+ outVect[T_C3] = inVect[C3];
+ break;
+ case T_D2:
+ outVect[T_D2] = inVect[D2];
+ break;
+ case T_F2:
+ outVect[T_F2] = inVect[F2];
+ break;
+
+ case T_G1:
+ outVect[T_G1] = inVect[G1];
+ break;
+ case T_I1:
+ outVect[T_I1] = inVect[I1];
+ break;
+
+ case T_G2:
+ outVect[T_G2] = inVect[G2];
+ break;
+ case T_I2:
+ outVect[T_I2] = inVect[I2];
+ break;
+
+ case T_H1:
+ outVect[T_H1] = inVect[H1];
+ break;
+ case T_H2:
+ outVect[T_H2] = inVect[H2];
+ break;
+ case T_H3:
+ outVect[T_H3] = inVect[H3];
+ break;
+ case T_H4:
+ outVect[T_H4] = inVect[H4];
+ break;
+ case T_H5:
+ outVect[T_H5] = inVect[H5];
+ break;
+ }
+ }
+ return PICO_OK;
+}/*pamReorgVector*/
+
+/**
+ * puts a PAM item into PU output buffer
+ * @param this : Pam object pointer
+ * @param outBuff : output buffer base pointer
+ * @param outWritePos : offset in output buffer
+ * @param *bytesWr : actual bytes written
+ * @return PICO_OK : put successful
+ * @return PICO_ERR_OTHER : errors on retrieving the PU pointer
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_put_item(register picodata_ProcessingUnit this,
+ picoos_uint8 *outBuff, picoos_uint16 outWritePos, picoos_uint8 *bytesWr)
+{
+ pam_subobj_t *pam;
+ picoos_uint8 *sDest, nI, nType, nIdx, fde;
+ picoos_uint32 pos, pos32;
+ picoos_int16 ft, dt;
+ picoos_uint16 uMinDur, uMaxDur;
+ pam = (pam_subobj_t *) this->subObj;
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ sDest = &(outBuff[outWritePos]);
+ sDest[0] = PICODATA_ITEM_PHONE; /*Item type*/
+ sDest[1] = pam->sPhFeats[P3]; /*phonetic id*/
+ sDest[2] = PICOPAM_NRSTPF; /*number of states per phone*/
+ sDest[3] = sizeof(picoos_uint16) * PICOPAM_NRSTPF * 3; /*size of the item*/
+ pos = 4;
+ /*make initial silence of sentence shorter (see also UpdateVector)*/
+ if ((pam->nCurrSyllable == 0) && (pam->nSyllPhoneme == 0)) {
+ for (nI = 0; nI < PICOPAM_NRSTPF - 1; nI++)
+ pam->numFramesState[nI] = 0;
+ pam->numFramesState[nI] = 2;
+ } else {
+ /*manage silence syllables with prescribed durations*/
+ pos32 = Min;
+ picoos_read_mem_pi_uint16(pam->sSyllFeats[pam->nCurrSyllable].phoneV,
+ &pos32, &uMinDur);
+ pos32 = Max;
+ picoos_read_mem_pi_uint16(pam->sSyllFeats[pam->nCurrSyllable].phoneV,
+ &pos32, &uMaxDur);
+
+ if (uMaxDur > 0) {
+ /* Select weights*/
+ nType = pam->sSyllFeats[pam->nCurrSyllable].phoneV[bnd];
+ switch (nType) {
+ case PICODATA_ITEMINFO1_BOUND_SBEG:
+ nIdx = PICOPAM_PWIDX_SBEG;
+ break;
+ case PICODATA_ITEMINFO1_BOUND_PHR1:
+ nIdx = PICOPAM_PWIDX_PHR1;
+ break;
+ case PICODATA_ITEMINFO1_BOUND_PHR2:
+ nIdx = PICOPAM_PWIDX_PHR2;
+ break;
+ case PICODATA_ITEMINFO1_BOUND_SEND:
+ case PICODATA_ITEMINFO1_BOUND_TERM:
+ nIdx = PICOPAM_PWIDX_SEND;
+ break;
+ default:
+ nIdx = PICOPAM_PWIDX_DEFA;
+ break;
+ }
+ fde = 2;
+ ft = 0;
+ dt = 0;
+ picodata_transformDurations(
+ fde, /* 2's exponent of frame duration in ms, e.g. 2 for 4ms, 3 for 8ms */
+ PICOPAM_NRSTPF, /* number of states per phone */
+ &(pam->numFramesState[0]), /* estimated durations */
+ pam->sil_weights[nIdx], /* integer weights */
+ uMinDur, /* minimum target duration in ms */
+ uMaxDur, /* maximum target duration in ms */
+ ft, /* factor to be multiplied to get the target */
+ &dt /* in/out, rest in ms */
+ );
+ }
+ }
+ /*put data*/
+ for (nI = 0; nI < PICOPAM_NRSTPF; nI++) {
+ picoos_write_mem_pi_uint16(sDest, &pos,
+ (picoos_uint16) pam->numFramesState[nI]);
+ picoos_write_mem_pi_uint16(sDest, &pos,
+ (picoos_uint16) pam->lf0Index[nI]);
+ picoos_write_mem_pi_uint16(sDest, &pos,
+ (picoos_uint16) pam->mgcIndex[nI]);
+ }
+ *bytesWr = sizeof(picodata_itemhead_t) + sizeof(picoos_uint16)
+ * PICOPAM_NRSTPF * 3;
+ return PICO_OK;
+}/*pam_put_item*/
+
+/**
+ * puts a non PAM (queued) item into PU output buffer
+ * @param qItem : pointer to item to put
+ * @param outBuff : output buffer base pointer
+ * @param outWritePos : offset in output buffer
+ * @param *bytesWr : actual bytes written
+ * @return PICO_OK : put successful
+ * @return PICO_ERR_OTHER : errors on retrieving the PU pointer
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_put_qItem(picoos_uint8 *qItem, picoos_uint8 *outBuff,
+ picoos_uint16 outWritePos, picoos_uint8 *bytesWr)
+{
+ picoos_uint8 *sDest, nI;
+ sDest = &(outBuff[outWritePos]);
+ *bytesWr = sizeof(picodata_itemhead_t);
+ for (nI = 0; nI < (sizeof(picodata_itemhead_t) + qItem[3]); nI++) {
+ sDest[nI] = qItem[nI];
+ }
+ *bytesWr = nI;
+ return PICO_OK;
+}/*pam_put_qItem*/
+
+/**
+ * tells if an item is a PAM command (except play)
+ * @param qItem : input item to test
+ * @return TRUE : qItem is a PAM command (except play)
+ * @return FALSE : qItem not a PAM command
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t is_pam_command(const picoos_uint8 * qItem)
+{
+ switch (qItem[0]) {
+
+ case PICODATA_ITEM_CMD:
+ switch (qItem[1]) {
+ case PICODATA_ITEMINFO1_CMD_FLUSH:
+ /* flush is for all PU's and as such it is also for PAM*/
+ case PICODATA_ITEMINFO1_CMD_PITCH:
+ case PICODATA_ITEMINFO1_CMD_SPEED:
+ return TRUE;
+ break;
+ default:
+ break;
+ }
+ }
+ return FALSE;
+}/*is_pam_command*/
+
+/**
+ * tells if an item is a PAM PLAY command
+ * @param qItem : input item to test
+ * @return TRUE : qItem is a PAM PLAY command
+ * @return FALSE : qItem not a PAM PLAY command
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t is_pam_play_command(picoos_uint8 *qItem)
+{
+ switch (qItem[0]) {
+
+ case PICODATA_ITEM_CMD:
+ switch (qItem[1]) {
+ case PICODATA_ITEMINFO1_CMD_PLAY:
+ if (qItem[2] == PICODATA_ITEMINFO2_CMD_TO_PAM)
+ return TRUE;
+ break;
+ default:
+ break;
+ }
+ }
+ return FALSE;
+}/*is_pam_play_command*/
+
+/**
+ * command processor for PAM pu
+ * @param this : Pam item subobject
+ * @param qItem : input item pointer
+ * @return PICOPAM_FLUSH_RECEIVED : when a FLUSH is received
+ * @return PICOPAM_CONTINUE : normal command processing
+ * @return PICODATA_PU_ERROR : errors in accessing data
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pamDoCommand(register picodata_ProcessingUnit this,
+ picoos_uint8 *qItem)
+{
+ pam_subobj_t *pam;
+ picoos_single fValue;
+ picoos_uint16 nValue;
+ picoos_uint32 nPos;
+ pam = (pam_subobj_t *) this->subObj;
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ if (qItem[0] == PICODATA_ITEM_CMD) {
+ switch (qItem[1]) {
+ case PICODATA_ITEMINFO1_CMD_FLUSH:
+ /* flush is for all PU's and as such it is also for PAM : implement the flush!!*/
+ pam_reset_processors(this);
+ pam->nLastAttachedItemId = pam->nCurrAttachedItem = 0;
+ pam->nAttachedItemsSize = 0;
+ return PICOPAM_FLUSH_RECEIVED;
+ break;
+
+ case PICODATA_ITEMINFO1_CMD_PITCH:
+ case PICODATA_ITEMINFO1_CMD_SPEED:
+ nPos = 4;
+ picoos_read_mem_pi_uint16(qItem, &nPos, &nValue);
+ if (qItem[2] == 'a') {
+ /*absloute modifier*/
+ fValue = (picoos_single) nValue / (picoos_single) 100.0f;
+ if (qItem[1] == PICODATA_ITEMINFO1_CMD_PITCH)
+ pam->pMod = fValue;
+ if (qItem[1] == PICODATA_ITEMINFO1_CMD_SPEED)
+ pam->dMod = (1.0f / fValue);
+ }
+ if (qItem[2] == 'r') {
+ /*relative modifier*/
+ fValue = (picoos_single) nValue / (picoos_single) 1000.0f;
+ if (qItem[1] == PICODATA_ITEMINFO1_CMD_PITCH)
+ pam->pMod *= (1.0f / fValue);
+ if (qItem[1] == PICODATA_ITEMINFO1_CMD_SPEED)
+ pam->dMod *= (1.0f / fValue);
+ }
+ return PICOPAM_CONTINUE;
+ break;
+
+ default:
+ break;
+ }/*end switch switch (qItem[1])*/
+ }/*end if (qItem[0]==PICODATA_ITEM_CMD)*/
+ return PICOPAM_CONTINUE;
+}/*pamDoCommand*/
+
+/**
+ * defines if an item has to be sent to following PUs
+ * @param qItem : input item pointer
+ * @return TRUE : item has to be transmitted to following PUs
+ * @return FALSE : item has to be consumed internallz on PAM
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t isItemToPut(picoos_uint8 *qItem)
+{
+ switch (qItem[0]) {
+ case PICODATA_ITEM_CMD:
+ /* is a command*/
+ if (PICODATA_ITEMINFO1_CMD_SPEED == qItem[1]) {
+ /* SPEED consumed here*/
+ return FALSE;
+ }
+ break;
+ case PICODATA_ITEM_BOUND:
+ switch (qItem[1]) {
+ case PICODATA_ITEMINFO1_BOUND_SBEG:
+ case PICODATA_ITEMINFO1_BOUND_PHR0:
+ case PICODATA_ITEMINFO1_BOUND_PHR1:
+ case PICODATA_ITEMINFO1_BOUND_PHR2:
+ case PICODATA_ITEMINFO1_BOUND_PHR3:
+ /*boudary items consumed here except SEND,TERM*/
+ return FALSE;
+ break;
+ default:
+ break;
+ }
+ break;
+ default:
+ break;
+ }
+ /*all other items not explicitly mentioned here
+ are transmitted to next PUs*/
+ return TRUE;
+}/*isItemToPut*/
+
+/**
+ * pushes a boundary TERM item into some buffer
+ * @param outBuff : output buffer base pointer
+ * @param outWritePos : offset in output buffer
+ * @param *bytesWr : actual bytes written
+ * @return PICO_OK
+ * @remarks used while forcing TERM input items in forward processing
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_put_term(picoos_uint8 *outBuff,
+ picoos_uint16 outWritePos, picoos_uint8 *bytesWr)
+{
+ picoos_uint8 *sDest;
+ sDest = &(outBuff[outWritePos]);
+ sDest[0] = PICODATA_ITEM_BOUND; /*Item type*/
+ sDest[1] = PICODATA_ITEMINFO1_BOUND_TERM;
+ sDest[2] = PICODATA_ITEMINFO2_BOUNDTYPE_T;
+ sDest[3] = 0; /*item size*/
+ *bytesWr = 4;
+ return PICO_OK;
+}/*pam_put_term*/
+
+/**
+ * translates one full phone into a PHONE Item including DT Dur, F0 and CEP trees feature generation and traversal
+ * @param this : Pam item subobject pointer
+ * @return PICO_OK : processing successful
+ * @return PICODATA_PU_ERROR : error accessing PAM object
+ * @return !=PICO_OK : processing errors
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pamPhoneProcess(register picodata_ProcessingUnit this)
+{
+ pam_subobj_t *pam;
+ pico_status_t sResult;
+ picokdt_classify_result_t dTreeResult;
+ picoos_uint8 nI, bWr;
+
+ pam = (pam_subobj_t *) this->subObj;
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ /*expands current phone in current syllable in the corresponding vector pam->sPhFeats[]*/
+ sResult = pam_expand_vector(this);
+ sResult = pamCompressVector(this);
+ sResult = pamReorgVector(this);
+
+ /*tree traversal for duration*/
+ if (!pam_do_tree(this, pam->dtdur, &(pam->sPhFeats[0]), PICOPAM_INVEC_SIZE,
+ &dTreeResult)) {
+ PICODBG_WARN(("problem using pam tree dtdur, using fallback value"));
+ dTreeResult.class = 0;
+ }
+ pam->durIndex = dTreeResult.class;
+ sResult = pam_get_duration(this, pam->durIndex, &(pam->phonDur),
+ &(pam->numFramesState[0]));
+
+ /*tree traversal for pitch*/
+ for (nI = 0; nI < PICOPAM_MAX_STATES_PER_PHONE; nI++) {
+ if (!pam_do_tree(this, pam->dtlfz[nI], &(pam->sPhFeats[0]),
+ PICOPAM_INVEC_SIZE, &dTreeResult)) {
+ PICODBG_WARN(("problem using pam tree lf0Tree, using fallback value"));
+ dTreeResult.class = 0;
+ }
+ pam->lf0Index[nI] = dTreeResult.class;
+ }
+
+ /*pdf access for pitch*/
+ for (nI = 0; nI < PICOPAM_MAX_STATES_PER_PHONE; nI++) {
+ sResult = pam_get_f0(this, &(pam->lf0Index[0]), nI, &(pam->phonF0[0]));
+ }
+
+ /*update vector with duration and pitch for cep tree traversal*/
+ sResult = pam_update_vector(this);
+ /*cep tree traversal*/
+ for (nI = 0; nI < PICOPAM_MAX_STATES_PER_PHONE; nI++) {
+
+ if (!pam_do_tree(this, pam->dtmgc[nI], &(pam->sPhFeats[0]),
+ PICOPAM_INVEC_SIZE, &dTreeResult)) {
+ PICODBG_WARN(("problem using pam tree lf0Tree, using fallback value"));
+ dTreeResult.class = 0;
+ }
+ pam->mgcIndex[nI] = dTreeResult.class;
+ }
+ /*put item to output buffer*/
+ sResult = pam_put_item(this, pam->outBuf, pam->outWritePos, &bWr);
+ if (sResult == PICO_OK)
+ pam->outWritePos += bWr;
+ else
+ return sResult;
+ return PICO_OK;
+}/*pamPhoneProcess*/
+
+/**
+ * manages first syllable attached items when seen before SBEG
+ * @param this : Pam item subobject pointer
+ * @return PICO_OK (0) : default return code --> means no more items to be processed before 1st syllable
+ * @return PICOPAM_GOTO_FEED : go to feed state after this
+ * @return PICOPAM_GOTO_SCHEDULE : flush received
+ * @return PICODATA_PU_ERROR : errors
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pamDoPreSyll(register picodata_ProcessingUnit this)
+{
+ pam_subobj_t *pam;
+ pico_status_t sResult;
+ picoos_uint8 bWr, nRc;
+ picoos_uint8 *qItem;
+ nRc = PICOPAM_PRE_SYLL_ENDED;
+ pam = (pam_subobj_t *) this->subObj;
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ /*regenerate initial items before the phonemes*/
+ if (((qItem = pamPopAttachedSy0(this)) != NULL) && !((qItem[0]
+ == PICODATA_ITEM_BOUND) && (qItem[1]
+ == PICODATA_ITEMINFO1_BOUND_SBEG))) {
+ if (isItemToPut(qItem)) {
+ pam_put_qItem(qItem, pam->outBuf, pam->outWritePos, &bWr);/*popped item has to be sent to next PU*/
+ pam->outWritePos += bWr;
+ nRc = PICOPAM_GOTO_FEED;
+ }
+
+ if (is_pam_command(qItem) == TRUE) {
+ nRc = pamDoCommand(this, qItem); /*popped item is a PAM command : do it NOW!!*/
+ if ((nRc == PICOPAM_FLUSH_RECEIVED) || (nRc == PICODATA_PU_ERROR)) {
+ /*FLUSH command RECEIVED or errors: stop ALL PROCESSING*/
+ return nRc;
+ }
+ }
+ pam->nCurrAttachedItem++;
+ if (nRc == 0)
+ return PICOPAM_CONTINUE;
+ else
+ return nRc;
+ }
+ /*SBEG item management*/
+ if ((qItem != NULL) && (qItem[0] == PICODATA_ITEM_BOUND) && (qItem[1]
+ == PICODATA_ITEMINFO1_BOUND_SBEG)) {
+ sResult = pam_put_qItem(qItem, pam->outBuf, pam->outWritePos, &bWr);
+ pam->outWritePos += bWr;
+ pam->nCurrAttachedItem++;
+ nRc = PICOPAM_GOTO_FEED;
+ }
+ return nRc;
+}/*pamDoPreSyll*/
+
+/**
+ * performs a step of the pam processing
+ * @param this : Pam item subobject pointer
+ * @param mode : mode for the PU
+ * @param *numBytesOutput : pointer to output number fo bytes produced
+ * @return PICODATA_PU_IDLE : nothing to do
+ * @return PICODATA_PU_BUSY : still tasks undergoing
+ * @return PICODATA_PU_ERROR : errors on processing
+ * @callgraph
+ * @callergraph
+ */
+static picodata_step_result_t pam_step(register picodata_ProcessingUnit this,
+ picoos_int16 mode, picoos_uint16 * numBytesOutput)
+{
+
+ register pam_subobj_t * pam;
+
+ pico_status_t sResult;
+ picoos_uint16 blen, numinb, numoutb;
+ pico_status_t rv;
+ picoos_uint8 bWr;
+ picoos_uint8 bForcedItem[4];
+ picoos_uint8 *qItem;
+
+ numinb = 0;
+ numoutb = 0;
+ rv = PICO_OK;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+ mode = mode; /* avoid warning "var not used in this function"*/
+ /*Init number of output bytes*/
+ *numBytesOutput = 0;
+
+ while (1) { /* exit via return */
+
+ PICODBG_DEBUG(("pam_step -- doing state %i",pam->procState));
+
+ switch (pam->procState) {
+
+ case PICOPAM_COLLECT:
+ /* *************** item collector ***********************************/
+ /*collecting items from the PU input buffer*/
+ sResult = picodata_cbGetItem(this->cbIn,
+ &(pam->inBuf[pam->inWritePos]), pam->inBufSize
+ - pam->inWritePos, &blen);
+ if (sResult != PICO_OK) {
+ if (sResult == PICO_EOF) {
+ /*no items available : remain in state 0 and return idle*/
+ return PICODATA_PU_IDLE;
+ } else {
+ /*errors : remain in state 0 and return error*/
+ PICODBG_DEBUG(("pam_step(PICOPAM_COLLECT) -- Errors on item buffer input, status: %d",sResult));
+ return PICODATA_PU_ERROR;
+ }
+ }
+
+ PICODBG_DEBUG(("pam_step -- got item, status: %d",sResult));
+ sResult = picodata_is_valid_item(
+ &(pam->inBuf[pam->inWritePos]), blen);
+ if (sResult != TRUE) {
+ /*input item is not valid : consume the input item and stay in COLLECT*/
+ pam->inWritePos += blen;
+ pam->inReadPos += blen;
+ if (pam->inReadPos >= pam->inWritePos) {
+ pam->inReadPos = 0;
+ pam->inWritePos = 0;
+ }PICODBG_DEBUG(("pam_step -- item is not valid, type: %d",pam->inBuf[pam->inWritePos]));
+ return PICODATA_PU_BUSY;
+ }
+
+ /*update input write pointer + move to "schedule" state*/
+ pam->inWritePos += blen;
+ pam->procState = PICOPAM_SCHEDULE;
+ return PICODATA_PU_BUSY;
+
+ case PICOPAM_SCHEDULE:
+ /* check out if more items are available */
+ if (pam->inReadPos >= pam->inWritePos) {
+ /*no more items : back to collect state*/
+ pam->procState = PICOPAM_COLLECT;
+ return PICODATA_PU_BUSY;
+ }
+ /* we have one full valid item, with len>0 starting at
+ pam->inBuf[pam->inReadPos]; here we decide how to elaborate it */
+
+ /* PLAY management */
+ if (is_pam_play_command(&(pam->inBuf[pam->inReadPos])) == TRUE) {
+ /*consume the input item : it has been managed*/
+ pam->inReadPos += pam->inBuf[pam->inReadPos + 3]
+ + sizeof(picodata_itemhead_t);
+ if (pam->inReadPos >= pam->inWritePos) {
+ pam->inReadPos = 0;
+ pam->inWritePos = 0;
+ }
+ /*stay in schedule*/
+ return PICODATA_PU_BUSY;
+ }
+
+ if (pam_check_immediate(this, &(pam->inBuf[pam->inReadPos]))) {
+ /* item has to be sent to next PU NOW : switch to "immediate" state */
+ pam->procState = PICOPAM_IMMEDIATE;
+ return PICODATA_PU_BUSY;
+ }
+ if (pamCheckResourceLimits(this, &(pam->inBuf[pam->inReadPos]))) {
+ /* item would not fit into local buffers -->> free some space -->>
+ switch to "force term" state */
+ pam->procState = PICOPAM_FORWARD_FORCE_TERM;
+ return PICODATA_PU_BUSY;
+ }
+
+ if (pam_deal_with(&(pam->inBuf[pam->inReadPos]))) {
+ /* item has to be managed by the "forward" state : switch to forward state*/
+ pam->procState = PICOPAM_FORWARD;
+ return PICODATA_PU_BUSY;
+ }
+
+ if (pam_hastobe_queued(this, &(pam->inBuf[pam->inReadPos]))) {
+ /* item is not for PAM so it has to be queued internally */
+ pam_queue(this, &(pam->inBuf[pam->inReadPos]));
+ /*consume the input item : it has been queued*/
+ pam->inReadPos += pam->inBuf[pam->inReadPos + 3]
+ + sizeof(picodata_itemhead_t);
+ if (pam->inReadPos >= pam->inWritePos) {
+ pam->inReadPos = 0;
+ pam->inWritePos = 0;
+ }
+ return PICODATA_PU_BUSY;
+ }
+ /*if we get here something wrong happened. Being the the item valid,
+ switch to "immediate" state -> send it to next PU -> */
+ PICODBG_DEBUG(("pam_step (PICOPAM_SCHEDULE) -- unexpected item is sent to next PU !!"));
+ pam->procState = PICOPAM_IMMEDIATE;
+ return PICODATA_PU_BUSY;
+ break; /*PICOPAM_SCHEDULE*/
+
+ case PICOPAM_FORWARD:
+ /*we have one full valid item, with len>0 starting at pam->inBuf[pam->inReadPos].
+ furthermore this item should be in the set {BOUND,SYLL}.
+ No other items should arrive here*/
+ sResult = pam_adapter_forward_step(this,
+ &(pam->inBuf[pam->inReadPos]));
+ /*decide if this item has to be queued for later re-synchronization
+ normally this is only done for SEND/TERM items*/
+ if (pam_hastobe_queued(this, &(pam->inBuf[pam->inReadPos]))) {
+ /*item has to be queued iternally in local storage*/
+ pam_queue(this, &(pam->inBuf[pam->inReadPos]));
+ }
+ /*now assign next state according to Forward results*/
+ switch (sResult) {
+ case PICOPAM_READY:
+ pam->needMoreInput = FALSE;
+ /*consume the input item : it has already been stored*/
+ pam->inReadPos += pam->inBuf[pam->inReadPos + 3]
+ + sizeof(picodata_itemhead_t);
+ if (pam->inReadPos >= pam->inWritePos) {
+ pam->inReadPos = 0;
+ pam->inWritePos = 0;
+ }
+ /*activate backward processing*/
+ sResult = pam_adapter_backward_step(this);
+ if (sResult == PICO_OK) {
+ pam->procState = PICOPAM_PROCESS;
+ return PICODATA_PU_BUSY;
+ } else {
+ PICODBG_DEBUG(("pam_step (PICOPAM_FORWARD) -- wrong return from BackwardStep: %d -- Buffered sentence will be discarded",sResult));
+ pam_reset_processors(this);
+ pam->nLastAttachedItemId = pam->nCurrAttachedItem
+ = 0;
+ pam->nAttachedItemsSize = 0;
+
+ pam->procState = PICOPAM_SCHEDULE;
+ return PICODATA_PU_BUSY;
+ }
+ break;
+
+ case PICOPAM_MORE:
+ pam->needMoreInput = TRUE;
+ /*consume the input item : it has already been stored*/
+ pam->inReadPos += pam->inBuf[pam->inReadPos + 3]
+ + sizeof(picodata_itemhead_t);
+ if (pam->inReadPos >= pam->inWritePos) {
+ /*input is finished and PAM need more data :
+ clenaup input buffer + switch state back to "schedule state"
+ */
+ pam->inReadPos = 0;
+ pam->inWritePos = 0;
+ pam->procState = PICOPAM_SCHEDULE;
+ return PICODATA_PU_ATOMIC;
+ } else {
+ /*input is not finished and need more data :
+ remain in state "PICOPAM_FORWARD" */
+ return PICODATA_PU_ATOMIC;
+ }
+ break;
+
+ case PICOPAM_NA:
+ default:
+ /*this item has not been stored in internal buffers:
+ assign this item to the management of
+ "immediate" state*/
+ pam->procState = PICOPAM_IMMEDIATE;
+ return PICODATA_PU_BUSY;
+ break;
+ } /*end switch sResult*/
+ break; /*PICOPAM_FORWARD*/
+
+ case PICOPAM_FORWARD_FORCE_TERM:
+ /*we have one full valid item, with len>0
+ starting at pam->inBuf[pam->inReadPos] but we decided
+ to force a TERM item before, without losing the item in
+ inBuf[inReadPos] : --> generate a TERM item and do the
+ forward processing */
+ pam_put_term(bForcedItem, 0, &bWr);
+ sResult = pam_adapter_forward_step(this, &(bForcedItem[0]));
+ switch (sResult) {
+ case PICOPAM_READY:
+ pam_queue(this, &(bForcedItem[0]));
+ /*activate backward processing*/
+ sResult = pam_adapter_backward_step(this);
+ if (sResult == PICO_OK) {
+ pam->procState = PICOPAM_PROCESS;
+ return PICODATA_PU_BUSY;
+ } else {
+ PICODBG_DEBUG(("pam_step (PICOPAM_FORWARD_FORCE_TERM) -- wrong return from BackwardStep: %d -- Buffered sentence will be discarded",sResult));
+ pam_reset_processors(this);
+ pam->nLastAttachedItemId = pam->nCurrAttachedItem
+ = 0;
+ pam->nAttachedItemsSize = 0;
+
+ pam->procState = PICOPAM_SCHEDULE;
+ return PICODATA_PU_BUSY;
+ }
+ break;
+
+ default:
+ PICODBG_DEBUG(("pam_step (PICOPAM_FORWARD_FORCE_TERM) -- Forced a TERM but processing do not appear to end -- Buffered sentence will be discarded",sResult));
+ pam_reset_processors(this);
+ pam->nLastAttachedItemId = pam->nCurrAttachedItem = 0;
+ pam->nAttachedItemsSize = 0;
+
+ pam->procState = PICOPAM_SCHEDULE;
+ return PICODATA_PU_BUSY;
+ break;
+
+ } /*end switch sResult*/
+ break; /*PICOPAM_FORWARD_FORCE_TERM*/
+
+ case PICOPAM_PROCESS:
+
+ if ((PICOPAM_FRAME_ITEM_SIZE + 4) > (pam->outBufSize
+ - pam->outWritePos)) {
+ /*WARNING (buffer overflow): leave status unchanged until output buffer free */
+ return PICODATA_PU_BUSY;
+ }
+
+ if (pam->nCurrSyllable == 0) {
+ sResult = pamDoPreSyll(this);
+ if (sResult == PICOPAM_GOTO_FEED) {
+ /*
+ items pushed to output buffer :
+ switch to "feed" but then back
+ to "process"
+ */
+ pam->retState = PICOPAM_PROCESS;
+ pam->procState = PICOPAM_FEED;
+ return PICODATA_PU_BUSY;
+ }
+ if (sResult == PICOPAM_CONTINUE) {
+ /*
+ items processed (maybe commands) :
+ return (maybe we need to process other
+ items in pre_syll) and then back to "process"
+ */
+ pam->retState = PICOPAM_PROCESS;
+ pam->procState = PICOPAM_PROCESS;
+ return PICODATA_PU_BUSY;
+ }
+
+ if ((sResult == PICOPAM_FLUSH_RECEIVED) || (sResult
+ == PICODATA_PU_ERROR)) {
+ /*
+ items processed were a flush or
+ problems found: switch to "schedule"
+ and abort all processing
+ */
+ pam->retState = PICOPAM_SCHEDULE;
+ pam->procState = PICOPAM_SCHEDULE;
+ return PICODATA_PU_BUSY;
+ }
+ if (sResult == PICOPAM_PRE_SYLL_ENDED) {
+ /*
+ we get here when pam->nCurrSyllable==0 and
+ no more items to be processed before the syllable
+ */
+ sResult = sResult;
+ }
+ }
+
+ if (pamHasToProcess(this)) {
+ if (pamPhoneProcess(this) == PICO_OK) {
+ sResult = pamUpdateProcess(this);
+ pam->procState = PICOPAM_FEED; /*switch to feed*/
+ return PICODATA_PU_BUSY;
+ } else {
+ PICODBG_DEBUG(("pam_step(PICOPAM_PROCESS) --- NULL return from pamPhoneProcess"));
+ return PICODATA_PU_ERROR;
+ }
+ }
+
+ if (pamHasToPop(this) != FALSE) {
+ if ((qItem = pamPopItem(this)) == NULL) {
+ PICODBG_DEBUG(("pam_step(PICOPAM_PROCESS) --- NULL return from pamPopItem"));
+ return PICODATA_PU_ERROR;
+ }
+
+ if (isItemToPut(qItem)) {
+ /*popped item has to be sent to next PU*/
+ sResult = pam_put_qItem(qItem, pam->outBuf,
+ pam->outWritePos, &bWr);
+ if (sResult != PICO_OK) {
+ PICODBG_DEBUG(("pam_step(PICOPAM_PROCESS) --- Error on writing item to output buffer"));
+ return PICODATA_PU_ERROR;
+ }
+ pam->outWritePos += bWr; /*item write ok*/
+ pam->procState = PICOPAM_FEED; /*switch to feed*/
+ }
+
+ /*moved command processing here (after pam_put_qItem) because of FLUSH command could erase
+ * the syllable structure and make it impossible to transmit the flush to other PUs*/
+ if (is_pam_command(qItem) == TRUE) {
+ sResult = pamDoCommand(this, qItem); /*popped item is a PAM command : do it NOW!!*/
+ if ((sResult == PICOPAM_FLUSH_RECEIVED) || (sResult
+ == PICODATA_PU_ERROR)) {
+ pam->retState = PICOPAM_SCHEDULE;
+ pam->procState = PICOPAM_SCHEDULE; /*switch to schedule */
+ return PICODATA_PU_BUSY;
+ }
+ }
+ /*update PAM status: if more items attached to the current syllable
+ stay in current syllable, otherwise move to next syllable and switch
+ to processing phones */
+ sResult = pamUpdateProcess(this); /*both "doCommand" or "put" : update PAM status*/
+ return PICODATA_PU_BUSY;
+ } else {
+ pam->procState = PICOPAM_SCHEDULE; /*switch to schedule */
+ return PICODATA_PU_BUSY;
+ }
+
+ break; /*PICOPAM_PROCESS*/
+
+ case PICOPAM_IMMEDIATE:
+ /* *** item is output NOW!!! */
+ /*context: full valid item, with len> starting at pam->inBuf[pam->inReadPos]*/
+ numinb = PICODATA_ITEM_HEADSIZE
+ + pam->inBuf[pam->inReadPos + 3];
+ sResult = picodata_copy_item(&(pam->inBuf[pam->inReadPos]),
+ numinb, &(pam->outBuf[pam->outWritePos]),
+ pam->outBufSize - pam->outWritePos, &numoutb);
+
+ if (sResult == PICO_OK) {
+ pam->inReadPos += numinb;
+ if (pam->inReadPos >= pam->inWritePos) {
+ pam->inReadPos = 0;
+ pam->inWritePos = 0;
+ pam->needMoreInput = FALSE;
+ }
+ pam->outWritePos += numoutb;
+ pam->procState = PICOPAM_FEED; /*switch to FEED state*/
+ pam->retState = PICOPAM_SCHEDULE; /*back to SCHEDULE after FEED*/
+ } else {
+ /*
+ PICO_EXC_BUF_IGNORE
+ PICO_EXC_BUF_UNDERFLOW
+ PICO_EXC_BUF_OVERFLOW
+ */
+ PICODBG_DEBUG(("pam_step(PICOPAM_IMMEDIATE) --- wrong return from picodata_copy_item:%d",sResult));
+ return PICODATA_PU_ERROR;
+ }
+ return PICODATA_PU_BUSY;
+ break; /*PICOPAM_IMMEDIATE*/
+
+ case PICOPAM_FEED:
+ /* *************** item output/feeding ***********************************/
+ /*feeding items to PU output buffer*/
+ sResult = picodata_cbPutItem(this->cbOut,
+ &(pam->outBuf[pam->outReadPos]), pam->outWritePos
+ - pam->outReadPos, &numoutb);
+ PICODBG_DEBUG(("pam_step -- put item, status: %d",sResult));
+ if (PICO_OK == sResult) {
+
+ PICODATA_INFO_ITEM(this->voice->kbArray[PICOKNOW_KBID_DBG],
+ (picoos_uint8 *)"pam: ",
+ pam->outBuf + pam->outReadPos, pam->outBufSize);
+
+ pam->outReadPos += numoutb;
+ *numBytesOutput = numoutb;
+ if (pam->outReadPos >= pam->outWritePos) {
+ /*reset the output pointers*/
+ pam->outReadPos = 0;
+ pam->outWritePos = 0;
+ /*switch to appropriate state*/
+ switch (pam->retState) {
+ case PICOPAM_IMMEDIATE:
+ pam->procState = PICOPAM_IMMEDIATE;
+ pam->retState = PICOPAM_SCHEDULE;
+ return PICODATA_PU_BUSY;
+ break;
+ case PICOPAM_PLAY:
+ pam->procState = PICOPAM_PLAY;
+ pam->retState = PICOPAM_SCHEDULE;
+ return PICODATA_PU_BUSY;
+ break;
+ default:
+ break;
+ }
+ /*Define next state
+ a)process (if current sentence has more data to process)
+ b)schedule (no more data to process in current sentence)
+ NOTE : case b)also happens when dealing with non BOUND/SYLL items*/
+ if ((pamHasToProcess(this)) || (pamHasToPop(this))) {
+ pam->procState = PICOPAM_PROCESS;
+ } else {
+ pam->nCurrSyllable = -1;
+ pam_reset_processors(this);
+ pam->nLastAttachedItemId = pam->nCurrAttachedItem
+ = 0;
+ pam->nAttachedItemsSize = 0;
+
+ pam->nSyllPhoneme = 0;
+ pam->procState = PICOPAM_SCHEDULE;
+ }
+ }
+ return PICODATA_PU_BUSY;
+
+ } else if (PICO_EXC_BUF_OVERFLOW == sResult) {
+
+ PICODBG_DEBUG(("pam_step ** feeding, overflow, PICODATA_PU_OUT_FULL"));
+ return PICODATA_PU_OUT_FULL;
+
+ } else if ((PICO_EXC_BUF_UNDERFLOW == sResult)
+ || (PICO_ERR_OTHER == sResult)) {
+
+ PICODBG_DEBUG(("pam_step ** feeding problem, discarding item"));
+ pam->outReadPos = 0;
+ pam->outWritePos = 0;
+ pam->procState = PICOPAM_COLLECT;
+ return PICODATA_PU_ERROR;
+
+ }
+ break; /*PICOPAM_FEED*/
+
+ default:
+ /*NOT feeding items*/
+ sResult = PICO_EXC_BUF_IGNORE;
+ break;
+ }/*end switch*/
+ return PICODATA_PU_BUSY; /*check if there is more data to process after feeding*/
+
+ }/*end while*/
+ return PICODATA_PU_IDLE;
+}/*pam_step*/
+
+/**
+ * performs one step of a PamTree
+ * @param this : Pam item subobject pointer
+ * @param dtpam : the Pam decision tree
+ * @param *invec : the input vector pointer
+ * @param inveclen : length of the input vector
+ * @param *dtres : the classification result
+ * @return dtres->set : the result of tree traversal
+ * @callgraph
+ * @callergraph
+ */
+static picoos_uint8 pam_do_tree(register picodata_ProcessingUnit this,
+ const picokdt_DtPAM dtpam, const picoos_uint8 *invec,
+ const picoos_uint8 inveclen, picokdt_classify_result_t *dtres)
+{
+ picoos_uint8 okay;
+
+ okay = TRUE;
+ /* construct input vector, which is set in dtpam */
+ if (!picokdt_dtPAMconstructInVec(dtpam, invec, inveclen)) {
+ /* error constructing invec */
+ PICODBG_WARN(("problem with invec"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_INVECTOR, NULL, NULL);
+ okay = FALSE;
+ }
+ /* classify */
+ if (okay && (!picokdt_dtPAMclassify(dtpam))) {
+ /* error doing classification */
+ PICODBG_WARN(("problem classifying"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_CLASSIFICATION, NULL,
+ NULL);
+ okay = FALSE;
+ }
+ /* decompose */
+ if (okay && (!picokdt_dtPAMdecomposeOutClass(dtpam, dtres))) {
+ /* error decomposing */
+ PICODBG_WARN(("problem decomposing"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_OUTVECTOR, NULL, NULL);
+ okay = FALSE;
+ }
+
+ PICODBG_TRACE(("dtpam output class: %d", dtres->class));
+
+ return dtres->set;
+}/*pam_do_tree*/
+
+/**
+ * returns the carrier vowel id inside a syllable
+ * @param this : Pam item subobject pointer
+ * @param item : the full syllable item
+ * @param *pos : pointer to the variable to receive the position of the carrier vowel
+ * @return the phonetic id for the carrier vowel inside the syllable
+ * @callgraph
+ * @callergraph
+ */
+static picoos_uint8 pam_get_vowel_name(register picodata_ProcessingUnit this,
+ picoos_uint8 *item, picoos_uint8 *pos)
+{
+ pam_subobj_t *pam;
+ picoos_uint8 *phon, nI, nCond1;
+ if (NULL == this || NULL == this->subObj) {
+ return 0;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+
+ if (item == NULL)
+ return 0;
+ if (item[3] == 0)
+ return 0;
+ phon = &item[4];
+ for (nI = 0; nI < item[3]; nI++) {
+ nCond1 = picoktab_isSyllCarrier(pam->tabphones, phon[nI]);
+ if (nCond1) {
+ *pos = nI;
+ return phon[nI];
+ }
+ }
+ return 0;
+}/*pam_get_vowel_name */
+
+/**
+ * returns the pause phone id in the current ph.alphabet
+ * @param this : Pam sub object pointer
+ * @return the (numeric) phonetic id of the pause phone in current phonetic alphabet
+ * @return 0 : errors on getting the pam subobject pointer
+ * @callgraph
+ * @callergraph
+ */
+static picoos_uint8 pam_get_pause_id(register picodata_ProcessingUnit this)
+{
+ picoos_uint8 nVal1;
+ /*picoos_uint8 nVal2; */
+ pam_subobj_t *pam;
+ if (NULL == this || NULL == this->subObj) {
+ return 0;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+ nVal1 = picoktab_getPauseID(pam->tabphones);
+ return nVal1;
+}/*pam_get_pause_id */
+
+/**
+ * returns the pam sentence type (declarative, interrogative...)
+ * @param iteminfo1 : the boundary item info 1
+ * @param iteminfo2 : the boundary item info 2
+ * @return the sentence type suitably encoded for trees
+ * @callgraph
+ * @callergraph
+ */
+static picoos_uint8 pam_map_sentence_type(picoos_uint8 iteminfo1,
+ picoos_uint8 iteminfo2)
+{
+ switch (iteminfo2) {
+ case PICODATA_ITEMINFO2_BOUNDTYPE_P:
+ return PICOPAM_DECLARATIVE;
+ case PICODATA_ITEMINFO2_BOUNDTYPE_T:
+ return PICOPAM_DECLARATIVE;
+ case PICODATA_ITEMINFO2_BOUNDTYPE_Q:
+ return PICOPAM_INTERROGATIVE;
+ case PICODATA_ITEMINFO2_BOUNDTYPE_E:
+ return PICOPAM_DECLARATIVE;
+ default:
+ return PICOPAM_DECLARATIVE;
+ }
+ iteminfo1 = iteminfo1; /* avoid warning "var not used in this function"*/
+ return PICOPAM_DECLARATIVE;
+}/*pam_map_sentence_type */
+
+/**
+ * returns the pam phrase type
+ * @param iteminfo1 : the boundary item info 1
+ * @param iteminfo2 : the boundary item info 2
+ * @return the phrase type suitably encoded for trees
+ * @callgraph
+ * @callergraph
+ */
+static picoos_uint8 pam_map_phrase_type(picoos_uint8 iteminfo1,
+ picoos_uint8 iteminfo2)
+{
+
+ switch (iteminfo2) {
+ case PICODATA_ITEMINFO2_BOUNDTYPE_P:
+ switch (iteminfo1) {
+ case PICODATA_ITEMINFO1_BOUND_PHR1:
+# ifdef PAM_PHR2_WITH_PR1
+ case PICODATA_ITEMINFO1_BOUND_PHR2:
+# endif
+ return PICOPAM_P; /*current_prhase type = "P" (encoded to 1) */
+ break;
+ case PICODATA_ITEMINFO1_BOUND_PHR3:
+# ifdef PAM_PHR2_WITH_PR3
+ case PICODATA_ITEMINFO1_BOUND_PHR2 :
+# endif
+ return PICOPAM_p; /*current_prhase type = "p" (encoded to 2) */
+ break;
+ case PICODATA_ITEMINFO1_BOUND_SBEG:
+ return PICOPAM_P; /*current_prhase type = "P" (encoded to 1) */
+ break;
+ default:
+ PICODBG_DEBUG(("Map pam_map_phrase_type : unexpected iteminfo1"));
+ return PICOPAM_P; /*current_prhase type = "P" (encoded to 1) */
+ break;
+ }
+ case PICODATA_ITEMINFO2_BOUNDTYPE_T:
+ return PICOPAM_T; /*current_prhase type = "T" (encoded to 0) */
+ break;
+ case PICODATA_ITEMINFO2_BOUNDTYPE_E:
+ return PICOPAM_T; /*current_prhase type = "T" (encoded to 0) */
+ break;
+ case PICODATA_ITEMINFO2_BOUNDTYPE_Q:
+ return PICOPAM_Y; /*current_prhase type = "T" (encoded to 0) */
+ break;
+ default:
+ PICODBG_DEBUG(("Map pam_map_phrase_type : unexpected iteminfo2"));
+ return PICOPAM_T; /*current_prhase type = "T" (encoded to 0) */
+ break;
+ }PICODBG_DEBUG(("Map pam_map_phrase_type : unexpected iteminfo2"));
+ return PICOPAM_T; /*current_prhase type = "T" (encoded to 0) */
+
+}/*pam_map_phrase_type */
+
+/**
+ * does the cleanup of the sub object processors flags at sentence start
+ * @param this : pointer to PAM PU sub object pointer
+ * @return PICO_OK : reset OK
+ * @return PICO_ERR_OTHER : errors on getting pam sub obj pointer
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_reset_processors(register picodata_ProcessingUnit this)
+{
+ pam_subobj_t *pam;
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+
+ pam->nCurrSyllable = -1;
+ pam->nTotalPhonemes = pam->nSyllPhoneme = pam->nCurrPhoneme
+ = pam->nTotalSyllables = pam->sType = pam->pType = 0;
+ pam->dRest = 0.0f;
+ /*set all to 0*/
+ pam->a3_overall_syllable = pam->a3_primary_phrase_syllable = pam->b4_b5_syllable =
+ pam->b6_b7_syllable = pam->b6_b7_state = pam->b8_b9_stressed_syllable =
+ pam->b10_b11_accented_syllable = pam->b12_b13_syllable = pam->b12_b13_state =
+ pam->b14_b15_syllable = pam->b14_b15_state = pam->b17_b19_syllable =
+ pam->b17_b19_state = pam->b18_b20_b21_syllable = pam->b18_b20_b21_state =
+ pam->c3_overall_syllable= pam->c3_primary_phrase_syllable = pam->d2_syllable_in_word =
+ pam->d2_prev_syllable_in_word = pam->d2_current_primary_phrase_word = pam->e1_syllable_word_start =
+ pam->e1_syllable_word_end= pam->e1_content = pam->e2_syllable_word_start =
+ pam->e2_syllable_word_end= pam->e3_e4_word = pam->e3_e4_state =
+ pam->e5_e6_content_word = pam->e5_e6_content = pam->e7_e8_word =
+ pam->e7_e8_content = pam->e7_e8_state = pam->e9_e11_word =
+ pam->e9_e11_saw_word = pam->e9_e11_state = pam->e10_e12_e13_word =
+ pam->e10_e12_e13_state = pam->e10_e12_e13_saw_word = pam->f2_overall_word =
+ pam->f2_word_syllable = pam->f2_next_word_syllable = pam->f2_current_primary_phrase_word =
+ pam->g1_current_secondary_phrase_syllable = pam->g1_current_syllable =
+ pam->g2_current_secondary_phrase_word = pam->g2_current_word =
+ pam->h1_current_secondary_phrase_syll = pam->h2_current_secondary_phrase_word =
+ pam->h3_h4_current_secondary_phrase_word = pam->h5_current_phrase_type =
+ pam->h5_syllable = pam->h5_state = pam->i1_secondary_phrase_syllable =
+ pam->i1_next_secondary_phrase_syllable = pam->i2_secondary_phrase_word =
+ pam->i2_next_secondary_phrase_word = pam->j1_utterance_syllable =
+ pam->j2_utterance_word = pam->j3_utterance_sec_phrases = 0;
+ /*Override 0 with 1*/
+ pam->b4_b5_syllable = pam->b17_b19_syllable = pam->b18_b20_b21_syllable =
+ pam->e9_e11_word = pam->e10_e12_e13_word = pam->e7_e8_word =
+ pam->h2_current_secondary_phrase_word = 1;
+ /*Override 0 with -1*/
+ pam->e1_syllable_word_start = pam->e1_syllable_word_end = pam->e2_syllable_word_start =
+ pam->e2_syllable_word_end = -1;
+
+ return PICO_OK;
+}/*pam_reset_processors*/
+
+/**
+ * does the cleanup of the sub object processors flags before the backward step
+ * @param this : pointer to PAM PU sub object pointer
+ * @return PICO_OK : reset OK
+ * @return PICO_ERR_OTHER : errors on getting pam sub obj pointer
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_reset_processors_back(
+ register picodata_ProcessingUnit this)
+{
+ pam_subobj_t *pam;
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+
+ /*set all to 0*/
+ pam->a3_overall_syllable
+ = pam->a3_primary_phrase_syllable
+ = pam->b4_b5_syllable
+ = pam->b6_b7_syllable
+ = pam->b6_b7_state
+ = pam->b8_b9_stressed_syllable
+ = pam->b10_b11_accented_syllable
+ = pam->b12_b13_syllable
+ = pam->b12_b13_state
+ = pam->b14_b15_syllable
+ = pam->b14_b15_state
+ = pam->b17_b19_syllable
+ = pam->b17_b19_state
+ = pam->b18_b20_b21_syllable
+ = pam->b18_b20_b21_state
+ = pam->c3_overall_syllable
+ = pam->c3_primary_phrase_syllable
+ = pam->d2_syllable_in_word
+ = pam->d2_prev_syllable_in_word
+ = pam->d2_current_primary_phrase_word
+ = pam->e1_syllable_word_start
+ = pam->e1_syllable_word_end
+ = pam->e1_content
+ = pam->e2_syllable_word_start
+ = pam->e2_syllable_word_end
+ = pam->e3_e4_word
+ = pam->e3_e4_state
+ = pam->e5_e6_content_word
+ = pam->e5_e6_content
+ = pam->e7_e8_word
+ = pam->e7_e8_content
+ = pam->e7_e8_state
+ = pam->e9_e11_word
+ = pam->e9_e11_saw_word
+ = pam->e9_e11_state
+ = pam->e10_e12_e13_word
+ = pam->e10_e12_e13_state
+ = pam->e10_e12_e13_saw_word
+ = pam->f2_overall_word
+ = pam->f2_word_syllable
+ = pam->f2_next_word_syllable
+ = pam->f2_current_primary_phrase_word
+ = pam->g1_current_secondary_phrase_syllable
+ = pam->g1_current_syllable
+ = pam->g2_current_secondary_phrase_word
+ = pam->g2_current_word
+ = pam->h1_current_secondary_phrase_syll
+ = pam->h2_current_secondary_phrase_word
+ = pam->h3_h4_current_secondary_phrase_word
+ = pam->h5_current_phrase_type
+ = pam->h5_state
+ = pam->i1_secondary_phrase_syllable
+ = pam->i1_next_secondary_phrase_syllable
+ = pam->i2_secondary_phrase_word
+ = pam->i2_next_secondary_phrase_word
+ = 0;
+ /*Override 0 with 1*/
+ pam->b4_b5_syllable = pam->b17_b19_syllable = pam->b18_b20_b21_syllable
+ = pam->e9_e11_word = pam->e10_e12_e13_word = pam->e7_e8_word
+ = pam->h2_current_secondary_phrase_word = 1;
+ /*Override 0 with -1*/
+ pam->e1_syllable_word_start = pam->e1_syllable_word_end
+ = pam->e2_syllable_word_start = pam->e2_syllable_word_end = -1;
+
+ return PICO_OK;
+}/*pam_reset_processors_back*/
+
+/**
+ * processes an input event for a specific feature
+ * @param this : pointer to PAM PU sub object pointer
+ * @param nFeat : feature column to process
+ * @param event_type : event id among syll/boundprim/boundsec/boundword
+ * @param direction : forward(0)/backward(1)
+ * @return PICO_OK : process OK
+ * @return PICO_ERR_OTHER : errors on getting pam sub obj pointer
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_process_event_feature(
+ register picodata_ProcessingUnit this, picoos_uint8 nFeat,
+ picoos_uint8 event_type, picoos_uint8 direction)
+{
+ picoos_uint8 sDest, nI;
+ picoos_uint16 syllCurr;
+ pam_subobj_t *pam;
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+ syllCurr = pam->nCurrSyllable;
+ switch (nFeat) {
+ case A3:
+ /*processor for A3*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ if ((pam->sSyllFeats[pam->nCurrSyllable].phoneV[P1]
+ == 1) || (pam->a3_primary_phrase_syllable >= 1)) {
+ if (pam->a3_overall_syllable < 1)
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[A3]
+ = 0;
+ else
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[A3]
+ = pam->sSyllFeats[pam->nCurrSyllable
+ - 1].phoneV[B3];
+ } else {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[A3] = 0;
+ }
+ pam->a3_primary_phrase_syllable++;
+ pam->a3_overall_syllable++;
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND) {
+ pam->a3_primary_phrase_syllable = 0;
+ }
+ break;
+ case PICOPAM_DIR_BACK:
+ /*do nothing*/
+ break;
+ }
+ break;
+ case B1:
+ case B2:
+ case B3:
+ /*done in createSyllable*/
+ break;
+ case B4:/*processor for B4,B5*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ sDest = B4;
+ break;
+ case PICOPAM_DIR_BACK:
+ sDest = B5;
+ break;
+ default:
+ sDest = B4;
+ break;
+ }
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[P1] == 0) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = pam->b4_b5_syllable;
+ pam->b4_b5_syllable++;
+ } else {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest] = 0;
+ }
+ }
+ if ((event_type == PICOPAM_EVENT_W_BOUND) || (event_type
+ == PICOPAM_EVENT_S_BOUND) || (event_type
+ == PICOPAM_EVENT_P_BOUND)) {
+ pam->b4_b5_syllable = 1;
+ }
+ break;
+ case B5:/*processor for B5 : done in B4*/
+ break;
+ case B6:/*processor for B6,B7*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ sDest = B6;
+ break;
+ case PICOPAM_DIR_BACK:
+ sDest = B7;
+ break;
+ default:
+ sDest = B6;
+ break;
+ }
+ switch (pam->b6_b7_state) {
+ case 0:
+ if (event_type == PICOPAM_EVENT_SYLL)
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = PICOPAM_DONT_CARE_VALUE;
+ if (event_type == PICOPAM_EVENT_S_BOUND) {
+ pam->b6_b7_syllable = 1;
+ pam->b6_b7_state = 1;
+ }
+ break;
+ case 1:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = pam->b6_b7_syllable;
+ pam->b6_b7_syllable++;
+ }
+ if (event_type == PICOPAM_EVENT_S_BOUND) {
+ pam->b6_b7_syllable = 1;
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND) {
+ pam->b6_b7_state = 0;
+ }
+ break;
+ default:
+ break;
+ }
+ break;
+ case B7:/*Done in B6*/
+ break;
+ case B8:/*processor for B8,B9*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ sDest = B8;
+ break;
+ case PICOPAM_DIR_BACK:
+ sDest = B9;
+ break;
+ default:
+ sDest = B8;
+ break;
+ }
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = pam->b8_b9_stressed_syllable;
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[B1] == 1)
+ pam->b8_b9_stressed_syllable++;
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND) {
+ pam->b8_b9_stressed_syllable = 0;
+ }
+
+ break;
+ case B9:/*done in B8*/
+ break;
+ case B10:/*processor for B10, B11*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ sDest = B10;
+ break;
+ case PICOPAM_DIR_BACK:
+ sDest = B11;
+ break;
+ default:
+ sDest = B10;
+ break;
+ }
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = pam->b10_b11_accented_syllable;
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[B2] == 1)
+ pam->b10_b11_accented_syllable++;
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND) {
+ pam->b10_b11_accented_syllable = 0;
+ }
+ break;
+ case B11:/*done in B10*/
+ break;
+ case B12:/*processor for B12,B13*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ sDest = B12;
+ break;
+ case PICOPAM_DIR_BACK:
+ sDest = B13;
+ break;
+ default:
+ sDest = B12;
+ break;
+ }
+ switch (pam->b12_b13_state) {
+ case 0:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[B1] == 0)
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = PICOPAM_DONT_CARE_VALUE;
+ else {
+ pam->b12_b13_syllable = 0;
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = PICOPAM_DONT_CARE_VALUE;
+ pam->b12_b13_state = 1;
+ }
+ }
+ break;
+ case 1:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = pam->b12_b13_syllable;
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[B1] == 1)
+ pam->b12_b13_syllable = 0;
+ else
+ pam->b12_b13_syllable++;
+ pam->b12_b13_state = 2;
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND)
+ pam->b12_b13_state = 0;
+ break;
+ case 2:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = pam->b12_b13_syllable;
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[B1] == 1)
+ pam->b12_b13_syllable = 0;
+ else
+ pam->b12_b13_syllable++;
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND)
+ pam->b12_b13_state = 0;
+
+ break;
+ default:
+ break;
+ }
+ break;
+ case B13:/*done in B12*/
+ break;
+
+ case B14:/*processor for B14, B15*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ sDest = B14;
+ break;
+ case PICOPAM_DIR_BACK:
+ sDest = B15;
+ break;
+ default:
+ sDest = B14;
+ break;
+ }
+ switch (pam->b14_b15_state) {
+ case 0:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[B2] == 0)
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = PICOPAM_DONT_CARE_VALUE;
+ else {
+ pam->b14_b15_syllable = 0;
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = PICOPAM_DONT_CARE_VALUE;
+ pam->b14_b15_state = 1;
+ }
+ }
+ break;
+ case 1:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = pam->b14_b15_syllable;
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[B2] == 1)
+ pam->b14_b15_syllable = 0;
+ else
+ pam->b14_b15_syllable++;
+ pam->b14_b15_state = 2;
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND) {
+ pam->b14_b15_state = 0;
+ }
+ break;
+ case 2:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = pam->b14_b15_syllable;
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[B2] == 1)
+ pam->b14_b15_syllable = 0;
+ else
+ pam->b14_b15_syllable++;
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND) {
+ pam->b14_b15_state = 0;
+ }
+ break;
+ default:
+ break;
+ }
+ break;
+ case B15:/*Processor for B15 : done in B14*/
+ break;
+ case B16:/*done in createSyllable*/
+ break;
+ case B17:/*processor for B17, B19 unified */
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ switch (pam->b17_b19_state) {
+ case 0:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B17]
+ = PICOPAM_DONT_CARE_VALUE;
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B19]
+ = pam->b17_b19_syllable;
+ pam->b17_b19_syllable++;
+ }
+ if (((event_type == PICOPAM_EVENT_P_BOUND)
+ || (event_type == PICOPAM_EVENT_S_BOUND))
+ && (pam->b17_b19_syllable > 1)) {
+ if (event_type == PICOPAM_EVENT_P_BOUND)
+ pam->b17_b19_syllable = 1;
+ pam->b17_b19_state = 1;
+ }
+ break;
+ case 1:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B17]
+ = pam->b17_b19_syllable;
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B19]
+ = PICOPAM_DONT_CARE_VALUE;
+ pam->b17_b19_syllable++;
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND) {
+ pam->b17_b19_syllable = 1;
+ }
+ break;
+ default:
+ break;
+ }
+ break;
+ case PICOPAM_DIR_BACK:
+ /*do nothing*/
+ break;
+ }
+ break;
+ case B18:/*processor for B18, B20, B21 unfied*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:/*do nothing*/
+ break;
+ case PICOPAM_DIR_BACK:
+ switch (pam->b18_b20_b21_state) {
+ case 0:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B18]
+ = PICOPAM_DONT_CARE_VALUE;
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[P6]
+ == PICOPAM_DECLARATIVE) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B20]
+ = pam->b18_b20_b21_syllable;
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B21]
+ = PICOPAM_DONT_CARE_VALUE;
+ } else {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B20]
+ = PICOPAM_DONT_CARE_VALUE;
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B21]
+ = pam->b18_b20_b21_syllable;
+ }
+ pam->b18_b20_b21_syllable++;
+ }
+ if (((event_type == PICOPAM_EVENT_P_BOUND)
+ || (event_type == PICOPAM_EVENT_S_BOUND))
+ && (pam->b18_b20_b21_syllable > 1)) {
+ if (event_type == PICOPAM_EVENT_P_BOUND)
+ pam->b18_b20_b21_syllable = 1;
+ pam->b18_b20_b21_state = 1;
+ }
+ break;
+ case 1:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B18]
+ = pam->b18_b20_b21_syllable;
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B20]
+ = PICOPAM_DONT_CARE_VALUE;
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B21]
+ = PICOPAM_DONT_CARE_VALUE;
+ pam->b18_b20_b21_syllable++;
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND) {
+ pam->b18_b20_b21_syllable = 1;
+ }
+ break;
+ default:
+ break;
+ }
+ break;
+ }
+ break;
+ case B19:/*processor for B19 : done in B17*/
+ break;
+ case B20:/*processor for B20 : done in B18*/
+ break;
+ case B21:/*processor for B21 : done in B18*/
+ break;
+ case C3:/*processor for C3*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ /*do nothing*/
+ break;
+ case PICOPAM_DIR_BACK:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ if ((pam->sSyllFeats[pam->nCurrSyllable].phoneV[P1]
+ == 1) || (pam->c3_primary_phrase_syllable >= 1)) {
+ if (pam->c3_overall_syllable < 1)
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[C3]
+ = 0;
+ else
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[C3]
+ = pam->sSyllFeats[pam->nCurrSyllable
+ + 1].phoneV[B3];
+ } else {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[C3] = 0;
+ }
+ pam->c3_primary_phrase_syllable++;
+ pam->c3_overall_syllable++;
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND) {
+ pam->c3_primary_phrase_syllable = 0;
+ }
+ break;
+ }
+ break;
+ case D2:/*processor for D2*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ if ((pam->sSyllFeats[pam->nCurrSyllable].phoneV[P1]
+ == 1) || (pam->d2_current_primary_phrase_word
+ >= 1))
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[D2]
+ = pam->d2_prev_syllable_in_word;
+ else
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[D2] = 0;
+
+ pam->d2_syllable_in_word++;
+ }
+ if ((event_type == PICOPAM_EVENT_W_BOUND) || (event_type
+ == PICOPAM_EVENT_S_BOUND) || (event_type
+ == PICOPAM_EVENT_P_BOUND)) {
+ pam->d2_current_primary_phrase_word = 1;
+ pam->d2_prev_syllable_in_word
+ = pam->d2_syllable_in_word;
+ pam->d2_syllable_in_word = 0;
+ /*pam->d2_current_primary_phrase_word++;*/
+ }
+ if ((event_type == PICOPAM_EVENT_P_BOUND)) {
+ pam->d2_current_primary_phrase_word = 0;
+ }
+ break;
+ case PICOPAM_DIR_BACK:
+ /*do nothing*/
+ break;
+ }
+ break;
+ case E1:/*processor for E1*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW: /*remember : content syllable indicator already on P5*/
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ if (pam->e1_syllable_word_start == -1)
+ pam->e1_syllable_word_start
+ = (picoos_int8) pam->nCurrSyllable;
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[P5] == 1)
+ pam->e1_content = 1;
+ pam->e1_syllable_word_end
+ = (picoos_int8) pam->nCurrSyllable;
+ }
+ if ((event_type == PICOPAM_EVENT_W_BOUND) || (event_type
+ == PICOPAM_EVENT_S_BOUND) || (event_type
+ == PICOPAM_EVENT_P_BOUND)) {
+ if ((pam->e1_syllable_word_start != -1)
+ && (pam->e1_syllable_word_end != -1)) {
+ for (nI = pam->e1_syllable_word_start; nI
+ <= pam->e1_syllable_word_end; nI++)
+ pam->sSyllFeats[nI].phoneV[E1]
+ = pam->e1_content;
+ }
+ pam->e1_content = 0;
+ pam->e1_syllable_word_start = -1;
+ pam->e1_syllable_word_end = -1;
+ }
+ break;
+ case PICOPAM_DIR_BACK:
+ /*do nothing*/
+ break;
+ }
+ break;
+ case E2:/*processor for E2*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ if (pam->e2_syllable_word_start == -1)
+ pam->e2_syllable_word_start
+ = (picoos_int8) pam->nCurrSyllable;
+ pam->e2_syllable_word_end
+ = (picoos_int8) pam->nCurrSyllable;
+ }
+ if ((event_type == PICOPAM_EVENT_W_BOUND) || (event_type
+ == PICOPAM_EVENT_S_BOUND) || (event_type
+ == PICOPAM_EVENT_P_BOUND)) {
+ if ((pam->e2_syllable_word_start != -1)
+ && (pam->e2_syllable_word_end != -1)) {
+ for (nI = pam->e2_syllable_word_start; nI
+ <= pam->e2_syllable_word_end; nI++)
+ pam->sSyllFeats[nI].phoneV[E2]
+ = pam->e2_syllable_word_end
+ - pam->e2_syllable_word_start
+ + 1;
+ }
+ pam->e1_content = 0;
+ pam->e2_syllable_word_start = -1;
+ pam->e2_syllable_word_end = -1;
+ }
+ break;
+ case PICOPAM_DIR_BACK:
+ break;
+ }
+ break;
+ case E3:/*processor for E3,E4*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ sDest = E3;
+ break;
+ case PICOPAM_DIR_BACK:
+ sDest = E4;
+ break;
+ default:
+ sDest = E3;
+ break;
+ }
+ switch (pam->e3_e4_state) {
+ case 0:
+ if (event_type == PICOPAM_EVENT_SYLL)
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = PICOPAM_DONT_CARE_VALUE;
+ if (event_type == PICOPAM_EVENT_S_BOUND) {
+ pam->e3_e4_word = 1;
+ pam->e3_e4_state = 1;
+ }
+ break;
+ case 1:
+ if (event_type == PICOPAM_EVENT_SYLL)
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = pam->e3_e4_word;
+ if (event_type == PICOPAM_EVENT_S_BOUND)
+ pam->e3_e4_word = 1;
+ if (event_type == PICOPAM_EVENT_W_BOUND)
+ pam->e3_e4_word++;
+ if (event_type == PICOPAM_EVENT_P_BOUND)
+ pam->e3_e4_state = 0;
+ break;
+ default:
+ break;
+ }
+ break;
+ case E4:/*processor for E4 : done in E3*/
+ break;
+ case E5:/*processor for E5,E6*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ sDest = E5;
+ break;
+ case PICOPAM_DIR_BACK:
+ sDest = E6;
+ break;
+ default:
+ sDest = E5;
+ break;
+ }
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = pam->e5_e6_content_word;
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[P5] == 1)
+ pam->e5_e6_content = 1;
+ }
+ if ((event_type == PICOPAM_EVENT_W_BOUND) || (event_type
+ == PICOPAM_EVENT_S_BOUND) || (event_type
+ == PICOPAM_EVENT_P_BOUND)) {
+ if (pam->e5_e6_content == 1)
+ pam->e5_e6_content_word++;
+ pam->e5_e6_content = 0;
+ if (event_type == PICOPAM_EVENT_P_BOUND)
+ pam->e5_e6_content_word = 0;
+ }
+ break;
+ case E6:/*processor for E6 : done in E5*/
+ break;
+ case E7:/*processor for E7,E8*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ sDest = E7;
+ break;
+ case PICOPAM_DIR_BACK:
+ sDest = E8;
+ break;
+ default:
+ sDest = E7;
+ break;
+ }
+ switch (pam->e7_e8_state) {
+ case 0:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = PICOPAM_DONT_CARE_VALUE;
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[P5] == 1)
+ pam->e7_e8_content = 1;
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND) {
+ pam->e7_e8_content = 0;
+ }
+
+ if ((event_type == PICOPAM_EVENT_W_BOUND) || (event_type
+ == PICOPAM_EVENT_S_BOUND)) {
+ if (pam->e7_e8_content == 1) {
+ pam->e7_e8_word = 0;
+ pam->e7_e8_content = 0;
+ pam->e7_e8_state = 1;
+ }
+ }
+ break;
+ case 1:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = pam->e7_e8_word;
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[P5] == 1)
+ pam->e7_e8_content = 1;
+ }
+ if ((event_type == PICOPAM_EVENT_W_BOUND) || (event_type
+ == PICOPAM_EVENT_S_BOUND)) {
+ if (pam->e7_e8_content == 1) {
+ pam->e7_e8_word = 0;
+ pam->e7_e8_content = 0;
+ } else {
+ pam->e7_e8_word++;
+ }
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND) {
+ pam->e7_e8_state = 0;
+ pam->e7_e8_content = 0; /*<<<<<< added */
+ }
+
+ default:
+ break;
+ }
+ break;
+ case E8:/*processor for E8 : done in E7*/
+ break;
+ case E9:
+ /*processor for E9, E11*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ switch (pam->e9_e11_state) {
+ case 0:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E9]
+ = PICOPAM_DONT_CARE_VALUE;
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E11]
+ = pam->e9_e11_word;
+ pam->e9_e11_saw_word = 1; /*new variable, needs to be initialized to 0*/
+ }
+ if (event_type == PICOPAM_EVENT_W_BOUND)
+ pam->e9_e11_word++;
+ if (((event_type == PICOPAM_EVENT_P_BOUND)
+ || (event_type == PICOPAM_EVENT_S_BOUND))
+ && (pam->e9_e11_saw_word == 1)) { /* modified*/
+ if (event_type == PICOPAM_EVENT_P_BOUND)
+ pam->e9_e11_word = 1;
+ else
+ pam->e9_e11_word++; /*modified*/
+ pam->e9_e11_state = 1;
+ }
+ break;
+ case 1:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E9]
+ = pam->e9_e11_word;
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E11]
+ = PICOPAM_DONT_CARE_VALUE;
+ }
+ if ((event_type == PICOPAM_EVENT_W_BOUND)
+ || (event_type == PICOPAM_EVENT_S_BOUND))
+ pam->e9_e11_word++;
+ if (event_type == PICOPAM_EVENT_P_BOUND)
+ pam->e9_e11_word = 1;
+ break;
+ default:
+ break;
+ }
+ break;
+ case PICOPAM_DIR_BACK:
+ /*do nothing*/
+ break;
+ }
+ break;
+ case E10:/*processor for E10, E12, E13 unified*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:/*do nothing*/
+ break;
+ case PICOPAM_DIR_BACK:
+ switch (pam->e10_e12_e13_state) {
+ case 0:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E10]
+ = PICOPAM_DONT_CARE_VALUE;
+ pam->e10_e12_e13_saw_word = 1;
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[P6]
+ == PICOPAM_DECLARATIVE) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E12]
+ = pam->e10_e12_e13_word;
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E13]
+ = PICOPAM_DONT_CARE_VALUE;
+ } else {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E12]
+ = PICOPAM_DONT_CARE_VALUE;
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E13]
+ = pam->e10_e12_e13_word;
+ }
+ }
+ if (event_type == PICOPAM_EVENT_W_BOUND)
+ pam->e10_e12_e13_word++;
+
+ /*if (((event_type==PICOPAM_EVENT_P_BOUND)||(event_type==PICOPAM_EVENT_S_BOUND))&&(pam->e10_e12_e13_word>1)) {*/
+ if (((event_type == PICOPAM_EVENT_P_BOUND)
+ || (event_type == PICOPAM_EVENT_S_BOUND))
+ && (pam->e10_e12_e13_saw_word > 0)) {
+ if (event_type == PICOPAM_EVENT_P_BOUND)
+ pam->e10_e12_e13_word = 1;
+ else
+ pam->e10_e12_e13_word++;
+ pam->e10_e12_e13_state = 1;
+ }
+ break;
+ case 1:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E10]
+ = pam->e10_e12_e13_word;
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E12]
+ = PICOPAM_DONT_CARE_VALUE;
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E13]
+ = PICOPAM_DONT_CARE_VALUE;
+ }
+ if ((event_type == PICOPAM_EVENT_W_BOUND)
+ || (event_type == PICOPAM_EVENT_S_BOUND))
+ pam->e10_e12_e13_word++;
+ if (event_type == PICOPAM_EVENT_P_BOUND)
+ pam->e10_e12_e13_word = 1;
+ break;
+ default:
+ break;
+ }
+ break;
+ }
+ break;
+
+ case E11:/*processor for E11 : done in E9*/
+ break;
+ case E12:/*processor for E12 : done in E10*/
+ break;
+ case E13:/*processor for E13 : done in E10*/
+ break;
+
+ case F2:
+ switch (direction) {
+ case PICOPAM_DIR_FORW:/*do nothing*/
+ break;
+ case PICOPAM_DIR_BACK:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ if (pam->f2_current_primary_phrase_word >= 1)/*at least second word in current primary phrase*/
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[F2]
+ = pam->f2_next_word_syllable;
+ else
+ /*first word in current primary phrase*/
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[F2] = 0;
+ pam->f2_word_syllable++;
+ }
+ if ((event_type == PICOPAM_EVENT_W_BOUND) || (event_type
+ == PICOPAM_EVENT_S_BOUND) || (event_type
+ == PICOPAM_EVENT_P_BOUND)) {/*word - end : switch*/
+ pam->f2_next_word_syllable = pam->f2_word_syllable;
+ pam->f2_word_syllable = 0;
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND)/*mark first word in current primary phrase*/
+ pam->f2_current_primary_phrase_word = 0;
+ else /*mark next word in current primary phrase(enables output in PICOPAM_EVENT_SYLL)*/
+ if ((event_type == PICOPAM_EVENT_W_BOUND) || (event_type
+ == PICOPAM_EVENT_S_BOUND))
+ pam->f2_current_primary_phrase_word++;
+ break;
+ }
+ break;
+ case G1:
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ if (pam->g1_current_secondary_phrase_syllable > 0)
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[G1]
+ = pam->g1_current_secondary_phrase_syllable;
+ else
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[G1] = 0;
+ pam->g1_current_syllable++;
+ }
+ if (event_type == PICOPAM_EVENT_S_BOUND) {
+ pam->g1_current_secondary_phrase_syllable
+ = pam->g1_current_syllable;
+ pam->g1_current_syllable = 0;
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND) {
+ pam->g1_current_secondary_phrase_syllable = 0;
+ pam->g1_current_syllable = 0;
+ }
+ case PICOPAM_DIR_BACK: /*do nothing*/
+ break;
+ }
+ break;
+ case G2:
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ if (pam->g2_current_secondary_phrase_word > 0)
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[G2]
+ = pam->g2_current_secondary_phrase_word;
+ else
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[G2] = 0;
+ }
+ if (event_type == PICOPAM_EVENT_W_BOUND)
+ pam->g2_current_word++;
+
+ if (event_type == PICOPAM_EVENT_S_BOUND) {
+ pam->g2_current_secondary_phrase_word
+ = pam->g2_current_word + 1;
+ pam->g2_current_word = 0;
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND) {
+ pam->g2_current_secondary_phrase_word = 0;
+ pam->g2_current_word = 0;
+ }
+ break;
+ case PICOPAM_DIR_BACK: /*do nothing*/
+ break;
+ }
+ break;
+ case H1:
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->h1_current_secondary_phrase_syll++;
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[H1]
+ = pam->h1_current_secondary_phrase_syll;
+ }
+ if ((event_type == PICOPAM_EVENT_S_BOUND) || (event_type
+ == PICOPAM_EVENT_P_BOUND))
+ pam->h1_current_secondary_phrase_syll = 0;
+ break;
+ case PICOPAM_DIR_BACK:
+ if (event_type == PICOPAM_EVENT_SYLL)
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[H1]
+ = pam->h1_current_secondary_phrase_syll;
+ if (event_type == PICOPAM_EVENT_S_BOUND)
+ pam->h1_current_secondary_phrase_syll
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[H1];
+ if (event_type == PICOPAM_EVENT_P_BOUND)
+ pam->h1_current_secondary_phrase_syll
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[H1];
+ break;
+ }
+ break;
+ case H2:
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[H2]
+ = pam->h2_current_secondary_phrase_word;
+ }
+ if (event_type == PICOPAM_EVENT_W_BOUND) {
+ pam->h2_current_secondary_phrase_word++;
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[H2]
+ = pam->h2_current_secondary_phrase_word;
+ }
+ if (event_type == PICOPAM_EVENT_S_BOUND) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[H2]
+ = pam->h2_current_secondary_phrase_word + 1;
+ pam->h2_current_secondary_phrase_word = 0;
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND) {
+ if (pam->nCurrSyllable > 1)
+ pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[H2]
+ = pam->h2_current_secondary_phrase_word + 1;
+ pam->h2_current_secondary_phrase_word = 0;
+ }
+ break;
+ case PICOPAM_DIR_BACK:
+ if (event_type == PICOPAM_EVENT_SYLL)
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[H2]
+ = pam->h2_current_secondary_phrase_word;
+ if (event_type == PICOPAM_EVENT_S_BOUND)
+ pam->h2_current_secondary_phrase_word
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[H2];
+ if (event_type == PICOPAM_EVENT_P_BOUND)
+ pam->h2_current_secondary_phrase_word
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[H2];
+ break;
+ }
+ break;
+ case H3:/*processor for H3,H4 unified */
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ sDest = H3;
+ break;
+ case PICOPAM_DIR_BACK:
+ sDest = H4;
+ break;
+ default:
+ sDest = H3;
+ break;
+ }
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[sDest]
+ = pam->h3_h4_current_secondary_phrase_word;
+ }
+ if ((event_type == PICOPAM_EVENT_S_BOUND) || (event_type
+ == PICOPAM_EVENT_P_BOUND))
+ pam->h3_h4_current_secondary_phrase_word++;
+ break;
+ case H4: /*processor for H4 : already in H3*/
+ break;
+
+ case H5:/*processor for H5*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ break;
+ case PICOPAM_DIR_BACK:
+ switch (pam->h5_state) {
+ case 0:
+ if (event_type == PICOPAM_EVENT_SYLL)
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[H5]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[H5];
+ if (event_type == PICOPAM_EVENT_S_BOUND) {
+ pam->h5_state = 1;
+ }
+ break;
+ case 1:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ if ((pam->sSyllFeats[pam->nCurrSyllable].phoneV[H5]
+ == PICOPAM_P)
+ && (pam->sSyllFeats[pam->nCurrSyllable].phoneV[P1]
+ == 0))
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[H5]
+ = PICOPAM_p;
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND) {
+ pam->h5_state = 0;
+ }
+ break;
+ default:
+ break;
+ }
+ break;
+
+ default:
+ break;
+ }
+ break;
+
+ case I1:/*processor for I1*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->i1_secondary_phrase_syllable++;
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[I1]
+ = pam->i1_secondary_phrase_syllable;
+ }
+ if ((event_type == PICOPAM_EVENT_S_BOUND) || (event_type
+ == PICOPAM_EVENT_P_BOUND))
+ pam->i1_secondary_phrase_syllable = 0;
+ break;
+ case PICOPAM_DIR_BACK:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[I1]
+ = pam->i1_next_secondary_phrase_syllable;
+ }
+ if (event_type == PICOPAM_EVENT_S_BOUND) {
+ pam->i1_next_secondary_phrase_syllable
+ = pam->i1_secondary_phrase_syllable;
+ pam->i1_secondary_phrase_syllable
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[I1];
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND) {
+ pam->i1_next_secondary_phrase_syllable = 0;
+ pam->i1_secondary_phrase_syllable
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[I1];
+ }
+ break;
+ }
+ break;
+ case I2: /*processor for I2*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[I2]
+ = pam->i2_secondary_phrase_word;
+ }
+ if (event_type == PICOPAM_EVENT_W_BOUND)
+ pam->i2_secondary_phrase_word++;
+
+ if ((event_type == PICOPAM_EVENT_P_BOUND) || (event_type
+ == PICOPAM_EVENT_S_BOUND))
+ pam->i2_secondary_phrase_word = 1;
+
+ break;
+ case PICOPAM_DIR_BACK:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[I2]
+ = pam->i2_next_secondary_phrase_word;
+ }
+ if (event_type == PICOPAM_EVENT_S_BOUND) {
+ pam->i2_next_secondary_phrase_word
+ = pam->i2_secondary_phrase_word;
+ pam->i2_secondary_phrase_word
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[I2];
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND) {
+ pam->i2_next_secondary_phrase_word = 0;
+ pam->i2_secondary_phrase_word
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[I2];
+ }
+ break;
+ }
+ break;
+ case J1: /*processor for J1 */
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ if (event_type == PICOPAM_EVENT_SYLL) {
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[P1] != 1)
+ pam->j1_utterance_syllable++;
+ }
+ break;
+ case PICOPAM_DIR_BACK:
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[J1]
+ = pam->j1_utterance_syllable;
+ break;
+ }
+ break;
+ case J2: /*processor for J2*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ if ((event_type == PICOPAM_EVENT_W_BOUND) || (event_type
+ == PICOPAM_EVENT_S_BOUND) || (event_type
+ == PICOPAM_EVENT_P_BOUND))
+ pam->j2_utterance_word++;
+ break;
+ case PICOPAM_DIR_BACK:
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[J2]
+ = pam->j2_utterance_word - 1;
+ break;
+ }
+ break;
+ case J3: /*processor for J3*/
+ switch (direction) {
+ case PICOPAM_DIR_FORW:
+ if (event_type == PICOPAM_EVENT_S_BOUND) {
+ pam->j3_utterance_sec_phrases++;
+ break;
+ }
+ if (event_type == PICOPAM_EVENT_P_BOUND) {
+ pam->j3_utterance_sec_phrases++;
+ break;
+ }
+ break;
+ case PICOPAM_DIR_BACK:
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[J3]
+ = pam->j3_utterance_sec_phrases - 1;
+ break;
+ }
+ break;
+ }
+ return PICO_OK;
+}/*pam_process_event_feature*/
+
+/**
+ * processes an input event spanning it to all column features
+ * @param this : pointer to PAM PU sub object pointer
+ * @param event_type : event id among syll/boundprim/boundsec/boundword
+ * @param direction : forward(0)/backward(1)
+ * @return PICO_OK : process OK
+ * @return PICO_ERR_OTHER : errors on getting pam sub obj pointer
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_process_event(register picodata_ProcessingUnit this,
+ picoos_uint8 event_type, picoos_uint8 direction)
+{
+ picoos_uint8 nFeat;
+ pico_status_t nResult;
+
+ pam_subobj_t *pam;
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+
+ if (direction == PICOPAM_DIR_FORW) {
+ if (event_type == PICOPAM_EVENT_P_BOUND)
+ /*primary boundary*/
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[P2] = 1;
+ if (event_type == PICOPAM_EVENT_S_BOUND)
+ /*secondary boundary*/
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[P3] = 1;
+ if (event_type == PICOPAM_EVENT_W_BOUND)
+ /*word boundary*/
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[P4] = 1;
+ }
+ for (nFeat = A3; nFeat <= J3; nFeat++) {
+ nResult = pam_process_event_feature(this, nFeat, event_type, direction);
+ if (nResult != PICO_OK)
+ return nResult;
+ }
+ return PICO_OK;
+}/*pam_process_event*/
+
+/**
+ * inserts a syllable inside the subobj sentence data struct.
+ * @param this : pointer to PAM PU sub object pointer
+ * @param syllType : the syllable type (pause/syllable)
+ * @param sContent : the item content
+ * @param sentType : the sentence type
+ * @param phType : the phrase type
+ * @param uBoundType : the boundary type (only for silence syllables)
+ * @param uMinDur, uMaxDur : the mimimum and maximum duration (only for silence syllables)
+ * @return PICO_OK : syllable creation successful
+ * @return PICO_ERR_OTHER : errors in one internal function (check_phones_size..)
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_create_syllable(register picodata_ProcessingUnit this,
+ picoos_uint8 syllType, picoos_uint8 *sContent, picoos_uint8 sentType,
+ picoos_uint8 phType, picoos_uint8 uBoundType, picoos_uint16 uMinDur,
+ picoos_uint16 uMaxDur)
+{
+ pam_subobj_t *pam;
+ picoos_uint8 nI;
+ picoos_uint8 pos;
+ /* picoos_uint8 *npUi16; */
+ picoos_uint32 pos32;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+ pos = 0;
+ /*check buffer full condition on number of syllables*/
+ if (check_syllables_size(pam, 1) != PICO_OK) {
+ return PICO_ERR_OTHER;
+ }
+
+ if (syllType == PICOPAM_SYLL_PAUSE) {
+ /*check buffer full condition on number of phonemes*/
+ if (check_phones_size(pam, 1) != PICO_OK) {
+ return PICO_ERR_OTHER;
+ }
+ }
+ if (syllType == PICOPAM_SYLL_SYLL) {
+ /*check item availability*/
+ if (sContent == NULL) {
+ return PICO_ERR_OTHER;
+ }
+ /*check buffer full condition on number of phonemes*/
+ if (check_phones_size(pam, sContent[3]) != PICO_OK) {
+ return PICO_ERR_OTHER;
+ }
+ }
+
+ /*open new syllable*/
+ pam->nCurrSyllable = pam->nCurrSyllable + 1;
+ /*cleanup*/
+ for (nI = 0; nI < PICOPAM_VECT_SIZE; nI++) {
+ if (pam->nCurrSyllable > 0) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[nI] = 0;
+ } else {
+ if ((nI >= ITM) && (nI <= itm)) {
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[nI] > 0) {
+ /*do not cleanup "attached item offset" fields (ITM, itm):
+ an already existing attached item could be lost*/
+ } else {
+ /*cleanup "attached item offset"*/
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[nI] = 0;
+ }
+ } else {
+ /*cleanup all fields except "attached item offset" (ITM, itm)*/
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[nI] = 0;
+ }
+ }
+ }
+ /*set minimum and maximum duration values*/
+ if ((uMinDur == 0) && (uMaxDur == 0) && (syllType == PICOPAM_SYLL_PAUSE)) {
+ /*both 0 : use default duration limits for boundaries*/
+ get_default_boundary_limit(uBoundType, &uMinDur, &uMaxDur);
+ }
+ if (uMinDur > 0) {
+ pos32 = Min;
+ picoos_write_mem_pi_uint16(pam->sSyllFeats[pam->nCurrSyllable].phoneV,
+ &pos32, uMinDur);
+ }
+ if (uMaxDur > 0) {
+ pos32 = Max;
+ picoos_write_mem_pi_uint16(pam->sSyllFeats[pam->nCurrSyllable].phoneV,
+ &pos32, uMaxDur);
+ }
+ /*END OF BREAK COMMAND SUPPORT*/
+
+ if (syllType == PICOPAM_SYLL_PAUSE) {
+ /*initialize a pause syllable*/
+ if (sentType == PICOPAM_DECLARATIVE)
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[P6]
+ = PICOPAM_DECLARATIVE;
+ if (sentType == PICOPAM_INTERROGATIVE)
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[P6]
+ = PICOPAM_INTERROGATIVE;
+
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[bnd] = uBoundType;
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[P1] = 1; /*this means the syllable contains a pause-silence*/
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[P8] = 1;
+
+ /*b1,b2,b9,b11,b13,b15,e1,e6,e8,e10 already set to 0*/
+
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B3]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[B4]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[B5]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[B6]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[B7]
+ = 1;
+
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B16]
+ = PICOPAM_PH_DONT_CARE_VAL; /*name of the vowel in the syllable = NONE */
+
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E2]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[E3]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[E4] = 1;
+
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[H5] = phType;
+
+ /*Store current phonetic codes in input phonetic string*/
+ pam->sPhIds[pam->nCurrPhoneme] = pam_get_pause_id(this);
+ picoos_mem_copy((void*) &pam->nCurrPhoneme,
+ &(pam->sSyllFeats[pam->nCurrSyllable].phoneV[FID]),
+ sizeof(pam->nCurrPhoneme));
+ pam->nCurrPhoneme++;
+ pam->nTotalPhonemes++;
+ /*add 1 to total number of syllables*/
+ pam->nTotalSyllables++;
+
+ return PICO_OK;
+ }
+ if (syllType == PICOPAM_SYLL_SYLL) {
+ /*initialize a real syllable*/
+ if (sContent[2] > PICODATA_ACC0)
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[P5] = 1; /*set content syllable indicator*/
+ if (sentType == PICOPAM_DECLARATIVE)
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[P6]
+ = PICOPAM_DECLARATIVE;
+ if (sentType == PICOPAM_INTERROGATIVE)
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[P6]
+ = PICOPAM_INTERROGATIVE;
+
+ if ((sContent[2] >= PICODATA_ACC1) && (sContent[2] <= PICODATA_ACC4))
+ /*stressed*/
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B1] = 1;
+
+ if ((sContent[2] >= PICODATA_ACC1) && (sContent[2] <= PICODATA_ACC2))
+ /*accented*/
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B2] = 1;
+
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B3] = sContent[3];/*len*/
+
+ if (pam->nCurrSyllable > 30)
+ pam->nCurrSyllable = pam->nCurrSyllable;
+
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B16] = pam_get_vowel_name(this,
+ sContent, &pos); /*name of the vowel in the syllable*/
+
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[P8] = pos; /*temp for storing the position of the vowel*/
+
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[H5] = phType;
+
+ /*Store current phonetic codes in input phonetic string*/
+ picoos_mem_copy((void*) &pam->nCurrPhoneme,
+ &(pam->sSyllFeats[pam->nCurrSyllable].phoneV[FID]),
+ sizeof(pam->nCurrPhoneme));
+ for (nI = 0; nI < sContent[3]; nI++)
+ pam->sPhIds[pam->nCurrPhoneme + nI] = sContent[4 + nI];
+ pam->nCurrPhoneme += nI;
+ pam->nTotalPhonemes += nI;
+ /*add 1 to total number of syllables*/
+ pam->nTotalSyllables++;
+ return PICO_OK;
+ }
+ /*if no SyllType has been identified -->> error*/
+ return PICO_ERR_OTHER;
+}/*pam_create_syllable*/
+
+/**
+ * performs the forward step of the PAM adapter
+ * @param this : pointer to PAM PU sub object pointer
+ * @param itemBase : pointer to current item
+ * @return PICOPAM_READY : forward step ok, the sentence is complete
+ * @return PICOPAM_MORE : forward step ok, but more data needed to complete the sentence
+ * @return PICO_ERR_OTHER : errors in one internal function (CreateSyllable..)
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_adapter_forward_step(
+ register picodata_ProcessingUnit this, picoos_uint8 *itemBase)
+{
+ register pam_subobj_t * pam;
+ pico_status_t sResult;
+ picoos_uint16 uMinDur, uMaxDur;
+ picoos_uint32 nPos;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+ uMinDur = uMaxDur = 0; /*default 0 : not initialized*/
+ switch (itemBase[0]) {
+ case PICODATA_ITEM_BOUND:
+ /*received a boundary item*/
+ switch (itemBase[1]) {
+ case PICODATA_ITEMINFO1_BOUND_SBEG:
+ case PICODATA_ITEMINFO1_BOUND_PHR1:
+#ifdef PAM_PHR2_WITH_PR1
+ case PICODATA_ITEMINFO1_BOUND_PHR2:
+#endif
+ case PICODATA_ITEMINFO1_BOUND_SEND:
+ case PICODATA_ITEMINFO1_BOUND_TERM:
+ if (itemBase[3] == 2 * sizeof(picoos_uint16)) {
+ /*only when the item len duration is equal to 2 int16 --> get the values*/
+ nPos = 4;
+ picoos_read_mem_pi_uint16(itemBase, &nPos, &uMinDur);
+ picoos_read_mem_pi_uint16(itemBase, &nPos, &uMaxDur);
+ }
+ break;
+ default:
+ break;
+ }
+ switch (itemBase[1]) {
+ case PICODATA_ITEMINFO1_BOUND_SBEG:
+ /* received a sentence init boundary */
+ pam_reset_processors(this); /*reset all processor variables*/
+ pam->sType
+ = pam_map_sentence_type(itemBase[1], itemBase[2]);
+ pam->pType = pam_map_phrase_type(itemBase[1], itemBase[2]);
+ /*create silence syll and process P_BOUND event*/
+ sResult = pam_create_syllable(this, PICOPAM_SYLL_PAUSE, NULL,
+ pam->sType, pam->pType, itemBase[1], uMinDur,
+ uMaxDur);
+ if (sResult != PICO_OK)
+ return sResult;
+ sResult = pam_process_event(this, PICOPAM_EVENT_P_BOUND,
+ PICOPAM_DIR_FORW);
+ if (sResult != PICO_OK)
+ return sResult;
+ return PICOPAM_MORE;
+ break;
+
+ case PICODATA_ITEMINFO1_BOUND_PHR1:
+#ifdef PAM_PHR2_WITH_PR1
+ case PICODATA_ITEMINFO1_BOUND_PHR2:
+#endif
+ /*received a primary boundary*/
+ pam->sType
+ = pam_map_sentence_type(itemBase[1], itemBase[2]);
+ pam->pType = pam_map_phrase_type(itemBase[1], itemBase[2]);
+ /*create silence syll and process P_BOUND event*/
+ sResult = pam_create_syllable(this, PICOPAM_SYLL_PAUSE, NULL,
+ pam->sType, pam->pType, itemBase[1], uMinDur,
+ uMaxDur);
+ if (sResult != PICO_OK)
+ return sResult;
+ sResult = pam_process_event(this, PICOPAM_EVENT_P_BOUND,
+ PICOPAM_DIR_FORW);
+ if (sResult != PICO_OK)
+ return sResult;
+ return PICOPAM_MORE;
+ break;
+
+#ifdef PAM_PHR2_WITH_PR3
+ case PICODATA_ITEMINFO1_BOUND_PHR2 :
+#endif
+ case PICODATA_ITEMINFO1_BOUND_PHR3:
+ /*received a secondary boundary*/
+ /*process S_BOUND event*/
+ sResult = pam_process_event(this, PICOPAM_EVENT_S_BOUND,
+ PICOPAM_DIR_FORW);
+ /*determine new sentence and Phrase types for following syllables*/
+ pam->sType
+ = pam_map_sentence_type(itemBase[1], itemBase[2]);
+ pam->pType = pam_map_phrase_type(itemBase[1], itemBase[2]);
+ if (sResult != PICO_OK)
+ return sResult;
+ return PICOPAM_MORE;
+ break;
+
+ case PICODATA_ITEMINFO1_BOUND_PHR0:
+ /*received a word end boundary*/
+ /*process W_BOUND event*/
+ sResult = pam_process_event(this, PICOPAM_EVENT_W_BOUND,
+ PICOPAM_DIR_FORW);
+ if (sResult != PICO_OK)
+ return sResult;
+ return PICOPAM_MORE;
+ break;
+
+ case PICODATA_ITEMINFO1_BOUND_SEND:
+ /*received a SEND boundary*/
+ /*insert a new silence syllable and process P_BOUND event*/
+ /*create silence syll and process P_BOUND event*/
+ sResult = pam_create_syllable(this, PICOPAM_SYLL_PAUSE, NULL,
+ pam->sType, pam->pType, itemBase[1], uMinDur,
+ uMaxDur);
+ if (sResult != PICO_OK)
+ return sResult;
+ sResult = pam_process_event(this, PICOPAM_EVENT_P_BOUND,
+ PICOPAM_DIR_FORW);
+ if (sResult != PICO_OK)
+ return sResult;
+ return PICOPAM_READY;
+ break;
+
+ case PICODATA_ITEMINFO1_BOUND_TERM:
+ /* received a flush boundary*/
+ if (pam->nCurrSyllable == -1) {
+ return PICOPAM_NA;
+ }
+ /*insert a new silence syllable and process P_BOUND event*/
+ /*create silence syll and process P_BOUND event*/
+ sResult = pam_create_syllable(this, PICOPAM_SYLL_PAUSE, NULL,
+ pam->sType, pam->pType, itemBase[1], uMinDur,
+ uMaxDur);
+ if (sResult != PICO_OK)
+ return sResult;
+ sResult = pam_process_event(this, PICOPAM_EVENT_P_BOUND,
+ PICOPAM_DIR_FORW);
+ if (sResult != PICO_OK)
+ return sResult;
+ return PICOPAM_READY;
+ break;
+
+ default:
+ /*boundary type not known*/
+ return PICOPAM_NA;
+ break;
+ }/*end switch (itemBase[1])*/
+ break; /*end case PICODATA_ITEM_BOUND*/
+
+ case PICODATA_ITEM_SYLLPHON:
+ /*received a syllable item*/
+ /* ------------------------------------------------------------------
+ following code has to be used if we do expect
+ SYLL items arrive even without SBEG items starting the sentence.
+ this may happen after a term has been issued to make room in local storage.
+ */
+ if (pam->nCurrSyllable == -1) {
+ pam_reset_processors(this);
+ /*insert an SBEG with sType and pType taken from previous sentence*/
+ sResult = pam_create_syllable(this, PICOPAM_SYLL_PAUSE, NULL,
+ pam->sType, pam->pType, PICODATA_ITEMINFO1_BOUND_SBEG,
+ 0, 0);
+ if (sResult != PICO_OK)
+ return sResult;
+ sResult = pam_process_event(this, PICOPAM_EVENT_P_BOUND,
+ PICOPAM_DIR_FORW);
+ if (sResult != PICO_OK)
+ return sResult;
+ }
+ /* ------------------------------------------------------------------*/
+ sResult = pam_create_syllable(this, PICOPAM_SYLL_SYLL, itemBase,
+ pam->sType, pam->pType, 0, 0, 0);
+ if (sResult != PICO_OK)
+ return sResult;
+ sResult = pam_process_event(this, PICOPAM_EVENT_SYLL,
+ PICOPAM_DIR_FORW);
+ if (sResult != PICO_OK)
+ return sResult;
+ return PICOPAM_MORE;
+ break;
+
+ default:
+ return PICOPAM_NA;
+ break;
+ }
+ return PICO_ERR_OTHER;
+}/*pam_adapter_forward_step*/
+
+/**
+ * performs the backward step of the PAM adapter
+ * @param this : pointer to PAM PU sub object pointer
+ * @return PICO_OK : backward step complete
+ * @return PICO_ERR_OTHER : errors on retrieving the PU pointer
+ * @remarks derived in some parts from the pam forward code
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_adapter_backward_step(
+ register picodata_ProcessingUnit this)
+{
+ register pam_subobj_t * pam;
+ picoos_uint8 nProcessed;
+ picoos_uint16 nSyll;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+
+ /*Resets the processors for the backward step*/
+ pam_reset_processors_back(this);
+ /*Do the backward step*/
+ nSyll = pam->nCurrSyllable;
+ while (pam->nCurrSyllable >= 0) {
+ nProcessed = 0;
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[P2] == 1) {
+ /*primary boundary*/
+ pam_process_event(this, PICOPAM_EVENT_P_BOUND, PICOPAM_DIR_BACK);
+ pam->nCurrSyllable--;
+ nProcessed = 1;
+ }
+ if ((nProcessed == 0)
+ && (pam->sSyllFeats[pam->nCurrSyllable].phoneV[P3] == 1)) {
+ /*secondary boundary*/
+ pam_process_event(this, PICOPAM_EVENT_S_BOUND, PICOPAM_DIR_BACK);
+ pam_process_event(this, PICOPAM_EVENT_SYLL, PICOPAM_DIR_BACK);
+ pam->nCurrSyllable--;
+ nProcessed = 1;
+ }
+ if ((nProcessed == 0)
+ && (pam->sSyllFeats[pam->nCurrSyllable].phoneV[P4] == 1)) {
+ /*word boundary*/
+ pam_process_event(this, PICOPAM_EVENT_W_BOUND, PICOPAM_DIR_BACK);
+ pam_process_event(this, PICOPAM_EVENT_SYLL, PICOPAM_DIR_BACK);
+ pam->nCurrSyllable--;
+ nProcessed = 1;
+ }
+ if (nProcessed == 0) {
+ /*non boundaried syllable*/
+ pam_process_event(this, PICOPAM_EVENT_SYLL, PICOPAM_DIR_BACK);
+ pam->nCurrSyllable--;
+ nProcessed = 0;
+ }
+ }/*end while (pam->nCurrSyllable>=0)*/
+ /*reset syllpointer to original value*/
+ pam->nCurrSyllable = nSyll;
+ /*Perform pause processing*/
+ pam_adapter_do_pauses(this);
+ pam->nCurrSyllable = 0;
+ pam->nSyllPhoneme = 0;
+
+ return PICO_OK;
+}/*pam_adapter_backward_step*/
+
+/**
+ * processes a pause (silence) syllable after backward processing
+ * @param this : pointer to PAM PU sub object pointer : processes a pause (silence) syllable after backward processing
+ * @return PICO_OK : backward step complete
+ * @return PICO_ERR_OTHER : errors on retrieving the PU pointer
+ * @remarks pam->nCurrSyllable should point to a pause item
+ * @remarks this function should be called after backward processing
+ * @remarks this function corresponds to initializing silence phonemes with
+ * @remarks values derived from previous or following syllables
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_do_pause(register picodata_ProcessingUnit this)
+{
+ picoos_uint16 syllCurr;
+ pam_subobj_t *pam;
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+ syllCurr = pam->nCurrSyllable;
+
+ /*processor for all features that can be inherited from previous syll (or word/phrase)*/
+ if (pam->nCurrSyllable > 0) {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[A3]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[B3];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B8]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[B8];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B10]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[B10];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B12]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[B12];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B14]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[B14];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B17]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[B17];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B19]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[B19];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B20]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[B20];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B21]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[B21];
+
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[D2]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[E2];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[G1]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[H1];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[G2]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[H2];
+
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E5]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[E5];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E7]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[E7];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E9]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[E9];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E11]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[E11];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E12]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[E12];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E13]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[E13];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[E13]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[E13];
+
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[H1]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[H1];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[H2]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[H2];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[H3]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[H3];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[H4]
+ = pam->sSyllFeats[pam->nCurrSyllable - 1].phoneV[H4];
+
+ } else {
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[A3]
+ =pam->sSyllFeats[pam->nCurrSyllable].phoneV[B8]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[B10]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[B12]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[B14]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[B17]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[B19]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[B20]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[B21]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[E5]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[E9]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[E11]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[E12]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[H1]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[H2]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[H3]
+ = 0;
+
+ /*init values different from 0*/
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[H4]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[J3];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[H5] = PICOPAM_p;
+
+ }
+
+ /*processor for all features that can be inherited from next syll (or word/phrase)*/
+ if (pam->nCurrSyllable < pam->nTotalSyllables - 1) {
+ /*non last syllable*/
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[C3]
+ = pam->sSyllFeats[pam->nCurrSyllable + 1].phoneV[B3];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[F2]
+ = pam->sSyllFeats[pam->nCurrSyllable + 1].phoneV[E2];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[I1]
+ = pam->sSyllFeats[pam->nCurrSyllable + 1].phoneV[H1];
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[I2]
+ = pam->sSyllFeats[pam->nCurrSyllable + 1].phoneV[H2];
+ } else {
+ /*last syllable*/
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[C3]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[F2]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[I1]
+ = pam->sSyllFeats[pam->nCurrSyllable].phoneV[I2]
+ = 0;
+ }
+
+ /*Other fixed values derived from de-facto standard*/
+ pam->sSyllFeats[pam->nCurrSyllable].phoneV[B18] = 0;
+
+ return PICO_OK;
+}/*pam_do_pause*/
+
+/**
+ * performs the initialization of pause "syllables"
+ * @param this : pointer to PAM PU sub object pointer : processes a pause (silence) syllable after backward processing
+ * @return PICO_OK : pause processing successful
+ * @return PICO_ERR_OTHER : errors on retrieving the PU pointer
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t pam_adapter_do_pauses(register picodata_ProcessingUnit this)
+{
+ register pam_subobj_t * pam;
+ picoos_uint16 nSyll;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ pam = (pam_subobj_t *) this->subObj;
+
+ /*Do the pause processing*/
+ nSyll = pam->nCurrSyllable;
+ while (pam->nCurrSyllable >= 0) {
+ if (pam->sSyllFeats[pam->nCurrSyllable].phoneV[P2] == 1) {
+ /*pause processing*/
+ pam_do_pause(this);
+ }
+ pam->nCurrSyllable--;
+ }/*end while (pam->nCurrSyllable>=0)*/
+ /*reset syllpointer to original value*/
+ pam->nCurrSyllable = nSyll;
+ return PICOPAM_READY;
+}/*pam_adapter_do_pauses*/
+
+#ifdef __cplusplus
+}
+#endif
+
+/* picopam.c end */
diff --git a/lib/picopam.h b/lib/picopam.h
new file mode 100644
index 0000000..18f7730
--- /dev/null
+++ b/lib/picopam.h
@@ -0,0 +1,79 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picopam.h
+ *
+ * Phonetic to Acoustic Mapping PU - Header file
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+/**
+ * @addtogroup picopam
+
+ * <b> Phonetic to acoustic mapping module </b>\n
+ *
+ * This module is responsible for mapping the phonetic domain features generated from text analysis
+ * into parametric representations suitable for signal generation. As such it is the interface
+ * between text analysis and signal generation
+ *
+ * Most the processing of PAM is logically splittable as follows
+ * - building a suitable symbolic feature vector set for the sentence
+ * - Feeding Decision Trees with the symbolic seqence vector set
+ * - Colecting the parametric output of the Decision trees into suitable items to be sent to following PU's
+ *
+ * To perform the decision tree feeding and output collection the PU uses an internal buffer. This
+ * buffer is used several times with different meanings.
+ * - While building the symbolic feature vector set for the sentence this data structure stores syllable relevant data.
+ * The corresponding phonetic data is stored outside as a single string of phonetic id's for all the sentence.
+ * - While feeding the decision trees the data structure is used to represent data for phonemes of the syllable
+ * - Addiotional data strucures are mantained to temporarily store items not pertaining to the PAM processing, for later resynchronization
+ *
+ * A quite detailed description of PAM processing is in the Know How section of the repository.
+ */
+
+#ifndef PICOPAM_H_
+#define PICOPAM_H_
+
+#include "picodata.h"
+#include "picokdt.h"
+#include "picokpdf.h"
+#include "picoktab.h"
+#include "picokdbg.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+/*------------------------------------------------------------------
+Service routines
+------------------------------------------------------------------*/
+picodata_ProcessingUnit picopam_newPamUnit(
+ picoos_MemoryManager mm, picoos_Common common,
+ picodata_CharBuffer cbIn, picodata_CharBuffer cbOut,
+ picorsrc_Voice voice);
+
+#ifdef __cplusplus
+}
+#endif
+#endif /*PICOPAM_H_*/
diff --git a/lib/picopltf.h b/lib/picopltf.h
new file mode 100644
index 0000000..0cf2659
--- /dev/null
+++ b/lib/picopltf.h
@@ -0,0 +1,73 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picopltf.h
+ *
+ * Header file (only) to define platform symbols.
+ *
+ * Refer to http://predef.sourceforge.net/ for a comprehensive list
+ * of pre-defined C/C++ compiler macros.
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ */
+
+#if !defined(__PICOPLTF_H__)
+#define __PICOPLTF_H__
+
+#define ENDIANNESS_BIG 1
+#define ENDIANNESS_LITTLE 2
+
+/* * platform identifiers ***/
+#define PICO_Windows 1 /* Windows */
+#define PICO_MacOSX 5 /* Macintosh OS X */
+#define PICO_Linux 7 /* Linux */
+
+/* * definition of current platform ***/
+#if !defined(PICO_PLATFORM)
+#if defined(_WIN32)
+#define PICO_PLATFORM PICO_Windows
+#elif defined(__APPLE__) && defined(__MACH__)
+#define PICO_PLATFORM PICO_MacOSX
+#elif defined(linux) || defined(__linux__) || defined(__linux)
+#define PICO_PLATFORM PICO_Linux
+#else
+#error PICO_PLATFORM not defined
+#endif
+#endif /* !defined(PICO_PLATFORM) */
+
+
+/* * symbol PICO_PLATFORM_STRING to define platform as string ***/
+#if (PICO_PLATFORM == PICO_Windows)
+#define PICO_PLATFORM_STRING "Windows"
+#elif (PICO_PLATFORM == PICO_MacOSX)
+#define PICO_PLATFORM_STRING "MacOSX"
+#elif (PICO_PLATFORM == PICO_Linux)
+#define PICO_PLATFORM_STRING "Linux"
+#elif (PICO_PLATFORM == PICO_GENERIC)
+#define PICO_PLATFORM_STRING "UnknownPlatform"
+#endif
+
+#if (PICO_PLATFORM == PICO_MacOSX)
+#define PICO_ENDIANNESS ENDIANNESS_BIG
+#else
+#define PICO_ENDIANNESS ENDIANNESS_LITTLE
+#endif
+
+#endif /* !defined(__PICOPLTF_H__) */
diff --git a/lib/picopr.c b/lib/picopr.c
new file mode 100644
index 0000000..f54734a
--- /dev/null
+++ b/lib/picopr.c
@@ -0,0 +1,3550 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picopr.c
+ *
+ * text preprocessor
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#include "picodefs.h"
+#include "picoos.h"
+#include "picobase.h"
+#include "picodbg.h"
+#include "picodata.h"
+#include "picokpr.h"
+#include "picopr.h"
+#include "picoktab.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* *****************************************************************************/
+/* constants */
+/* *****************************************************************************/
+
+#define PR_TRACE_MEM FALSE
+#define PR_TRACE_MAX_MEM FALSE
+#define PR_TRACE_PATHCOST TRUE
+
+#define PR_WORK_MEM_SIZE 10000
+#define PR_DYN_MEM_SIZE 7000
+
+#define PR_ENABLED TRUE
+
+#define PR_MAX_NR_ITERATIONS 1000;
+
+#define SPEC_CHAR "\\/"
+
+#define PICO_ERR_CONTEXT_NOT_FOUND PICO_ERR_OTHER
+#define PICO_ERR_MAX_PREPROC_PATH_LEN_REACHED PICO_ERR_OTHER
+
+#define IN_BUF_SIZE 255
+#define OUT_BUF_SIZE IN_BUF_SIZE + 3 * PICODATA_ITEM_HEADSIZE + 3
+
+#define PR_MAX_NR_PREPROC (1 + PICOKNOW_MAX_NUM_UTPP)
+
+#define PR_MAX_PATH_LEN 130
+#define PR_MAX_DATA_LEN IN_BUF_SIZE
+#define PR_MAX_DATA_LEN_Z PR_MAX_DATA_LEN + 1 /* all strings in picopr should use this constant
+ to ensure zero termination */
+#define PR_COST_INIT 100000
+#define PR_COST 10
+#define PR_EOL '\n'
+
+/* Bit mask constants for token sets with parameters */
+#define PR_TSE_MASK_OUT (1<<PR_TSEOut)
+#define PR_TSE_MASK_MIN (1<<PR_TSEMin)
+#define PR_TSE_MASK_MAX (1<<PR_TSEMax)
+#define PR_TSE_MASK_LEN (1<<PR_TSELen)
+#define PR_TSE_MASK_VAL (1<<PR_TSEVal)
+#define PR_TSE_MASK_STR (1<<PR_TSEStr)
+#define PR_TSE_MASK_HEAD (1<<PR_TSEHead)
+#define PR_TSE_MASK_MID (1<<PR_TSEMid)
+#define PR_TSE_MASK_TAIL (1<<PR_TSETail)
+#define PR_TSE_MASK_PROD (1<<PR_TSEProd)
+#define PR_TSE_MASK_PRODEXT (1<<PR_TSEProdExt)
+#define PR_TSE_MASK_VAR (1<<PR_TSEVar)
+#define PR_TSE_MASK_LEX (1<<PR_TSELex)
+#define PR_TSE_MASK_COST (1<<PR_TSECost)
+#define PR_TSE_MASK_ID (1<<PR_TSEID)
+#define PR_TSE_MASK_DUMMY1 (1<<PR_TSEDummy1)
+#define PR_TSE_MASK_DUMMY2 (1<<PR_TSEDummy2)
+#define PR_TSE_MASK_DUMMY3 (1<<PR_TSEDummy3)
+
+/* Bit mask constants for token sets without parameters */
+#define PR_TSE_MASK_BEGIN (1<<PR_TSEBegin)
+#define PR_TSE_MASK_END (1<<PR_TSEEnd)
+#define PR_TSE_MASK_SPACE (1<<PR_TSESpace)
+#define PR_TSE_MASK_DIGIT (1<<PR_TSEDigit)
+#define PR_TSE_MASK_LETTER (1<<PR_TSELetter)
+#define PR_TSE_MASK_CHAR (1<<PR_TSEChar)
+#define PR_TSE_MASK_SEQ (1<<PR_TSESeq)
+#define PR_TSE_MASK_CMPR (1<<PR_TSECmpr)
+#define PR_TSE_MASK_NLZ (1<<PR_TSENLZ)
+#define PR_TSE_MASK_ROMAN (1<<PR_TSERoman)
+#define PR_TSE_MASK_CI (1<<PR_TSECI)
+#define PR_TSE_MASK_CIS (1<<PR_TSECIS)
+#define PR_TSE_MASK_AUC (1<<PR_TSEAUC)
+#define PR_TSE_MASK_ALC (1<<PR_TSEALC)
+#define PR_TSE_MASK_SUC (1<<PR_TSESUC)
+#define PR_TSE_MASK_ACCEPT (1<<PR_TSEAccept)
+#define PR_TSE_MASK_NEXT (1<<PR_TSENext)
+#define PR_TSE_MASK_ALTL (1<<PR_TSEAltL)
+#define PR_TSE_MASK_ALTR (1<<PR_TSEAltR)
+
+#define PR_FIRST_TSE_WP PR_TSEOut
+
+#define PR_SMALLER 1
+#define PR_EQUAL 0
+#define PR_LARGER 2
+
+#define PR_SPELL_WITH_SENTENCE_BREAK -2
+#define PR_SPELL_WITH_PHRASE_BREAK -1
+#define PR_SPELL 0
+
+#define PICO_SPEED_MIN 20
+#define PICO_SPEED_MAX 500
+#define PICO_SPEED_DEFAULT 100
+#define PICO_SPEED_FACTOR_MIN 500
+#define PICO_SPEED_FACTOR_MAX 2000
+
+#define PICO_PITCH_MIN 50
+#define PICO_PITCH_MAX 200
+#define PICO_PITCH_DEFAULT 100
+#define PICO_PITCH_FACTOR_MIN 500
+#define PICO_PITCH_FACTOR_MAX 2000
+#define PICO_PITCH_ADD_MIN -100
+#define PICO_PITCH_ADD_MAX 100
+#define PICO_PITCH_ADD_DEFAULT 0
+
+#define PICO_VOLUME_MIN 0
+#define PICO_VOLUME_MAX 500
+#define PICO_VOLUME_DEFAULT 100
+#define PICO_VOLUME_FACTOR_MIN 500
+#define PICO_VOLUME_FACTOR_MAX 2000
+
+#define PICO_CONTEXT_DEFAULT "DEFAULT"
+
+#define PICO_PARAGRAPH_PAUSE_DUR 500
+
+
+/* *****************************************************************************/
+/* types */
+/* *****************************************************************************/
+
+typedef enum {PR_OStr, PR_OVar, PR_OItem, PR_OSpell, PR_ORomanToCard, PR_OVal,
+ PR_OLeft, PR_ORight, PR_ORLZ, PR_OIgnore, PR_OPitch, PR_OSpeed,
+ PR_OVolume, PR_OVoice, PR_OContext, PR_OPhonSVOXPA, PR_OPhonSAMPA,
+ PR_OPlay, PR_OUseSig, PR_OGenFile, PR_OAudioEdit, PR_OPara,
+ PR_OSent, PR_OBreak, PR_OMark, PR_OConcat, PR_OLast} pr_OutType;
+
+typedef enum {PR_TSEBegin, PR_TSEEnd, PR_TSESpace, PR_TSEDigit, PR_TSELetter, PR_TSEChar, PR_TSESeq,
+ PR_TSECmpr, PR_TSENLZ, PR_TSERoman, PR_TSECI, PR_TSECIS, PR_TSEAUC, PR_TSEALC, PR_TSESUC,
+ PR_TSEAccept, PR_TSENext, PR_TSEAltL, PR_TSEAltR} pr_TokSetEleNP;
+
+typedef enum {PR_TSEOut, PR_TSEMin, PR_TSEMax, PR_TSELen, PR_TSEVal, PR_TSEStr, PR_TSEHead, PR_TSEMid,
+ PR_TSETail, PR_TSEProd, PR_TSEProdExt, PR_TSEVar, PR_TSELex, PR_TSECost, PR_TSEID,
+ PR_TSEDummy1, PR_TSEDummy2, PR_TSEDummy3} pr_TokSetEleWP;
+
+typedef enum {PR_GSNoPreproc, PR_GS_START, PR_GSContinue, PR_GSNeedToken, PR_GSNotFound, PR_GSFound} pr_GlobalState;
+
+typedef enum {PR_LSError, PR_LSInit, PR_LSGetToken, PR_LSGetToken2, PR_LSMatch, PR_LSGoBack,
+ PR_LSGetProdToken, PR_LSInProd, PR_LSGetProdContToken, PR_LSInProdCont, PR_LSGetNextToken,
+ PR_LSGetAltToken} pr_LocalState;
+
+typedef enum {PR_MSNotMatched, PR_MSMatched, PR_MSMatchedContinue, PR_MSMatchedMulti} pr_MatchState;
+
+typedef struct pr_Prod * pr_ProdList;
+typedef struct pr_Prod {
+ picokpr_Preproc rNetwork;
+ picokpr_ProdArrOffset rProdOfs;
+ pr_ProdList rNext;
+} pr_Prod;
+
+typedef struct pr_Context * pr_ContextList;
+typedef struct pr_Context {
+ picoos_uchar * rContextName;
+ pr_ProdList rProdList;
+ pr_ContextList rNext;
+} pr_Context;
+
+/* *****************************************************************************/
+/* used, but to be checked */
+
+#define MaxNrShortStrParams 2
+#define MaxPhoneLen 14
+#define ShortStrParamLen (2 * MaxPhoneLen)
+typedef picoos_uchar ShortStrParam[ShortStrParamLen];
+
+
+typedef struct pr_ioItem * pr_ioItemPtr;
+typedef struct pr_ioItem {
+ pr_ioItemPtr next;
+ picoos_int32 val;
+ struct picodata_itemhead head;
+ picoos_uchar * strci;
+ picoos_uchar * strcis;
+ picoos_bool alc;
+ picoos_bool auc;
+ picoos_bool suc;
+ picoos_uchar data[PR_MAX_DATA_LEN_Z];
+} pr_ioItem;
+
+typedef struct pr_ioItem2 {
+ pr_ioItemPtr next;
+ picoos_int32 val;
+ struct picodata_itemhead head;
+ picoos_uchar * strci;
+ picoos_uchar * strcis;
+ picoos_bool alc;
+ picoos_bool auc;
+ picoos_bool suc;
+} pr_ioItem2;
+
+#define PR_IOITEM_MIN_SIZE sizeof(pr_ioItem2)
+
+typedef picoos_uint32 pr_MemState;
+typedef enum {pr_DynMem, pr_WorkMem} pr_MemTypes;
+
+/* *****************************************************************************/
+
+typedef struct pr_OutItemVar * pr_OutItemVarPtr;
+struct pr_OutItemVar {
+ pr_ioItemPtr first;
+ pr_ioItemPtr last;
+ picoos_int32 id;
+ pr_OutItemVarPtr next;
+};
+
+
+struct pr_WorkItem {
+ pr_ioItemPtr rit;
+};
+typedef pr_ioItemPtr pr_WorkItems[PR_MAX_PATH_LEN+1];
+
+struct pr_PathEle {
+ picokpr_Preproc rnetwork;
+ picoos_int16 ritemid;
+ picoos_int16 rcompare;
+ picoos_int16 rdepth;
+ picokpr_TokArrOffset rtok;
+ picokpr_StrArrOffset rprodname;
+ picoos_int32 rprodprefcost;
+ pr_LocalState rlState;
+};
+
+typedef struct pr_Path {
+ picoos_int32 rcost;
+ picoos_int32 rlen;
+ struct pr_PathEle rele[PR_MAX_PATH_LEN];
+} pr_Path;
+
+/* *****************************************************************************/
+
+/** subobject : PreprocUnit
+ * shortcut : pr
+ */
+typedef struct pr_subobj
+{
+ pr_ioItemPtr rinItemList;
+ pr_ioItemPtr rlastInItem;
+ pr_ioItemPtr routItemList;
+ pr_ioItemPtr rlastOutItem;
+ pr_GlobalState rgState;
+ pr_Path ractpath;
+ pr_Path rbestpath;
+ picoos_int32 rnritems;
+ pr_WorkItems ritems;
+ picoos_int32 rignore;
+ picoos_int32 spellMode;
+ picoos_int32 maxPathLen;
+ picoos_bool insidePhoneme;
+
+ picoos_uint8 inBuf[IN_BUF_SIZE+PICODATA_ITEM_HEADSIZE]; /* internal input buffer */
+ picoos_uint16 inBufLen;
+
+ picoos_uint8 outBuf[OUT_BUF_SIZE]; /* internal output buffer */
+ picoos_uint16 outReadPos; /* next pos to read from outBuf */
+ picoos_uint16 outWritePos; /* next pos to write to outBuf */
+
+ picokpr_Preproc preproc[PR_MAX_NR_PREPROC];
+ pr_ContextList ctxList;
+ pr_ProdList prodList;
+
+ pr_ContextList actCtx;
+ picoos_bool actCtxChanged;
+
+ picoos_uchar tmpStr1[PR_MAX_DATA_LEN_Z];
+ picoos_uchar tmpStr2[PR_MAX_DATA_LEN_Z];
+
+ picoos_uint8 pr_WorkMem[PR_WORK_MEM_SIZE];
+ picoos_uint32 workMemTop;
+ picoos_uint32 maxWorkMemTop;
+ picoos_uint8 pr_DynMem[PR_DYN_MEM_SIZE];
+ picoos_MemoryManager dynMemMM;
+ picoos_int32 dynMemSize;
+ picoos_int32 maxDynMemSize;
+
+ picoos_bool outOfMemory;
+
+ picoos_bool forceOutput;
+ picoos_int16 nrIterations;
+
+ picoos_uchar lspaces[128];
+ picoos_uchar saveFile[IN_BUF_SIZE];
+
+ pr_ioItem tmpItem;
+
+ picotrns_SimpleTransducer transducer;
+
+ /* kbs */
+
+ picoktab_Graphs graphs;
+ picokfst_FST xsampa_parser;
+ picokfst_FST svoxpa_parser;
+ picokfst_FST xsampa2svoxpa_mapper;
+
+} pr_subobj_t;
+
+/* *****************************************************************************/
+/* prototypes */
+
+static void pr_getOutputItemList (picodata_ProcessingUnit this, pr_subobj_t * pr,
+ picokpr_Preproc network,
+ picokpr_OutItemArrOffset outitem,
+ pr_OutItemVarPtr vars,
+ pr_ioItemPtr * first, pr_ioItemPtr * last);
+
+/* *****************************************************************************/
+
+#define pr_iABS(X) (((X) < 0) ? (-(X)) : (X))
+
+/* *****************************************************************************/
+/* module internal memory managment for dynamic and working memory using memory
+ partitions allocated with pr_subobj_t.
+ Dynamic memory is allocated in pr_subobj_t->pr_DynMem. Dynamic memory has
+ to be deallocated again with pr_DEALLOCATE.
+ Working memory is allocated in pr_subobj_t->pr_WorkMem. Working memory is stack
+ based and may not to be deallocated with pr_DEALLOCATE, but with pr_resetMemState
+ to a state previously saved with pr_getMemState.
+*/
+
+static void pr_ALLOCATE (picodata_ProcessingUnit this, pr_MemTypes mType, void * * adr, unsigned int byteSize)
+ /* allocates 'byteSize' bytes in the memery partition given by 'mType' */
+{
+ pr_subobj_t * pr = (pr_subobj_t *) this->subObj;
+ picoos_int32 incrUsedBytes, prevmaxDynMemSize;
+
+ if (mType == pr_WorkMem) {
+ if ((pr->workMemTop + byteSize) < PR_WORK_MEM_SIZE) {
+ (*adr) = (void *)(&(pr->pr_WorkMem[pr->workMemTop]));
+ byteSize = ((byteSize + PICOOS_ALIGN_SIZE - 1) / PICOOS_ALIGN_SIZE) * PICOOS_ALIGN_SIZE;
+ pr->workMemTop += byteSize;
+#if PR_TRACE_MEM
+ PICODBG_INFO(("pr_WorkMem: +%u, tot:%i of %i", byteSize, pr->workMemTop, PR_WORK_MEM_SIZE));
+#endif
+
+ if (pr->workMemTop > pr->maxWorkMemTop) {
+ pr->maxWorkMemTop = pr->workMemTop;
+#if PR_TRACE_MAX_MEM
+ PICODBG_INFO(("new max pr_WorkMem: %i of %i", pr->workMemTop, PR_WORK_MEM_SIZE));
+#endif
+ }
+ }
+ else {
+ (*adr) = NULL;
+ PICODBG_ERROR(("pr out of working memory"));
+ picoos_emRaiseException(this->common->em, PICO_EXC_OUT_OF_MEM, (picoos_char *)"pr out of dynamic memory", (picoos_char *)"");
+ pr->outOfMemory = TRUE;
+ }
+ }
+ else if (mType == pr_DynMem) {
+ (*adr) = picoos_allocate(pr->dynMemMM, byteSize);
+ if ((*adr) != NULL) {
+ prevmaxDynMemSize = pr->maxDynMemSize;
+ picoos_getMemUsage(pr->dynMemMM, 1, &pr->dynMemSize, &incrUsedBytes, &pr->maxDynMemSize);
+#if PR_TRACE_MEM
+ PICODBG_INFO(("pr_DynMem : +%i, tot:%i of %i", incrUsedBytes, pr->dynMemSize, PR_DYN_MEM_SIZE));
+#endif
+
+#if PR_TRACE_MAX_MEM
+ if (pr->maxDynMemSize > prevmaxDynMemSize) {
+ PICODBG_INFO(("new max pr_DynMem : %i of %i", pr->maxDynMemSize, PR_DYN_MEM_SIZE));
+ }
+#endif
+ }
+ else {
+ PICODBG_ERROR(("pr out of dynamic memory"));
+ picoos_emRaiseException(this->common->em, PICO_EXC_OUT_OF_MEM, (picoos_char *)"pr out of dynamic memory", (picoos_char *)"");
+ pr->outOfMemory = TRUE;
+ }
+ }
+ else {
+ (*adr) = NULL;
+ }
+}
+
+
+static void pr_DEALLOCATE (picodata_ProcessingUnit this, pr_MemTypes mType, void * * adr)
+{
+ pr_subobj_t * pr = (pr_subobj_t *) this->subObj;
+ picoos_int32 incrUsedBytes;
+ if (mType == pr_WorkMem) {
+ PICODBG_INFO(("not possible; use pr_resetMemState instead"));
+ }
+ else if (mType == pr_DynMem) {
+ picoos_deallocate(pr->dynMemMM, &(*adr));
+ picoos_getMemUsage(pr->dynMemMM, 1, &pr->dynMemSize, &incrUsedBytes, &pr->maxDynMemSize);
+#if PR_TRACE_MEM
+ PICODBG_INFO(("pr_DynMem : %i, tot:%i of %i: adr: %u", incrUsedBytes, pr->dynMemSize, PR_DYN_MEM_SIZE, *adr));
+#endif
+ }
+ else {
+ (*adr) = NULL;
+ }
+}
+
+
+static void pr_getMemState(picodata_ProcessingUnit this, pr_MemTypes mType, picoos_uint32 *lmemState)
+{
+ pr_subobj_t * pr = (pr_subobj_t *) this->subObj;
+ mType = mType; /* avoid warning "var not used in this function"*/
+ *lmemState = pr->workMemTop;
+}
+
+
+static void pr_resetMemState(picodata_ProcessingUnit this, pr_MemTypes mType, picoos_uint32 lmemState)
+{
+ pr_subobj_t * pr = (pr_subobj_t *) this->subObj;
+
+#if PR_TRACE_MEM
+ PICODBG_INFO(("pr_WorkMem: -%i, tot:%i of %i", pr->workMemTop-lmemState, lmemState, PR_WORK_MEM_SIZE));
+#endif
+ mType = mType; /* avoid warning "var not used in this function"*/
+ pr->workMemTop = lmemState;
+}
+
+
+/* *****************************************************************************/
+/* string operations */
+
+static picoos_int32 pr_strlen(const picoos_uchar * str)
+{
+ picoos_int32 i;
+
+ i=0;
+ while ((i<PR_MAX_DATA_LEN) && (str[i] != 0)) {
+ i++;
+ }
+ return i;
+}
+
+
+static picoos_uint32 pr_strcpy(picoos_uint8 * dest, const picoos_uint8 * src)
+{
+ picoos_int32 i;
+
+ i = 0;
+ while ((i<PR_MAX_DATA_LEN) && (src[i] != 0)) {
+ dest[i] = src[i];
+ i++;
+ }
+ dest[i] = 0;
+ return i;
+}
+
+
+static picoos_uint32 pr_strcat(picoos_uint8 * dest, const picoos_uint8 * src)
+{
+ picoos_int32 i, j;
+
+ i = 0;
+ while ((i<PR_MAX_DATA_LEN) && (dest[i] != 0)) {
+ i++;
+ }
+ j = 0;
+ while ((i<PR_MAX_DATA_LEN) && (j<PR_MAX_DATA_LEN) && (src[j] != 0)) {
+ dest[i] = src[j];
+ i++;
+ j++;
+ }
+ dest[i] = 0;
+ return i;
+}
+
+
+static void pr_getTermPartStr (picoos_uchar string[], picoos_int32 * ind, picoos_uchar termCh, picoos_uchar str[], picoos_bool * done)
+{
+ int j;
+ picoos_bool done1;
+
+ done1 = TRUE;
+ j = 0;
+ while ((*ind < PR_MAX_DATA_LEN) && (string[*ind] != termCh) && (string[*ind] != 0)) {
+ if (j < PR_MAX_DATA_LEN) {
+ str[j] = string[*ind];
+ j++;
+ } else {
+ done1 = FALSE;
+ }
+ (*ind)++;
+ }
+ if (j < PR_MAX_DATA_LEN) {
+ str[j] = 0;
+ }
+ *done = ((*ind < PR_MAX_DATA_LEN) && (string[*ind] == termCh));
+ if (*done) {
+ (*ind)++;
+ }
+ *done = *done && done1;
+}
+
+
+static picoos_int32 pr_removeSubstring (int pos, int len, unsigned char str[])
+{
+ int i;
+ int length;
+
+ length = pr_strlen(str);
+ if (pos >= length) {
+ return length;
+ } else {
+ i = pos + len;
+ while (i < length) {
+ str[pos] = str[i];
+ i++;
+ pos++;
+ }
+ str[pos] = 0;
+ return pos;
+ }
+}
+
+
+static picoos_bool pr_strEqual(picoos_uchar * str1, picoos_uchar * str2)
+{
+ return (picoos_strcmp((picoos_char *)str1, (picoos_char *)str2) == 0);
+}
+
+
+static void pr_int_to_string(picoos_int32 n, picoos_uchar * str, picoos_int32 maxstrlen)
+{
+ picoos_int32 i, len;
+ picoos_bool negative=FALSE;
+
+ len = 0;
+ str[0] = 0;
+ if (n<0) {
+ negative = TRUE;
+ n = -n;
+ len++;
+ }
+ i = n;
+
+ while (i>0) {
+ i = i / 10;
+ len++;
+ }
+
+ if (len<maxstrlen) {
+ str[len] = 0;
+ i = n;
+ while ((i>0) && (len>0)) {
+ len--;
+ str[len] = i % 10 + '0';
+ i = i / 10;
+ }
+ if (negative) {
+ len--;
+ str[len] = '-';
+ }
+ }
+}
+/* *****************************************************************************/
+
+static void pr_firstLetterToLowerCase (const picoos_uchar src[], picoos_uchar dest[])
+{
+
+ picoos_int32 i;
+ picoos_int32 j;
+ picoos_int32 l;
+ picoos_bool done;
+
+ i = 0;
+ j = 0;
+ l = picobase_det_utf8_length(src[0]);
+ while ((i < l) && (j < PR_MAX_DATA_LEN)) {
+ dest[j] = src[i];
+ i++;
+ j++;
+ }
+ if (j < PR_MAX_DATA_LEN) {
+ dest[j] = 0;
+ }
+ picobase_lowercase_utf8_str(dest, (picoos_char*)dest, PR_MAX_DATA_LEN, &done);
+ j = picobase_det_utf8_length(dest[0]);
+ l = pr_strlen(src);
+ while ((i < l) && (j < PR_MAX_DATA_LEN)) {
+ dest[j] = src[i];
+ i++;
+ j++;
+ }
+ dest[j] = 0;
+}
+
+
+static picoos_int32 tok_tokenDigitStrToInt (picodata_ProcessingUnit this, pr_subobj_t * pr, picoos_uchar stokenStr[])
+{
+ picoos_uint32 i;
+ picoos_uint32 l;
+ picoos_int32 id;
+ picoos_int32 val;
+ picoos_uint32 n;
+ picobase_utf8char utf8char;
+
+ val = 0;
+ i = 0;
+ l = pr_strlen(stokenStr);
+ while (i < l) {
+ picobase_get_next_utf8char(stokenStr, PR_MAX_DATA_LEN, & i, utf8char);
+ id = picoktab_graphOffset(pr->graphs, utf8char);
+ if (id > 0) {
+ if (picoktab_getIntPropValue(pr->graphs, id, &n)) {
+ val = (10 * val) + n;
+ } else {
+ val = ((10 * val) + (int)((int)utf8char[0] - (int)'0'));
+ }
+ } else if ((utf8char[0] >= '0') && (utf8char[0] <= '9')) {
+ val = 10 * val + ((int)utf8char[0] - (int)'0');
+ }
+ }
+ return val;
+}
+
+
+static picoos_bool pr_isLatinNumber (picoos_uchar str[], picoos_int32 * val)
+{
+
+ picoos_uint32 li;
+ picoos_uint32 llen;
+ picoos_uchar lact;
+ picoos_uchar lnext;
+ picoos_uchar lprev;
+ picoos_uchar llatinI;
+ picoos_uchar llatinV;
+ picoos_uchar llatinX;
+ picoos_uchar llatinL;
+ picoos_uchar llatinC;
+ picoos_uchar llatinD;
+ picoos_uchar llatinM;
+ picoos_int32 lseq;
+ picobase_utf8char utf8;
+
+ *val = 0;
+ llen = picobase_utf8_length(str, PR_MAX_DATA_LEN);
+ if (llen > 0) {
+ li = 0;
+ picobase_get_next_utf8char(str, PR_MAX_DATA_LEN, & li,utf8);
+ if (picobase_is_utf8_uppercase(utf8, PICOBASE_UTF8_MAXLEN)) {
+ llatinI = 'I';
+ llatinV = 'V';
+ llatinX = 'X';
+ llatinL = 'L';
+ llatinC = 'C';
+ llatinD = 'D';
+ llatinM = 'M';
+ } else {
+ llatinI = 'i';
+ llatinV = 'v';
+ llatinX = 'x';
+ llatinL = 'l';
+ llatinC = 'c';
+ llatinD = 'd';
+ llatinM = 'm';
+ }
+ lseq = 1000;
+ li = 0;
+ while (li < llen) {
+ if (li > 0) {
+ lprev = str[li - 1];
+ } else {
+ lprev = 0;
+ }
+ lact = str[li];
+ if (li < (llen - 1)) {
+ lnext = str[li + 1];
+ } else {
+ lnext = 0;
+ }
+ if ((lseq > 1) && (lact == llatinI)) {
+ if ((lprev != lact) && (lseq >= 4)) {
+ if (lnext == llatinV) {
+ *val = *val + 4;
+ li++;
+ lseq = 1;
+ } else if (lnext == llatinX) {
+ *val = *val + 9;
+ li++;
+ lseq = 1;
+ } else {
+ *val = *val + 1;
+ lseq = 3;
+ }
+ } else {
+ *val = *val + 1;
+ lseq = lseq - 1;
+ }
+ } else if ((lseq > 5) && (lact == llatinV)) {
+ *val = *val + 5;
+ lseq = 5;
+ } else if ((lseq > 10) && (lact == llatinX)) {
+ if ((lprev != lact) && (lseq >= 40)) {
+ if (lnext == llatinL) {
+ *val = *val + 40;
+ li++;
+ lseq = 10;
+ } else if (lnext == llatinC) {
+ *val = *val + 90;
+ li++;
+ lseq = 10;
+ } else {
+ *val = *val + 10;
+ lseq = 30;
+ }
+ } else {
+ *val = *val + 10;
+ lseq = lseq - 10;
+ }
+ } else if ((lseq > 50) && (lact == llatinL)) {
+ *val = *val + 50;
+ lseq = 50;
+ } else if ((lseq > 100) && (lact == llatinC)) {
+ if ((lprev != lact) && (lseq >= 400)) {
+ if (lnext == llatinD) {
+ *val = *val + 400;
+ li++;
+ lseq = 100;
+ } else if (lnext == llatinM) {
+ *val = *val + 900;
+ li++;
+ lseq = 100;
+ } else {
+ *val = *val + 100;
+ lseq = 300;
+ }
+ } else {
+ *val = *val + 100;
+ lseq = lseq - 100;
+ }
+ } else if ((lseq > 500) && (lact == llatinD)) {
+ *val = *val + 500;
+ lseq = 500;
+ } else if ((lseq >= 1000) && (lact == llatinM)) {
+ *val = *val + 1000;
+ } else {
+ return FALSE;
+ }
+ li++;
+ }
+ }
+ return TRUE;
+}
+
+
+static picoos_bool pr_isSUC (picoos_uchar str[])
+{
+
+ picoos_int32 li;
+ picoos_bool lis;
+ picobase_utf8char lutf;
+ picoos_int32 lj;
+ picoos_int32 ll;
+ picoos_bool luc;
+
+ li = 0;
+ lis = TRUE;
+ luc = TRUE;
+ while (lis && (li < PR_MAX_DATA_LEN) && (str[li] != 0)) {
+ lj = 0;
+ ll = picobase_det_utf8_length(str[li]);
+ while (lj < ll) {
+ lutf[lj] = str[li];
+ lj++;
+ li++;
+ }
+ lutf[lj] = 0;
+ if (luc) {
+ lis = lis && picobase_is_utf8_uppercase(lutf,PICOBASE_UTF8_MAXLEN+1);
+ } else {
+ lis = lis && picobase_is_utf8_lowercase(lutf,PICOBASE_UTF8_MAXLEN+1);
+ }
+ luc = FALSE;
+ }
+ return lis;
+}
+
+/* *****************************************************************************/
+
+static picoos_bool pr_isCmdType (pr_ioItemPtr it, picoos_uint8 type)
+{
+ if ((it != NULL) && (it->head.type == PICODATA_ITEM_CMD) && (it->head.info1 == type)) {
+ return TRUE;
+ } else {
+ return FALSE;
+ }
+}
+
+
+static picoos_bool pr_isCmdInfo2 (pr_ioItemPtr it, picoos_uint8 info2)
+{
+ if ((it != NULL) && (it->head.type == PICODATA_ITEM_CMD) && (it->head.info2 == info2)) {
+ return TRUE;
+ } else {
+ return FALSE;
+ }
+}
+
+
+static void pr_initPathEle (struct pr_PathEle * ele)
+{
+ ele->rnetwork = NULL;
+ ele->rtok = 0;
+ ele->ritemid = -1;
+ ele->rdepth = 1;
+ ele->rlState = PR_LSInit;
+ ele->rcompare = -1;
+ ele->rprodname = 0;
+ ele->rprodprefcost = 0;
+}
+
+/* *****************************************************************************/
+
+static void pr_disposeProdList (register picodata_ProcessingUnit this, pr_ProdList * prodList)
+{
+ pr_ProdList p;
+
+ while ((*prodList) != NULL) {
+ p = (*prodList);
+ (*prodList) = (*prodList)->rNext;
+ picoos_deallocate(this->common->mm, (void *) &p);
+ }
+}
+
+
+static pico_Status pr_addContext (register picodata_ProcessingUnit this, pr_subobj_t * pr, pr_ContextList * ctxList, picokpr_VarStrPtr contextNamePtr, picokpr_VarStrPtr netNamePtr, picokpr_VarStrPtr prodNamePtr)
+{
+ picokpr_Preproc net;
+ pr_ContextList ctx;
+ pr_ProdList prod;
+ int i;
+ picokpr_VarStrPtr strp;
+ picoos_int32 lprodarrlen;
+
+ ctx = (*ctxList);
+ while ((ctx != NULL) && !(pr_strEqual(contextNamePtr, ctx->rContextName))) {
+ ctx = ctx->rNext;
+ }
+ if (ctx == NULL) {
+ ctx = picoos_allocate(this->common->mm, sizeof(pr_Context));
+ if (ctx == NULL) {
+ return PICO_EXC_OUT_OF_MEM;
+ }
+ ctx->rNext = (*ctxList);
+ ctx->rProdList = NULL;
+ ctx->rContextName = contextNamePtr;
+ (*ctxList) = ctx;
+ }
+ i = 0;
+ net = pr->preproc[i];
+ while ((i<PR_MAX_NR_PREPROC) && (net != NULL) && !(pr_strEqual(netNamePtr, picokpr_getPreprocNetName(net)))) {
+ i++;
+ net = pr->preproc[i];
+ }
+ if (net != NULL) {
+ i = 0;
+ strp = picokpr_getVarStrPtr(net, picokpr_getProdNameOfs(net, i));
+ lprodarrlen = picokpr_getProdArrLen(net);
+ while ((i < lprodarrlen) && !(pr_strEqual(prodNamePtr, strp))) {
+ i++;
+ if (i < lprodarrlen) {
+ strp = picokpr_getVarStrPtr(net, picokpr_getProdNameOfs(net, i));
+ }
+ }
+ if (i < lprodarrlen) {
+ prod = picoos_allocate(this->common->mm, sizeof(pr_Prod));
+ if (prod == NULL) {
+ return PICO_EXC_OUT_OF_MEM;
+ }
+ prod->rNetwork = net;
+ prod->rProdOfs = i;
+ prod->rNext = ctx->rProdList;
+ ctx->rProdList = prod;
+ }
+ }
+ return PICO_OK;
+}
+
+
+static pico_Status pr_createContextList (register picodata_ProcessingUnit this)
+{
+ pr_subobj_t * pr = (pr_subobj_t *) this->subObj;
+ picokpr_VarStrPtr ctxNamePtr;
+ picokpr_VarStrPtr netNamePtr;
+ picokpr_VarStrPtr prodNamePtr;
+ picoos_int32 p, i, n;
+ pico_Status status;
+
+ pr->ctxList = NULL;
+ for (p=0; p<PR_MAX_NR_PREPROC; p++) {
+ if (pr->preproc[p] != NULL) {
+ n = picokpr_getCtxArrLen(pr->preproc[p]);
+ for (i = 1; i<n; i++) {
+ ctxNamePtr = picokpr_getVarStrPtr(pr->preproc[p], picokpr_getCtxCtxNameOfs(pr->preproc[p], i));
+ netNamePtr = picokpr_getVarStrPtr(pr->preproc[p], picokpr_getCtxNetNameOfs(pr->preproc[p], i));
+ prodNamePtr = picokpr_getVarStrPtr(pr->preproc[p], picokpr_getCtxProdNameOfs(pr->preproc[p], i));
+ status = pr_addContext(this, pr, &pr->ctxList, ctxNamePtr,netNamePtr, prodNamePtr);
+ if (status != PICO_OK) {
+ return status;
+ }
+ }
+ }
+ }
+ return PICO_OK;
+}
+
+
+static void pr_disposeContextList (register picodata_ProcessingUnit this)
+{
+ pr_subobj_t * pr = (pr_subobj_t *) this->subObj;
+ pr_ContextList c;
+
+ while (pr->ctxList != NULL) {
+ c = pr->ctxList;
+ pr->ctxList = pr->ctxList->rNext;
+ pr_disposeProdList(this, & c->rProdList);
+ picoos_deallocate(this->common->mm, (void *) &c);
+ }
+}
+
+
+static pr_ContextList pr_findContext (pr_ContextList contextList, picoos_uchar contextName[])
+{
+ pr_ContextList context;
+
+ context = contextList;
+ while ((context != NULL) && !(pr_strEqual(context->rContextName,contextName))) {
+ context = context->rNext;
+ }
+ return context;
+}
+
+
+static void pr_setContext (register picodata_ProcessingUnit this, pr_subobj_t * pr, picoos_uchar context[])
+{
+
+ pr_ContextList ctx;
+
+ ctx = pr_findContext(pr->ctxList,context);
+ if (ctx != NULL) {
+ pr->actCtx = ctx;
+ pr->actCtxChanged = TRUE;
+ } else {
+ PICODBG_WARN(("context '%s' not found; no change",context));
+ picoos_emRaiseWarning(this->common->em, PICO_ERR_CONTEXT_NOT_FOUND, (picoos_char*)"context '%s' not found; no change",(picoos_char*)context);
+ }
+}
+
+/* *****************************************************************************/
+/* item handling routines */
+
+
+static picoos_uint32 pr_copyData(picoos_uint8 * dest, const picoos_uint8 * src, picoos_int32 nrBytes, picoos_bool zeroTerm)
+{
+ picoos_int32 i=0;
+
+ if ((src != NULL) && (dest != NULL)) {
+ i = 0;
+ while ((i<nrBytes) && (i<PR_MAX_DATA_LEN)) {
+ dest[i] = src[i];
+ i++;
+ }
+ if (zeroTerm) {
+ dest[i] = 0;
+ }
+ }
+ return i;
+}
+
+
+static void pr_initItem(picodata_ProcessingUnit this, pr_ioItem * item)
+{
+ item->next = NULL;
+ item->val = 0;
+ item->head.len = 0;
+ item->strci = NULL;
+ item->strcis = NULL;
+ item->suc = FALSE;
+ item->alc = FALSE;
+ item->auc = FALSE;
+}
+
+
+static void pr_newItem (picodata_ProcessingUnit this, pr_MemTypes mType, pr_ioItemPtr * item, picoos_uint8 itemType, picoos_int32 size, picoos_bool inItem)
+{
+ pr_subobj_t * pr = (pr_subobj_t *) this->subObj;
+
+ if (mType == pr_WorkMem) {
+ pr_ALLOCATE(this, mType, (void * *) & (*item),PR_IOITEM_MIN_SIZE+size+1);
+ if (pr->outOfMemory) return;
+ pr_initItem(this, *item);
+ }
+ else if ((mType == pr_DynMem) && inItem) {
+ pr_ALLOCATE(this, mType, (void * *) & (*item), PR_IOITEM_MIN_SIZE+3*size+3);
+ if (pr->outOfMemory) return;
+ pr_initItem(this, *item);
+ if (itemType == PICODATA_ITEM_TOKEN) {
+ (*item)->strci = &((*item)->data[size+1]);
+ (*item)->strcis = &((*item)->data[2*size+2]);
+ (*item)->strci[0] = 0;
+ (*item)->strcis[0] = 0;
+ }
+ }
+ else if ((mType == pr_DynMem) && !inItem) {
+ pr_ALLOCATE(this, mType, (void * *) & (*item), PR_IOITEM_MIN_SIZE+size+1);
+ if (pr->outOfMemory) return;
+ pr_initItem(this, *item);
+ }
+
+ (*item)->data[0] = 0;
+}
+
+
+static void pr_copyItemContent (picodata_ProcessingUnit this, pr_ioItem * inItem, pr_ioItem * outItem)
+{
+ if (outItem != NULL) {
+ outItem->next = inItem->next;
+ outItem->val = inItem->val;
+ outItem->head = inItem->head;
+ outItem->suc = inItem->suc;
+ outItem->alc = inItem->alc;
+ outItem->auc = inItem->auc;
+ if (inItem->head.len > 0 ) {
+ pr_copyData(outItem->data, inItem->data, inItem->head.len, /*zeroTerm*/TRUE);
+ pr_copyData(outItem->strci, inItem->strci, inItem->head.len, /*zeroTerm*/TRUE);
+ pr_copyData(outItem->strcis, inItem->strcis, inItem->head.len, /*zeroTerm*/TRUE);
+ }
+ }
+}
+
+
+static void pr_copyItem (picodata_ProcessingUnit this, pr_MemTypes mType, pr_ioItemPtr inItem, pr_ioItemPtr * outItem)
+{
+ pr_subobj_t * pr = (pr_subobj_t *) this->subObj;
+
+ if (inItem != NULL) {
+ pr_newItem(this, mType,& (*outItem), inItem->head.type, inItem->head.len, FALSE);
+ if (pr->outOfMemory) return;
+ pr_copyItemContent(this, inItem, *outItem);
+ }
+ else {
+ outItem = NULL;
+ }
+}
+
+
+static void pr_startItemList (pr_ioItemPtr * firstItem, pr_ioItemPtr * lastItem)
+{
+ *firstItem = NULL;
+ *lastItem = NULL;
+}
+
+
+static void pr_appendItem (picodata_ProcessingUnit this, pr_ioItemPtr * firstItem, pr_ioItemPtr * lastItem, pr_ioItemPtr item)
+{
+ if (item != NULL) {
+ item->next = NULL;
+ if ((*lastItem) == NULL) {
+ *firstItem = item;
+ } else {
+ (*lastItem)->next = item;
+ }
+ (*lastItem) = item;
+ }
+}
+
+
+static void pr_disposeItem (picodata_ProcessingUnit this, pr_ioItemPtr * item)
+{
+ if ((*item) != NULL) {
+ pr_DEALLOCATE(this, pr_DynMem, (void * *) & (*item));
+ }
+}
+
+
+static void pr_putItem (picodata_ProcessingUnit this, pr_subobj_t * pr,
+ pr_ioItemPtr * first, pr_ioItemPtr * last,
+ picoos_uint8 itemType, picoos_uint8 info1, picoos_uint8 info2,
+ picoos_uint16 val,
+ picoos_uchar str[])
+{
+ picoos_int32 i;
+ pr_ioItemPtr item;
+
+ pr->tmpItem.next = NULL;
+ pr->tmpItem.val = 0;
+ pr->tmpItem.head.type = itemType;
+ pr->tmpItem.head.info1 = info1;
+ pr->tmpItem.head.info2 = info2;
+
+ pr_initItem(this, &pr->tmpItem);
+ switch (itemType) {
+ case PICODATA_ITEM_CMD:
+ switch (info1) {
+ case PICODATA_ITEMINFO1_CMD_CONTEXT:
+ case PICODATA_ITEMINFO1_CMD_VOICE:
+ case PICODATA_ITEMINFO1_CMD_MARKER:
+ case PICODATA_ITEMINFO1_CMD_PLAY:
+ case PICODATA_ITEMINFO1_CMD_SAVE:
+ case PICODATA_ITEMINFO1_CMD_UNSAVE:
+ case PICODATA_ITEMINFO1_CMD_PROSDOMAIN:
+ pr->tmpItem.head.len = picoos_strlen((picoos_char*)str);
+ for (i=0; i<pr->tmpItem.head.len; i++) {
+ pr->tmpItem.data[i] = str[i];
+ }
+ pr_copyItem(this, pr_WorkMem, &pr->tmpItem, & item);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & (*first),& (*last),item);
+ break;
+ case PICODATA_ITEMINFO1_CMD_IGNSIG:
+ case PICODATA_ITEMINFO1_CMD_IGNORE:
+ case PICODATA_ITEMINFO1_CMD_FLUSH:
+ pr->tmpItem.head.len = 0;
+ pr_copyItem(this, pr_WorkMem, &pr->tmpItem, & item);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & (*first),& (*last),item);
+ break;
+ case PICODATA_ITEMINFO1_CMD_SPEED:
+ case PICODATA_ITEMINFO1_CMD_PITCH:
+ case PICODATA_ITEMINFO1_CMD_VOLUME:
+ case PICODATA_ITEMINFO1_CMD_SPELL:
+ case PICODATA_ITEMINFO1_CMD_SIL:
+ pr->tmpItem.head.len = 2;
+ pr->tmpItem.data[0] = val % 256;
+ pr->tmpItem.data[1] = val / 256;
+ pr_copyItem(this, pr_WorkMem, &pr->tmpItem, & item);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & (*first),& (*last),item);
+ break;
+ case PICODATA_ITEMINFO1_CMD_PHONEME:
+ PICODBG_WARN(("phoneme command not yet implemented"));
+ break;
+ default:
+ PICODBG_WARN(("pr_putItem: unknown command type"));
+ }
+ break;
+ case PICODATA_ITEM_TOKEN:
+ pr->tmpItem.head.len = picoos_strlen((picoos_char*)str);
+ for (i=0; i<pr->tmpItem.head.len; i++) {
+ pr->tmpItem.data[i] = str[i];
+ }
+ pr_copyItem(this, pr_WorkMem, &pr->tmpItem, & item);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & (*first),& (*last),item);
+ break;
+ default:
+ PICODBG_WARN(("pr_putItem: unknown item type"));
+ }
+}
+
+
+static void pr_appendItemToOutItemList (picodata_ProcessingUnit this, pr_subobj_t * pr,
+ pr_ioItemPtr * firstItem, pr_ioItemPtr * lastItem, pr_ioItemPtr item)
+{
+ pr_ioItemPtr litem;
+ picoos_int32 li;
+ picoos_int32 li2;
+ picoos_int32 lid;
+ picoos_int32 ln;
+ picoos_int32 ln2;
+ picoos_uint8 ltype;
+ picoos_int8 lsubtype;
+ picoos_uchar lstr[10];
+ picoos_bool ldone;
+
+ item->next = NULL;
+ if (((pr->spellMode != 0) && (item->head.type == PICODATA_ITEM_TOKEN) && (item->head.info1 != PICODATA_ITEMINFO1_TOKTYPE_SPACE))) {
+ li = 0;
+ ln = pr_strlen(item->data);
+ while (li < ln) {
+ ln2 = picobase_det_utf8_length(item->data[li]);
+ for (li2 = 0; li2<ln2; li2++) {
+ lstr[li2] = item->data[li];
+ li++;
+ }
+ lstr[ln2] = 0;
+ lid = picoktab_graphOffset(pr->graphs, lstr);
+ if ((lid > 0) && picoktab_getIntPropTokenType(pr->graphs, lid, <ype) &&
+ ((ltype == PICODATA_ITEMINFO1_TOKTYPE_LETTERV) /*|| (ltype == PICODATA_ITEMINFO1_TOKTYPE_DIGIT)*/)) {
+ ln2 = pr_strcat(lstr,(picoos_uchar*)SPEC_CHAR);
+ picoktab_getIntPropTokenSubType(pr->graphs,lid, &lsubtype);
+ }
+ else {
+ ltype = PICODATA_ITEMINFO1_TOKTYPE_UNDEFINED;
+ lsubtype = -(1);
+ }
+ pr_newItem(this, pr_DynMem,& litem, PICODATA_ITEM_TOKEN, ln2, /*inItem*/FALSE);
+ if (pr->outOfMemory) return;
+ litem->head.type = PICODATA_ITEM_TOKEN;
+ litem->head.info1 = item->head.info1;
+ litem->head.info2 = item->head.info2;
+ pr_strcpy(litem->data, lstr);
+ litem->data[ln2] = 0;
+ litem->head.len = ln2;
+ if (litem->head.info1 == PICODATA_ITEMINFO1_TOKTYPE_DIGIT) {
+ litem->val = tok_tokenDigitStrToInt(this, pr, litem->data);
+ } else {
+ litem->val = 0;
+ }
+ picobase_lowercase_utf8_str(litem->data,litem->strci,PR_MAX_DATA_LEN, &ldone);
+ pr_firstLetterToLowerCase(litem->data,litem->strcis);
+ litem->alc = picobase_is_utf8_lowercase(litem->data,PR_MAX_DATA_LEN);
+ litem->auc = picobase_is_utf8_uppercase(litem->data,PR_MAX_DATA_LEN);
+ litem->suc = pr_isSUC(litem->data);
+
+ pr_appendItem(this, firstItem, lastItem, litem);
+ if (pr->spellMode == PR_SPELL_WITH_SENTENCE_BREAK) {
+ pr_newItem(this, pr_DynMem,& litem, PICODATA_ITEM_TOKEN, 2, /*inItem*/FALSE);
+ if (pr->outOfMemory) return;
+ litem->head.type = PICODATA_ITEM_TOKEN;
+ litem->head.info1 = PICODATA_ITEMINFO1_TOKTYPE_CHAR;
+ litem->head.info2 = -1;
+ litem->head.len = 1;
+ litem->data[0] = ',';
+ litem->data[1] = 0;
+ litem->strci[0] = ',';
+ litem->strci[1] = 0;
+ litem->strcis[0] = ',';
+ litem->strcis[1] = 0;
+ litem->val = 0;
+ pr_appendItem(this, firstItem, lastItem, litem);
+ } else if (pr->spellMode == PR_SPELL_WITH_SENTENCE_BREAK) {
+ pr_newItem(this, pr_DynMem,& litem, PICODATA_ITEM_CMD, 0, /*inItem*/FALSE);
+ if (pr->outOfMemory) return;
+ litem->head.type = PICODATA_ITEM_CMD;
+ litem->head.info1 = PICODATA_ITEMINFO1_CMD_FLUSH;
+ litem->head.info2 = PICODATA_ITEMINFO2_NA;
+ litem->head.len = 0;
+ pr_appendItem(this, firstItem, lastItem,litem);
+ } else if (pr->spellMode > 0) {
+ pr_newItem(this, pr_DynMem,& litem, PICODATA_ITEM_CMD, 2, /*inItem*/FALSE);
+ if (pr->outOfMemory) return;
+ litem->head.type = PICODATA_ITEM_CMD;
+ litem->head.info1 = PICODATA_ITEMINFO1_CMD_SIL;
+ litem->head.info2 = PICODATA_ITEMINFO2_NA;
+ litem->head.len = 2;
+ litem->data[0] = pr->spellMode % 256;
+ litem->data[1] = pr->spellMode / 256;
+ pr_appendItem(this, firstItem, lastItem, litem);
+ }
+ }
+ pr_disposeItem(this, & item);
+ } else if (pr_isCmdType(item, PICODATA_ITEMINFO1_CMD_SPELL) && pr_isCmdInfo2(item, PICODATA_ITEMINFO2_CMD_START)) {
+ pr->spellMode = item->data[0]+256*item->data[1];
+ pr_disposeItem(this, & item);
+ } else if (pr_isCmdType(item, PICODATA_ITEMINFO1_CMD_SPELL) && pr_isCmdInfo2(item, PICODATA_ITEMINFO2_CMD_END)) {
+ pr->spellMode = 0;
+ pr_disposeItem(this, & item);
+ } else {
+ pr_appendItem(this, firstItem,lastItem,item);
+ }
+}
+
+
+/* *****************************************************************************/
+
+static pr_OutItemVarPtr pr_findVariable (pr_OutItemVarPtr vars, picoos_int32 id)
+{
+ while ((vars != NULL) && (vars->id != id)) {
+ vars = vars->next;
+ }
+ if ((vars != NULL)) {
+ return vars;
+ } else {
+ return NULL;
+ }
+}
+
+
+static void pr_genCommands (picodata_ProcessingUnit this, pr_subobj_t * pr,
+ picokpr_Preproc network, picokpr_OutItemArrOffset outitem, pr_OutItemVarPtr vars, pr_ioItemPtr * first, pr_ioItemPtr * last)
+{
+
+ pr_ioItemPtr litem;
+ pr_OutItemVarPtr lvar;
+ picoos_uint8 lcmd;
+ picoos_uint8 linfo2;
+ picoos_bool ldone;
+#if 0
+ picoos_int32 lphontype;
+#endif
+ picokpr_VarStrPtr lstrp;
+ picoos_int32 lnum;
+ pr_ioItemPtr lf;
+ pr_ioItemPtr ll;
+ picoos_int32 lf0beg;
+ picoos_int32 lf0end;
+ ShortStrParam lxfadebeg;
+ ShortStrParam lxfadeend;
+ picoos_bool lout;
+ picoos_int32 ltype;
+ picoos_int32 argOfs;
+ picoos_int32 nextOfs;
+ picoos_int32 nextOfs2;
+ picoos_int32 nextOfs3;
+ picoos_int32 nextOfs4;
+ picoos_uchar alphabet[32];
+
+ lcmd = 0;
+ lnum = 0;
+ litem = NULL;
+ ltype = picokpr_getOutItemType(network, outitem);
+ switch (ltype) {
+ case PR_OIgnore:
+ if (picokpr_getOutItemVal(network, outitem) == 0) {
+ pr_putItem(this, pr, & (*first),& (*last),PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_IGNORE,PICODATA_ITEMINFO2_CMD_START,0,(picoos_uchar*)"");
+ } else {
+ pr_putItem(this, pr, & (*first),& (*last),PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_IGNORE,PICODATA_ITEMINFO2_CMD_END,0,(picoos_uchar*)"");
+ }
+ break;
+ case PR_OPitch: case PR_OSpeed: case PR_OVolume:
+ switch (ltype) {
+ case PR_OPitch:
+ lcmd = PICODATA_ITEMINFO1_CMD_PITCH;
+ lnum = PICO_PITCH_DEFAULT;
+ break;
+ case PR_OSpeed:
+ lcmd = PICODATA_ITEMINFO1_CMD_SPEED;
+ lnum = PICO_SPEED_DEFAULT;
+ break;
+ case PR_OVolume:
+ lcmd = PICODATA_ITEMINFO1_CMD_VOLUME;
+ lnum = PICO_VOLUME_DEFAULT;
+ break;
+ default:
+ break;
+ }
+ if ((picokpr_getOutItemArgOfs(network, outitem) != 0)) {
+ switch (picokpr_getOutItemType(network, picokpr_getOutItemArgOfs(network, outitem))) {
+ case PR_OVal:
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD, lcmd,PICODATA_ITEMINFO2_CMD_ABSOLUTE, picokpr_getOutItemVal(network, picokpr_getOutItemArgOfs(network, outitem)),(picoos_uchar*)"");
+ break;
+ case PR_OVar:
+ lvar = pr_findVariable(vars,picokpr_getOutItemVal(network, picokpr_getOutItemArgOfs(network, outitem)));
+ if ((lvar != NULL) && (lvar->first != NULL) && (lvar->first->head.type == PICODATA_ITEM_TOKEN)) {
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD,lcmd,PICODATA_ITEMINFO2_CMD_ABSOLUTE,picoos_atoi((picoos_char*)lvar->first->data),(picoos_uchar*)"");
+ }
+ break;
+ default:
+ pr_startItemList(& lf,& ll);
+ pr_getOutputItemList(this, pr, network,picokpr_getOutItemArgOfs(network, outitem),vars,& lf,& ll);
+ if (pr->outOfMemory) return;
+ if (((lf != NULL) && (lf->head.type == PICODATA_ITEM_TOKEN))) {
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD,lcmd,PICODATA_ITEMINFO2_CMD_ABSOLUTE,picoos_atoi((picoos_char*)lf->data),(picoos_uchar*)"");
+ }
+ break;
+ }
+ } else {
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD,lcmd,PICODATA_ITEMINFO2_CMD_ABSOLUTE,lnum,(picoos_uchar*)"");
+ }
+ break;
+
+ case PR_OPhonSVOXPA: case PR_OPhonSAMPA:
+ if (picokpr_getOutItemArgOfs(network, outitem) != 0) {
+ if (ltype == PR_OPhonSVOXPA) {
+ picoos_strlcpy(alphabet, PICODATA_SVOXPA, sizeof(alphabet));
+ }
+ else {
+ picoos_strlcpy(alphabet, PICODATA_SAMPA, sizeof(alphabet));
+ }
+ pr_startItemList(& lf,& ll);
+ pr_getOutputItemList(this, pr, network,picokpr_getOutItemArgOfs(network, outitem),vars,& lf,& ll);
+ if (pr->outOfMemory) return;
+ if (lf != NULL) {
+ ldone = FALSE;
+ if (lf->head.type == PICODATA_ITEM_TOKEN) {
+ if (picodata_mapPAStrToPAIds(pr->transducer, this->common, pr->xsampa_parser, pr->svoxpa_parser, pr->xsampa2svoxpa_mapper, lf->data, alphabet, pr->tmpStr1, sizeof(pr->tmpStr1)-1) == PICO_OK) {
+ pr_putItem(this, pr, & (*first),& (*last), PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_PHONEME,
+ PICODATA_ITEMINFO2_CMD_START, 0, pr->tmpStr1);
+ ldone = TRUE;
+ }
+ else {
+ PICODBG_WARN(("cannot map phonetic string '%s'; synthesizeing text instead", lf->data));
+ picoos_emRaiseWarning(this->common->em, PICO_ERR_OTHER,(picoos_char*)"", (picoos_char*)"cannot map phonetic string '%s'; synthesizing text instead", lf->data);
+ }
+ }
+ if (ldone) {
+ lf = lf->next;
+ while (lf != NULL) {
+ if (lf->head.type == PICODATA_ITEM_TOKEN) {
+ pr_putItem(this, pr, & (*first),& (*last), PICODATA_ITEM_TOKEN, PICODATA_ITEMINFO1_CMD_PHONEME,
+ PICODATA_ITEMINFO2_CMD_END, 0, (picoos_char *)"");
+ }
+ lf = lf->next;
+ }
+ pr_putItem(this, pr, & (*first),& (*last), PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_PHONEME,
+ PICODATA_ITEMINFO2_CMD_END, 0, (picoos_char *)"");
+ }
+ }
+ }
+ break;
+
+ case PR_OSent:
+ pr_putItem(this, pr, & (*first),& (*last), PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_FLUSH, PICODATA_ITEMINFO2_NA, 0, (picoos_uchar*)"");
+ break;
+ case PR_OPara:
+ pr_putItem(this, pr, & (*first),& (*last), PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_FLUSH, PICODATA_ITEMINFO2_NA, 0, (picoos_uchar*)"");
+ if (picokpr_getOutItemVal(network, outitem) == 1) {
+ pr_putItem(this, pr, & (*first),& (*last), PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_SIL, PICODATA_ITEMINFO2_NA, PICO_PARAGRAPH_PAUSE_DUR, (picoos_uchar*)"");
+ }
+ break;
+ case PR_OBreak:
+ if ((picokpr_getOutItemArgOfs(network, outitem) != 0)) {
+ switch (picokpr_getOutItemType(network, picokpr_getOutItemArgOfs(network, outitem))) {
+ case PR_OVal:
+ pr_putItem(this, pr, & (*first),& (*last), PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_SIL, PICODATA_ITEMINFO2_NA, picokpr_getOutItemVal(network, picokpr_getOutItemArgOfs(network, outitem)), (picoos_uchar*)"");
+ break;
+ case PR_OVar:
+ lvar = pr_findVariable(vars,picokpr_getOutItemVal(network, picokpr_getOutItemArgOfs(network, outitem)));
+ if ((lvar != NULL) && (lvar->first != NULL) && (lvar->first->head.type == PICODATA_ITEM_TOKEN)) {
+ pr_putItem(this, pr, & (*first),& (*last), PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_SIL, PICODATA_ITEMINFO2_NA, picoos_atoi((picoos_char*)lvar->first->data), (picoos_uchar*)"");
+ }
+ break;
+ default:
+ pr_startItemList(& lf,& ll);
+ pr_getOutputItemList(this, pr, network,picokpr_getOutItemArgOfs(network, outitem),vars,& lf,& ll);
+ if (pr->outOfMemory) return;
+ if (((lf != NULL) && (lf->head.type == PICODATA_ITEM_TOKEN))) {
+ pr_putItem(this, pr, & (*first),& (*last), PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_SIL, PICODATA_ITEMINFO2_NA, picoos_atoi((picoos_char*)lf->data), (picoos_uchar*)"");
+ }
+ break;
+ }
+ }
+ break;
+ case PR_OVoice: case PR_OContext: case PR_OMark:
+ if (picokpr_getOutItemType(network, outitem) == PR_OVoice) {
+ lcmd = PICODATA_ITEMINFO1_CMD_VOICE;
+ pr->tmpStr1[0] = 0;
+ lnum = 1;
+ } else if (picokpr_getOutItemType(network, outitem) == PR_OContext) {
+ lcmd = PICODATA_ITEMINFO1_CMD_CONTEXT;
+ pr_strcpy(pr->tmpStr1, (picoos_uchar*)PICO_CONTEXT_DEFAULT);
+ lnum = 1;
+ } else if ((picokpr_getOutItemType(network, outitem) == PR_OMark)) {
+ lcmd = PICODATA_ITEMINFO1_CMD_MARKER;
+ pr->tmpStr1[0] = 0;
+ lnum = 0;
+ }
+ if (picokpr_getOutItemArgOfs(network, outitem) != 0) {
+ switch (picokpr_getOutItemType(network, picokpr_getOutItemArgOfs(network, outitem))) {
+ case PR_OVar:
+ lvar = pr_findVariable(vars,picokpr_getOutItemVal(network, picokpr_getOutItemArgOfs(network, outitem)));
+ if (lvar != NULL) {
+ litem = lvar->first;
+ }
+ pr->tmpStr1[0] = 0;
+ while (litem != NULL) {
+ if (litem->head.type == PICODATA_ITEM_TOKEN) {
+ pr_strcat(pr->tmpStr1,litem->data);
+ }
+ litem = litem->next;
+ }
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD, lcmd,lnum,0,pr->tmpStr1);
+ break;
+ case PR_OStr:
+ if (picokpr_getOutItemStrOfs(network, picokpr_getOutItemArgOfs(network, outitem)) != 0) {
+ lstrp = picokpr_getOutItemStr(network, picokpr_getOutItemArgOfs(network, outitem));
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD,lcmd,lnum,0,lstrp);
+ }
+ break;
+ default:
+ pr_startItemList(& lf,& ll);
+ pr_getOutputItemList(this, pr, network,picokpr_getOutItemArgOfs(network, outitem),vars,& lf,& ll);
+ if (pr->outOfMemory) return;
+ if (((lf != NULL) && (lf->head.type == PICODATA_ITEM_TOKEN))) {
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD,lcmd,lnum,0,lf->data);
+ }
+ break;
+ }
+ } else {
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD,lcmd,lnum,0,pr->tmpStr1);
+ }
+ break;
+ case PR_OGenFile:
+ if (picokpr_getOutItemArgOfs(network, outitem) != 0) {
+ lcmd = PICODATA_ITEMINFO1_CMD_SAVE;
+ } else {
+ lcmd = PICODATA_ITEMINFO1_CMD_UNSAVE;
+ }
+ pr->tmpStr1[0] = 0;
+ lnum = 0;
+ if (picokpr_getOutItemArgOfs(network, outitem) != 0) {
+ switch (picokpr_getOutItemType(network, picokpr_getOutItemArgOfs(network, outitem))) {
+ case PR_OVar:
+ lvar = pr_findVariable(vars,picokpr_getOutItemVal(network, picokpr_getOutItemArgOfs(network, outitem)));
+ if (lvar != NULL) {
+ litem = lvar->first;
+ }
+ pr->tmpStr1[0] = 0;
+ while (litem != NULL) {
+ if (litem->head.type == PICODATA_ITEM_TOKEN) {
+ pr_strcat(pr->tmpStr1,litem->data);
+ }
+ litem = litem->next;
+ }
+ if ((lnum = picodata_getPuTypeFromExtension(pr->tmpStr1, /*input*/FALSE)) != PICODATA_ITEMINFO2_CMD_TO_UNKNOWN) {
+ if (pr->saveFile[0] != 0) {
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_UNSAVE,
+ picodata_getPuTypeFromExtension(pr->saveFile, /*input*/FALSE),0,pr->saveFile);
+ }
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD, lcmd,lnum,0,pr->tmpStr1);
+ pr_strcpy(pr->saveFile, pr->tmpStr1);
+ }
+ break;
+ case PR_OStr:
+ if (picokpr_getOutItemStrOfs(network, picokpr_getOutItemArgOfs(network, outitem)) != 0) {
+ lstrp = picokpr_getOutItemStr(network, picokpr_getOutItemArgOfs(network, outitem));
+ if ((lnum = picodata_getPuTypeFromExtension(lstrp, /*input*/FALSE)) != PICODATA_ITEMINFO2_CMD_TO_UNKNOWN) {
+ if (pr->saveFile[0] != 0) {
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_UNSAVE,
+ picodata_getPuTypeFromExtension(pr->saveFile, /*input*/FALSE),0,pr->saveFile);
+ }
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD, lcmd,lnum,0,lstrp);
+ pr_strcpy(pr->saveFile, lstrp);
+ }
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD,lcmd,lnum,0,lstrp);
+ }
+ break;
+ default:
+ pr_startItemList(& lf,& ll);
+ pr_getOutputItemList(this, pr, network,picokpr_getOutItemArgOfs(network, outitem),vars,& lf,& ll);
+ if (pr->outOfMemory) return;
+ if (((lf != NULL) && (lf->head.type == PICODATA_ITEM_TOKEN))) {
+ if ((lnum = picodata_getPuTypeFromExtension(lf->data, /*input*/FALSE)) != PICODATA_ITEMINFO2_CMD_TO_UNKNOWN) {
+ if (pr->saveFile[0] != 0) {
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_UNSAVE,
+ picodata_getPuTypeFromExtension(pr->saveFile, /*input*/FALSE),0,pr->saveFile);
+ }
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD, lcmd,lnum,0,lf->data);
+ pr_strcpy(pr->saveFile, lf->data);
+ }
+ }
+ break;
+ }
+/*
+ } else {
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD,lcmd,lnum,0,pr->tmpStr1);
+*/
+ }
+ break;
+ case PR_OUseSig: case PR_OPlay:
+ lout = FALSE;
+ lf0beg = -(1);
+ lf0end = -(1);
+ lxfadebeg[0] = 0;
+ lxfadeend[0] = 0;
+ pr->tmpStr1[0] = 0;
+ if ((picokpr_getOutItemType(network, outitem) == PR_OUseSig)) {
+ lcmd = PICODATA_ITEMINFO1_CMD_IGNSIG;
+ } else {
+ lcmd = PICODATA_ITEMINFO1_CMD_IGNORE;
+ }
+ if (picokpr_getOutItemArgOfs(network, outitem) != 0) {
+ linfo2 = PICODATA_ITEMINFO2_CMD_START;
+ } else {
+ linfo2 = PICODATA_ITEMINFO2_CMD_END;
+ }
+ if (picokpr_getOutItemArgOfs(network, outitem) != 0) {
+ switch (picokpr_getOutItemType(network, picokpr_getOutItemArgOfs(network, outitem))) {
+ case PR_OVar:
+ lvar = pr_findVariable(vars,picokpr_getOutItemVal(network, picokpr_getOutItemArgOfs(network, outitem)));
+ if (lvar != NULL) {
+ litem = lvar->first;
+ }
+ pr->tmpStr1[0] = 0;
+ while (litem != NULL) {
+ if (litem->head.type == PICODATA_ITEM_TOKEN) {
+ pr_strcat(pr->tmpStr1,litem->data);
+ }
+ litem = litem->next;
+ }
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_PLAY,
+ picodata_getPuTypeFromExtension(pr->tmpStr1, /*input*/TRUE),0, pr->tmpStr1);
+ lout = TRUE;
+ break;
+ case PR_OStr:
+ if (picokpr_getOutItemStrOfs(network, picokpr_getOutItemArgOfs(network, outitem)) != 0) {
+ lstrp = picokpr_getOutItemStr(network, picokpr_getOutItemArgOfs(network, outitem));
+ pr_strcpy(pr->tmpStr1, lstrp);
+ lout = TRUE;
+ }
+ break;
+ default:
+ pr_startItemList(& lf,& ll);
+ pr_getOutputItemList(this, pr, network,picokpr_getOutItemArgOfs(network, outitem),vars,& lf,& ll);
+ if (pr->outOfMemory) return;
+ if ((lf != NULL) && (lf->head.type == PICODATA_ITEM_TOKEN)) {
+ pr_strcpy(pr->tmpStr1, lf->data);
+ lout = TRUE;
+ }
+ break;
+ }
+ }
+ argOfs = picokpr_getOutItemArgOfs(network, outitem);
+ if (argOfs != 0) {
+ nextOfs = picokpr_getOutItemNextOfs(network, outitem);
+ if (nextOfs != 0) {
+ if (picokpr_getOutItemType(network, nextOfs) == PR_OVal) {
+ lf0beg = picokpr_getOutItemVal(network, nextOfs);
+ }
+ nextOfs2 = picokpr_getOutItemNextOfs(network, nextOfs);
+ if (nextOfs2 != 0) {
+ if (picokpr_getOutItemType(network, nextOfs2) == PR_OVal) {
+ lf0end = picokpr_getOutItemVal(network, nextOfs2);
+ }
+ nextOfs3 = picokpr_getOutItemNextOfs(network, nextOfs2);
+ if (nextOfs3 != 0) {
+ if ((picokpr_getOutItemType(network, nextOfs3) == PR_OStr) && (picokpr_getOutItemStrOfs(network, nextOfs3) != 0)) {
+ lstrp = picokpr_getOutItemStr(network, nextOfs3);
+ pr_strcpy(lxfadebeg, lstrp);
+ }
+ nextOfs4 = picokpr_getOutItemNextOfs(network, nextOfs3);
+ if (nextOfs4 != 0) {
+ if ((picokpr_getOutItemType(network, nextOfs4) == PR_OStr) && (picokpr_getOutItemStrOfs(network, nextOfs4) != 0)) {
+ lstrp = picokpr_getOutItemStr(network, nextOfs4);
+ pr_strcpy(lxfadeend, lstrp);
+ }
+ }
+ }
+ }
+ }
+ }
+ if (lout) {
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD,PICODATA_ITEMINFO1_CMD_PLAY,
+ picodata_getPuTypeFromExtension(pr->tmpStr1, /*input*/TRUE),0,pr->tmpStr1);
+ }
+ pr_putItem(this, pr,& (*first),& (*last),PICODATA_ITEM_CMD,lcmd,linfo2,0,(picoos_uchar*)"");
+ break;
+ default:
+ PICODBG_INFO(("unknown command"));
+ break;
+ }
+}
+
+
+static void pr_getOutputItemList (picodata_ProcessingUnit this,
+ pr_subobj_t * pr,
+ picokpr_Preproc network,
+ picokpr_OutItemArrOffset outitem,
+ pr_OutItemVarPtr vars,
+ pr_ioItemPtr * first,
+ pr_ioItemPtr * last)
+{
+
+ picokpr_OutItemArrOffset lo;
+ picoos_int32 llen;
+ picoos_int32 llen2;
+ picokpr_VarStrPtr lstrp;
+ picoos_int32 lval32;
+ picoos_int32 li;
+ picoos_int32 li2;
+ picoos_int32 ln;
+ picoos_int32 ln2;
+ pr_ioItemPtr litem2;
+ pr_ioItemPtr lf;
+ pr_ioItemPtr ll;
+ picoos_int32 lid;
+ picoos_uint8 ltype;
+ picoos_int8 lsubtype;
+ pr_OutItemVarPtr lvar;
+ picoos_int32 lspellmode;
+ picoos_int32 largOfs;
+ picoos_int32 lnextOfs;
+
+
+ lo = outitem;
+ while (lo != 0) {
+ switch (picokpr_getOutItemType(network, lo)) {
+ case PR_OStr:
+ lstrp = picokpr_getOutItemStr(network, lo);
+ if (pr->outOfMemory) return;
+ pr_initItem(this, &pr->tmpItem);
+ pr->tmpItem.head.type = PICODATA_ITEM_TOKEN;
+ pr->tmpItem.head.info1 = PICODATA_ITEMINFO1_TOKTYPE_UNDEFINED;
+ pr->tmpItem.head.info2 = -1;
+ pr->tmpItem.head.len = pr_strcpy(pr->tmpItem.data, lstrp);
+ pr_copyItem(this, pr_WorkMem, &pr->tmpItem, &litem2);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & (*first),& (*last),litem2);
+ break;
+ case PR_OVar:
+ lvar = pr_findVariable(vars,picokpr_getOutItemVal(network, lo));
+ if (lvar != NULL) {
+ lf = lvar->first;
+ } else {
+ lf = NULL;
+ }
+ while (lf != NULL) {
+ pr_copyItem(this, pr_WorkMem,& (*lf),& litem2);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & (*first),& (*last),litem2);
+ lf = lf->next;
+ }
+ break;
+ case PR_OSpell:
+ lspellmode = PR_SPELL;
+ largOfs = picokpr_getOutItemArgOfs(network, lo);
+ if ((largOfs!= 0) && ((lnextOfs = picokpr_getOutItemNextOfs(network, largOfs)) != 0)) {
+ lspellmode = picokpr_getOutItemVal(network, lnextOfs);
+ }
+ pr_startItemList(& lf,& ll);
+ pr_getOutputItemList(this, pr, network,largOfs,vars,& lf,& ll);
+ if (pr->outOfMemory) return;
+ while (lf != NULL) {
+ switch (lf->head.type) {
+ case PICODATA_ITEM_TOKEN:
+ li = 0;
+ ln = pr_strlen(lf->data);
+ while (li < ln) {
+ pr_initItem(this, &pr->tmpItem);
+ if (pr->outOfMemory) return;
+ pr->tmpItem.head.type = PICODATA_ITEM_TOKEN;
+ pr->tmpItem.head.info1 = lf->head.info1;
+ pr->tmpItem.head.info2 = lf->head.info2;
+ pr->tmpItem.head.len = picobase_det_utf8_length(lf->data[li]);
+ for (li2 = 0; li2 < pr->tmpItem.head.len; li2++) {
+ pr->tmpItem.data[li2] = lf->data[li];
+ li++;
+ }
+ pr->tmpItem.data[pr->tmpItem.head.len] = 0;
+ pr->tmpItem.val = 0;
+ lid = picoktab_graphOffset(pr->graphs,pr->tmpItem.data);
+ if (lid > 0) {
+ if (picoktab_getIntPropTokenType(pr->graphs, lid, <ype)) {
+ if ((ltype == PICODATA_ITEMINFO1_TOKTYPE_LETTERV)/* || (ltype == PICODATA_ITEMINFO1_TOKTYPE_DIGIT)*/) {
+ pr->tmpItem.head.len = pr_strcat(pr->tmpItem.data,(picoos_uchar*)SPEC_CHAR);
+ }
+ }
+ picoktab_getIntPropTokenSubType(pr->graphs, lid, &lsubtype);
+ } else {
+ ltype = PICODATA_ITEMINFO1_TOKTYPE_UNDEFINED;
+ lsubtype = -(1);
+ }
+ pr_copyItem(this, pr_WorkMem, &pr->tmpItem, &litem2);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & (*first),& (*last), litem2);
+ if (lspellmode == PR_SPELL_WITH_PHRASE_BREAK) {
+ pr_initItem(this, &pr->tmpItem);
+ pr->tmpItem.head.type = PICODATA_ITEM_TOKEN;
+ pr->tmpItem.head.info1 = PICODATA_ITEMINFO1_TOKTYPE_CHAR;
+ pr->tmpItem.head.info2 = lsubtype;
+ pr->tmpItem.head.len = 1;
+ pr->tmpItem.data[0] = ',';
+ pr->tmpItem.data[1] = 0;
+ pr->tmpItem.val = 0;
+ pr_copyItem(this, pr_WorkMem, &pr->tmpItem, &litem2);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & (*first),& (*last),litem2);
+ } else if (lspellmode == PR_SPELL_WITH_SENTENCE_BREAK) {
+ pr_initItem(this, &pr->tmpItem);
+ pr->tmpItem.head.type = PICODATA_ITEM_CMD;
+ pr->tmpItem.head.info1 = PICODATA_ITEMINFO1_CMD_FLUSH;
+ pr->tmpItem.head.info2 = PICODATA_ITEMINFO2_NA;
+ pr->tmpItem.head.len = 0;
+ pr_copyItem(this, pr_WorkMem, &pr->tmpItem, &litem2);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & (*first),& (*last),litem2);
+ } else if (lspellmode > 0) {
+ pr_initItem(this, &pr->tmpItem);
+ pr->tmpItem.head.type = PICODATA_ITEM_TOKEN;
+ pr->tmpItem.head.info1 = PICODATA_ITEMINFO1_TOKTYPE_CHAR;
+ pr->tmpItem.head.info2 = lsubtype;
+ pr->tmpItem.head.len = 1;
+ pr->tmpItem.data[0] = ',';
+ pr->tmpItem.data[1] = 0;
+ pr->tmpItem.val = 0;
+ pr_copyItem(this, pr_WorkMem, &pr->tmpItem, &litem2);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & (*first),& (*last),litem2);
+ }
+ }
+ break;
+ default:
+ pr_copyItem(this, pr_WorkMem,& (*lf),& litem2);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & (*first),& (*last),litem2);
+ break;
+ }
+ ll = lf;
+ lf = lf->next;
+ ll->next = NULL;
+ }
+ break;
+ case PR_OConcat:
+ pr_startItemList(& lf,& ll);
+ pr_getOutputItemList(this, pr, network,picokpr_getOutItemArgOfs(network, lo),vars,& lf,& ll);
+ if (pr->outOfMemory) return;
+ pr_initItem(this, &pr->tmpItem);
+ pr->tmpItem.head.type = PICODATA_ITEM_TOKEN;
+ pr->tmpItem.head.info1 = PICODATA_ITEMINFO1_TOKTYPE_UNDEFINED;
+ pr->tmpItem.head.info2 = -(1);
+ pr->tmpItem.head.len = 0;
+ pr->tmpItem.data[0] = 0;
+ pr->tmpItem.val = 0;
+ while (lf != NULL) {
+ switch (lf->head.type) {
+ case PICODATA_ITEM_TOKEN:
+ pr->tmpItem.head.len = pr_strcat(pr->tmpItem.data,lf->data);
+ break;
+ case PICODATA_ITEM_CMD:
+ pr_copyItem(this, pr_WorkMem, &pr->tmpItem, &litem2);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & (*first),& (*last),litem2);
+
+ pr_copyItem(this, pr_WorkMem, lf, &litem2);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & (*first),& (*last),litem2);
+
+ pr_initItem(this, &pr->tmpItem);
+ pr->tmpItem.head.type = PICODATA_ITEM_TOKEN;
+ pr->tmpItem.head.info1 = PICODATA_ITEMINFO1_TOKTYPE_UNDEFINED;
+ pr->tmpItem.head.info2 = -(1);
+ pr->tmpItem.head.len = 0;
+ pr->tmpItem.data[0] = 0;
+ pr->tmpItem.val = 0;
+ break;
+ default:
+ break;
+ }
+ lf = lf->next;
+ }
+ pr_copyItem(this, pr_WorkMem, &pr->tmpItem, &litem2);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & (*first),& (*last),litem2);
+ break;
+ case PR_ORomanToCard:
+ pr_startItemList(& lf,& ll);
+ pr_getOutputItemList(this, pr, network,picokpr_getOutItemArgOfs(network, lo),vars,& lf,& ll);
+ if (pr->outOfMemory) return;
+ if ((lf != NULL) && (lf->head.type == PICODATA_ITEM_TOKEN)) {
+ pr_initItem(this, &pr->tmpItem);
+ pr_copyItemContent(this, lf, &pr->tmpItem);
+ if (pr_isLatinNumber(lf->data, & lval32)) {
+ pr_int_to_string(lval32, (picoos_char *)pr->tmpItem.data, PR_MAX_DATA_LEN_Z);
+ pr->tmpItem.head.len = pr_strlen(pr->tmpItem.data);
+ pr->tmpItem.val = lval32;
+ }
+ pr_copyItem(this, pr_WorkMem, &pr->tmpItem, &litem2);
+ pr_appendItem(this, & (*first),& (*last),litem2);
+ }
+ break;
+ case PR_OVal:
+ break;
+ case PR_OLeft:
+ pr_startItemList(& lf,& ll);
+ pr_getOutputItemList(this, pr, network,picokpr_getOutItemNextOfs(network, picokpr_getOutItemArgOfs(network, lo)),vars,& lf,& ll);
+ if (pr->outOfMemory) return;
+ if ((lf != NULL) && (lf->head.type == PICODATA_ITEM_TOKEN)) {
+ pr_initItem(this, &pr->tmpItem);
+ pr_copyItemContent(this, lf, &pr->tmpItem);
+ llen = lf->head.len;
+ llen2 = picobase_utf8_length(pr->tmpItem.data,PR_MAX_DATA_LEN);
+ ln = 0;
+ ln2 = 0;
+ largOfs = picokpr_getOutItemVal(network, picokpr_getOutItemArgOfs(network, lo));
+ while ((ln < llen) && (ln2 < llen2) && (ln2 < largOfs)) {
+ ln = ln + picobase_det_utf8_length(pr->tmpItem.data[ln]);
+ ln2++;
+ }
+ pr->tmpItem.data[ln] = 0;
+ pr->tmpItem.head.len = ln;
+ pr_copyItem(this, pr_WorkMem, &pr->tmpItem, &litem2);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & (*first),& (*last),litem2);
+ }
+ break;
+ case PR_ORight:
+ pr_startItemList(& lf,& ll);
+ pr_getOutputItemList(this, pr, network,picokpr_getOutItemNextOfs(network, picokpr_getOutItemArgOfs(network, lo)),vars,& lf,& ll);
+ if (pr->outOfMemory) return;
+ if ((lf != NULL) && (lf->head.type == PICODATA_ITEM_TOKEN)) {
+ pr_initItem(this, &pr->tmpItem);
+ pr_copyItemContent(this, lf, & pr->tmpItem);
+ llen = lf->head.len;
+ llen2 = picobase_utf8_length(pr->tmpItem.data,PR_MAX_DATA_LEN);
+ ln = 0;
+ ln2 = 0;
+ while ((ln < llen) && (ln2 < llen2) && (ln2 < (llen2 - picokpr_getOutItemVal(network, picokpr_getOutItemArgOfs(network, lo))))) {
+ ln = ln + picobase_det_utf8_length(pr->tmpItem.data[ln]);
+ ln2++;
+ }
+ pr->tmpItem.head.len = pr_removeSubstring(0,ln,pr->tmpItem.data);
+ pr_copyItem(this, pr_WorkMem, &pr->tmpItem, & litem2);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & (*first),& (*last),litem2);
+ }
+ break;
+ case PR_OItem:
+ pr_startItemList(& lf,& ll);
+ pr_getOutputItemList(this, pr, network,picokpr_getOutItemNextOfs(network, picokpr_getOutItemArgOfs(network, lo)),vars,& lf,& ll);
+ if (pr->outOfMemory) return;
+ ln = picokpr_getOutItemVal(network, picokpr_getOutItemArgOfs(network, lo));
+ li = 1;
+ while ((li < ln) && (lf != NULL)) {
+ lf = lf->next;
+ li++;
+ }
+ if ((lf != NULL) && (li == ln) && (lf->head.type == PICODATA_ITEM_TOKEN)) {
+ pr_copyItem(this, pr_WorkMem, lf, & litem2);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & (*first),& (*last),litem2);
+ }
+ break;
+ case PR_ORLZ:
+ pr_startItemList(& lf,& ll);
+ pr_getOutputItemList(this, pr, network, picokpr_getOutItemArgOfs(network, lo),vars,& lf,& ll);
+ if (pr->outOfMemory) return;
+ if ((lf != NULL) && (lf->head.type == PICODATA_ITEM_TOKEN)) {
+ pr_initItem(this, &pr->tmpItem);
+ pr_copyItemContent(this, lf, & pr->tmpItem);
+ li = 0;
+ while ((li < lf->head.len) && (pr->tmpItem.data[li] == '0')) {
+ li++;
+ }
+ pr->tmpItem.head.len = pr_removeSubstring(0,li,pr->tmpItem.data);
+ pr_copyItem(this, pr_WorkMem, &pr->tmpItem, & litem2);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & (*first),& (*last),litem2);
+ }
+ break;
+ case PR_OIgnore: case PR_OPitch: case PR_OSpeed: case PR_OVolume: case PR_OPhonSVOXPA: case PR_OPhonSAMPA: case PR_OBreak: case PR_OMark: case PR_OPara: case PR_OSent: case PR_OPlay:
+ case PR_OUseSig: case PR_OGenFile: case PR_OAudioEdit: case PR_OContext: case PR_OVoice:
+ pr_genCommands(this, pr, network,lo,vars,& (*first),& (*last));
+ if (pr->outOfMemory) return;
+ break;
+ default:
+ PICODBG_INFO(("unkown command"));
+ break;
+ }
+ lo = picokpr_getOutItemNextOfs(network, lo);
+ }
+}
+
+
+static picoos_int32 pr_attrVal (picokpr_Preproc network, picokpr_TokArrOffset tok, pr_TokSetEleWP type)
+{
+
+ pr_TokSetEleWP tse;
+ picoos_int32 n;
+ picokpr_TokSetWP set;
+
+ n = 0;
+ tse = PR_FIRST_TSE_WP;
+ set = picokpr_getTokSetWP(network, tok);
+ while (tse < type) {
+ if (((1<<tse) & set) != 0) {
+ n++;
+ }
+ tse = (pr_TokSetEleWP)((picoos_int32)tse+1);
+ }
+ return picokpr_getAttrValArrInt32(network, picokpr_getTokAttribOfs(network, tok) + n);
+}
+
+
+static void pr_getOutput (picodata_ProcessingUnit this, pr_subobj_t * pr,
+ picoos_int32 * i, picoos_int32 d, pr_ioItemPtr * o, pr_ioItemPtr * ol)
+{
+
+ register struct pr_PathEle * with__0;
+ pr_OutItemVarPtr lvars;
+ pr_OutItemVarPtr lvar;
+ pr_ioItemPtr lit;
+ pr_ioItemPtr ldit;
+ pr_ioItemPtr ldlit;
+ picoos_bool lfirst;
+ pr_ioItemPtr lcopy;
+ picokpr_TokSetWP wpset;
+ picokpr_TokSetNP npset;
+ picoos_int32 li;
+
+ lfirst = TRUE;
+ (*i)++;
+ lit = NULL;
+ lvars = NULL;
+ ldit = NULL;
+ ldlit = NULL;
+ while ((*i) < pr->rbestpath.rlen) {
+ with__0 = & pr->rbestpath.rele[*i];
+ li = 0;
+ if (*i > 0) {
+ while ((li < 127) && (li < pr->rbestpath.rele[*i].rdepth)) {
+ pr->lspaces[li++] = ' ';
+ }
+ }
+ pr->lspaces[li] = 0;
+ if (with__0->rprodname != 0) {
+ PICODBG_INFO(("pp path :%s%s(", pr->lspaces, picokpr_getVarStrPtr(with__0->rnetwork,with__0->rprodname)));
+ }
+ if ((pr->ritems[with__0->ritemid+1] != NULL) && (pr->ritems[with__0->ritemid+1]->head.type == PICODATA_ITEM_TOKEN)) {
+ PICODBG_INFO(("pp in (1): %s'%s'", pr->lspaces, pr->ritems[with__0->ritemid+1]->data));
+ }
+ if ((pr->ritems[with__0->ritemid+1] != NULL)) {
+ while ((pr->rinItemList != NULL) && (pr->rinItemList != pr->ritems[with__0->ritemid+1]) && (pr->rinItemList->head.type != PICODATA_ITEM_TOKEN)) {
+ lit = pr->rinItemList;
+ pr->rinItemList = pr->rinItemList->next;
+ lit->next = NULL;
+ pr_copyItem(this, pr_WorkMem,& (*lit),& lcopy);
+ if (pr->outOfMemory) return;
+ pr_disposeItem(this, & lit);
+ pr_appendItem(this, & (*o),& (*ol),lcopy);
+ }
+ if (pr->rinItemList != NULL) {
+ lit = pr->rinItemList;
+ pr->rinItemList = pr->rinItemList->next;
+ lit->next = NULL;
+ } else {
+ lit = NULL;
+ }
+ }
+ wpset = picokpr_getTokSetWP(with__0->rnetwork, with__0->rtok);
+ npset = picokpr_getTokSetNP(with__0->rnetwork, with__0->rtok);
+
+ if ((PR_TSE_MASK_PROD & wpset) != 0) {
+ if ((PR_TSE_MASK_VAR & wpset) != 0) {
+ lvar = pr_findVariable(lvars,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEVar));
+ if (lvar == NULL) {
+ pr_ALLOCATE(this, pr_WorkMem, (void *) & lvar,(sizeof (*lvar)));
+ lvar->next = lvars;
+ lvar->id = pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEVar);
+ lvars = lvar;
+ }
+ pr_startItemList(& lvar->first,& lvar->last);
+ pr_getOutput(this, pr, & (*i),(d + 1),& lvar->first,& lvar->last);
+ if (pr->outOfMemory) return;
+ } else {
+ pr_startItemList(& ldit,& ldlit);
+ pr_getOutput(this, pr, & (*i),(d + 1),& ldit,& ldlit);
+ if (pr->outOfMemory) return;
+ }
+ (*i)++;
+ } else if ((PR_TSE_MASK_VAR & wpset) != 0) {
+ lvar = pr_findVariable(lvars,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEVar));
+ if (lvar == NULL) {
+ pr_ALLOCATE(this, pr_WorkMem, (void *) & lvar,(sizeof (*lvar)));
+ lvar->next = lvars;
+ lvar->id = pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEVar);
+ lvars = lvar;
+ }
+ if (((PR_TSE_MASK_LEX & wpset) != 0) && ((PR_TSE_MASK_LETTER & npset) == 0)) {
+ if (lfirst) {
+ pr_newItem(this, pr_WorkMem,& lit, PICODATA_ITEM_TOKEN, sizeof(struct pr_ioItem), /*inItem*/FALSE);
+ if (pr->outOfMemory) return;
+ lit->head.type = PICODATA_ITEM_TOKEN;
+ lit->head.info1 = pr->ritems[with__0->ritemid+1]->head.info1;
+ lit->head.info2 = pr->ritems[with__0->ritemid+1]->head.info2;
+ if (pr->ritems[with__0->ritemid+1]->head.info1 == PICODATA_ITEMINFO1_TOKTYPE_SPACE) {
+ lit->head.len = pr_strcpy(lit->data, (picoos_uchar*)"_");
+ } else {
+ lit->head.len = pr_strcpy(lit->data, pr->ritems[with__0->ritemid+1]->data);
+ }
+ lvar->first = lit;
+ lvar->last = lit;
+ lfirst = FALSE;
+ } else {
+ if ((pr->ritems[with__0->ritemid+1]->head.info1 == PICODATA_ITEMINFO1_TOKTYPE_SPACE)) {
+ lvar->last->head.len = pr_strcat(lvar->last->data,(picoos_uchar*)"_");
+ } else {
+ lvar->last->head.len = pr_strcat(lvar->last->data,pr->ritems[with__0->ritemid+1]->data);
+ }
+ lvar->last->head.info1 = PICODATA_ITEMINFO1_TOKTYPE_UNDEFINED;
+ lvar->last->head.info2 = -(1);
+ }
+ } else {
+ lvar->first = pr->ritems[with__0->ritemid+1];
+ lvar->last = pr->ritems[with__0->ritemid+1];
+ }
+ (*i)++;
+ } else if ((PR_TSE_MASK_OUT & wpset) != 0) {
+ pr_getOutputItemList(this, pr, with__0->rnetwork,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEOut),lvars,& (*o),& (*ol));
+ if (pr->outOfMemory) return;
+ (*i)++;
+ } else if (((*i) < (pr->rbestpath.rlen - 1)) && (d != pr->rbestpath.rele[(*i) + 1].rdepth)) {
+ if ((*i > 0) && (with__0->rdepth-1) == pr->rbestpath.rele[*i + 1].rdepth) {
+ li = 0;
+ while ((li < 127) && (li < with__0->rdepth-1)) {
+ pr->lspaces[li++] = ' ';
+ }
+ pr->lspaces[li] = 0;
+ PICODBG_INFO(("pp path :%s)", pr->lspaces));
+ }
+ return;
+ } else {
+ (*i)++;
+ }
+ if ((PR_TSE_MASK_LEX & wpset) == 0) {
+ lfirst = TRUE;
+ }
+ }
+ li = 0;
+ while ((li < 127) && (li < pr->rbestpath.rele[*i].rdepth-1)) {
+ pr->lspaces[li++] = ' ';
+ }
+ pr->lspaces[li] = 0;
+ PICODBG_INFO(("pp path :%s)", pr->lspaces));
+}
+
+
+static void pr_outputPath (picodata_ProcessingUnit this, pr_subobj_t * pr)
+{
+ register struct pr_PathEle * with__0;
+ picoos_int32 li;
+ pr_ioItemPtr lf;
+ pr_ioItemPtr ll;
+ pr_ioItemPtr lit;
+ pr_ioItemPtr lit2;
+ pr_MemState lmemState;
+ picoos_bool lastPlayFileFound;
+
+ pr_getMemState(this, pr_WorkMem, & lmemState);
+ lf = NULL;
+ ll = NULL;
+ li = -(1);
+ pr_getOutput(this, pr, & li,1,& lf,& ll);
+ if (pr->outOfMemory) return;
+ lastPlayFileFound = TRUE;
+ while (lf != NULL) {
+ lit = lf;
+ lf = lf->next;
+ lit->next = NULL;
+ if (pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_PLAY)) {
+ lastPlayFileFound = picoos_FileExists(this->common, (picoos_char*)lit->data);
+ if (!lastPlayFileFound) {
+ PICODBG_WARN(("file '%s' not found; synthesizing enclosed text instead", lit->data));
+ picoos_emRaiseWarning(this->common->em, PICO_EXC_CANT_OPEN_FILE, (picoos_char*)"", (picoos_char*)"file '%s' not found; synthesizing enclosed text instead",lit->data);
+ }
+ }
+ if (pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_PHONEME) && pr_isCmdInfo2(lit, PICODATA_ITEMINFO2_CMD_START)) {
+ pr->insidePhoneme = TRUE;
+ } else if (pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_PHONEME)&& pr_isCmdInfo2(lit, PICODATA_ITEMINFO2_CMD_END)) {
+ pr->insidePhoneme = FALSE;
+ }
+ if ((pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_PLAY) && !lastPlayFileFound)) {
+ } else if (pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_IGNORE) && pr_isCmdInfo2(lit, PICODATA_ITEMINFO2_CMD_START)) {
+ if (lastPlayFileFound) {
+ pr->rignore++;
+ }
+ } else if (pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_IGNORE) && pr_isCmdInfo2(lit, PICODATA_ITEMINFO2_CMD_END)) {
+ if (lastPlayFileFound) {
+ if (pr->rignore > 0) {
+ pr->rignore--;
+ }
+ }
+ } else if (pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_IGNSIG) && pr_isCmdInfo2(lit, PICODATA_ITEMINFO2_CMD_START) && !lastPlayFileFound) {
+ } else if (pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_IGNSIG) && pr_isCmdInfo2(lit, PICODATA_ITEMINFO2_CMD_END) && !lastPlayFileFound) {
+ } else if (pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_CONTEXT)) {
+ if (pr->rignore <= 0) {
+ pr_setContext(this, pr, lit->data);
+ }
+ } else if (pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_VOICE)) {
+ if (pr->rignore <= 0) {
+ pr_copyItem(this, pr_DynMem,lit,& lit2);
+ if (pr->outOfMemory) return;
+ pr_appendItem(this, & pr->routItemList,& pr->rlastOutItem, lit2);
+ }
+ } else {
+ if ((pr->rignore <= 0) && !(pr->insidePhoneme && (pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_PLAY) ||
+ pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_IGNSIG)))) {
+ PICODBG_INFO(("pp out(1): '%s'", lit->data));
+ pr_copyItem(this, pr_DynMem,lit,& lit2);
+ if (pr->outOfMemory) return;
+ pr_appendItemToOutItemList(this, pr, & pr->routItemList,& pr->rlastOutItem,lit2);
+ if (pr->outOfMemory) return;
+ }
+ }
+ }
+ for (li = 0; li<pr->rbestpath.rlen; li++) {
+ with__0 = & pr->rbestpath.rele[li];
+ pr_disposeItem(this, & pr->ritems[with__0->ritemid+1]);
+ }
+ pr_resetMemState(this, pr_WorkMem, lmemState);
+}
+
+/* *****************************************************************************/
+
+static void pr_compare (picoos_uchar str1lc[], picoos_uchar str2[], picoos_int16 * result)
+{
+
+ picoos_int32 i;
+ picoos_int32 j;
+ picoos_int32 l;
+ picoos_uint32 pos;
+ picoos_bool finished1;
+ picoos_bool finished2;
+ picoos_bool done;
+ picobase_utf8char utf8char;
+
+ pos = 0;
+ l = picobase_get_next_utf8char(str2, PR_MAX_DATA_LEN, & pos,utf8char);
+ picobase_lowercase_utf8_str(utf8char,(picoos_char*)utf8char,PICOBASE_UTF8_MAXLEN+1, &done);
+ l = picobase_det_utf8_length(utf8char[0]);
+ j = 0;
+ i = 0;
+ while ((i < PR_MAX_DATA_LEN) && (str1lc[i] != 0) && (l > 0) && (j <= 3) && (str1lc[i] == utf8char[j])) {
+ i++;
+ j++;
+ if (j >= l) {
+ l = picobase_get_next_utf8char(str2, PR_MAX_DATA_LEN, & pos,utf8char);
+ picobase_lowercase_utf8_str(utf8char,(picoos_char*)utf8char,PICOBASE_UTF8_MAXLEN+1, &done);
+ l = picobase_det_utf8_length(utf8char[0]);
+ j = 0;
+ }
+ }
+ finished1 = (i >= PR_MAX_DATA_LEN) || (str1lc[i] == 0);
+ finished2 = (j > 3) || (utf8char[j] == 0);
+ if (finished1 && finished2) {
+ *result = PR_EQUAL;
+ } else if (finished1) {
+ *result = PR_SMALLER;
+ } else if (finished2) {
+ *result = PR_LARGER;
+ } else {
+ if (str1lc[i] < utf8char[j]) {
+ *result = PR_SMALLER;
+ } else {
+ *result = PR_LARGER;
+ }
+ }
+}
+
+
+static picoos_bool pr_hasToken (picokpr_TokSetWP * tswp, picokpr_TokSetNP * tsnp)
+{
+ return (((( PR_TSE_MASK_SPACE | PR_TSE_MASK_DIGIT | PR_TSE_MASK_LETTER | PR_TSE_MASK_SEQ
+ | PR_TSE_MASK_CHAR | PR_TSE_MASK_BEGIN | PR_TSE_MASK_END) & (*tsnp)) != 0) ||
+ ((PR_TSE_MASK_LEX & (*tswp)) != 0));
+}
+
+
+static picoos_bool pr_getNextToken (picodata_ProcessingUnit this, pr_subobj_t * pr)
+{
+ register struct pr_PathEle * with__0;
+ picoos_int32 len;
+ picokpr_TokSetNP npset;
+
+ len = pr->ractpath.rlen;
+ with__0 = & pr->ractpath.rele[len - 1];
+ npset = picokpr_getTokSetNP(with__0->rnetwork, with__0->rtok);
+ if ((len > 0) && (pr->ractpath.rlen < PR_MAX_PATH_LEN) && ((PR_TSE_MASK_NEXT & npset) != 0)) {
+ pr_initPathEle(& pr->ractpath.rele[len]);
+ pr->ractpath.rele[len].rnetwork = with__0->rnetwork;
+ pr->ractpath.rele[len].rtok = picokpr_getTokNextOfs(with__0->rnetwork, with__0->rtok);
+ pr->ractpath.rele[len].rdepth = with__0->rdepth;
+ pr->ractpath.rlen++;
+ return TRUE;
+ } else {
+ if (len >= PR_MAX_PATH_LEN) {
+ PICODBG_INFO(("max path len reached (pr_getNextToken)"));
+ }
+ return FALSE;
+ }
+}
+
+
+static picoos_bool pr_getAltToken (picodata_ProcessingUnit this, pr_subobj_t * pr)
+{
+ register struct pr_PathEle * with__0;
+ picokpr_TokArrOffset lTok;
+ picokpr_TokSetNP npset;
+
+
+ with__0 = & pr->ractpath.rele[pr->ractpath.rlen - 1];
+ if ((pr->ractpath.rlen > 0) && (pr->ractpath.rlen < PR_MAX_PATH_LEN)) {
+ npset = picokpr_getTokSetNP(with__0->rnetwork, with__0->rtok);
+ if (with__0->rcompare == PR_SMALLER) {
+ if ((PR_TSE_MASK_ALTL & npset) != 0) {
+ lTok = picokpr_getTokAltLOfs(with__0->rnetwork, with__0->rtok);
+ } else {
+ return FALSE;
+ }
+ } else {
+ if ((PR_TSE_MASK_ALTR & npset) != 0) {
+ lTok = picokpr_getTokAltROfs(with__0->rnetwork, with__0->rtok);
+ } else {
+ return FALSE;
+ }
+ }
+ with__0->rlState = PR_LSInit;
+ with__0->rtok = lTok;
+ with__0->ritemid = -1;
+ with__0->rcompare = -1;
+ with__0->rprodname = 0;
+ with__0->rprodprefcost = 0;
+ return TRUE;
+ } else {
+ if (pr->ractpath.rlen >= PR_MAX_PATH_LEN) {
+ PICODBG_INFO(("max path len reached (pr_getAltToken)"));
+ }
+ return FALSE;
+ }
+}
+
+
+static picoos_bool pr_findProduction (picodata_ProcessingUnit this, pr_subobj_t * pr,
+ picoos_uchar str[], picokpr_Preproc * network, picokpr_TokArrOffset * tokOfs)
+{
+ picoos_bool found;
+ picoos_int32 p;
+ picoos_int32 ind;
+ picoos_int32 i;
+ picoos_bool done;
+ picokpr_VarStrPtr lstrp;
+ picoos_int32 lprodarrlen;
+
+ ind = 0;
+ pr_getTermPartStr(str,& ind,'.',pr->tmpStr1,& done);
+ pr_getTermPartStr(str,& ind,'.',pr->tmpStr2,& done);
+ found = FALSE;
+
+ for (p=0; p<PR_MAX_NR_PREPROC; p++) {
+ if (!found && (pr->preproc[p] != NULL)) {
+ if (pr_strEqual(pr->tmpStr1, picokpr_getPreprocNetName(pr->preproc[p]))) {
+ i = 0;
+ lprodarrlen = picokpr_getProdArrLen(pr->preproc[p]);
+ while (!found && (i <= (lprodarrlen - 1))) {
+ lstrp = picokpr_getVarStrPtr(pr->preproc[p],picokpr_getProdNameOfs(pr->preproc[p], i));
+ if (pr_strEqual(pr->tmpStr2, lstrp)) {
+ *network = pr->preproc[p];
+ *tokOfs = picokpr_getProdATokOfs(pr->preproc[p], i);
+ return TRUE;
+ }
+ i++;
+ }
+ }
+ }
+ }
+ return FALSE;
+}
+
+
+static picoos_bool pr_getProdToken (picodata_ProcessingUnit this, pr_subobj_t * pr)
+{
+ register struct pr_PathEle * with__0;
+ picokpr_VarStrPtr lstrp;
+ picokpr_TokSetWP wpset;
+
+ if ((pr->ractpath.rlen > 0) && (pr->ractpath.rlen < PR_MAX_PATH_LEN)) {
+ with__0 = & pr->ractpath.rele[pr->ractpath.rlen - 1];
+ wpset = picokpr_getTokSetWP(with__0->rnetwork, with__0->rtok);
+ if ((PR_TSE_MASK_PROD & wpset) != 0) {
+ if ((PR_TSE_MASK_PRODEXT & wpset) != 0) {
+ pr_initPathEle(& pr->ractpath.rele[pr->ractpath.rlen]);
+ lstrp = picokpr_getVarStrPtr(with__0->rnetwork, pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEProdExt));
+ if (pr_findProduction(this, pr, lstrp,& pr->ractpath.rele[pr->ractpath.rlen].rnetwork,& pr->ractpath.rele[pr->ractpath.rlen].rtok)) {
+ with__0->rprodname = picokpr_getProdNameOfs(with__0->rnetwork, pr_attrVal(with__0->rnetwork, with__0->rtok,PR_TSEProd));
+ with__0->rprodprefcost = picokpr_getProdPrefCost(with__0->rnetwork, pr_attrVal(with__0->rnetwork,with__0->rtok,PR_TSEProd));
+ pr->ractpath.rele[pr->ractpath.rlen].rdepth = with__0->rdepth + 1;
+ pr->ractpath.rlen++;
+ return TRUE;
+ } else {
+ return FALSE;
+ }
+ } else {
+ pr_initPathEle(& pr->ractpath.rele[pr->ractpath.rlen]);
+ pr->ractpath.rele[pr->ractpath.rlen].rnetwork = with__0->rnetwork;
+ pr->ractpath.rele[pr->ractpath.rlen].rtok = picokpr_getProdATokOfs(with__0->rnetwork, pr_attrVal(with__0->rnetwork, with__0->rtok,PR_TSEProd));
+ with__0->rprodname = picokpr_getProdNameOfs(with__0->rnetwork, pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEProd));
+ with__0->rprodprefcost = picokpr_getProdPrefCost(with__0->rnetwork, pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEProd));
+ pr->ractpath.rele[pr->ractpath.rlen].rdepth = with__0->rdepth + 1;
+ pr->ractpath.rlen++;
+ return TRUE;
+ }
+ }
+ }
+ if (pr->ractpath.rlen >= PR_MAX_PATH_LEN) {
+ PICODBG_INFO(("max path len reached (pr_getProdToken)"));
+ }
+ return FALSE;
+}
+
+
+static picoos_bool pr_getProdContToken (picodata_ProcessingUnit this, pr_subobj_t * pr)
+{
+ picoos_int32 li;
+
+ li = pr->ractpath.rlen - 1;
+ while ((li > 0) && !((pr->ractpath.rele[li].rdepth == (pr->ractpath.rele[pr->ractpath.rlen - 1].rdepth - 1)) && ((PR_TSE_MASK_PROD &picokpr_getTokSetWP(pr->ractpath.rele[li].rnetwork, pr->ractpath.rele[li].rtok)) != 0))) {
+ li--;
+ }
+ if (((li >= 0) && (pr->ractpath.rlen < PR_MAX_PATH_LEN) && (PR_TSE_MASK_NEXT &picokpr_getTokSetNP(pr->ractpath.rele[li].rnetwork, pr->ractpath.rele[li].rtok)) != 0)) {
+ pr_initPathEle(& pr->ractpath.rele[pr->ractpath.rlen]);
+ pr->ractpath.rele[pr->ractpath.rlen].rnetwork = pr->ractpath.rele[li].rnetwork;
+ pr->ractpath.rele[pr->ractpath.rlen].rtok = picokpr_getTokNextOfs(pr->ractpath.rele[li].rnetwork, pr->ractpath.rele[li].rtok);
+ pr->ractpath.rele[pr->ractpath.rlen].rdepth = pr->ractpath.rele[li].rdepth;
+ pr->ractpath.rlen++;
+ return TRUE;
+ } else {
+ if (pr->ractpath.rlen >= PR_MAX_PATH_LEN) {
+ PICODBG_INFO(("max path len reached (pr_getProdContToken)"));
+ }
+ return FALSE;
+ }
+}
+
+/* *****************************************************************************/
+
+static picoos_bool pr_getTopLevelToken (picodata_ProcessingUnit this, pr_subobj_t * pr, picoos_bool firstprod)
+{
+ if (firstprod) {
+ if (pr->actCtx != NULL) {
+ pr->prodList = pr->actCtx->rProdList;
+ } else {
+ pr->prodList = NULL;
+ }
+ } else if (pr->prodList != NULL) {
+ pr->prodList = pr->prodList->rNext;
+ }
+ if ((pr->prodList != NULL) && (pr->prodList->rProdOfs != 0) && (picokpr_getProdATokOfs(pr->prodList->rNetwork, pr->prodList->rProdOfs) != 0)) {
+ pr_initPathEle(& pr->ractpath.rele[pr->ractpath.rlen]);
+ pr->ractpath.rele[pr->ractpath.rlen].rdepth = 1;
+ pr->ractpath.rele[pr->ractpath.rlen].rnetwork = pr->prodList->rNetwork;
+ pr->ractpath.rele[pr->ractpath.rlen].rtok = picokpr_getProdATokOfs(pr->prodList->rNetwork, pr->prodList->rProdOfs);
+ pr->ractpath.rele[pr->ractpath.rlen].rlState = PR_LSInit;
+ pr->ractpath.rele[pr->ractpath.rlen].rcompare = -1;
+ pr->ractpath.rele[pr->ractpath.rlen].rprodname = picokpr_getProdNameOfs(pr->prodList->rNetwork, pr->prodList->rProdOfs);
+ pr->ractpath.rele[pr->ractpath.rlen].rprodprefcost = picokpr_getProdPrefCost(pr->prodList->rNetwork, pr->prodList->rProdOfs);
+ pr->ractpath.rlen++;
+ return TRUE;
+ } else {
+ return FALSE;
+ }
+}
+
+
+static picoos_bool pr_getToken (picodata_ProcessingUnit this, pr_subobj_t * pr)
+{
+ picoos_int32 ln;
+ picoos_int32 lid;
+
+ ln = (pr->ractpath.rlen - 2);
+ while ((ln >= 0) && (pr->ractpath.rele[ln].ritemid == -1)) {
+ ln = ln - 1;
+ }
+ if (ln >= 0) {
+ lid = pr->ractpath.rele[ln].ritemid + 1;
+ } else {
+ lid = 0;
+ }
+ if (lid < pr->rnritems) {
+ pr->ractpath.rele[pr->ractpath.rlen - 1].ritemid = lid;
+ } else {
+ pr->ractpath.rele[pr->ractpath.rlen - 1].ritemid = -1;
+ }
+ return (lid < pr->rnritems);
+}
+
+
+static picoos_bool pr_getNextMultiToken (picodata_ProcessingUnit this, pr_subobj_t * pr)
+{
+ picoos_int32 len;
+
+ len = pr->ractpath.rlen;
+ if ((len > 0) && (len < PR_MAX_PATH_LEN)) {
+ pr->ractpath.rele[len].rtok = pr->ractpath.rele[len - 1].rtok;
+ pr->ractpath.rele[len].ritemid = -(1);
+ pr->ractpath.rele[len].rcompare = pr->ractpath.rele[len - 1].rcompare;
+ pr->ractpath.rele[len].rdepth = pr->ractpath.rele[len - 1].rdepth;
+ pr->ractpath.rele[len].rlState = PR_LSInit;
+ pr->ractpath.rlen++;
+ return TRUE;
+ } else {
+ if (len >= PR_MAX_PATH_LEN) {
+ PICODBG_INFO(("max path len reached (pr_getNextMultiToken)"));
+ }
+ return FALSE;
+ }
+ return FALSE;
+}
+
+
+static pr_MatchState pr_matchMultiToken (picodata_ProcessingUnit this, pr_subobj_t * pr,
+ picokpr_TokSetNP npset, picokpr_TokSetWP wpset)
+{
+ picoos_bool lcontinue=FALSE;
+ picoos_bool lmatch=FALSE;
+
+ if (lmatch) {
+ return PR_MSMatchedMulti;
+ } else if (lcontinue) {
+ return PR_MSMatchedContinue;
+ } else {
+ return PR_MSNotMatched;
+ }
+ pr = pr; /* avoid warning "var not used in this function"*/
+ npset = npset; /* avoid warning "var not used in this function"*/
+ wpset = wpset; /* avoid warning "var not used in this function"*/
+
+}
+
+
+static pr_MatchState pr_matchTokensSpace (picodata_ProcessingUnit this, pr_subobj_t * pr, picoos_int32 cmpres,
+ picokpr_TokSetNP npset, picokpr_TokSetWP wpset)
+{
+ register struct pr_PathEle * with__0;
+ picoos_int32 llen;
+ picoos_int32 lulen;
+ picoos_int32 li;
+ picokpr_VarStrPtr lstrp;
+ picoos_int32 leol;
+
+ with__0 = & pr->ractpath.rele[pr->ractpath.rlen - 1];
+ if ((PR_TSE_MASK_SPACE & npset) == 0) {
+ return PR_MSNotMatched;
+ }
+ lstrp = (picokpr_VarStrPtr)&pr->ritems[with__0->ritemid+1]->data;
+ lulen = picobase_utf8_length(lstrp,PR_MAX_DATA_LEN);
+ if (((PR_TSE_MASK_LEN & wpset) != 0) && (lulen != pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSELen))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_MIN & wpset) != 0) && (lulen < pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEMin))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_MAX & wpset) != 0) && (lulen > pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEMax))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_STR & wpset) != 0) && (cmpres != PR_EQUAL)) {
+ return PR_MSNotMatched;
+ }
+ if ((PR_TSE_MASK_VAL & wpset) != 0) {
+ leol = 0;
+ llen = pr_strlen(lstrp);
+ for (li = 0; li < llen; li++) {
+ if (lstrp[li] == PR_EOL) {
+ leol++;
+ }
+ }
+ if (leol != pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEVal)) {
+ return PR_MSNotMatched;
+ }
+ }
+ if (((PR_TSE_MASK_ID & wpset) != 0) && (pr->ritems[with__0->ritemid+1]->head.info2 != pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEID))) {
+ return PR_MSNotMatched;
+ }
+ return PR_MSMatched;
+}
+
+
+static pr_MatchState pr_matchTokensDigit (picodata_ProcessingUnit this, pr_subobj_t * pr, picoos_int32 cmpres,
+ picokpr_TokSetNP npset, picokpr_TokSetWP wpset)
+{
+ register struct pr_PathEle * with__0;
+ picoos_int32 lulen;
+ picoos_int32 lval;
+ picokpr_VarStrPtr lstrp;
+
+ with__0 = & pr->ractpath.rele[pr->ractpath.rlen - 1];
+ if ((PR_TSE_MASK_DIGIT & npset) == 0) {
+ return PR_MSNotMatched;
+ }
+ lstrp = (picokpr_VarStrPtr)&pr->ritems[with__0->ritemid+1]->data;
+ lulen = picobase_utf8_length(lstrp,PR_MAX_DATA_LEN);
+ if ((((PR_TSE_MASK_LEN & wpset) != 0) && (lulen != pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSELen)))) {
+ return PR_MSNotMatched;
+ }
+ lval = pr->ritems[with__0->ritemid+1]->val;
+ if (((PR_TSE_MASK_MIN & wpset) != 0) && (lval < pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEMin))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_MAX & wpset) != 0) && (lval > pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEMax))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_STR & wpset) != 0) && (cmpres != PR_EQUAL)) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_VAL & wpset) != 0) && (lval != pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEVal))) {
+ return PR_MSNotMatched;
+ }
+ if ((((PR_TSE_MASK_NLZ & npset) != 0) && lstrp[0] == '0')) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_HEAD & wpset) != 0) && !(picokpr_isEqualHead(with__0->rnetwork,lstrp,PR_MAX_DATA_LEN,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEHead)))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_MID & wpset) != 0) && !(picokpr_isEqualMid(with__0->rnetwork,lstrp,PR_MAX_DATA_LEN,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEMid)))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_TAIL & wpset) != 0) && !(picokpr_isEqualTail(with__0->rnetwork,lstrp,PR_MAX_DATA_LEN,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSETail)))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_ID & wpset) != 0) && (pr->ritems[with__0->ritemid+1]->head.info2 != pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEID))) {
+ return PR_MSNotMatched;
+ }
+ return PR_MSMatched;
+}
+
+
+static pr_MatchState pr_matchTokensSeq (picodata_ProcessingUnit this, pr_subobj_t * pr, picoos_int32 cmpres,
+ picokpr_TokSetNP npset, picokpr_TokSetWP wpset)
+{
+
+ register struct pr_PathEle * with__0;
+ picoos_int32 lulen;
+ picokpr_VarStrPtr lstrp;
+
+ with__0 = & pr->ractpath.rele[pr->ractpath.rlen - 1];
+
+ if (!((PR_TSE_MASK_SEQ & npset) != 0)) {
+ return PR_MSNotMatched;
+ }
+ lstrp = (picokpr_VarStrPtr)(void *) &pr->ritems[with__0->ritemid+1]->data;
+ lulen = picobase_utf8_length(lstrp,PR_MAX_DATA_LEN);
+ if (((PR_TSE_MASK_LEN & wpset) != 0) && (lulen != pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSELen))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_MIN & wpset) != 0) && (lulen < pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEMin))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_MAX & wpset) != 0) && (lulen > pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEMax))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_STR & wpset) != 0) && (cmpres != PR_EQUAL)) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_HEAD & wpset) != 0) && !(picokpr_isEqualHead(with__0->rnetwork,lstrp,PR_MAX_DATA_LEN,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEHead)))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_MID & wpset) != 0) && !(picokpr_isEqualMid(with__0->rnetwork,lstrp,PR_MAX_DATA_LEN ,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEMid)))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_TAIL & wpset) != 0) && !(picokpr_isEqualTail(with__0->rnetwork,lstrp,PR_MAX_DATA_LEN,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSETail)))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_ID & wpset) != 0) && (pr->ritems[with__0->ritemid+1]->head.info2 != pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEID))) {
+ return PR_MSNotMatched;
+ }
+ return PR_MSMatched;
+}
+
+
+static pr_MatchState pr_matchTokensChar (picodata_ProcessingUnit this, pr_subobj_t * pr, picoos_int32 cmpres,
+ picokpr_TokSetNP npset, picokpr_TokSetWP wpset)
+{
+ register struct pr_PathEle * with__0;
+
+ with__0 = & pr->ractpath.rele[pr->ractpath.rlen - 1];
+
+ if (!((PR_TSE_MASK_CHAR & npset) != 0)) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_STR & wpset) != 0) && (cmpres != PR_EQUAL)) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_ID & wpset) != 0) && (pr->ritems[with__0->ritemid+1]->head.info2 != pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEID))) {
+ return PR_MSNotMatched;
+ }
+ return PR_MSMatched;
+}
+
+
+static pr_MatchState pr_matchTokensLetter (picodata_ProcessingUnit this, pr_subobj_t * pr, picoos_int32 cmpres,
+ picokpr_TokSetNP npset, picokpr_TokSetWP wpset)
+{
+
+ register struct pr_PathEle * with__0;
+ picoos_int32 lulen;
+ picoos_int32 lromanval;
+
+ with__0 = & pr->ractpath.rele[pr->ractpath.rlen - 1];
+
+ if ( !((PR_TSE_MASK_LETTER & npset) != 0)) {
+ return PR_MSNotMatched;
+ }
+ lulen = picobase_utf8_length(pr->ritems[with__0->ritemid+1]->data, PR_MAX_DATA_LEN);
+ if (((PR_TSE_MASK_LEN & wpset) != 0) && (lulen != pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSELen))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_MIN & wpset) != 0) && (lulen < pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEMin))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_MAX & wpset) != 0) && (lulen > pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEMax))) {
+ return PR_MSNotMatched;
+ }
+ if ((PR_TSE_MASK_CI & npset) != 0) {
+ if (((PR_TSE_MASK_STR & wpset) != 0) && (cmpres != PR_EQUAL)) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_HEAD & wpset) != 0) && !(picokpr_isEqualHead(with__0->rnetwork,pr->ritems[with__0->ritemid+1]->strci,PR_MAX_DATA_LEN,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEHead)))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_MID & wpset) != 0) && !(picokpr_isEqualMid(with__0->rnetwork,pr->ritems[with__0->ritemid+1]->strci,PR_MAX_DATA_LEN,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEMid)))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_TAIL & wpset) != 0) && !(picokpr_isEqualTail(with__0->rnetwork,pr->ritems[with__0->ritemid+1]->strci,PR_MAX_DATA_LEN,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSETail)))) {
+ return PR_MSNotMatched;
+ }
+ } else if ((PR_TSE_MASK_CIS & npset) != 0) {
+ if (((PR_TSE_MASK_STR & wpset) != 0) && !(picokpr_isEqual(with__0->rnetwork,pr->ritems[with__0->ritemid+1]->strcis,PR_MAX_DATA_LEN,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEStr)))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_HEAD & wpset) != 0) && !(picokpr_isEqualHead(with__0->rnetwork,pr->ritems[with__0->ritemid+1]->strcis,PR_MAX_DATA_LEN,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEHead)))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_MID & wpset) != 0) && !(picokpr_isEqualMid(with__0->rnetwork,pr->ritems[with__0->ritemid+1]->strcis,PR_MAX_DATA_LEN,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEMid)))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_TAIL & wpset) != 0) && !(picokpr_isEqualTail(with__0->rnetwork,pr->ritems[with__0->ritemid+1]->strcis,PR_MAX_DATA_LEN,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSETail)))) {
+ return PR_MSNotMatched;
+ }
+ } else {
+ if (((PR_TSE_MASK_STR & wpset) != 0) && !(picokpr_isEqual(with__0->rnetwork,pr->ritems[with__0->ritemid+1]->data,PR_MAX_DATA_LEN,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEStr)))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_HEAD & wpset) != 0) && !(picokpr_isEqualHead(with__0->rnetwork,pr->ritems[with__0->ritemid+1]->data,PR_MAX_DATA_LEN,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEHead)))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_MID & wpset) != 0) && !(picokpr_isEqualMid(with__0->rnetwork,pr->ritems[with__0->ritemid+1]->data,PR_MAX_DATA_LEN,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEMid)))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_TAIL & wpset) != 0) && !(picokpr_isEqualTail(with__0->rnetwork,pr->ritems[with__0->ritemid+1]->data,PR_MAX_DATA_LEN,pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSETail)))) {
+ return PR_MSNotMatched;
+ }
+ }
+ if (((PR_TSE_MASK_AUC & npset) != 0) && !(pr->ritems[with__0->ritemid+1]->auc)) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_ALC & npset) != 0) && !(pr->ritems[with__0->ritemid+1]->alc)) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_SUC & npset) != 0) && !(pr->ritems[with__0->ritemid+1]->suc)) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_ROMAN & npset) != 0) && !(pr_isLatinNumber(pr->ritems[with__0->ritemid+1]->data,& lromanval))) {
+ return PR_MSNotMatched;
+ }
+ if (((PR_TSE_MASK_ID & wpset) != 0) && (pr->ritems[with__0->ritemid+1]->head.info2 != pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEID))) {
+ return PR_MSNotMatched;
+ }
+ return PR_MSMatched;
+}
+
+
+static pr_MatchState pr_matchTokensBegin (picodata_ProcessingUnit this, pr_subobj_t * pr,
+ picokpr_TokSetNP npset, picokpr_TokSetWP wpset)
+{
+ npset = npset; /* avoid warning "var not used in this function"*/
+ wpset = wpset; /* avoid warning "var not used in this function"*/
+ if ((PR_TSE_MASK_BEGIN &picokpr_getTokSetNP(pr->ractpath.rele[pr->ractpath.rlen - 1].rnetwork, pr->ractpath.rele[pr->ractpath.rlen - 1].rtok)) != 0) {
+ return PR_MSMatched;
+ } else {
+ return PR_MSNotMatched;
+ }
+}
+
+
+
+static pr_MatchState pr_matchTokensEnd (picodata_ProcessingUnit this, pr_subobj_t * pr,
+ picokpr_TokSetNP npset, picokpr_TokSetWP wpset)
+{
+ npset = npset; /* avoid warning "var not used in this function"*/
+ wpset = wpset; /* avoid warning "var not used in this function"*/
+ if ((PR_TSE_MASK_END &picokpr_getTokSetNP(pr->ractpath.rele[pr->ractpath.rlen - 1].rnetwork, pr->ractpath.rele[pr->ractpath.rlen - 1].rtok)) != 0) {
+ return PR_MSMatched;
+ } else {
+ return PR_MSNotMatched;
+ }
+}
+
+
+static pr_MatchState pr_matchTokens (picodata_ProcessingUnit this, pr_subobj_t * pr, picoos_int16 * cmpres)
+{
+
+ register struct pr_PathEle * with__0;
+ picokpr_VarStrPtr lstrp;
+ picokpr_TokSetNP npset;
+ picokpr_TokSetWP wpset;
+
+ with__0 = & pr->ractpath.rele[pr->ractpath.rlen - 1];
+ npset = picokpr_getTokSetNP(with__0->rnetwork, with__0->rtok);
+ wpset = picokpr_getTokSetWP(with__0->rnetwork, with__0->rtok);
+
+ *cmpres = PR_EQUAL;
+ if ((PR_TSE_MASK_STR & wpset) != 0) {
+ lstrp = picokpr_getVarStrPtr(with__0->rnetwork, pr_attrVal(with__0->rnetwork, with__0->rtok, PR_TSEStr));
+ pr_compare(pr->ritems[with__0->ritemid+1]->strci,lstrp,cmpres);
+ }
+ if (((PR_TSE_MASK_LEX & wpset) == PR_TSE_MASK_LEX) && ((PR_TSE_MASK_LETTER & npset) == 0)) {
+ return pr_matchMultiToken(this, pr, npset, wpset);
+ } else {
+ switch (pr->ritems[with__0->ritemid+1]->head.info1) {
+ case PICODATA_ITEMINFO1_TOKTYPE_BEGIN:
+ return pr_matchTokensBegin(this, pr, npset, wpset);
+ break;
+ case PICODATA_ITEMINFO1_TOKTYPE_END:
+ return pr_matchTokensEnd(this, pr, npset, wpset);
+ break;
+ case PICODATA_ITEMINFO1_TOKTYPE_SPACE:
+ return pr_matchTokensSpace(this, pr, *cmpres, npset, wpset);
+ break;
+ case PICODATA_ITEMINFO1_TOKTYPE_DIGIT:
+ return pr_matchTokensDigit(this, pr, *cmpres, npset, wpset);
+ break;
+ case PICODATA_ITEMINFO1_TOKTYPE_LETTER:
+ return pr_matchTokensLetter(this, pr, *cmpres, npset, wpset);
+ break;
+ case PICODATA_ITEMINFO1_TOKTYPE_SEQ:
+ return pr_matchTokensSeq(this, pr, *cmpres, npset, wpset);
+ break;
+ case PICODATA_ITEMINFO1_TOKTYPE_CHAR:
+ return pr_matchTokensChar(this, pr, *cmpres, npset, wpset);
+ break;
+ default:
+ PICODBG_INFO(("pr_matchTokens: unknown token type"));
+ return PR_MSNotMatched;
+ break;
+ }
+ }
+}
+
+
+static void pr_calcPathCost (struct pr_Path * path)
+{
+ picoos_int32 li;
+ picoos_bool lfirst;
+ picokpr_TokSetWP wpset;
+ picokpr_TokSetNP npset;
+#if PR_TRACE_PATHCOST
+ picoos_uchar str[1000];
+ picoos_uchar * strp;
+#endif
+
+#if PR_TRACE_PATHCOST
+ str[0] = 0;
+#endif
+
+ lfirst = TRUE;
+ path->rcost = PR_COST_INIT;
+ for (li = 0; li < path->rlen; li++) {
+ if (li == 0) {
+ path->rcost = path->rcost + path->rele[li].rprodprefcost;
+ }
+ wpset = picokpr_getTokSetWP(path->rele[li].rnetwork, path->rele[li].rtok);
+ npset = picokpr_getTokSetNP(path->rele[li].rnetwork, path->rele[li].rtok);
+ if ((PR_TSE_MASK_COST & wpset) != 0) {
+ if (((PR_TSE_MASK_LEX & wpset) == PR_TSE_MASK_LEX) && ((PR_TSE_MASK_LETTER & npset) == 0)) {
+ if (lfirst) {
+ path->rcost = path->rcost - PR_COST + pr_attrVal(path->rele[li].rnetwork, path->rele[li].rtok, PR_TSECost);
+ } else {
+ path->rcost = path->rcost - PR_COST;
+ }
+ lfirst = FALSE;
+ } else {
+ path->rcost = path->rcost - PR_COST + pr_attrVal(path->rele[li].rnetwork, path->rele[li].rtok, PR_TSECost);
+ lfirst = TRUE;
+ }
+ } else if (pr_hasToken(& wpset,& npset)) {
+ path->rcost = path->rcost - PR_COST;
+ }
+#if PR_TRACE_PATHCOST
+ if ((path->rele[li].rprodname != 0)) {
+ strp = picokpr_getVarStrPtr(path->rele[li].rnetwork, path->rele[li].rprodname);
+ picoos_strcat(str, (picoos_char *)" ");
+ picoos_strcat(str, strp);
+ }
+#endif
+ }
+#if PR_TRACE_PATHCOST
+ PICODBG_INFO(("pp cost: %i %s", path->rcost, str));
+#endif
+}
+
+
+void pr_processToken (picodata_ProcessingUnit this, pr_subobj_t * pr)
+{
+ register struct pr_PathEle * with__0;
+ picoos_bool ldummy;
+ picoos_int32 li;
+ picokpr_TokSetNP npset;
+ picokpr_TokSetWP wpset;
+
+ do {
+ pr->rgState = PR_GSContinue;
+ if ((pr->ractpath.rlen == 0)) {
+ if (pr_getTopLevelToken(this, pr, FALSE)) {
+ pr->rgState = PR_GSContinue;
+ } else if (pr->rbestpath.rlen == 0) {
+ pr->rgState = PR_GSNotFound;
+ } else {
+ pr->rgState = PR_GSFound;
+ }
+ } else {
+ if (pr->maxPathLen < pr->ractpath.rlen) {
+ pr->maxPathLen = pr->ractpath.rlen;
+ }
+ with__0 = & pr->ractpath.rele[pr->ractpath.rlen - 1];
+ switch (with__0->rlState) {
+ case PR_LSInit:
+ npset = picokpr_getTokSetNP(with__0->rnetwork, with__0->rtok);
+ wpset = picokpr_getTokSetWP(with__0->rnetwork, with__0->rtok);
+ if ((PR_TSE_MASK_ACCEPT & npset) != 0){
+ if (with__0->rdepth == 1) {
+ pr_calcPathCost(&pr->ractpath);
+ if ((pr->rbestpath.rlen == 0) || (pr->ractpath.rcost < pr->rbestpath.rcost)) {
+ pr->rbestpath.rlen = pr->ractpath.rlen;
+ pr->rbestpath.rcost = pr->ractpath.rcost;
+ for (li = 0; li < pr->ractpath.rlen; li++) {
+ pr->rbestpath.rele[li] = pr->ractpath.rele[li];
+ }
+ }
+ with__0->rlState = PR_LSGetNextToken;
+ } else {
+ with__0->rlState = PR_LSGetProdContToken;
+ }
+ } else if ((PR_TSE_MASK_PROD & wpset) != 0) {
+ with__0->rlState = PR_LSGetProdToken;
+ } else if ((PR_TSE_MASK_OUT & wpset) != 0) {
+ with__0->rlState = PR_LSGetNextToken;
+ } else if (pr_hasToken(& wpset,& npset)) {
+ with__0->rlState = PR_LSGetToken;
+ } else {
+ with__0->rlState = PR_LSGetNextToken;
+ }
+ break;
+ case PR_LSGetProdToken:
+ with__0->rlState = PR_LSGetAltToken;
+ ldummy = pr_getProdToken(this, pr);
+ break;
+ case PR_LSGetProdContToken:
+ with__0->rlState = PR_LSGetAltToken;
+ ldummy = pr_getProdContToken(this, pr);
+ break;
+ case PR_LSGoBack:
+ pr->ractpath.rlen--;
+ break;
+ case PR_LSGetToken:
+ if (pr_getToken(this, pr)) {
+ with__0->rlState = PR_LSMatch;
+ } else if (pr->forceOutput) {
+ with__0->rlState = PR_LSGetAltToken;
+ } else {
+ with__0->rlState = PR_LSGetToken2;
+ pr->rgState = PR_GSNeedToken;
+ }
+ break;
+ case PR_LSGetToken2:
+ if (pr_getToken(this, pr)) {
+ with__0->rlState = PR_LSMatch;
+ } else {
+ with__0->rlState = PR_LSGoBack;
+ }
+ break;
+ case PR_LSMatch:
+ switch (pr_matchTokens(this, pr, & with__0->rcompare)) {
+ case PR_MSMatched:
+ with__0->rlState = PR_LSGetNextToken;
+ break;
+ case PR_MSMatchedContinue:
+ with__0->rlState = PR_LSGetAltToken;
+ ldummy = pr_getNextMultiToken(this, pr);
+ break;
+ case PR_MSMatchedMulti:
+ with__0->rlState = PR_LSGetNextToken;
+ ldummy = pr_getNextMultiToken(this, pr);
+ break;
+ default:
+ with__0->rlState = PR_LSGetAltToken;
+ break;
+ }
+ break;
+ case PR_LSGetNextToken:
+ with__0->rlState = PR_LSGetAltToken;
+ ldummy = pr_getNextToken(this, pr);
+ break;
+ case PR_LSGetAltToken:
+ with__0->rlState = PR_LSGoBack;
+ ldummy = pr_getAltToken(this, pr);
+ break;
+ default:
+ PICODBG_INFO(("unhandled local state"));
+ break;
+ }
+ }
+ pr->nrIterations--;
+ } while ((pr->rgState == PR_GSContinue) && (pr->nrIterations > 0));
+}
+
+
+void pr_process (picodata_ProcessingUnit this, pr_subobj_t * pr)
+{
+ switch (pr->rgState) {
+ case PR_GS_START:
+ case PR_GSFound:
+ case PR_GSNotFound:
+ pr->ractpath.rlen = 0;
+ pr->ractpath.rcost = PR_COST_INIT;
+ pr->rbestpath.rlen = 0;
+ pr->rbestpath.rcost = PR_COST_INIT;
+ if (pr_getTopLevelToken(this, pr, TRUE)) {
+ pr->rgState = PR_GSContinue;
+ } else {
+ pr->rgState = PR_GSNotFound;
+ }
+ break;
+ case PR_GSContinue:
+ pr_processToken(this, pr);
+ break;
+ case PR_GSNeedToken:
+ pr->rgState = PR_GSContinue;
+ break;
+ default:
+ pr->rgState = PR_GS_START;
+ break;
+ }
+}
+
+
+static void pr_prepareItem (picodata_ProcessingUnit this, pr_subobj_t * pr, pr_ioItemPtr item)
+{
+ pr->ritems[pr->rnritems + 1] = item;
+ pr->rnritems++;
+}
+
+
+static void pr_processItems (picodata_ProcessingUnit this, pr_subobj_t * pr)
+{
+ pr_ioItemPtr lit;
+ pr_MemState lmemState;
+
+ pr_getMemState(this, pr_WorkMem,& lmemState);
+
+ while ((pr->rinItemList != NULL) && (pr->rinItemList->head.type != PICODATA_ITEM_TOKEN)) {
+ lit = pr->rinItemList;
+ PICODBG_INFO(("pp in (0)"));
+ PICODBG_INFO(("pp out(0)"));
+ pr->rinItemList = pr->rinItemList->next;
+ lit->next = NULL;
+ if (pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_PHONEME) && pr_isCmdInfo2(lit, PICODATA_ITEMINFO2_CMD_START)) {
+ pr->insidePhoneme = TRUE;
+ } else if (pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_PHONEME) && pr_isCmdInfo2(lit, PICODATA_ITEMINFO2_CMD_END)) {
+ pr->insidePhoneme = FALSE;
+ }
+ if (pr->insidePhoneme && (pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_PLAY) || pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_IGNSIG))) {
+ pr_disposeItem(this, & lit);
+ } else if (pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_CONTEXT)) {
+ pr_setContext(this, pr, lit->data);
+ pr_disposeItem(this, & lit);
+ } else if (pr->rignore <= 0) {
+ pr_appendItemToOutItemList(this, pr, & pr->routItemList,& pr->rlastOutItem,lit);
+ if (pr->outOfMemory) return;
+ } else {
+ pr_disposeItem(this, & lit);
+ }
+ pr->rgState = PR_GS_START;
+ }
+ if (pr->rinItemList != NULL) {
+ pr_process(this, pr);
+ if (pr->rgState == PR_GSNotFound) {
+ lit = pr->rinItemList;
+ pr->rinItemList = pr->rinItemList->next;
+ lit->next = NULL;
+ PICODBG_INFO(("pp in (2): '%s'", lit->data));
+ if (pr->rignore <= 0) {
+ PICODBG_INFO(("pp out(2): '%s'", lit->data));
+ }
+
+ if (pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_PHONEME) && pr_isCmdInfo2(lit, PICODATA_ITEMINFO2_CMD_START)) {
+ pr->insidePhoneme = TRUE;
+ } else if (pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_PHONEME) && pr_isCmdInfo2(lit, PICODATA_ITEMINFO2_CMD_END)) {
+ pr->insidePhoneme = FALSE;
+ }
+ if (((pr->rignore <= 0) && !((pr->insidePhoneme && (pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_PLAY) || pr_isCmdType(lit,PICODATA_ITEMINFO1_CMD_IGNSIG)))))) {
+ pr_appendItemToOutItemList(this, pr, & pr->routItemList,& pr->rlastOutItem,lit);
+ if (pr->outOfMemory) return;
+ } else {
+ pr_disposeItem(this, & lit);
+ }
+ pr->rgState = PR_GS_START;
+ pr->rnritems = 0;
+ } else if (pr->rgState == PR_GSFound) {
+ pr_outputPath(this, pr);
+ if (pr->outOfMemory) return;
+ pr->rgState = PR_GS_START;
+ pr->rnritems = 0;
+ }
+ }
+ if (pr->rinItemList == NULL) {
+ pr->rlastInItem = NULL;
+ } else if (pr->rnritems == 0) {
+ lit = pr->rinItemList;
+ while (lit != NULL) {
+ if (lit->head.type == PICODATA_ITEM_TOKEN) {
+ pr_prepareItem(this, pr, lit);
+ }
+ lit = lit->next;
+ }
+ }
+ pr_resetMemState(this, pr_WorkMem,lmemState);
+}
+
+
+
+extern void pr_treatItem (picodata_ProcessingUnit this, pr_subobj_t * pr, pr_ioItemPtr item)
+{
+ pr_ioItemPtr lit;
+
+ pr_startItemList(& pr->routItemList,& pr->rlastOutItem);
+
+ if (!PR_ENABLED || (pr->rgState == PR_GSNoPreproc)) {
+ /* preprocessing disabled or no preproc networks available:
+ append items directly to output item list */
+ PICODBG_INFO(("pp in (3): '%s'", item->data));
+ PICODBG_INFO(("pp out(3): '%s'", item->data));
+ pr_appendItemToOutItemList(this, pr, & pr->routItemList,& pr->rlastOutItem,item);
+ } else {
+
+ if (pr->actCtxChanged) {
+ pr->rgState = PR_GS_START;
+ pr->ractpath.rcost = PR_COST_INIT;
+ pr->ractpath.rlen = 0;
+ pr->rbestpath.rcost = PR_COST_INIT;
+ pr->rbestpath.rlen = 0;
+ pr->prodList = NULL;
+ pr->rnritems = 0;
+ pr->actCtxChanged = FALSE;
+ }
+ if (pr_isCmdType(item , PICODATA_ITEMINFO1_CMD_CONTEXT) || pr_isCmdType(item, PICODATA_ITEMINFO1_CMD_FLUSH)) {
+ /* context switch or flush: force processing and empty input item list */
+ pr->forceOutput = TRUE;
+ }
+ pr_appendItem(this, & pr->rinItemList,& pr->rlastInItem, item);
+ if (pr->rnritems == 0) {
+ lit = pr->rinItemList;
+ while (lit != NULL) {
+ if (lit->head.type == PICODATA_ITEM_TOKEN) {
+ pr_prepareItem(this, pr, lit);
+ }
+ lit = lit->next;
+ }
+ } else if (item->head.type == PICODATA_ITEM_TOKEN) {
+ pr_prepareItem(this, pr, item);
+ }
+ }
+}
+
+/* *****************************************************************************/
+/* *****************************************************************************/
+/* *****************************************************************************/
+
+
+pico_status_t prReset(register picodata_ProcessingUnit this)
+{
+ picoos_int32 i;
+ pr_subobj_t * pr;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ pr = (pr_subobj_t *) this->subObj;
+
+ pr->rinItemList = NULL;
+ pr->rlastInItem = NULL;
+ pr->routItemList = NULL;
+ pr->rlastOutItem = NULL;
+ pr->ractpath.rcost = PR_COST_INIT;
+ pr->ractpath.rlen = 0;
+ pr->rbestpath.rcost = PR_COST_INIT;
+ pr->rbestpath.rlen = 0;
+ pr->rnritems = 0;
+ pr->ritems[0] = NULL;
+ pr->rignore = 0;
+ pr->spellMode = 0;
+ pr->maxPathLen = 0;
+ pr->insidePhoneme = FALSE;
+ pr->saveFile[0] = 0;
+
+ pr->outReadPos = 0;
+ pr->outWritePos = 0;
+ pr->inBufLen = 0;
+
+ pr->rgState = PR_GSNoPreproc;
+ for (i=0; i<PR_MAX_NR_PREPROC; i++) {
+ if (pr->preproc[i] != NULL) {
+ pr->rgState = PR_GS_START;
+ }
+ }
+ pr->actCtx = pr_findContext(pr->ctxList, (picoos_uchar*)PICO_CONTEXT_DEFAULT);
+ pr->actCtxChanged = FALSE;
+ pr->prodList = NULL;
+
+ if (((picoos_uint32)pr->pr_WorkMem % PICOOS_ALIGN_SIZE) == 0) {
+ pr->workMemTop = 0;
+ }
+ else {
+ pr->workMemTop = PICOOS_ALIGN_SIZE - ((picoos_uint32)pr->pr_WorkMem % PICOOS_ALIGN_SIZE);
+ }
+ pr->maxWorkMemTop=0;
+ pr->dynMemSize=0;
+ pr->maxDynMemSize=0;
+ /* this is ok to be in 'initialize' because it is a private memory within pr. Creating a new mm
+ * here amounts to resetting this internal memory
+ */
+ pr->dynMemMM = picoos_newMemoryManager((void *)pr->pr_DynMem, PR_DYN_MEM_SIZE,
+ /*enableMemProt*/ FALSE);
+ pr->outOfMemory = FALSE;
+
+ pr->forceOutput = FALSE;
+
+
+ pr->xsampa_parser = picokfst_getFST(this->voice->kbArray[PICOKNOW_KBID_FST_XSAMPA_PARSE]);
+
+ pr->svoxpa_parser = picokfst_getFST(this->voice->kbArray[PICOKNOW_KBID_FST_SVOXPA_PARSE]);
+
+ pr->xsampa2svoxpa_mapper = picokfst_getFST(this->voice->kbArray[PICOKNOW_KBID_FST_XSAMPA2SVOXPA]);
+
+
+
+ return PICO_OK;
+}
+
+
+pico_status_t prInitialize(register picodata_ProcessingUnit this)
+{
+/*
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+*/
+ return prReset(this);
+}
+
+
+pico_status_t prTerminate(register picodata_ProcessingUnit this)
+{
+ return PICO_OK;
+}
+
+picodata_step_result_t prStep(register picodata_ProcessingUnit this, picoos_int16 mode, picoos_uint16 * numBytesOutput);
+
+pico_status_t prSubObjDeallocate(register picodata_ProcessingUnit this,
+ picoos_MemoryManager mm)
+{
+ pr_subobj_t * pr;
+
+ if (NULL != this) {
+ pr = (pr_subobj_t *) this->subObj;
+ mm = mm; /* avoid warning "var not used in this function"*/
+ PICODBG_INFO(("max pr_WorkMem: %i of %i", pr->maxWorkMemTop, PR_WORK_MEM_SIZE));
+ PICODBG_INFO(("max pr_DynMem: %i of %i", pr->maxDynMemSize, PR_DYN_MEM_SIZE));
+
+ pr_disposeContextList(this);
+ picoos_deallocate(this->common->mm, (void *) &this->subObj);
+ }
+ return PICO_OK;
+}
+
+picodata_ProcessingUnit picopr_newPreprocUnit(picoos_MemoryManager mm, picoos_Common common,
+ picodata_CharBuffer cbIn, picodata_CharBuffer cbOut,
+ picorsrc_Voice voice)
+{
+ picoos_int32 i;
+ pr_subobj_t * pr;
+
+
+ picodata_ProcessingUnit this = picodata_newProcessingUnit(mm, common, cbIn, cbOut, voice);
+ if (this == NULL) {
+ return NULL;
+ }
+
+ this->initialize = prInitialize;
+ PICODBG_DEBUG(("set this->step to prStep"));
+ this->step = prStep;
+ this->terminate = prTerminate;
+ this->subDeallocate = prSubObjDeallocate;
+ this->subObj = picoos_allocate(mm, sizeof(pr_subobj_t));
+#if PR_TRACE_MEM || PR_TRACE_MAX_MEM
+ PICODBG_INFO(("preproc alloc: %i", sizeof(pr_subobj_t)));
+ PICODBG_INFO(("max dyn size: %i", PR_MAX_PATH_LEN*((((PR_IOITEM_MIN_SIZE+2) + PICOOS_ALIGN_SIZE - 1) / PICOOS_ALIGN_SIZE) * PICOOS_ALIGN_SIZE + 16)));
+#endif
+ if (this->subObj == NULL) {
+ picoos_deallocate(mm, (void *)&this);
+ return NULL;
+ }
+ pr = (pr_subobj_t *) this->subObj;
+
+ pr->graphs = picoktab_getGraphs(this->voice->kbArray[PICOKNOW_KBID_TAB_GRAPHS]);
+ pr->preproc[0] = picokpr_getPreproc(this->voice->kbArray[PICOKNOW_KBID_TPP_MAIN]);
+ for (i=0; i<PICOKNOW_MAX_NUM_UTPP; i++) {
+ pr->preproc[1+i] = picokpr_getPreproc(this->voice->kbArray[PICOKNOW_KBID_TPP_USER_1+i]);
+ }
+
+ if (pr_createContextList(this) != PICO_OK) {
+ pr_disposeContextList(this);
+ picoos_deallocate(mm, (void *)&this);
+ return NULL;
+ }
+ prInitialize(this);
+ return this;
+}
+
+/**
+ * fill up internal buffer
+ */
+picodata_step_result_t prStep(register picodata_ProcessingUnit this,
+ picoos_int16 mode, picoos_uint16 * numBytesOutput)
+{
+ register pr_subobj_t * pr;
+ pr_ioItemPtr it;
+ picoos_int32 len, i;
+ pico_status_t rv;
+ picoos_int32 id;
+ picoos_uint8 info1;
+ picoos_uint8 info2;
+ picoos_int32 nrUtfChars;
+ picoos_uint32 pos;
+ picobase_utf8char inUtf8char, outUtf8char;
+ picoos_int32 inUtf8charlen, outUtf8charlen;
+ picoos_int32 lenpos;
+ picoos_bool ldone;
+ picoos_bool split;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ pr = (pr_subobj_t *) this->subObj;
+
+ if (pr->outOfMemory) return PICODATA_PU_ERROR;
+
+ mode = mode; /* avoid warning "var not used in this function"*/
+ pr->nrIterations = PR_MAX_NR_ITERATIONS;
+
+ *numBytesOutput = 0;
+ while (1) { /* exit via return */
+ if ((pr->outWritePos - pr->outReadPos) > 0) {
+ /* deliver the data in the output buffer */
+ if (picodata_cbPutItem(this->cbOut, &pr->outBuf[pr->outReadPos], pr->outWritePos - pr->outReadPos, numBytesOutput) == PICO_OK) {
+ pr->outReadPos += *numBytesOutput;
+ if (pr->outWritePos == pr->outReadPos) {
+ pr->outWritePos = 0;
+ pr->outReadPos = 0;
+ }
+ }
+ else {
+ return PICODATA_PU_OUT_FULL;
+ }
+ }
+ else if (pr->routItemList != NULL) {
+ /* there are item(s) in the output item list, move them to the output buffer */
+ it = pr->routItemList;
+ pr->routItemList = pr->routItemList->next;
+ if (pr->routItemList == NULL) {
+ pr->rlastOutItem = NULL;
+ }
+ if (it->head.type == PICODATA_ITEM_TOKEN) {
+ if ((it->head.info1 != PICODATA_ITEMINFO1_TOKTYPE_SPACE) && (it->head.len > 0)) {
+ nrUtfChars = picobase_utf8_length(it->data, PR_MAX_DATA_LEN);
+ if ((nrUtfChars == 1)
+ && (((id = picoktab_graphOffset(pr->graphs, it->data)) > 0))
+ && picoktab_getIntPropPunct(pr->graphs, id, &info1, &info2)) {
+ /* single punctuation chars have to be delivered as PICODATA_ITEM_PUNC items
+ instead as PICODATA_ITEM_WORDGRAPH items */
+ pr->outBuf[pr->outWritePos++] = PICODATA_ITEM_PUNC;
+ pr->outBuf[pr->outWritePos++] = info1;
+ pr->outBuf[pr->outWritePos++] = info2;
+ pr->outBuf[pr->outWritePos++] = 0;
+ PICODATA_INFO_ITEM(this->voice->kbArray[PICOKNOW_KBID_DBG],
+ (picoos_uint8 *)"pr: ", pr->outBuf, pr->outWritePos);
+ }
+ else {
+ /* do subgraphs substitutions and deliver token items as PICODATA_ITEM_WORDGRAPH
+ items to the output buffer */
+ split = FALSE;
+ pr->outBuf[pr->outWritePos++] = PICODATA_ITEM_WORDGRAPH;
+ pr->outBuf[pr->outWritePos++] = PICODATA_ITEMINFO1_NA;
+ pr->outBuf[pr->outWritePos++] = PICODATA_ITEMINFO2_NA;
+ lenpos=pr->outWritePos;
+ pr->outBuf[pr->outWritePos++] = 0;
+ pos = 0;
+ len = pr_strlen(it->data);
+ while (pos < (picoos_uint32)len) {
+ if (picobase_get_next_utf8char(it->data, it->head.len, &pos, inUtf8char)) {
+ if (inUtf8char[0] <= 32) {
+ /* do not add whitespace characters to the output buffer,
+ but initiate token splitting instead
+
+ */
+ split = TRUE;
+ }
+ else if (((id = picoktab_graphOffset(pr->graphs, inUtf8char)) > 0) && picoktab_getStrPropGraphsubs1(pr->graphs, id, outUtf8char)) {
+ if (split) {
+ /* split the token, eg. start a new item */
+ pr->outBuf[pr->outWritePos++] = PICODATA_ITEM_WORDGRAPH;
+ pr->outBuf[pr->outWritePos++] = PICODATA_ITEMINFO1_NA;
+ pr->outBuf[pr->outWritePos++] = PICODATA_ITEMINFO2_NA;
+ lenpos=pr->outWritePos;
+ pr->outBuf[pr->outWritePos++] = 0;
+ }
+ outUtf8charlen = picobase_det_utf8_length(outUtf8char[0]);
+ for (i=0; i<outUtf8charlen; i++) {
+ pr->outBuf[pr->outWritePos++] = outUtf8char[i];
+ pr->outBuf[lenpos]++;
+ }
+ if (picoktab_getStrPropGraphsubs2(pr->graphs, id, outUtf8char)) {
+ outUtf8charlen = picobase_det_utf8_length(outUtf8char[0]);
+ for (i=0; i<outUtf8charlen; i++) {
+ pr->outBuf[pr->outWritePos++] = outUtf8char[i];
+ pr->outBuf[lenpos]++;
+ }
+ }
+ split = FALSE;
+ }
+ else {
+ if (split) {
+ /* split the token, eg. start a new item */
+ pr->outBuf[pr->outWritePos++] = PICODATA_ITEM_WORDGRAPH;
+ pr->outBuf[pr->outWritePos++] = PICODATA_ITEMINFO1_NA;
+ pr->outBuf[pr->outWritePos++] = PICODATA_ITEMINFO2_NA;
+ lenpos=pr->outWritePos;
+ pr->outBuf[pr->outWritePos++] = 0;
+ }
+ inUtf8charlen = picobase_det_utf8_length(inUtf8char[0]);
+ for (i=0; i<inUtf8charlen; i++) {
+ pr->outBuf[pr->outWritePos++] = inUtf8char[i];
+ pr->outBuf[lenpos]++;
+ }
+ split = FALSE;
+ }
+ }
+ }
+ PICODATA_INFO_ITEM(this->voice->kbArray[PICOKNOW_KBID_DBG],
+ (picoos_uint8 *)"pr: ", pr->outBuf, pr->outWritePos);
+ }
+ }
+ }
+ else {
+ /* handle all other item types and put them to the output buffer */
+ pr->outBuf[pr->outWritePos++] = it->head.type;
+ pr->outBuf[pr->outWritePos++] = it->head.info1;
+ pr->outBuf[pr->outWritePos++] = it->head.info2;
+ pr->outBuf[pr->outWritePos++] = it->head.len;
+ for (i=0; i<it->head.len; i++) {
+ pr->outBuf[pr->outWritePos++] = it->data[i];
+ }
+ PICODATA_INFO_ITEM(this->voice->kbArray[PICOKNOW_KBID_DBG],
+ (picoos_uint8 *)"pr: ", pr->outBuf, pr->outWritePos);
+ }
+ pr_disposeItem(this, &it);
+ }
+ else if (pr->forceOutput) {
+ pr_processItems(this, pr);
+ if (pr->rinItemList == NULL) {
+ pr->forceOutput = FALSE;
+ }
+ }
+ else if ((pr->rgState != PR_GSNeedToken) && (pr->rinItemList != NULL)) {
+ pr_processItems(this, pr);
+ }
+ else if (pr->inBufLen > 0) {
+ /* input data is available in the input buffer, copy it to an input item
+ and treat it */
+ if (pr->dynMemSize < (45*PR_DYN_MEM_SIZE / 100)) {
+ pr_newItem(this, pr_DynMem, &it, pr->inBuf[0], pr->inBuf[3], /*inItem*/TRUE);
+ if (pr->outOfMemory) return PICODATA_PU_ERROR;
+ it->head.type = pr->inBuf[0];
+ it->head.info1 = pr->inBuf[1];
+ it->head.info2 = pr->inBuf[2];
+ it->head.len = pr->inBuf[3];
+ for (i=0; i<pr->inBuf[3]; i++) {
+ it->data[i] = pr->inBuf[4+i];
+ }
+ it->data[pr->inBuf[3]] = 0;
+ if ((pr->inBuf[0] == PICODATA_ITEM_TOKEN) && ((pr->inBuf[1] == PICODATA_ITEMINFO1_TOKTYPE_DIGIT))) {
+ it->val = tok_tokenDigitStrToInt(this, pr, it->data);
+ } else {
+ it->val = 0;
+ }
+ if (pr->inBuf[0] == PICODATA_ITEM_TOKEN) {
+ picobase_lowercase_utf8_str(it->data,it->strci,PR_MAX_DATA_LEN, &ldone);
+ pr_firstLetterToLowerCase(it->data,it->strcis);
+ it->alc = picobase_is_utf8_lowercase(it->data,PR_MAX_DATA_LEN);
+ it->auc = picobase_is_utf8_uppercase(it->data,PR_MAX_DATA_LEN);
+ it->suc = pr_isSUC(it->data);
+ }
+
+ pr_treatItem(this, pr, it);
+ if (pr->outOfMemory) return PICODATA_PU_ERROR;
+ pr_processItems(this, pr);
+ pr->inBufLen = 0;
+ }
+ else {
+ pr->forceOutput = TRUE;
+ }
+ }
+ else {
+ /* there is not data in the output buffer and there is no data in the output item list, so
+ check whether input data is available */
+ rv = picodata_cbGetItem(this->cbIn, pr->inBuf, IN_BUF_SIZE+PICODATA_ITEM_HEADSIZE, &pr->inBufLen);
+ if (PICO_OK == rv) {
+ } else if (PICO_EOF == rv) {
+ /* there was no item in the char buffer */
+ return PICODATA_PU_IDLE;
+ } else if ((PICO_EXC_BUF_UNDERFLOW == rv) || (PICO_EXC_BUF_OVERFLOW == rv)) {
+ pr->inBufLen = 0;
+ PICODBG_ERROR(("problem getting item"));
+ picoos_emRaiseException(this->common->em, rv, NULL, NULL);
+ return PICODATA_PU_ERROR;
+ } else {
+ pr->inBufLen = 0;
+ PICODBG_ERROR(("problem getting item, unhandled"));
+ picoos_emRaiseException(this->common->em, rv, NULL, NULL);
+ return PICODATA_PU_ERROR;
+ }
+ }
+#if PR_TRACE_MEM
+ PICODBG_INFO(("memory: dyn=%u, work=%u", pr->dynMemSize, pr->workMemTop));
+#endif
+ if (pr->nrIterations <= 0) {
+ return PICODATA_PU_BUSY;
+ }
+ } /* while */
+ return PICODATA_PU_ERROR;
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+
+
+/* end */
diff --git a/lib/picopr.h b/lib/picopr.h
new file mode 100644
index 0000000..3ade5c9
--- /dev/null
+++ b/lib/picopr.h
@@ -0,0 +1,63 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picopr.h
+ *
+ * text preprocessor PU - include file
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+/**
+ * @addtogroup picopr
+ *
+ * <b> Text Pre Processing module </b>\n
+ *
+*/
+#ifndef PICOPR_H_
+#define PICOPR_H_
+
+#include "picoos.h"
+#include "picodata.h"
+#include "picorsrc.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+picodata_ProcessingUnit picopr_newPreprocUnit(
+ picoos_MemoryManager mm,
+ picoos_Common common,
+ picodata_CharBuffer cbIn,
+ picodata_CharBuffer cbOut,
+ picorsrc_Voice voice);
+
+#define PICOPR_OUTBUF_SIZE 256
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif /*PICOPR_H_*/
diff --git a/lib/picorsrc.c b/lib/picorsrc.c
new file mode 100644
index 0000000..d6e1e51
--- /dev/null
+++ b/lib/picorsrc.c
@@ -0,0 +1,1028 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picorsrc.c
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#include "picodefs.h"
+#include "picoos.h"
+#include "picodbg.h"
+
+/* knowledge layer */
+#include "picoknow.h"
+
+#include "picokdt.h"
+#include "picoklex.h"
+#include "picokfst.h"
+#include "picokpdf.h"
+#include "picoktab.h"
+#include "picokpr.h"
+
+#include "picorsrc.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+#if defined(PICO_DEBUG)
+#include "picokdbg.h"
+#endif
+
+
+/** object : Resource
+ * shortcut : rsrc
+ *
+ */
+typedef struct picorsrc_resource {
+ picoos_uint32 magic; /* magic number used to validate handles */
+ /* next connects all active resources of a resource manager and the garbaged resources of the manager's free list */
+ picorsrc_Resource next;
+ picorsrc_resource_type_t type;
+ picorsrc_resource_name_t name;
+ picoos_int8 lockCount; /* count of current subscribers of this resource */
+ picoos_File file;
+ picoos_uint8 * raw_mem; /* pointer to allocated memory. NULL if preallocated. */
+ /* picoos_uint32 size; */
+ picoos_uint8 * start; /* start of content (after header) */
+ picoknow_KnowledgeBase kbList;
+} picorsrc_resource_t;
+
+
+#define MAGIC_MASK 0x7049634F /* pIcO */
+
+#define SET_MAGIC_NUMBER(res) \
+ (res)->magic = ((picoos_uint32) (res)) ^ MAGIC_MASK
+
+#define CHECK_MAGIC_NUMBER(res) \
+ ((res)->magic == (((picoos_uint32) (res)) ^ MAGIC_MASK))
+
+
+
+/**
+ * Returns non-zero if 'this' is a valid resource handle, zero otherwise.
+ */
+picoos_int16 picoctrl_isValidResourceHandle(picorsrc_Resource this)
+{
+ return (this != NULL) && CHECK_MAGIC_NUMBER(this);
+}
+
+
+static picorsrc_Resource picorsrc_newResource(picoos_MemoryManager mm)
+{
+ picorsrc_Resource this = picoos_allocate(mm, sizeof(*this));
+ if (NULL != this) {
+ SET_MAGIC_NUMBER(this);
+ /* initialize */
+ this->name[0] = NULLC;
+ /* picoos_strlcpy(this->name, name,PICORSRC_MAX_RSRC_NAME_SIZ); */
+ this->next = NULL;
+ this->type = PICORSRC_TYPE_NULL;
+ this->lockCount = 0;
+ this->file = NULL;
+ this->raw_mem = NULL;
+ this->start = NULL;
+ this->kbList = NULL;
+ /* this->size=0; */
+ }
+ return this;
+}
+
+static void picorsrc_disposeResource(picoos_MemoryManager mm, picorsrc_Resource * this)
+{
+ if (NULL != (*this)) {
+ (*this)->magic ^= 0xFFFEFDFC;
+ /* we have to explicitly free 'raw_mem' here because in testing
+ scenarios (where memory protection functionality is enabled)
+ it might be allocated aside from normal memory */
+ if ((*this)->raw_mem != NULL) {
+ picoos_deallocProtMem(mm, (void *) &(*this)->raw_mem);
+ }
+ picoos_deallocate(mm,(void * *)this);
+ }
+}
+
+
+
+
+static void picorsrc_initializeVoice(picorsrc_Voice this)
+{
+ picoos_uint16 i;
+ if (NULL != this) {
+ /* initialize */
+ for (i=0; i<PICORSRC_KB_ARRAY_SIZE; i++) {
+ this->kbArray[i] = NULL;
+ }
+ this->numResources = 0;
+ this->next = NULL;
+ }
+}
+
+static picorsrc_Voice picorsrc_newVoice(picoos_MemoryManager mm)
+{
+ picorsrc_Voice this = (picorsrc_Voice) picoos_allocate(mm,sizeof(*this));
+ picorsrc_initializeVoice(this);
+ return this;
+}
+
+/*
+static void picorsrc_disposeVoice(picoos_MemoryManager mm, picorsrc_Voice * this)
+{
+ if (NULL != (*this)) {
+
+ picoos_deallocate(mm,(void *)this);
+ }
+}
+*/
+
+
+/** object : VoiceDefinition
+ * shortcut : vdef
+ *
+ */
+
+typedef struct picorsrc_voice_definition * picorsrc_VoiceDefinition;
+
+typedef struct picorsrc_voice_definition {
+ picoos_char voiceName[PICO_MAX_VOICE_NAME_SIZE];
+ picoos_uint8 numResources;
+ picorsrc_resource_name_t resourceName[PICO_MAX_NUM_RSRC_PER_VOICE];
+ picorsrc_VoiceDefinition next;
+} picorsrc_voice_definition_t;
+
+
+static picorsrc_VoiceDefinition picorsrc_newVoiceDefinition(picoos_MemoryManager mm)
+{
+ /* picoos_uint8 i; */
+ picorsrc_VoiceDefinition this = (picorsrc_VoiceDefinition) picoos_allocate(mm,sizeof(*this));
+ if (NULL != this) {
+ /* initialize */
+ this->voiceName[0] = NULLC;
+ this->numResources = 0;
+ /*
+ for (i=0; i < PICO_MAX_NUM_RSRC_PER_VOICE; i++) {
+ this->resourceName[i][0] = NULLC;
+ }
+ */
+ this->next = NULL;
+ }
+ return this;
+}
+
+/*
+static void picorsrc_disposeVoiceDefinition(picoos_MemoryManager mm, picorsrc_VoiceDefinition * this)
+{
+ if (NULL != (*this)) {
+
+ picoos_deallocate(mm,(void *)this);
+ }
+}
+*/
+
+
+
+/** object : ResourceManager
+ * shortcut : rm
+ *
+ */
+typedef struct picorsrc_resource_manager {
+ picoos_Common common;
+ picoos_uint16 numResources;
+ picorsrc_Resource resources, freeResources;
+ picoos_uint16 numVoices;
+ picorsrc_Voice voices, freeVoices;
+ picoos_uint16 numVdefs;
+ picorsrc_VoiceDefinition vdefs, freeVdefs;
+ picoos_uint16 numKbs;
+ picoknow_KnowledgeBase freeKbs;
+ picoos_header_string_t tmpHeader;
+} picorsrc_resource_manager_t;
+
+pico_status_t picorsrc_createDefaultResource(picorsrc_ResourceManager this /*,
+ picorsrc_Resource * resource */);
+
+
+picorsrc_ResourceManager picorsrc_newResourceManager(picoos_MemoryManager mm, picoos_Common common /* , picoos_char * configFile */)
+{
+ picorsrc_ResourceManager this = picoos_allocate(mm,sizeof(*this));
+ if (NULL != this) {
+ /* initialize */
+ this->common = common;
+ this->numResources = 0;
+ this->resources = NULL;
+ this->freeResources = NULL;
+ this->numVoices = 0;
+ this->voices = NULL;
+ this->freeVoices = NULL;
+ this->numVdefs = 0;
+ this->vdefs = NULL;
+ this->freeVdefs = NULL;
+ }
+ return this;
+}
+
+void picorsrc_disposeResourceManager(picoos_MemoryManager mm, picorsrc_ResourceManager * this)
+{
+ if (NULL != (*this)) {
+ /* terminate */
+ picoos_deallocate(mm,(void *)this);
+ }
+}
+
+
+/* ******* accessing resources **************************************/
+
+
+static pico_status_t findResource(picorsrc_ResourceManager this, picoos_char * resourceName, picorsrc_Resource * rsrc) {
+ picorsrc_Resource r;
+ if (NULL == this) {
+ return PICO_ERR_NULLPTR_ACCESS;
+ }
+ r = this->resources;
+ while (NULL != r && (0 != picoos_strcmp(r->name,resourceName))) {
+ r = r->next;
+ }
+ *rsrc = r;
+ return PICO_OK;
+}
+
+static picoos_uint8 isResourceLoaded(picorsrc_ResourceManager this, picoos_char * resourceName) {
+ picorsrc_Resource res;
+
+ if (PICO_OK == findResource(this, resourceName,&res)){
+ return (NULL != res);
+ } else {
+ return FALSE;
+ }
+ }
+
+static pico_status_t parse_resource_name(picoos_char * fileName)
+{
+ PICODBG_DEBUG(("analysing file name %s",fileName));
+ if (picoos_has_extension(fileName,
+ (picoos_char *)PICO_BIN_EXTENSION)) {
+ return PICO_OK;
+ } else {
+ return PICO_EXC_UNEXPECTED_FILE_TYPE;
+ }
+}
+
+
+
+static pico_status_t readHeader(picorsrc_ResourceManager this,
+ picoos_FileHeader header, picoos_uint32 * headerlen, picoos_File file)
+{
+
+ picoos_uint16 hdrlen1;
+ picoos_uint32 n;
+ pico_status_t status;
+
+
+ /* read PICO header */
+ status = picoos_readPicoHeader(file, headerlen);
+ if (PICO_OK == status) {
+
+ } else {
+ return picoos_emRaiseException(this->common->em,status,NULL,(picoos_char *)"problem reading file header");
+ }
+ /* read header length (excluding length itself) */
+ status = picoos_read_pi_uint16(file,&hdrlen1);
+ PICODBG_DEBUG(("got header size %d",hdrlen1));
+
+ if (PICO_OK == status) {
+ *headerlen += 2;
+ status = (hdrlen1 <= PICOOS_MAX_HEADER_STRING_LEN-1) ? PICO_OK : PICO_ERR_OTHER;
+ if (PICO_OK == status) {
+ n = hdrlen1;
+ if (picoos_ReadBytes(file, (picoos_uint8 *) this->tmpHeader, &n) && hdrlen1 == n) {
+ this->tmpHeader[hdrlen1] = NULLC;
+ *headerlen += hdrlen1;
+ PICODBG_DEBUG(("got header <%s>",this->tmpHeader));
+
+ status = PICO_OK;
+ } else {
+ status = PICO_ERR_OTHER;
+ }
+ }
+ if (PICO_OK == status) {
+ status = picoos_hdrParseHeader(header, this->tmpHeader);
+ }
+ }
+ return status;
+}
+
+static pico_status_t picorsrc_createKnowledgeBase(
+ picorsrc_ResourceManager this,
+ picoos_uint8 * data,
+ picoos_uint32 size,
+ picoknow_kb_id_t kbid,
+ picoknow_KnowledgeBase * kb)
+{
+ (*kb) = picoknow_newKnowledgeBase(this->common->mm);
+ if (NULL == (*kb)) {
+ return PICO_EXC_OUT_OF_MEM;
+ }
+ (*kb)->base = data;
+ (*kb)->size = size;
+ (*kb)->id = kbid;
+ switch (kbid) {
+ case PICOKNOW_KBID_TPP_MAIN:
+ case PICOKNOW_KBID_TPP_USER_1:
+ case PICOKNOW_KBID_TPP_USER_2:
+ return picokpr_specializePreprocKnowledgeBase(*kb, this->common);
+ break;
+ case PICOKNOW_KBID_TAB_GRAPHS:
+ return picoktab_specializeGraphsKnowledgeBase(*kb, this->common);
+ break;
+ case PICOKNOW_KBID_TAB_PHONES:
+ return picoktab_specializePhonesKnowledgeBase(*kb, this->common);
+ break;
+ case PICOKNOW_KBID_TAB_POS:
+ return picoktab_specializePosKnowledgeBase(*kb, this->common);
+ break;
+ case PICOKNOW_KBID_FIXED_IDS:
+ return picoktab_specializeIdsKnowledgeBase(*kb, this->common);
+ break;
+ case PICOKNOW_KBID_LEX_MAIN:
+ case PICOKNOW_KBID_LEX_USER_1:
+ case PICOKNOW_KBID_LEX_USER_2:
+ return picoklex_specializeLexKnowledgeBase(*kb, this->common);
+ break;
+ case PICOKNOW_KBID_DT_POSP:
+ return picokdt_specializeDtKnowledgeBase(*kb, this->common,
+ PICOKDT_KDTTYPE_POSP);
+ break;
+ case PICOKNOW_KBID_DT_POSD:
+ return picokdt_specializeDtKnowledgeBase(*kb, this->common,
+ PICOKDT_KDTTYPE_POSD);
+ break;
+ case PICOKNOW_KBID_DT_G2P:
+ return picokdt_specializeDtKnowledgeBase(*kb, this->common,
+ PICOKDT_KDTTYPE_G2P);
+ break;
+ case PICOKNOW_KBID_DT_PHR:
+ return picokdt_specializeDtKnowledgeBase(*kb, this->common,
+ PICOKDT_KDTTYPE_PHR);
+ break;
+ case PICOKNOW_KBID_DT_ACC:
+ return picokdt_specializeDtKnowledgeBase(*kb, this->common,
+ PICOKDT_KDTTYPE_ACC);
+ break;
+ case PICOKNOW_KBID_FST_SPHO_1:
+ case PICOKNOW_KBID_FST_SPHO_2:
+ case PICOKNOW_KBID_FST_SPHO_3:
+ case PICOKNOW_KBID_FST_SPHO_4:
+ case PICOKNOW_KBID_FST_SPHO_5:
+ case PICOKNOW_KBID_FST_SPHO_6:
+ case PICOKNOW_KBID_FST_SPHO_7:
+ case PICOKNOW_KBID_FST_SPHO_8:
+ case PICOKNOW_KBID_FST_SPHO_9:
+ case PICOKNOW_KBID_FST_SPHO_10:
+ case PICOKNOW_KBID_FST_WPHO_1:
+ case PICOKNOW_KBID_FST_WPHO_2:
+ case PICOKNOW_KBID_FST_WPHO_3:
+ case PICOKNOW_KBID_FST_WPHO_4:
+ case PICOKNOW_KBID_FST_WPHO_5:
+ case PICOKNOW_KBID_FST_SVOXPA_PARSE:
+ case PICOKNOW_KBID_FST_XSAMPA_PARSE:
+ case PICOKNOW_KBID_FST_XSAMPA2SVOXPA:
+
+ return picokfst_specializeFSTKnowledgeBase(*kb, this->common);
+ break;
+
+ case PICOKNOW_KBID_DT_DUR:
+ case PICOKNOW_KBID_DT_LFZ1:
+ case PICOKNOW_KBID_DT_LFZ2:
+ case PICOKNOW_KBID_DT_LFZ3:
+ case PICOKNOW_KBID_DT_LFZ4:
+ case PICOKNOW_KBID_DT_LFZ5:
+ case PICOKNOW_KBID_DT_MGC1:
+ case PICOKNOW_KBID_DT_MGC2:
+ case PICOKNOW_KBID_DT_MGC3:
+ case PICOKNOW_KBID_DT_MGC4:
+ case PICOKNOW_KBID_DT_MGC5:
+ return picokdt_specializeDtKnowledgeBase(*kb, this->common,
+ PICOKDT_KDTTYPE_PAM);
+ break;
+ case PICOKNOW_KBID_PDF_DUR:
+ return picokpdf_specializePdfKnowledgeBase(*kb, this->common,
+ PICOKPDF_KPDFTYPE_DUR);
+
+ break;
+ case PICOKNOW_KBID_PDF_LFZ:
+ return picokpdf_specializePdfKnowledgeBase(*kb, this->common,
+ PICOKPDF_KPDFTYPE_MUL);
+ break;
+ case PICOKNOW_KBID_PDF_MGC:
+ return picokpdf_specializePdfKnowledgeBase(*kb, this->common,
+ PICOKPDF_KPDFTYPE_MUL);
+ break;
+ case PICOKNOW_KBID_PDF_PHS:
+ return picokpdf_specializePdfKnowledgeBase(*kb, this->common,
+ PICOKPDF_KPDFTYPE_PHS);
+ break;
+
+
+
+#if defined(PICO_DEBUG)
+ case PICOKNOW_KBID_DBG:
+ return picokdbg_specializeDbgKnowledgeBase(*kb, this->common);
+ break;
+#endif
+
+ default:
+ break;
+ }
+ return PICO_OK;
+}
+
+
+static pico_status_t picorsrc_releaseKnowledgeBase(
+ picorsrc_ResourceManager this,
+ picoknow_KnowledgeBase * kb)
+{
+ (*kb) = NULL;
+ return PICO_OK;
+}
+
+static pico_status_t picorsrc_getKbList(picorsrc_ResourceManager this,
+ picoos_uint8 * data,
+ picoos_uint32 datalen,
+ picoknow_KnowledgeBase * kbList)
+{
+
+ pico_status_t status = PICO_OK;
+ picoos_uint32 curpos = 0, offset, size;
+ picoos_uint8 i, numKbs, kbid;
+ picoos_char str[PICOKNOW_MAX_KB_NAME_SIZ];
+ picoknow_KnowledgeBase kb;
+
+ *kbList = NULL;
+ datalen = datalen;
+ /* read number of fields */
+ numKbs = data[curpos++];
+ PICODBG_DEBUG(("number of kbs (unrestricted) = %i",numKbs));
+ status = (numKbs <= PICOKNOW_MAX_NUM_RESOURCE_KBS) ? PICO_OK : PICO_EXC_FILE_CORRUPT;
+ /* read in all kb names */
+ PICODBG_DEBUG(("number of kbs = %i",numKbs));
+ i = 0;
+ while ((PICO_OK == status) && (i++ < numKbs)) {
+ status = (picoos_get_str((picoos_char *)data,&curpos,str,PICOOS_MAX_FIELD_STRING_LEN)) ? PICO_OK : PICO_EXC_FILE_CORRUPT;
+ PICODBG_DEBUG(("contains knowledge base %s (status: %i)",str, status));
+ }
+ /* consume termination of last str */
+ curpos++;
+ i = 0;
+ while ((PICO_OK == status) && (i++ < numKbs)) {
+ kbid = data[curpos++];
+ PICODBG_DEBUG(("got kb id %i, curpos now %i",kbid, curpos));
+ status = picoos_read_mem_pi_uint32(data,&curpos,&offset);
+ PICODBG_DEBUG(("got kb offset %i, curpos now %i",offset, curpos));
+ status = picoos_read_mem_pi_uint32(data,&curpos,&size);
+ PICODBG_DEBUG(("got kb size %i, curpos now %i",size, curpos));
+ if (PICO_OK == status) {
+ if (0 == offset) {
+ /* currently we consider a kb mentioned in resource but with offset 0 (no knowledge) as
+ * different form a kb not mentioned at all. We might reconsider that later. */
+ PICODBG_DEBUG((" kb (id %i) is mentioned but empty (base:%i, size:%i)",kb->id, kb->base, kb->size));
+ status = picorsrc_createKnowledgeBase(this, NULL, size, (picoknow_kb_id_t)kbid, &kb);
+ } else {
+ status = picorsrc_createKnowledgeBase(this, data+offset, size, (picoknow_kb_id_t)kbid, &kb);
+ }
+ PICODBG_DEBUG(("found kb (id %i) starting at %i with size %i",kb->id, kb->base, kb->size));
+ if (PICO_OK == status) {
+ kb->next = *kbList;
+ *kbList = kb;
+ }
+ }
+ }
+ if (PICO_OK != status) {
+ kb = *kbList;
+ while (NULL != kb) {
+ picorsrc_releaseKnowledgeBase(this,&kb);
+ }
+ }
+
+ return status;
+
+}
+
+/* load resource file. the type of resource file etc. are in the header,
+ * then follows the directory, then the knowledge bases themselves (as byte streams) */
+
+pico_status_t picorsrc_loadResource(picorsrc_ResourceManager this,
+ picoos_char * fileName, picorsrc_Resource * resource)
+{
+ picorsrc_Resource res;
+ picoos_uint32 headerlen, len,maxlen;
+ picoos_file_header_t header;
+ picoos_uint8 rem;
+ pico_status_t status = PICO_OK;
+
+ if (resource == NULL) {
+ return PICO_ERR_NULLPTR_ACCESS;
+ } else {
+ *resource = NULL;
+ }
+
+ res = picorsrc_newResource(this->common->mm);
+
+ if (NULL == res) {
+ return picoos_emRaiseException(this->common->em,PICO_EXC_OUT_OF_MEM,NULL,NULL);
+ }
+
+ if (PICO_MAX_NUM_RESOURCES <= this->numResources) {
+ picoos_deallocate(this->common->mm, (void *) &res);
+ return picoos_emRaiseException(this->common->em,PICO_EXC_MAX_NUM_EXCEED,NULL,(picoos_char *)"no more than %i resources",PICO_MAX_NUM_RESOURCES);
+ }
+
+ /* ***************** parse file name for file type and parameters */
+
+ if (PICO_OK != parse_resource_name(fileName)) {
+ picoos_deallocate(this->common->mm, (void *) &res);
+ return PICO_EXC_UNEXPECTED_FILE_TYPE;
+ }
+
+ /* ***************** get header info */
+
+ /* open binary file for reading (no key, nrOfBufs, bufSize) */
+ PICODBG_DEBUG(("trying to open file %s",fileName));
+ if (!picoos_OpenBinary(this->common, &res->file, fileName)) {
+ /* open didn't succeed */
+ status = PICO_EXC_CANT_OPEN_FILE;
+ PICODBG_ERROR(("can't open file %s",fileName));
+ picoos_emRaiseException(this->common->em, PICO_EXC_CANT_OPEN_FILE,
+ NULL, (picoos_char *) "%s", fileName);
+ }
+ if (PICO_OK == status) {
+ status = readHeader(this, &header, &headerlen, res->file);
+ /* res->file now positioned at first pos after header */
+ }
+
+ /* ***************** check header values */
+ if (PICO_OK == status && isResourceLoaded(this, header.field[PICOOS_HEADER_NAME].value)) {
+ /* lingware is allready loaded, do nothing */
+ PICODBG_WARN((">>> lingware '%s' allready loaded",header.field[PICOOS_HEADER_NAME].value));
+ picoos_emRaiseWarning(this->common->em,PICO_WARN_RESOURCE_DOUBLE_LOAD,NULL,(picoos_char *)"%s",header.field[PICOOS_HEADER_NAME].value);
+ status = PICO_WARN_RESOURCE_DOUBLE_LOAD;
+ }
+
+ if (PICO_OK == status) {
+ /* get data length */
+ status = picoos_read_pi_uint32(res->file, &len);
+ PICODBG_DEBUG(("found net resource len of %i",len));
+ /* allocate memory */
+ if (PICO_OK == status) {
+ PICODBG_TRACE((">>> 2"));
+ maxlen = len + PICOOS_ALIGN_SIZE; /* once would be sufficient? */
+ res->raw_mem = picoos_allocProtMem(this->common->mm, maxlen);
+ /* res->size = maxlen; */
+ status = (NULL == res->raw_mem) ? PICO_EXC_OUT_OF_MEM : PICO_OK;
+ }
+ if (PICO_OK == status) {
+ rem = (picoos_uint32) res->raw_mem % PICOOS_ALIGN_SIZE;
+ if (rem > 0) {
+ res->start = res->raw_mem + (PICOOS_ALIGN_SIZE - rem);
+ } else {
+ res->start = res->raw_mem;
+ }
+
+ /* read file contents into memory */
+ status = (picoos_ReadBytes(res->file, res->start, &len)) ? PICO_OK
+ : PICO_ERR_OTHER;
+ /* resources are read-only; the following write protection
+ has an effect in test configurations only */
+ picoos_protectMem(this->common->mm, res->start, len, /*enable*/TRUE);
+ }
+ /* note resource unique name */
+ if (PICO_OK == status) {
+ if (picoos_strlcpy(res->name,header.field[PICOOS_HEADER_NAME].value,PICORSRC_MAX_RSRC_NAME_SIZ) < PICORSRC_MAX_RSRC_NAME_SIZ) {
+ PICODBG_DEBUG(("assigned name %s to resource",res->name));
+ status = PICO_OK;
+ } else {
+ status = PICO_ERR_INDEX_OUT_OF_RANGE;
+ PICODBG_ERROR(("failed assigning name %s to resource",
+ res->name));
+ picoos_emRaiseException(this->common->em,
+ PICO_ERR_INDEX_OUT_OF_RANGE, NULL,
+ (picoos_char *)"resource %s",res->name);
+ }
+ }
+
+ /* get resource type */
+ if (PICO_OK == status) {
+ if (!picoos_strcmp(header.field[PICOOS_HEADER_CONTENT_TYPE].value, PICORSRC_FIELD_VALUE_TEXTANA)) {
+ res->type = PICORSRC_TYPE_TEXTANA;
+ } else if (!picoos_strcmp(header.field[PICOOS_HEADER_CONTENT_TYPE].value, PICORSRC_FIELD_VALUE_SIGGEN)) {
+ res->type = PICORSRC_TYPE_SIGGEN;
+ } else if (!picoos_strcmp(header.field[PICOOS_HEADER_CONTENT_TYPE].value, PICORSRC_FIELD_VALUE_SIGGEN)) {
+ res->type = PICORSRC_TYPE_USER_LEX;
+ } else if (!picoos_strcmp(header.field[PICOOS_HEADER_CONTENT_TYPE].value, PICORSRC_FIELD_VALUE_SIGGEN)) {
+ res->type = PICORSRC_TYPE_USER_PREPROC;
+ } else {
+ res->type = PICORSRC_TYPE_OTHER;
+ }
+ }
+
+ if (PICO_OK == status) {
+ /* create kb list from resource */
+ status = picorsrc_getKbList(this, res->start, len, &res->kbList);
+ }
+ }
+
+ if (status == PICO_OK) {
+ /* add resource to rm */
+ res->next = this->resources;
+ this->resources = res;
+ this->numResources++;
+ *resource = res;
+ PICODBG_DEBUG(("done loading resource %s from %s", res->name, fileName));
+ } else {
+ picorsrc_disposeResource(this->common->mm, &res);
+ PICODBG_ERROR(("failed to load resource"));
+ }
+
+ if (status < 0) {
+ return status;
+ } else {
+ return PICO_OK;
+ }
+}
+
+static pico_status_t picorsrc_releaseKbList(picorsrc_ResourceManager this, picoknow_KnowledgeBase * kbList)
+{
+ picoknow_KnowledgeBase kbprev, kb;
+ kb = *kbList;
+ while (NULL != kb) {
+ kbprev = kb;
+ kb = kb->next;
+ picoknow_disposeKnowledgeBase(this->common->mm,&kbprev);
+ }
+ *kbList = NULL;
+ return PICO_OK;
+}
+
+/* unload resource file. (if resource file is busy, warn and don't unload) */
+pico_status_t picorsrc_unloadResource(picorsrc_ResourceManager this, picorsrc_Resource * resource) {
+
+ picorsrc_Resource r1, r2, rsrc;
+
+ if (resource == NULL) {
+ return PICO_ERR_NULLPTR_ACCESS;
+ } else {
+ rsrc = *resource;
+ }
+
+ if (rsrc->lockCount > 0) {
+ return PICO_EXC_RESOURCE_BUSY;
+ }
+ /* terminate */
+ if (rsrc->file != NULL) {
+ picoos_CloseBinary(this->common, &rsrc->file);
+ }
+ if (NULL != rsrc->raw_mem) {
+ picoos_deallocProtMem(this->common->mm, (void *) &rsrc->raw_mem);
+ PICODBG_DEBUG(("deallocated raw mem"));
+ }
+
+ r1 = NULL;
+ r2 = this->resources;
+ while (r2 != NULL && r2 != rsrc) {
+ r1 = r2;
+ r2 = r2->next;
+ }
+ if (NULL == r1) {
+ this->resources = rsrc->next;
+ } else if (NULL == r2) {
+ /* didn't find resource in rm! */
+ return PICO_ERR_OTHER;
+ } else {
+ r1->next = rsrc->next;
+ }
+
+ if (NULL != rsrc->kbList) {
+ picorsrc_releaseKbList(this, &rsrc->kbList);
+ }
+
+ picoos_deallocate(this->common->mm,(void **)resource);
+ this->numResources--;
+
+ return PICO_OK;
+}
+
+
+pico_status_t picorsrc_createDefaultResource(picorsrc_ResourceManager this
+ /*, picorsrc_Resource * resource */)
+{
+ picorsrc_Resource res;
+ pico_status_t status = PICO_OK;
+
+
+ /* *resource = NULL; */
+
+ if (PICO_MAX_NUM_RESOURCES <= this->numResources) {
+ return picoos_emRaiseException(this->common->em,PICO_EXC_MAX_NUM_EXCEED,NULL,(picoos_char *)"no more than %i resources",PICO_MAX_NUM_RESOURCES);
+ }
+
+ res = picorsrc_newResource(this->common->mm);
+
+ if (NULL == res) {
+ return picoos_emRaiseException(this->common->em,PICO_EXC_OUT_OF_MEM,NULL,NULL);
+ }
+
+ if (picoos_strlcpy(res->name,PICOKNOW_DEFAULT_RESOURCE_NAME,PICORSRC_MAX_RSRC_NAME_SIZ) < PICORSRC_MAX_RSRC_NAME_SIZ) {
+ PICODBG_DEBUG(("assigned name %s to default resource",res->name));
+ status = PICO_OK;
+ } else {
+ PICODBG_ERROR(("failed assigning name %s to default resource",res->name));
+ status = PICO_ERR_INDEX_OUT_OF_RANGE;
+ }
+ status = picorsrc_createKnowledgeBase(this, NULL, 0, (picoknow_kb_id_t)PICOKNOW_KBID_FIXED_IDS, &res->kbList);
+
+ if (PICO_OK == status) {
+ res->next = this->resources;
+ this->resources = res;
+ this->numResources++;
+ /* *resource = res; */
+
+ }
+
+
+ return status;
+
+}
+
+pico_status_t picorsrc_rsrcGetName(picorsrc_Resource this,
+ picoos_char * name, picoos_uint32 maxlen) {
+ if (!picoctrl_isValidResourceHandle(this)) {
+ return PICO_ERR_INVALID_ARGUMENT;
+ }
+ picoos_strlcpy(name, this->name,maxlen);
+ return PICO_OK;
+}
+
+
+/* ******* accessing voice definitions **************************************/
+
+
+static pico_status_t findVoiceDefinition(picorsrc_ResourceManager this,
+ const picoos_char * voiceName, picorsrc_VoiceDefinition * vdef)
+{
+ picorsrc_VoiceDefinition v;
+ PICODBG_DEBUG(("finding voice name %s",voiceName));
+ if (NULL == this) {
+ return PICO_ERR_NULLPTR_ACCESS;
+ }
+ v = this->vdefs;
+ while (NULL != v && (0 != picoos_strcmp(v->voiceName,voiceName))) {
+ PICODBG_DEBUG(("%s doesnt match",v->voiceName));
+ v = v->next;
+ }
+ *vdef = v;
+ if (v == NULL) {
+ PICODBG_DEBUG(("didnt find voice name %s",voiceName));
+ } else {
+ PICODBG_DEBUG(("found voice name %s",voiceName));
+ }
+ return PICO_OK;
+}
+
+
+pico_status_t picorsrc_addResourceToVoiceDefinition(picorsrc_ResourceManager this,
+ picoos_char * voiceName, picoos_char * resourceName)
+{
+ picorsrc_VoiceDefinition vdef;
+
+ if (NULL == this) {
+ PICODBG_ERROR(("this is NULL"));
+ return PICO_ERR_NULLPTR_ACCESS;
+ }
+ if ((PICO_OK == findVoiceDefinition(this,voiceName,&vdef)) && (NULL != vdef)) {
+ if (PICO_MAX_NUM_RSRC_PER_VOICE <= vdef->numResources) {
+ return picoos_emRaiseException(this->common->em,PICO_EXC_MAX_NUM_EXCEED,NULL,(picoos_char *)"no more than %i resources per voice",PICO_MAX_NUM_RSRC_PER_VOICE);
+ }
+ if (picoos_strlcpy(vdef->resourceName[vdef->numResources++], resourceName,
+ PICORSRC_MAX_RSRC_NAME_SIZ) < PICORSRC_MAX_RSRC_NAME_SIZ) {
+ PICODBG_DEBUG(("vdef added resource '%s' to voice '%s'",resourceName,voiceName));
+ return PICO_OK;
+ } else {
+ PICODBG_ERROR(("illegal name (%s)",resourceName));
+ return picoos_emRaiseException(this->common->em,PICO_EXC_NAME_ILLEGAL,NULL,(picoos_char *)"%s",resourceName);
+ }
+
+ } else {
+ return picoos_emRaiseException(this->common->em,PICO_EXC_NAME_UNDEFINED,NULL,(picoos_char *)"%s",voiceName);
+ }
+}
+
+
+pico_status_t picorsrc_createVoiceDefinition(picorsrc_ResourceManager this,
+ picoos_char * voiceName)
+{
+ picorsrc_VoiceDefinition vdef;
+
+ if (NULL == this) {
+ PICODBG_ERROR(("this is NULL"));
+ return PICO_ERR_NULLPTR_ACCESS;
+ }
+ if ((PICO_OK == findVoiceDefinition(this,voiceName,&vdef)) && (NULL != vdef)) {
+ PICODBG_ERROR(("voice %s allready defined",voiceName));
+ return picoos_emRaiseException(this->common->em,PICO_EXC_NAME_CONFLICT,NULL,NULL);
+ }
+ if (PICO_MAX_NUM_VOICE_DEFINITIONS <= this->numVdefs) {
+ PICODBG_ERROR(("max number of vdefs exceeded (%i)",this->numVdefs));
+ return picoos_emRaiseException(this->common->em,PICO_EXC_MAX_NUM_EXCEED,NULL,(picoos_char *)"no more than %i voice definitions",PICO_MAX_NUM_VOICE_DEFINITIONS);
+ }
+ if (NULL == this->freeVdefs) {
+ vdef = picorsrc_newVoiceDefinition(this->common->mm);
+ } else {
+ vdef = this->freeVdefs;
+ this->freeVdefs = vdef->next;
+ vdef->voiceName[0] = NULLC;
+ vdef->numResources = 0;
+ vdef->next = NULL;
+ }
+ if (NULL == vdef) {
+ return picoos_emRaiseException(this->common->em,PICO_EXC_OUT_OF_MEM,NULL,NULL);
+ }
+ if (picoos_strlcpy(vdef->voiceName, voiceName,
+ PICO_MAX_VOICE_NAME_SIZE) < PICO_MAX_VOICE_NAME_SIZE) {
+ vdef->next = this->vdefs;
+ this->vdefs = vdef;
+ this->numVdefs++;
+ if (PICO_OK != picorsrc_addResourceToVoiceDefinition(this,voiceName,PICOKNOW_DEFAULT_RESOURCE_NAME)) {
+ return picoos_emRaiseException(this->common->em,PICO_ERR_OTHER,NULL,(picoos_char *)"problem loading default resource %s",voiceName);
+ }
+ PICODBG_DEBUG(("vdef created (%s)",voiceName));
+ return PICO_OK;
+ } else {
+ PICODBG_ERROR(("illegal name (%s)",voiceName));
+ return picoos_emRaiseException(this->common->em,PICO_EXC_NAME_ILLEGAL,NULL,(picoos_char *)"%s",voiceName);
+ }
+}
+
+
+pico_status_t picorsrc_releaseVoiceDefinition(picorsrc_ResourceManager this,
+ picoos_char *voiceName)
+{
+ picorsrc_VoiceDefinition v, l;
+
+ if (this == NULL) {
+ return PICO_ERR_NULLPTR_ACCESS;
+ }
+
+ l = NULL;
+ v = this->vdefs;
+ while ((v != NULL) && (picoos_strcmp(v->voiceName, voiceName) != 0)) {
+ l = v;
+ v = v->next;
+ }
+ if (v != NULL) {
+ /* remove v from vdefs list */
+ if (l != NULL) {
+ l->next = v->next;
+ } else {
+ this->vdefs = v->next;
+ }
+ /* insert v at head of freeVdefs list */
+ v->next = this->freeVdefs;
+ this->freeVdefs = v;
+ this->numVdefs--;
+ return PICO_OK;
+ } else {
+ /* we should rather return a warning, here */
+ /* return picoos_emRaiseException(this->common->em,PICO_EXC_NAME_UNDEFINED,"%s", NULL); */
+ return PICO_OK;
+ }
+}
+
+
+
+/* ******* accessing voices **************************************/
+
+
+/* create voice, given a voice name. the corresponding lock counts are incremented */
+
+pico_status_t picorsrc_createVoice(picorsrc_ResourceManager this, const picoos_char * voiceName, picorsrc_Voice * voice) {
+
+ picorsrc_VoiceDefinition vdef;
+ picorsrc_Resource rsrc;
+ picoos_uint8 i, required;
+ picoknow_KnowledgeBase kb;
+ /* pico_status_t status = PICO_OK; */
+
+ PICODBG_DEBUG(("creating voice %s",voiceName));
+
+ if (NULL == this) {
+ PICODBG_ERROR(("this is NULL"));
+ return PICO_ERR_NULLPTR_ACCESS;
+
+ }
+ /* check number of voices */
+ if (PICORSRC_MAX_NUM_VOICES <= this->numVoices) {
+ PICODBG_ERROR(("PICORSRC_MAX_NUM_VOICES exceeded"));
+ return picoos_emRaiseException(this->common->em,PICO_EXC_MAX_NUM_EXCEED,NULL,(picoos_char *)"no more than %i voices",PICORSRC_MAX_NUM_VOICES);
+ }
+
+ /* find voice definition for that name */
+ if (!(PICO_OK == findVoiceDefinition(this,voiceName,&vdef)) || (NULL == vdef)) {
+ PICODBG_ERROR(("no voice definition for %s",voiceName));
+ return picoos_emRaiseException(this->common->em,PICO_EXC_NAME_UNDEFINED,NULL,(picoos_char *)"voice definition %s",voiceName);
+
+ }
+ PICODBG_DEBUG(("found voice definition for %s",voiceName));
+
+ /* check that resources are loaded */
+ for (i = 0; i < vdef->numResources; i++) {
+ required = (NULLC != vdef->resourceName[i][0]);
+ if (required && !isResourceLoaded(this,vdef->resourceName[i])) {
+ PICODBG_ERROR(("resource missing"));
+ return picoos_emRaiseException(this->common->em,PICO_EXC_RESOURCE_MISSING,NULL,(picoos_char *)"resource %s for voice %s",vdef->resourceName[i],voiceName);
+ }
+ }
+
+ /* allocate new voice */
+ if (NULL == this->freeVoices) {
+ *voice = picorsrc_newVoice(this->common->mm);
+ } else {
+ *voice = this->freeVoices;
+ this->freeVoices = (*voice)->next;
+ picorsrc_initializeVoice(*voice);
+ }
+ if (*voice == NULL) {
+ return picoos_emRaiseException(this->common->em, PICO_EXC_OUT_OF_MEM, NULL, NULL);
+ }
+ this->numVoices++;
+
+ /* copy resource kb pointers into kb array of voice */
+ for (i = 0; i < vdef->numResources; i++) {
+ required = (NULLC != vdef->resourceName[i][0]);
+ if (required) {
+ findResource(this,vdef->resourceName[i],&rsrc);
+ (*voice)->resourceArray[(*voice)->numResources++] = rsrc;
+ rsrc->lockCount++;
+ kb = rsrc->kbList;
+ while (NULL != kb) {
+ if (NULL != (*voice)->kbArray[kb->id]) {
+ picoos_emRaiseWarning(this->common->em,PICO_WARN_KB_OVERWRITE,NULL, (picoos_char *)"%i", kb->id);
+ PICODBG_WARN(("overwriting knowledge base of id %i", kb->id));
+
+ }
+ PICODBG_DEBUG(("setting knowledge base of id %i", kb->id));
+
+ (*voice)->kbArray[kb->id] = kb;
+ kb = kb->next;
+ }
+ }
+ } /* for */
+
+ return PICO_OK;
+}
+
+/* dispose voice. the corresponding lock counts are decremented. */
+
+pico_status_t picorsrc_releaseVoice(picorsrc_ResourceManager this, picorsrc_Voice * voice)
+{
+ picoos_uint16 i;
+ picorsrc_Voice v = *voice;
+ if (NULL == this || NULL == v) {
+ return PICO_ERR_NULLPTR_ACCESS;
+ }
+ for (i = 0; i < v->numResources; i++) {
+ v->resourceArray[i]->lockCount--;
+ }
+ v->next = this->freeVoices;
+ this->freeVoices = v;
+ this->numVoices--;
+
+ return PICO_OK;
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+/* end picorsrc.c */
diff --git a/lib/picorsrc.h b/lib/picorsrc.h
new file mode 100644
index 0000000..4dfb19e
--- /dev/null
+++ b/lib/picorsrc.h
@@ -0,0 +1,185 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picorsrc.h
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+/**
+ * @addtogroup picorsrc
+ *
+ * <b> Pico Resource Management module </b>\n
+ *
+*/
+
+#ifndef PICORSRC_H_
+#define PICORSRC_H_
+
+#include "picodefs.h"
+#include "picoos.h"
+#include "picoknow.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+#define PICORSRC_MAX_RSRC_NAME_SIZ PICO_MAX_RESOURCE_NAME_SIZE /* including terminating NULLC */
+
+#define PICORSRC_MAX_NUM_VOICES 64
+
+/* size of kb array of a voice */
+#define PICORSRC_KB_ARRAY_SIZE 64
+
+typedef picoos_char picorsrc_resource_name_t[PICORSRC_MAX_RSRC_NAME_SIZ];
+
+typedef enum picorsrc_resource_type {
+ PICORSRC_TYPE_NULL,
+ PICORSRC_TYPE_TEXTANA,
+ PICORSRC_TYPE_SIGGEN,
+ PICORSRC_TYPE_USER_LEX,
+ PICORSRC_TYPE_USER_PREPROC,
+ PICORSRC_TYPE_OTHER
+} picorsrc_resource_type_t;
+
+
+#define PICORSRC_FIELD_VALUE_TEXTANA (picoos_char *) "TEXTANA"
+#define PICORSRC_FIELD_VALUE_SIGGEN (picoos_char *) "SIGGEN"
+#define PICORSRC_FIELD_VALUE_USERLEX (picoos_char *) "USERLEX"
+#define PICORSRC_FIELD_VALUE_USERTPP (picoos_char *) "USERTPP"
+
+
+
+typedef struct picorsrc_resource_manager * picorsrc_ResourceManager;
+typedef struct picorsrc_voice * picorsrc_Voice;
+typedef struct picorsrc_resource * picorsrc_Resource;
+
+
+/* **************************************************************************
+ *
+ * file name extensions
+ *
+ ****************************************************************************/
+
+#define PICO_BIN_EXTENSION ".bin"
+#define PICO_INPLACE_EXTENSION ".inp"
+
+
+
+/* **************************************************************************
+ *
+ * construct/destruct resource manager
+ *
+ ****************************************************************************/
+
+/* create resource manager, given a config file name (or default name, if empty) */
+
+picorsrc_ResourceManager picorsrc_newResourceManager(picoos_MemoryManager mm, picoos_Common common /* , picoos_char * configFile */);
+
+void picorsrc_disposeResourceManager(picoos_MemoryManager mm, picorsrc_ResourceManager * this);
+
+
+/* **************************************************************************
+ *
+ * resources
+ *
+ ****************************************************************************/
+
+/**
+ * Returns non-zero if 'resource' is a valid resource handle, zero otherwise.
+ */
+picoos_int16 picoctrl_isValidResourceHandle(picorsrc_Resource resource);
+
+/* load resource file. the type of resource file, magic numbers, checksum etc. are in the header, then follows the directory
+ * (with fixed structure per resource type), then the knowledge bases themselves (as byte streams) */
+pico_status_t picorsrc_loadResource(picorsrc_ResourceManager this,
+ picoos_char * fileName, picorsrc_Resource * resource);
+
+/* unload resource file. (warn if resource file is busy) */
+pico_status_t picorsrc_unloadResource(picorsrc_ResourceManager this, picorsrc_Resource * rsrc);
+
+
+pico_status_t picorsrc_createDefaultResource(picorsrc_ResourceManager this /*,
+ picorsrc_Resource * resource */);
+
+
+pico_status_t picorsrc_rsrcGetName(picorsrc_Resource resource,
+ picoos_char * name, picoos_uint32 maxlen);
+
+/* **************************************************************************
+ *
+ * voice definitions
+ *
+ ****************************************************************************/
+
+
+pico_status_t picorsrc_createVoiceDefinition(picorsrc_ResourceManager this,
+ picoos_char * voiceName);
+
+
+pico_status_t picorsrc_releaseVoiceDefinition(picorsrc_ResourceManager this,
+ picoos_char * voiceName);
+
+pico_status_t picorsrc_addResourceToVoiceDefinition(picorsrc_ResourceManager this,
+ picoos_char * voiceName, picoos_char * resourceName);
+
+/* **************************************************************************
+ *
+ * voices
+ *
+ ****************************************************************************/
+
+/** object : Voice
+ * shortcut : voice
+ *
+ */
+
+typedef struct picorsrc_voice {
+
+ picorsrc_Voice next;
+
+ picoknow_KnowledgeBase kbArray[PICORSRC_KB_ARRAY_SIZE];
+
+ picoos_uint8 numResources;
+
+ picorsrc_Resource resourceArray[PICO_MAX_NUM_RSRC_PER_VOICE];
+
+
+} picorsrc_voice_t;
+
+
+
+/* create voice, given a voice name. the corresponding lock counts are incremented */
+pico_status_t picorsrc_createVoice(picorsrc_ResourceManager this, const picoos_char * voiceName, picorsrc_Voice * voice);
+
+/* dispose voice. the corresponding lock counts are decremented. */
+pico_status_t picorsrc_releaseVoice(picorsrc_ResourceManager this, picorsrc_Voice * voice);
+
+#ifdef __cplusplus
+}
+#endif
+
+
+
+#endif /*PICORSRC_H_*/
diff --git a/lib/picosa.c b/lib/picosa.c
new file mode 100644
index 0000000..3147c76
--- /dev/null
+++ b/lib/picosa.c
@@ -0,0 +1,1738 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picosa.c
+ *
+ * sentence analysis - POS disambiguation
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#include "picoos.h"
+#include "picodbg.h"
+#include "picobase.h"
+#include "picokdt.h"
+#include "picoklex.h"
+#include "picoktab.h"
+#include "picokfst.h"
+#include "picotrns.h"
+#include "picodata.h"
+#include "picosa.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* PU saStep states */
+#define SA_STEPSTATE_COLLECT 0
+#define SA_STEPSTATE_PROCESS_POSD 10
+#define SA_STEPSTATE_PROCESS_WPHO 11
+#define SA_STEPSTATE_PROCESS_TRNS_PARSE 12
+#define SA_STEPSTATE_PROCESS_TRNS_FST 13
+#define SA_STEPSTATE_FEED 2
+
+#define SA_MAX_ALTDESC_SIZE (30*(PICOTRNS_MAX_NUM_POSSYM + 2))
+
+#define SA_MSGSTR_SIZE 32
+
+/* subobject : SentAnaUnit
+ * shortcut : sa
+ * context size : one phrase, max. 30 non-PUNC items, for non-processed items
+ * one item if internal input empty
+ */
+
+/** @addtogroup picosa
+
+ internal buffers:
+
+ - headx: array for extended item heads of fixed size (head plus
+ index for content, plus two fields for boundary strength/type)
+
+ - cbuf1, cbuf2: buffers for item contents (referenced by index in
+ headx). Future: replace these two buffers by a single double-sided
+ buffer (double shrink-grow type)
+
+ 0. bottom up filling of items in headx and cbuf1
+
+ 1. POS disambiguation (right-to-left, top-to-bottom):
+ - number and sequence of items unchanged
+ - item content can only get smaller (reducing nr of results in WORDINDEX)
+ -> info stays in "headx, cbuf1" and changed in place \n
+ WORDGRAPH(POSes,NA)graph -> WORDGRAPH(POS,NA)graph \n
+ WORDINDEX(POSes,NA)POS1ind1...POSNindN -> WORDINDEX(POS,NA)POS|ind \n
+
+ 2. lex-index lookup and G2P (both directions possible, left-to-right done):
+ - number and sequence of items unchanged, item head info and content
+ changes
+ -> headx changed in place; cbuf1 to cbuf2 \n
+ WORDGRAPH(POS,NA)graph -> WORDPHON(POS,NA)phon \n
+ WORDINDEX(POS,NA)POS|ind -> WORDPHON(POS,NA)phon \n
+
+ 3. phrasing (right-to-left):
+
+ Previous (before introducing SBEG)\n
+ ----------------------------------
+ 1| 2| 3| 4| \n
+ e.g. from WP WP WP WP WP PUNC WP WP PUNC WP WP WP PUNC FLUSH \n
+ e.g. to BINIT WP WP WP BPHR3 WP WP BPHR1 WP WP BSEND WP WP WP BSEND BTERM \n
+ |1 |2 |3 |4 \n
+
+ 3-level bound state: to keep track of bound strength from end of
+ previous punc-phrase, then BOUND item output as first item
+ (strength from prev punc-phrase and type from current
+ punc-phrase).
+
+ trailing PUNC item bound states
+ INIT SEND PHR1
+ PUNC(SENTEND, T) B(I,T)>SEND B(S,T)>SEND B(P1,T)>SEND
+ PUNC(SENTEND, Q) B(I,Q)>SEND B(S,Q)>SEND B(P1,Q)>SEND
+ PUNC(SENTEND, E) B(I,E)>SEND B(S,E)>SEND B(P1,E)>SEND
+ PUNC(PHRASEEND, P) B(I,P)>PHR1 B(S,P)>PHR1 B(P1,P)>PHR1
+ PUNC(PHRASEEND, FORC) B(I,P)>PHR1 B(S,P)>PHR1 B(P1,P)>PHR1
+ PUNC(FLUSH, T) B(I,T).. B(S,T).. B(P1,T)..
+ B(T,NA) B(T,NA) B(T,NA)
+ >INIT >INIT >INIT
+
+ PHR2/3 case:
+ trailing PUNC item bound states
+ INIT SEND PHR1
+ PUNC(SENTEND, T) B(I,P)B(P,T)>SEND B(S,P)B(P,T)>SEND B(P1,P)B(P,T)>SEND
+ PUNC(SENTEND, Q) B(I,P)B(P,Q)>SEND B(S,P)B(P,Q)>SEND B(P1,P)B(P,Q)>SEND
+ PUNC(SENTEND, E) B(I,P)B(P,E)>SEND B(S,P)B(P,E)>SEND B(P1,P)B(P,E)>SEND
+ PUNC(PHRASEEND, P) B(I,P)B(P,P)>PHR1 B(S,P)B(P,P)>PHR1 B(P1,P)B(P,P)>PHR1
+ PUNC(PHREND, FORC) B(I,P)B(P,P)>PHR1 B(S,P)B(P,P)>PHR1 B(P1,P)B(P,P)>PHR1
+ PUNC(FLUSH, T) B(I,P)B(P,T).. B(S,T)B(P,T).. B(P1,T)B(P,T)..
+ B(T,NA) B(T,NA) B(T,NA)
+ >INIT >INIT >INIT
+
+ Current
+ --------
+ e.g. from WP WP WP WP WP PUNC WP WP PUNC WP WP WP PUNC FLUSH
+ e.g. to BSBEG WP WP WP BPHR3 WP WP BPHR1 WP WP BSEND BSBEG WP WP WP BSEND BTERM
+ |1 |2 |3 |4
+
+ 2-level bound state: The internal buffer contains one primary phrase (sometimes forced, if buffer
+ allmost full), with the trailing PUNCT item included (last item).
+ If the trailing PUNC is a a primary phrase separator, the
+ item is not output, but instead, the bound state is set to PPHR, so that the correct BOUND can
+ be output at the start of the next primary phrase.
+ Otherwise,
+ the item is converted to the corresponding BOUND and output. the bound state is set to SSEP,
+ so that a BOUND of type SBEG is output at the start of the next primary phrase.
+
+ trailing PUNC item bound states
+ SSEP PPHR
+ PUNC(SENTEND, X) B(B,X)>SSEP B(P1,X)>SSEP (X = T | Q | E)
+ PUNC(FLUSH, T) B(B,T)>SSEP* B(P1,T)>SSEP
+ PUNC(PHRASEEND, P) B(B,P)>PPHR B(P1,P)>PPHR
+ PUNC(PHRASEEND, FORC) B(B,P)>PPHR B(P1,P)>PPHR
+
+* If more than one sentence separators follow each other (e.g. SEND-FLUSH, SEND-SEND) then
+ all but the first will be treated as an (empty) phrase containing just this item.
+ If this (single) item is a flush, creation of SBEG is suppressed.
+
+
+ - dtphr phrasing tree (rather subphrasing tree it should be called)
+ determines
+ BOUND_PHR2
+ BOUND_PHR3
+ - boundary strenghts are determined for every word (except the
+ first one) from right-to-left. The boundary types mark the phrase
+ type of the phrase following the boundary.
+ - number of items actually changed (new BOUND items added): because
+ of fixed size without content, two fields are contained in headx
+ to indicate if a BOUND needs to be added to the LEFT of the item.
+ -> headx further extended with boundary strength and type info to
+ indicate that to the left of the headx ele a BOUND needs to be
+ inserted when outputting.
+
+ 4. accentuation:
+ - number of items unchanged, content unchanged, only head info changes
+ -> changed in place in headx
+*/
+
+
+typedef struct {
+ picodata_itemhead_t head;
+ picoos_uint16 cind;
+} picosa_headx_t;
+
+
+typedef struct sa_subobj {
+ picoos_uint8 procState; /* for next processing step decision */
+
+ picoos_uint8 inspaceok; /* flag: headx/cbuf1 has space for an item */
+ picoos_uint8 needsmoreitems; /* flag: need more items */
+ picoos_uint8 phonesTransduced; /* flag: */
+
+ picoos_uint8 tmpbuf[PICODATA_MAX_ITEMSIZE]; /* tmp. location for an item */
+
+ picosa_headx_t headx[PICOSA_MAXNR_HEADX];
+ picoos_uint16 headxBottom; /* bottom */
+ picoos_uint16 headxLen; /* length, 0 if empty */
+
+ picoos_uint8 cbuf1[PICOSA_MAXSIZE_CBUF];
+ picoos_uint16 cbuf1BufSize; /* actually allocated size */
+ picoos_uint16 cbuf1Len; /* length, 0 if empty */
+
+ picoos_uint8 cbuf2[PICOSA_MAXSIZE_CBUF];
+ picoos_uint16 cbuf2BufSize; /* actually allocated size */
+ picoos_uint16 cbuf2Len; /* length, 0 if empty */
+
+ picotrns_possym_t phonBufA[PICOTRNS_MAX_NUM_POSSYM+1];
+ picotrns_possym_t phonBufB[PICOTRNS_MAX_NUM_POSSYM+1];
+ picotrns_possym_t * phonBuf;
+ picotrns_possym_t * phonBufOut;
+ picoos_uint16 phonReadPos, phonWritePos; /* next pos to read from phonBufIn, next pos to write to phonBufIn */
+ picoos_uint16 nextReadPos; /* position of (potential) next item to read from */
+
+
+ /* buffer for internal calculation of transducer */
+ picotrns_AltDesc altDescBuf;
+ /* the number of AltDesc in the buffer */
+ picoos_uint16 maxAltDescLen;
+
+ /* tab knowledge base */
+ picoktab_Graphs tabgraphs;
+ picoktab_Phones tabphones;
+ picoktab_Pos tabpos;
+ picoktab_FixedIds fixedIds;
+
+ /* dtposd knowledge base */
+ picokdt_DtPosD dtposd;
+
+ /* dtg2p knowledge base */
+ picokdt_DtG2P dtg2p;
+
+ /* lex knowledge base */
+ picoklex_Lex lex;
+
+ /* ulex knowledge bases */
+ picoos_uint8 numUlex;
+ picoklex_Lex ulex[PICOKNOW_MAX_NUM_ULEX];
+
+ /* fst knowledge bases */
+ picoos_uint8 numFsts;
+ picokfst_FST fst[PICOKNOW_MAX_NUM_WPHO_FSTS];
+ picoos_uint8 curFst; /* the fst to be applied next */
+
+
+} sa_subobj_t;
+
+
+static pico_status_t saInitialize(register picodata_ProcessingUnit this) {
+ sa_subobj_t * sa;
+ picoos_uint16 i;
+ picokfst_FST fst;
+ picoknow_kb_id_t fstKbIds[PICOKNOW_MAX_NUM_WPHO_FSTS] = PICOKNOW_KBID_WPHO_ARRAY;
+ picoklex_Lex ulex;
+ picoknow_kb_id_t ulexKbIds[PICOKNOW_MAX_NUM_ULEX] = PICOKNOW_KBID_ULEX_ARRAY;
+
+ PICODBG_DEBUG(("calling"));
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(this->common->em,
+ PICO_ERR_NULLPTR_ACCESS, NULL, NULL);
+ }
+ sa = (sa_subobj_t *) this->subObj;
+
+ /* sa->common = this->common; */
+
+ sa->procState = SA_STEPSTATE_COLLECT;
+
+ sa->inspaceok = TRUE;
+ sa->needsmoreitems = TRUE;
+
+ sa->headxBottom = 0;
+ sa->headxLen = 0;
+ sa->cbuf1BufSize = PICOSA_MAXSIZE_CBUF;
+ sa->cbuf2BufSize = PICOSA_MAXSIZE_CBUF;
+ sa->cbuf1Len = 0;
+ sa->cbuf2Len = 0;
+
+ /* init headx, cbuf1, cbuf2 */
+ for (i = 0; i < PICOSA_MAXNR_HEADX; i++){
+ sa->headx[i].head.type = 0;
+ sa->headx[i].head.info1 = PICODATA_ITEMINFO1_NA;
+ sa->headx[i].head.info2 = PICODATA_ITEMINFO2_NA;
+ sa->headx[i].head.len = 0;
+ sa->headx[i].cind = 0;
+ }
+ for (i = 0; i < PICOSA_MAXSIZE_CBUF; i++) {
+ sa->cbuf1[i] = 0;
+ sa->cbuf2[i] = 0;
+ }
+
+
+ /* possym buffer */
+ sa->phonesTransduced = FALSE;
+ sa->phonBuf = sa->phonBufA;
+ sa->phonBufOut = sa->phonBufB;
+ sa->phonReadPos = 0;
+ sa->phonWritePos = 0;
+ sa->nextReadPos = 0;
+
+ /* kb fst[] */
+ sa->numFsts = 0;
+ for (i = 0; i<PICOKNOW_MAX_NUM_WPHO_FSTS; i++) {
+ fst = picokfst_getFST(this->voice->kbArray[fstKbIds[i]]);
+ if (NULL != fst) {
+ sa->fst[sa->numFsts++] = fst;
+ }
+ }
+ sa->curFst = 0;
+ PICODBG_DEBUG(("got %i fsts", sa->numFsts));
+ /* kb fixedIds */
+ sa->fixedIds = picoktab_getFixedIds(this->voice->kbArray[PICOKNOW_KBID_FIXED_IDS]);
+
+ /* kb tabgraphs */
+ sa->tabgraphs =
+ picoktab_getGraphs(this->voice->kbArray[PICOKNOW_KBID_TAB_GRAPHS]);
+ if (sa->tabgraphs == NULL) {
+ return picoos_emRaiseException(this->common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ PICODBG_DEBUG(("got tabgraphs"));
+
+ /* kb tabphones */
+ sa->tabphones =
+ picoktab_getPhones(this->voice->kbArray[PICOKNOW_KBID_TAB_PHONES]);
+ if (sa->tabphones == NULL) {
+ return picoos_emRaiseException(this->common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ PICODBG_DEBUG(("got tabphones"));
+
+#ifdef PICO_DEBU
+ {
+ picoos_uint16 itmp;
+ for (itmp = 0; itmp < 256; itmp++) {
+ if (picoktab_hasVowelProp(sa->tabphones, itmp)) {
+ PICODBG_DEBUG(("tabphones hasVowel: %d", itmp));
+ }
+ if (picoktab_hasDiphthProp(sa->tabphones, itmp)) {
+ PICODBG_DEBUG(("tabphones hasDiphth: %d", itmp));
+ }
+ if (picoktab_hasGlottProp(sa->tabphones, itmp)) {
+ PICODBG_DEBUG(("tabphones hasGlott: %d", itmp));
+ }
+ if (picoktab_hasNonsyllvowelProp(sa->tabphones, itmp)) {
+ PICODBG_DEBUG(("tabphones hasNonsyllvowel: %d", itmp));
+ }
+ if (picoktab_hasSyllconsProp(sa->tabphones, itmp)) {
+ PICODBG_DEBUG(("tabphones hasSyllcons: %d", itmp));
+ }
+ if (picoktab_isPrimstress(sa->tabphones, itmp)) {
+ PICODBG_DEBUG(("tabphones isPrimstress: %d", itmp));
+ }
+ if (picoktab_isSecstress(sa->tabphones, itmp)) {
+ PICODBG_DEBUG(("tabphones isSecstress: %d", itmp));
+ }
+ if (picoktab_isSyllbound(sa->tabphones, itmp)) {
+ PICODBG_DEBUG(("tabphones isSyllbound: %d", itmp));
+ }
+ if (picoktab_isPause(sa->tabphones, itmp)) {
+ PICODBG_DEBUG(("tabphones isPause: %d", itmp));
+ }
+ }
+
+ PICODBG_DEBUG(("tabphones primstressID: %d",
+ picoktab_getPrimstressID(sa->tabphones)));
+ PICODBG_DEBUG(("tabphones secstressID: %d",
+ picoktab_getSecstressID(sa->tabphones)));
+ PICODBG_DEBUG(("tabphones syllboundID: %d",
+ picoktab_getSyllboundID(sa->tabphones)));
+ PICODBG_DEBUG(("tabphones pauseID: %d",
+ picoktab_getPauseID(sa->tabphones)));
+ }
+#endif
+
+ /* kb tabpos */
+ sa->tabpos =
+ picoktab_getPos(this->voice->kbArray[PICOKNOW_KBID_TAB_POS]);
+ if (sa->tabpos == NULL) {
+ return picoos_emRaiseException(this->common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ PICODBG_DEBUG(("got tabpos"));
+
+ /* kb dtposd */
+ sa->dtposd = picokdt_getDtPosD(this->voice->kbArray[PICOKNOW_KBID_DT_POSD]);
+ if (sa->dtposd == NULL) {
+ return picoos_emRaiseException(this->common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ PICODBG_DEBUG(("got dtposd"));
+
+ /* kb dtg2p */
+ sa->dtg2p = picokdt_getDtG2P(this->voice->kbArray[PICOKNOW_KBID_DT_G2P]);
+ if (sa->dtg2p == NULL) {
+ return picoos_emRaiseException(this->common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ PICODBG_DEBUG(("got dtg2p"));
+
+ /* kb lex */
+ sa->lex = picoklex_getLex(this->voice->kbArray[PICOKNOW_KBID_LEX_MAIN]);
+ if (sa->lex == NULL) {
+ return picoos_emRaiseException(this->common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ PICODBG_DEBUG(("got lex"));
+
+ /* kb ulex[] */
+ sa->numUlex = 0;
+ for (i = 0; i<PICOKNOW_MAX_NUM_ULEX; i++) {
+ ulex = picoklex_getLex(this->voice->kbArray[ulexKbIds[i]]);
+ if (NULL != ulex) {
+ sa->ulex[sa->numUlex++] = ulex;
+ }
+ }
+ PICODBG_DEBUG(("got %i user lexica", sa->numUlex));
+
+ return PICO_OK;
+}
+
+static picodata_step_result_t saStep(register picodata_ProcessingUnit this,
+ picoos_int16 mode,
+ picoos_uint16 *numBytesOutput);
+
+static pico_status_t saTerminate(register picodata_ProcessingUnit this) {
+ return PICO_OK;
+}
+
+static pico_status_t saSubObjDeallocate(register picodata_ProcessingUnit this,
+ picoos_MemoryManager mm) {
+ sa_subobj_t * sa;
+ if (NULL != this) {
+ sa = (sa_subobj_t *) this->subObj;
+ picotrns_deallocate_alt_desc_buf(mm,&sa->altDescBuf);
+ picoos_deallocate(mm, (void *) &this->subObj);
+ }
+ return PICO_OK;
+}
+
+
+picodata_ProcessingUnit picosa_newSentAnaUnit(picoos_MemoryManager mm,
+ picoos_Common common,
+ picodata_CharBuffer cbIn,
+ picodata_CharBuffer cbOut,
+ picorsrc_Voice voice) {
+ picodata_ProcessingUnit this;
+ sa_subobj_t * sa;
+ this = picodata_newProcessingUnit(mm, common, cbIn, cbOut, voice);
+ if (this == NULL) {
+ return NULL;
+ }
+
+ this->initialize = saInitialize;
+ PICODBG_DEBUG(("set this->step to saStep"));
+ this->step = saStep;
+ this->terminate = saTerminate;
+ this->subDeallocate = saSubObjDeallocate;
+
+ this->subObj = picoos_allocate(mm, sizeof(sa_subobj_t));
+ if (this->subObj == NULL) {
+ picoos_deallocate(mm, (void *)&this);
+ picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM, NULL, NULL);
+ return NULL;
+ }
+
+ sa = (sa_subobj_t *) this->subObj;
+
+ sa->altDescBuf = picotrns_allocate_alt_desc_buf(mm, SA_MAX_ALTDESC_SIZE, &sa->maxAltDescLen);
+ if (NULL == sa->altDescBuf) {
+ picotrns_deallocate_alt_desc_buf(mm,&sa->altDescBuf);
+ picoos_deallocate(mm, (void *)&sa);
+ picoos_deallocate(mm, (void *)&this);
+ picoos_emRaiseException(common->em,PICO_EXC_OUT_OF_MEM, NULL, NULL);
+ }
+
+
+ saInitialize(this);
+ return this;
+}
+
+
+/* ***********************************************************************/
+/* PROCESS_POSD disambiguation functions */
+/* ***********************************************************************/
+
+/* find next POS to the right of 'ind' and return its POS and index */
+static picoos_uint8 saPosDItemSeqGetPosRight(register picodata_ProcessingUnit this,
+ register sa_subobj_t *sa,
+ const picoos_uint16 ind,
+ const picoos_uint16 top,
+ picoos_uint16 *rightind) {
+ picoos_uint8 val;
+ picoos_int32 i;
+
+ val = PICOKDT_EPSILON;
+ for (i = ind + 1; ((val == PICOKDT_EPSILON) && (i < top)); i++) {
+ if ((sa->headx[i].head.type == PICODATA_ITEM_WORDGRAPH) ||
+ (sa->headx[i].head.type == PICODATA_ITEM_WORDINDEX) ||
+ (sa->headx[i].head.type == PICODATA_ITEM_WORDPHON) ) {
+ val = sa->headx[i].head.info1;
+ }
+ }
+ *rightind = i - 1;
+ return val;
+}
+
+
+/* left-to-right, for each WORDGRAPH/WORDINDEX/WORDPHON do posd */
+static pico_status_t saDisambPos(register picodata_ProcessingUnit this,
+ register sa_subobj_t *sa) {
+ picokdt_classify_result_t dtres;
+ picoos_uint8 half_nratt_posd = PICOKDT_NRATT_POSD >> 1;
+ picoos_uint16 valbuf[PICOKDT_NRATT_POSD]; /* only [0..half_nratt_posd] can be >2^8 */
+ picoos_uint16 prevout; /* direct dt output (hist.) or POS of prev word */
+ picoos_uint16 lastprev3; /* last index of POS(es) found to the left */
+ picoos_uint16 curPOS; /* POS(es) of current word */
+ picoos_int32 first; /* index of first item with POS(es) */
+ picoos_int32 ci;
+ picoos_uint8 okay; /* two uses: processing okay and lexind resovled */
+ picoos_uint8 i;
+ picoos_uint16 inval;
+ picoos_uint16 fallback;
+
+ /* set initial values */
+ okay = TRUE;
+ prevout = PICOKDT_HISTORY_ZERO;
+ curPOS = PICODATA_ITEMINFO1_ERR;
+ first = 0;
+
+ while ((first < sa->headxLen) &&
+ (sa->headx[first].head.type != PICODATA_ITEM_WORDGRAPH) &&
+ (sa->headx[first].head.type != PICODATA_ITEM_WORDINDEX) &&
+ (sa->headx[first].head.type != PICODATA_ITEM_WORDPHON)) {
+ first++;
+ }
+ if (first >= sa->headxLen) {
+ /* phrase not containing an item with POSes info, e.g. single flush */
+ PICODBG_DEBUG(("no item with POSes found"));
+ return PICO_OK;
+ }
+
+ lastprev3 = first;
+
+ for (i = 0; i <= half_nratt_posd; i++) {
+ valbuf[i] = PICOKDT_HISTORY_ZERO;
+ }
+ /* set POS(es) of current word, will be shifted afterwards */
+ valbuf[half_nratt_posd+1] = sa->headx[first].head.info1;
+ for (i = half_nratt_posd+2; i < PICOKDT_NRATT_POSD; i++) {
+ /* find next POS to the right and set valbuf[i] */
+ valbuf[i] = saPosDItemSeqGetPosRight(this, sa, lastprev3, sa->headxLen, &lastprev3);
+ }
+
+ PICODBG_TRACE(("headxLen: %d", sa->headxLen));
+
+ /* process from left to right all items in headx */
+ for (ci = first; ci < sa->headxLen; ci++) {
+ okay = TRUE;
+
+ PICODBG_TRACE(("iter: %d, type: %c", ci, sa->headx[ci].head.type));
+
+ /* if not (WORDGRAPH or WORDINDEX) */
+ if ((sa->headx[ci].head.type != PICODATA_ITEM_WORDGRAPH) &&
+ (sa->headx[ci].head.type != PICODATA_ITEM_WORDINDEX) &&
+ (sa->headx[ci].head.type != PICODATA_ITEM_WORDPHON)) {
+ continue;
+ }
+
+ PICODBG_TRACE(("iter: %d, curPOS: %d", ci, sa->headx[ci].head.info1));
+
+ /* no continue so far => at [ci] we have a WORDGRAPH / WORDINDEX item */
+ /* shift all elements one position to the left */
+ /* shift predicted values (history) */
+ for (i=1; i<half_nratt_posd; i++) {
+ valbuf[i-1] = valbuf[i];
+ }
+ /* insert previously predicted value (now history) */
+ valbuf[half_nratt_posd-1] = prevout;
+ /* shift not yet predicted values */
+ for (i=half_nratt_posd+1; i<PICOKDT_NRATT_POSD; i++) {
+ valbuf[i-1] = valbuf[i];
+ }
+ /* find next POS to the right and set valbuf[PICOKDT_NRATT_POSD-1] */
+ valbuf[PICOKDT_NRATT_POSD-1] = saPosDItemSeqGetPosRight(this, sa, lastprev3, sa->headxLen, &lastprev3);
+
+ /* just to be on the safe side; the following should never happen */
+ if (sa->headx[ci].head.info1 != valbuf[half_nratt_posd]) {
+ PICODBG_WARN(("syncing POS"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_INVECTOR,
+ NULL, NULL);
+ valbuf[half_nratt_posd] = sa->headx[ci].head.info1;
+ }
+
+ curPOS = valbuf[half_nratt_posd];
+
+ /* Check if POS disambiguation not needed */
+ if (picoktab_isUniquePos(sa->tabpos, (picoos_uint8) curPOS)) {
+ /* not needed */
+ inval = 0;
+ fallback = 0;
+ if (!picokdt_dtPosDreverseMapOutFixed(sa->dtposd, curPOS,
+ &prevout, &fallback)) {
+ if (fallback) {
+ prevout = fallback;
+
+ } else {
+ PICODBG_ERROR(("problem doing reverse output mapping"));
+ prevout = curPOS;
+ }
+ }
+ PICODBG_DEBUG(("keeping: %d", sa->headx[ci].head.info1));
+ continue;
+ }
+
+ /* assuming PICOKDT_NRATT_POSD == 7 */
+ PICODBG_DEBUG(("%d: [%d %d %d %d %d %d %d]",
+ ci, valbuf[0], valbuf[1], valbuf[2],
+ valbuf[3], valbuf[4], valbuf[5], valbuf[6]));
+
+ /* no continue so far => POS disambiguation needed */
+ /* construct input vector, which is set in dtposd */
+ if (!picokdt_dtPosDconstructInVec(sa->dtposd, valbuf)) {
+ /* error constructing invec */
+ PICODBG_WARN(("problem with invec"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_INVECTOR,
+ NULL, NULL);
+ okay = FALSE;
+ }
+ /* classify */
+ if (okay && (!picokdt_dtPosDclassify(sa->dtposd, &prevout))) {
+ /* error doing classification */
+ PICODBG_WARN(("problem classifying"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_CLASSIFICATION,
+ NULL, NULL);
+ okay = FALSE;
+ }
+ /* decompose */
+ if (okay && (!picokdt_dtPosDdecomposeOutClass(sa->dtposd, &dtres))) {
+ /* error decomposing */
+ PICODBG_WARN(("problem decomposing"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_OUTVECTOR,
+ NULL, NULL);
+ okay = FALSE;
+ }
+ if (okay && dtres.set) {
+ PICODBG_DEBUG(("in: %d, out: %d", valbuf[3], dtres.class));
+ } else {
+ PICODBG_WARN(("problem disambiguating POS"));
+ dtres.class = PICODATA_ITEMINFO1_ERR;
+ }
+
+ if (dtres.class > 255) {
+ PICODBG_WARN(("dt result outside valid range, setting pos to ERR"));
+ dtres.class = PICODATA_ITEMINFO1_ERR;
+ }
+
+ sa->headx[ci].head.info1 = (picoos_uint8)dtres.class;
+ if (sa->headx[ci].head.type == PICODATA_ITEM_WORDINDEX) {
+ /* find pos/ind entry in cbuf matching unique,
+ disambiguated POS, adapt current headx cind/len
+ accordingly */
+ PICODBG_DEBUG(("select phon based on POS disambiguation"));
+ okay = FALSE;
+ for (i = 0; i < sa->headx[ci].head.len; i += PICOKLEX_POSIND_SIZE) {
+ PICODBG_DEBUG(("comparing POS at cind + %d", i));
+ if (picoktab_isPartOfPosGroup(sa->tabpos,
+ (picoos_uint8)dtres.class,
+ sa->cbuf1[sa->headx[ci].cind + i])) {
+ PICODBG_DEBUG(("found match for entry %d",
+ i/PICOKLEX_POSIND_SIZE + 1));
+ sa->headx[ci].cind += i;
+ okay = TRUE;
+ break;
+ }
+ }
+ /* not finding a match is possible if posd predicts a POS that
+ is not part of any of the input POSes -> no warning */
+#if defined(PICO_DEBUG)
+ if (!okay) {
+ PICODBG_DEBUG(("no match found, selecting 1st entry"));
+ }
+#endif
+ sa->headx[ci].head.len = PICOKLEX_POSIND_SIZE;
+ }
+ }
+ return PICO_OK;
+}
+
+
+/* ***********************************************************************/
+/* PROCESS_WPHO functions, copy, lexindex, and g2p */
+/* ***********************************************************************/
+
+/* ************** copy ***************/
+
+static pico_status_t saCopyItemContent1to2(register picodata_ProcessingUnit this,
+ register sa_subobj_t *sa,
+ picoos_uint16 ind) {
+ picoos_uint16 i;
+ picoos_uint16 cind1;
+
+ /* set headx.cind, and copy content, head unchanged */
+ cind1 = sa->headx[ind].cind;
+ sa->headx[ind].cind = sa->cbuf2Len;
+
+ /* check cbufLen */
+ if (sa->headx[ind].head.len > (sa->cbuf2BufSize - sa->cbuf2Len)) {
+ sa->headx[ind].head.len = sa->cbuf2BufSize - sa->cbuf2Len;
+ PICODBG_WARN(("phones skipped"));
+ picoos_emRaiseWarning(this->common->em,
+ PICO_WARN_INCOMPLETE, NULL, NULL);
+ if (sa->headx[ind].head.len == 0) {
+ sa->headx[ind].cind = 0;
+ }
+ }
+
+ for (i = 0; i < sa->headx[ind].head.len; i++) {
+ sa->cbuf2[sa->cbuf2Len] = sa->cbuf1[cind1 + i];
+ sa->cbuf2Len++;
+ }
+
+ PICODBG_DEBUG(("%c item, len: %d",
+ sa->headx[ind].head.type, sa->headx[ind].head.len));
+
+ return PICO_OK;
+}
+
+
+/* ************** lexindex ***************/
+
+static pico_status_t saLexIndLookup(register picodata_ProcessingUnit this,
+ register sa_subobj_t *sa,
+ picoklex_Lex lex,
+ picoos_uint16 ind) {
+ picoos_uint8 pos;
+ picoos_uint8 *phones;
+ picoos_uint8 plen;
+ picoos_uint16 i;
+
+ if (picoklex_lexIndLookup(lex, &(sa->cbuf1[sa->headx[ind].cind + 1]),
+ PICOKLEX_IND_SIZE, &pos, &phones, &plen)) {
+ sa->headx[ind].cind = sa->cbuf2Len;
+
+ /* check cbufLen */
+ if (plen > (sa->cbuf2BufSize - sa->cbuf2Len)) {
+ plen = sa->cbuf2BufSize - sa->cbuf2Len;
+ PICODBG_WARN(("phones skipped"));
+ picoos_emRaiseWarning(this->common->em,
+ PICO_WARN_INCOMPLETE, NULL, NULL);
+ if (plen == 0) {
+ sa->headx[ind].cind = 0;
+ }
+ }
+
+ /* set item head, info1, info2 unchanged */
+ sa->headx[ind].head.type = PICODATA_ITEM_WORDPHON;
+ sa->headx[ind].head.len = plen;
+
+ for (i = 0; i < plen; i++) {
+ sa->cbuf2[sa->cbuf2Len] = phones[i];
+ sa->cbuf2Len++;
+ }
+
+ PICODBG_DEBUG(("%c item, pos: %d, plen: %d",
+ PICODATA_ITEM_WORDPHON, pos, plen));
+
+ } else {
+ PICODBG_WARN(("lexIndLookup problem"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_PU_IRREG_ITEM,
+ NULL, NULL);
+ }
+ return PICO_OK;
+}
+
+
+
+/* ************** g2p ***************/
+
+
+/* Name : saGetNvowel
+ Function: returns vowel info in a word or word seq
+ Input : sInChar the grapheme string to be converted in phoneme
+ inLen number of bytes in grapheme buffer
+ inPos start position of current grapheme (0..inLen-1)
+ Output : nVow number of vowels in the word
+ nVord vowel order in the word
+ Returns : TRUE: processing successful; FALSE: errors
+*/
+static picoos_uint8 saGetNrVowel(register picodata_ProcessingUnit this,
+ register sa_subobj_t *sa,
+ const picoos_uint8 *sInChar,
+ const picoos_uint16 inLen,
+ const picoos_uint8 inPos,
+ picoos_uint8 *nVow,
+ picoos_uint8 *nVord) {
+ picoos_uint32 nCount;
+ picoos_uint32 pos;
+ picoos_uint8 cstr[PICOBASE_UTF8_MAXLEN + 1];
+
+ /*defaults*/
+ *nVow = 0;
+ *nVord = 0;
+ /*1:check wether the current char is a vowel*/
+ pos = inPos;
+ if (!picobase_get_next_utf8char(sInChar, inLen, &pos, cstr) ||
+ !picoktab_hasVowellikeProp(sa->tabgraphs, cstr, PICOBASE_UTF8_MAXLEN)) {
+ return FALSE;
+ }
+ /*2:count number of vowels in current word and find vowel order*/
+ for (nCount = 0; nCount < inLen; ) {
+ if (!picobase_get_next_utf8char(sInChar, inLen, &nCount, cstr)) {
+ return FALSE;
+ }
+ if (picoktab_hasVowellikeProp(sa->tabgraphs, cstr,
+ PICOBASE_UTF8_MAXLEN)) {
+ (*nVow)++;
+ if (nCount == pos) {
+ (*nVord) = (*nVow);
+ }
+ }
+ }
+ return TRUE;
+}
+
+
+/* do g2p for a full word, right-to-left */
+static picoos_uint8 saDoG2P(register picodata_ProcessingUnit this,
+ register sa_subobj_t *sa,
+ const picoos_uint8 *graph,
+ const picoos_uint8 graphlen,
+ const picoos_uint8 pos,
+ picoos_uint8 *phones,
+ const picoos_uint16 phonesmaxlen,
+ picoos_uint16 *plen) {
+ picoos_uint16 outNp1Ch; /*last 3 outputs produced*/
+ picoos_uint16 outNp2Ch;
+ picoos_uint16 outNp3Ch;
+ picoos_uint8 nPrimary;
+ picoos_uint8 nCount;
+ picoos_uint32 utfpos;
+ picoos_uint16 nOutVal;
+ picoos_uint8 okay;
+ picoos_uint16 phonesind;
+ picoos_uint8 nrvow;
+ picoos_uint8 ordvow;
+ picokdt_classify_vecresult_t dtresv;
+ picoos_uint16 i;
+
+ *plen = 0;
+ okay = TRUE;
+
+ /* use sa->tmpbuf[PICOSA_MAXITEMSIZE] to temporarly store the
+ phones which are predicted in reverse order. Once all are
+ available put them in phones in usuable order. phonesind is
+ used to fille item in reverse order starting at the end of
+ tmpbuf. */
+ phonesind = PICOSA_MAXITEMSIZE - 1;
+
+ /* prepare the data for loop operations */
+ outNp1Ch = PICOKDT_HISTORY_ZERO;
+ outNp2Ch = PICOKDT_HISTORY_ZERO;
+ outNp3Ch = PICOKDT_HISTORY_ZERO;
+
+ /* inner loop */
+ nPrimary = 0;
+
+ /* ************************************************/
+ /* go backward grapheme by grapheme, it's utf8... */
+ /* ************************************************/
+
+ /* set start nCount to position of start of last utfchar */
+ /* ! watch out! somethimes starting at 1, sometimes at 0,
+ ! sometimes counting per byte, sometimes per UTF8 char */
+ /* nCount is (start position + 1) of utf8 char */
+ utfpos = graphlen;
+ if (picobase_get_prev_utf8charpos(graph, 0, &utfpos)) {
+ nCount = utfpos + 1;
+ } else {
+ /* should not occurr */
+ PICODBG_ERROR(("invalid utf8 string, graphlen: %d", graphlen));
+ return FALSE;
+ }
+
+ while (nCount > 0) {
+ PICODBG_TRACE(("right-to-left g2p, count: %d", nCount));
+ okay = TRUE;
+
+ if (!saGetNrVowel(this, sa, graph, graphlen, nCount-1, &nrvow,
+ &ordvow)) {
+ nrvow = 0;
+ ordvow = 0;
+ }
+
+ /* prepare input vector, set inside tree object invec,
+ * g2pBuildVector will call the constructInVec tree method */
+ if (!picokdt_dtG2PconstructInVec(sa->dtg2p,
+ graph, /*grapheme start*/
+ graphlen, /*grapheme length*/
+ nCount-1, /*grapheme current position*/
+ pos, /*Word POS*/
+ nrvow, /*nr vowels if vowel, 0 else */
+ ordvow, /*ord of vowel if vowel, 0 el*/
+ &nPrimary, /*primary stress flag*/
+ outNp1Ch, /*Right phoneme context +1*/
+ outNp2Ch, /*Right phoneme context +2*/
+ outNp3Ch)) { /*Right phon context +3*/
+ /*Errors in preparing the input vector : skip processing*/
+ PICODBG_WARN(("problem with invec"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_INVECTOR,
+ NULL, NULL);
+ okay = FALSE;
+ }
+
+ /* classify using the invec in the tree object and save the direct
+ tree output also in the tree object */
+ if (okay && (!picokdt_dtG2Pclassify(sa->dtg2p, &nOutVal))) {
+ /* error doing classification */
+ PICODBG_WARN(("problem classifying"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_CLASSIFICATION,
+ NULL, NULL);
+ okay = FALSE;
+ }
+
+ /* decompose the invec in the tree object and return result in dtresv */
+ if (okay && (!picokdt_dtG2PdecomposeOutClass(sa->dtg2p, &dtresv))) {
+ /* error decomposing */
+ PICODBG_WARN(("problem decomposing"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_OUTVECTOR,
+ NULL, NULL);
+ okay = FALSE;
+ }
+
+ if (okay) {
+ if ((dtresv.nr == 0) || (dtresv.classvec[0] == PICOKDT_EPSILON)) {
+ /* no phones to be added */
+ PICODBG_TRACE(("epsilon, no phone added %c", graph[nCount-1]));
+ ;
+ } else {
+ /* add decomposed output to tmpbuf, reverse order */
+ for (i = dtresv.nr; ((((PICOSA_MAXITEMSIZE - 1) -
+ phonesind)<phonesmaxlen) &&
+ (i > 0)); ) {
+ i--;
+ PICODBG_TRACE(("%c %d",graph[nCount-1],dtresv.classvec[i]));
+ if (dtresv.classvec[i] > 255) {
+ PICODBG_WARN(("dt result outside valid range, "
+ "skipping phone"));
+ continue;
+ }
+ sa->tmpbuf[phonesind--] = (picoos_uint8)dtresv.classvec[i];
+ if (!nPrimary) {
+ if (picoktab_isPrimstress(sa->tabphones,
+ (picoos_uint8)dtresv.classvec[i])) {
+ nPrimary = 1;
+ }
+ }
+ (*plen)++;
+ }
+ if (i > 0) {
+ PICODBG_WARN(("phones skipped"));
+ picoos_emRaiseWarning(this->common->em,
+ PICO_WARN_INCOMPLETE, NULL, NULL);
+ }
+ }
+ }
+
+ /*shift tree output history and update*/
+ outNp3Ch = outNp2Ch;
+ outNp2Ch = outNp1Ch;
+ outNp1Ch = nOutVal;
+
+ /* go backward one utf8 char */
+ /* nCount is in +1 domain */
+ if (nCount <= 1) {
+ /* end of str */
+ nCount = 0;
+ } else {
+ utfpos = nCount - 1;
+ if (!picobase_get_prev_utf8charpos(graph, 0, &utfpos)) {
+ /* should not occur */
+ PICODBG_ERROR(("invalid utf8 string, utfpos: %d", utfpos));
+ return FALSE;
+ } else {
+ nCount = utfpos + 1;
+ }
+ }
+ }
+
+ /* a must be: (PICOSA_MAXITEMSIZE-1) - phonesind == *plen */
+ /* now that we have all phone IDs, copy in correct order to phones */
+ /* phonesind point to next free slot in the reverse domainn,
+ ie. inc first */
+ phonesind++;
+ for (i = 0; i < *plen; i++, phonesind++) {
+ phones[i] = sa->tmpbuf[phonesind];
+ }
+ return TRUE;
+}
+
+
+/* item in headx[ind]/cbuf1, out: modified headx and cbuf2 */
+
+static pico_status_t saGraphemeToPhoneme(register picodata_ProcessingUnit this,
+ register sa_subobj_t *sa,
+ picoos_uint16 ind) {
+ picoos_uint16 plen;
+
+ PICODBG_TRACE(("starting g2p"));
+
+ if (saDoG2P(this, sa, &(sa->cbuf1[sa->headx[ind].cind]),
+ sa->headx[ind].head.len, sa->headx[ind].head.info1,
+ &(sa->cbuf2[sa->cbuf2Len]), (sa->cbuf2BufSize - sa->cbuf2Len),
+ &plen)) {
+
+ /* check of cbuf2Len done in saDoG2P, phones skipped if needed */
+ if (plen > 255) {
+ PICODBG_WARN(("maximum number of phones exceeded (%d), skipping",
+ plen));
+ plen = 255;
+ }
+
+ /* set item head, info1, info2 unchanged */
+ sa->headx[ind].head.type = PICODATA_ITEM_WORDPHON;
+ sa->headx[ind].head.len = (picoos_uint8)plen;
+ sa->headx[ind].cind = sa->cbuf2Len;
+ sa->cbuf2Len += plen;
+ PICODBG_DEBUG(("%c item, plen: %d",
+ PICODATA_ITEM_WORDPHON, plen));
+ } else {
+ PICODBG_WARN(("problem doing g2p"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_PU_IRREG_ITEM,
+ NULL, NULL);
+ }
+ return PICO_OK;
+}
+
+
+/* ***********************************************************************/
+/* extract phonemes of an item into a phonBuf */
+/* ***********************************************************************/
+
+static pico_status_t saAddPhoneme(register sa_subobj_t *sa, picoos_uint16 pos, picoos_uint16 sym) {
+ /* picoos_uint8 plane, unshifted; */
+
+ /* just for debuging */
+ /*
+ unshifted = picotrns_unplane(sym,&plane);
+ PICODBG_DEBUG(("adding %i/%i (%c on plane %i) at phonBuf[%i]",pos,sym,unshifted,plane,sa->phonWritePos));
+ */
+ if (PICOTRNS_MAX_NUM_POSSYM <= sa->phonWritePos) {
+ /* not an error! */
+ PICODBG_DEBUG(("couldn't add because phon buffer full"));
+ return PICO_EXC_BUF_OVERFLOW;
+ } else {
+ sa->phonBuf[sa->phonWritePos].pos = pos;
+ sa->phonBuf[sa->phonWritePos].sym = sym;
+ sa->phonWritePos++;
+ return PICO_OK;
+ }
+}
+
+/*
+static pico_status_t saAddStartPhoneme(register sa_subobj_t *sa) {
+ return saAddPhoneme(sa, PICOTRNS_POS_IGNORE,
+ (PICOKFST_PLANE_INTERN << 8) + sa->fixedIds->phonStartId);
+}
+
+
+static pico_status_t saAddTermPhoneme(register sa_subobj_t *sa) {
+ return saAddPhoneme(sa, PICOTRNS_POS_IGNORE,
+ (PICOKFST_PLANE_INTERN << 8) + sa->fixedIds->phonTermId);
+}
+
+*/
+
+static pico_status_t saExtractPhonemes(register picodata_ProcessingUnit this,
+ register sa_subobj_t *sa, picoos_uint16 pos,
+ picodata_itemhead_t* head, const picoos_uint8* content)
+{
+ pico_status_t rv= PICO_OK;
+ picoos_uint8 i;
+ picoos_int16 fstSymbol;
+#if defined(PICO_DEBUG)
+ picoos_char msgstr[SA_MSGSTR_SIZE];
+#endif
+
+ PICODBG_TRACE(("doing item %s",
+ picodata_head_to_string(head,msgstr,SA_MSGSTR_SIZE)));
+ /*
+ Items considered in a transduction are WORDPHON item. its starting offset within the inBuf is given as
+ 'pos'.
+ Elements that go into the transduction receive "their" position in the buffer.
+ */
+ sa->phonWritePos = 0;
+ /* WORDPHON(POS,WACC)phon */
+ rv = saAddPhoneme(sa, PICOTRNS_POS_IGNORE,
+ (PICOKFST_PLANE_INTERN << 8) + sa->fixedIds->phonStartId);
+ for (i = 0; i < head->len; i++) {
+ fstSymbol = /* (PICOKFST_PLANE_PHONEMES << 8) + */content[i];
+ /* */
+ PICODBG_TRACE(("adding phoneme %c",fstSymbol));
+ rv = saAddPhoneme(sa, pos+PICODATA_ITEM_HEADSIZE+i, fstSymbol);
+ }
+ rv = saAddPhoneme(sa, PICOTRNS_POS_IGNORE,
+ (PICOKFST_PLANE_INTERN << 8) + sa->fixedIds->phonTermId);
+ sa->nextReadPos = pos + PICODATA_ITEM_HEADSIZE + head->len;
+ return rv;
+}
+
+
+#define SA_POSSYM_OK 0
+#define SA_POSSYM_OUT_OF_RANGE 1
+#define SA_POSSYM_END 2
+#define SA_POSSYM_INVALID -3
+/* *readPos is the next position in phonBuf to be read, and *writePos is the first position not to be read (may be outside
+ * buf).
+ * 'rangeEnd' is the first possym position outside the desired range.
+ * Possible return values:
+ * SA_POSSYM_OK : 'pos' and 'sym' are set to the read possym, *readPos is advanced
+ * SA_POSSYM_OUT_OF_RANGE : pos is out of range. 'pos' is set to that of the read possym, 'sym' is undefined
+ * SA_POSSYM_UNDERFLOW : no more data in buf. 'pos' is set to PICOTRNS_POS_INVALID, 'sym' is undefined
+ * SA_POSSYM_INVALID : "strange" pos. 'pos' is set to PICOTRNS_POS_INVALID, 'sym' is undefined
+ */
+static pico_status_t getNextPosSym(sa_subobj_t * sa, picoos_int16 * pos, picoos_int16 * sym,
+ picoos_int16 rangeEnd) {
+ /* skip POS_IGNORE */
+ while ((sa->phonReadPos < sa->phonWritePos) && (PICOTRNS_POS_IGNORE == sa->phonBuf[sa->phonReadPos].pos)) {
+ PICODBG_DEBUG(("ignoring phone at sa->phonBuf[%i] because it has pos==IGNORE",sa->phonReadPos));
+ sa->phonReadPos++;
+ }
+ if ((sa->phonReadPos < sa->phonWritePos)) {
+ *pos = sa->phonBuf[sa->phonReadPos].pos;
+ if ((PICOTRNS_POS_INSERT == *pos) || ((0 <= *pos) && (*pos < rangeEnd))) {
+ *sym = sa->phonBuf[sa->phonReadPos++].sym;
+ return SA_POSSYM_OK;
+ } else if (*pos < 0){ /* *pos is "strange" (e.g. POS_INVALID) */
+ return SA_POSSYM_INVALID;
+ } else {
+ return SA_POSSYM_OUT_OF_RANGE;
+ }
+ } else {
+ /* no more possyms to read */
+ *pos = PICOTRNS_POS_INVALID;
+ return SA_POSSYM_END;
+ }
+}
+
+
+
+
+/* ***********************************************************************/
+/* saStep function */
+/* ***********************************************************************/
+
+/*
+complete phrase processed in one step, if not fast enough -> rework
+
+init, collect into internal buffer, process, and then feed to
+output buffer
+
+init state: INIT ext ext
+state trans: in hc1 hc2 out
+
+INIT | putItem = 0 0 +1 | BUSY -> COLL (put B-SBEG item,
+ set do-init to false)
+
+ inspace-ok-hc1
+ needs-more-items-(phrase-or-flush)
+COLL1 |getItems -n +n 0 1 | ATOMIC -> PPOSD (got items,
+ if flush set do-init)
+COLL2 |getItems -n +n 1 0 | ATOMIC -> PPOSD (got items, forced)
+COLL3 |getItems -n +n 1 1 | IDLE (got items, need more)
+COLL4 |getItems = = 1 1 | IDLE (got no items)
+
+PPOSD | posd = ~n~n | BUSY -> PWP (posd done)
+PWP | lex/g2p = ~n-n 0+n | BUSY -> PPHR (lex/g2p done)
+PPHR | phr = -n 0 +m=n | BUSY -> PACC (phr done, m>=n)
+PACC | acc = 0 0 ~m=n | BUSY -> FEED (acc done)
+
+ doinit-flag
+FEED | putItems 0 0 0 -m-n +m 0 | BUSY -> COLL (put items)
+FEED | putItems 0 0 0 -m-n +m 1 | BUSY -> INIT (put items)
+FEED | putItems 0 0 0 -d-d +d | OUT_FULL (put some items)
+*/
+
+static picodata_step_result_t saStep(register picodata_ProcessingUnit this,
+ picoos_int16 mode,
+ picoos_uint16 *numBytesOutput) {
+ register sa_subobj_t *sa;
+ pico_status_t rv = PICO_OK;
+ pico_status_t rvP = PICO_OK;
+ picoos_uint16 blen = 0;
+ picoos_uint16 clen = 0;
+ picoos_uint16 i;
+ picoklex_Lex lex;
+
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ sa = (sa_subobj_t *) this->subObj;
+ mode = mode; /* avoid warning "var not used in this function"*/
+ *numBytesOutput = 0;
+ while (1) { /* exit via return */
+ PICODBG_DEBUG(("doing state %i, hLen|c1Len|c2Len: %d|%d|%d",
+ sa->procState, sa->headxLen, sa->cbuf1Len,
+ sa->cbuf2Len));
+
+ switch (sa->procState) {
+
+ /* *********************************************************/
+ /* collect state: get item(s) from charBuf and store in
+ * internal buffers, need a complete punctuation-phrase
+ */
+ case SA_STEPSTATE_COLLECT:
+
+ while (sa->inspaceok && sa->needsmoreitems
+ && (PICO_OK ==
+ (rv = picodata_cbGetItem(this->cbIn, sa->tmpbuf,
+ PICOSA_MAXITEMSIZE, &blen)))) {
+ rvP = picodata_get_itemparts(sa->tmpbuf,
+ PICOSA_MAXITEMSIZE,
+ &(sa->headx[sa->headxLen].head),
+ &(sa->cbuf1[sa->cbuf1Len]),
+ sa->cbuf1BufSize-sa->cbuf1Len,
+ &clen);
+ if (rvP != PICO_OK) {
+ PICODBG_ERROR(("problem getting item parts"));
+ picoos_emRaiseException(this->common->em, rvP,
+ NULL, NULL);
+ return PICODATA_PU_ERROR;
+ }
+
+ /* if CMD(...FLUSH...) -> PUNC(...FLUSH...),
+ construct PUNC-FLUSH item in headx */
+ if ((sa->headx[sa->headxLen].head.type ==
+ PICODATA_ITEM_CMD) &&
+ (sa->headx[sa->headxLen].head.info1 ==
+ PICODATA_ITEMINFO1_CMD_FLUSH)) {
+ sa->headx[sa->headxLen].head.type =
+ PICODATA_ITEM_PUNC;
+ sa->headx[sa->headxLen].head.info1 =
+ PICODATA_ITEMINFO1_PUNC_FLUSH;
+ sa->headx[sa->headxLen].head.info2 =
+ PICODATA_ITEMINFO2_PUNC_SENT_T;
+ sa->headx[sa->headxLen].head.len = 0;
+ }
+
+ /* convert opening phoneme command to WORDPHON
+ * and assign user-POS XX to it (Bug 432) */
+ sa->headx[sa->headxLen].cind = sa->cbuf1Len;
+ /* maybe overwritten later */
+ if ((sa->headx[sa->headxLen].head.type ==
+ PICODATA_ITEM_CMD) &&
+ (sa->headx[sa->headxLen].head.info1 ==
+ PICODATA_ITEMINFO1_CMD_PHONEME)&&
+ (sa->headx[sa->headxLen].head.info2 ==
+ PICODATA_ITEMINFO2_CMD_START)) {
+ picoos_uint8 i;
+ picoos_uint8 wordsep = picoktab_getWordboundID(sa->tabphones);
+ PICODBG_INFO(("wordsep id is %i",wordsep));
+ sa->headx[sa->headxLen].head.type = PICODATA_ITEM_WORDPHON;
+ sa->headx[sa->headxLen].head.info1 = PICODATA_POS_XX;
+ sa->headx[sa->headxLen].head.info2 = PICODATA_ITEMINFO2_NA;
+ /* cut off additional words */
+ i = 0;
+ while ((i < sa->headx[sa->headxLen].head.len) && (wordsep != sa->cbuf1[sa->headx[sa->headxLen].cind+i])) {
+ PICODBG_INFO(("accepting phoneme %i",sa->cbuf1[sa->headx[sa->headxLen].cind+i]));
+
+ i++;
+ }
+ if (i < sa->headx[sa->headxLen].head.len) {
+ PICODBG_INFO(("cutting off superfluous phonetic words at %i",i));
+ sa->headx[sa->headxLen].head.len = i;
+ }
+ }
+
+ /* check/set needsmoreitems */
+ if (sa->headx[sa->headxLen].head.type ==
+ PICODATA_ITEM_PUNC) {
+ sa->needsmoreitems = FALSE;
+ }
+
+ /* check/set inspaceok, keep spare slot for forcing */
+ if ((sa->headxLen >= (PICOSA_MAXNR_HEADX - 2)) ||
+ ((sa->cbuf1BufSize - sa->cbuf1Len) <
+ PICOSA_MAXITEMSIZE)) {
+ sa->inspaceok = FALSE;
+ }
+
+ if (clen > 0) {
+ sa->headx[sa->headxLen].cind = sa->cbuf1Len;
+ sa->cbuf1Len += clen;
+ } else {
+ sa->headx[sa->headxLen].cind = 0;
+ }
+ sa->headxLen++;
+ }
+
+ if (!sa->needsmoreitems) {
+ /* 1, phrase buffered */
+ sa->procState = SA_STEPSTATE_PROCESS_POSD;
+ return PICODATA_PU_ATOMIC;
+ } else if (!sa->inspaceok) {
+ /* 2, forced phrase end */
+ /* at least one slot is still free, use it to
+ force a trailing PUNC item */
+ sa->headx[sa->headxLen].head.type = PICODATA_ITEM_PUNC;
+ sa->headx[sa->headxLen].head.info1 =
+ PICODATA_ITEMINFO1_PUNC_PHRASEEND;
+ sa->headx[sa->headxLen].head.info2 =
+ PICODATA_ITEMINFO2_PUNC_PHRASE_FORCED;
+ sa->headx[sa->headxLen].head.len = 0;
+ sa->needsmoreitems = FALSE; /* not really needed for now */
+ sa->headxLen++;
+ PICODBG_WARN(("forcing phrase end, added PUNC_PHRASEEND"));
+ picoos_emRaiseWarning(this->common->em,
+ PICO_WARN_FALLBACK, NULL,
+ (picoos_char *)"forced phrase end");
+ sa->procState = SA_STEPSTATE_PROCESS_POSD;
+ return PICODATA_PU_ATOMIC;
+ } else if (rv == PICO_EOF) {
+ /* 3, 4 */
+ return PICODATA_PU_IDLE;
+ } else if ((rv == PICO_EXC_BUF_UNDERFLOW) ||
+ (rv == PICO_EXC_BUF_OVERFLOW)) {
+ /* error, no valid item in cb (UNDER) */
+ /* or tmpbuf not large enough, not possible (OVER) */
+ /* no exception raised, left for ctrl to handle */
+ PICODBG_ERROR(("buffer under/overflow, rv: %d", rv));
+ return PICODATA_PU_ERROR;
+ } else {
+ /* error, only possible if cbGetItem implementation
+ changes without this function being adapted*/
+ PICODBG_ERROR(("untreated return value, rv: %d", rv));
+ return PICODATA_PU_ERROR;
+ }
+ break;
+
+
+ /* *********************************************************/
+ /* process posd state: process items in headx/cbuf1
+ * and change in place
+ */
+ case SA_STEPSTATE_PROCESS_POSD:
+ /* ensure there is an item in inBuf */
+ if (sa->headxLen > 0) {
+ /* we have a phrase in headx, cbuf1 (can be
+ single PUNC item without POS), do pos disamb */
+ if (PICO_OK != saDisambPos(this, sa)) {
+ picoos_emRaiseException(this->common->em,
+ PICO_ERR_OTHER, NULL, NULL);
+ return PICODATA_PU_ERROR;
+ }
+ sa->procState = SA_STEPSTATE_PROCESS_WPHO;
+
+ } else if (sa->headxLen == 0) { /* no items in inBuf */
+ PICODBG_WARN(("no items in inBuf"));
+ sa->procState = SA_STEPSTATE_COLLECT;
+ return PICODATA_PU_BUSY;
+ }
+
+#if defined (PICO_DEBUG)
+ if (1) {
+ picoos_uint8 i, j, ittype;
+ for (i = 0; i < sa->headxLen; i++) {
+ ittype = sa->headx[i].head.type;
+ PICODBG_INFO_CTX();
+ PICODBG_INFO_MSG(("sa-d: ("));
+ PICODBG_INFO_MSG(("'%c',", ittype));
+ if ((32 <= sa->headx[i].head.info1) &&
+ (sa->headx[i].head.info1 < 127) &&
+ (ittype != PICODATA_ITEM_WORDGRAPH) &&
+ (ittype != PICODATA_ITEM_WORDINDEX)) {
+ PICODBG_INFO_MSG(("'%c',",sa->headx[i].head.info1));
+ } else {
+ PICODBG_INFO_MSG(("%3d,", sa->headx[i].head.info1));
+ }
+ if ((32 <= sa->headx[i].head.info2) &&
+ (sa->headx[i].head.info2 < 127)) {
+ PICODBG_INFO_MSG(("'%c',",sa->headx[i].head.info2));
+ } else {
+ PICODBG_INFO_MSG(("%3d,", sa->headx[i].head.info2));
+ }
+ PICODBG_INFO_MSG(("%3d)", sa->headx[i].head.len));
+
+ for (j = 0; j < sa->headx[i].head.len; j++) {
+ if ((ittype == PICODATA_ITEM_WORDGRAPH) ||
+ (ittype == PICODATA_ITEM_CMD)) {
+ PICODBG_INFO_MSG(("%c",
+ sa->cbuf1[sa->headx[i].cind+j]));
+ } else {
+ PICODBG_INFO_MSG(("%4d",
+ sa->cbuf1[sa->headx[i].cind+j]));
+ }
+ }
+ PICODBG_INFO_MSG(("\n"));
+ }
+ }
+#endif
+
+ break;
+
+
+ /* *********************************************************/
+ /* process wpho state: process items in headx/cbuf1 and modify
+ * headx in place and fill cbuf2
+ */
+ case SA_STEPSTATE_PROCESS_WPHO:
+ /* ensure there is an item in inBuf */
+ if (sa->headxLen > 0) {
+ /* we have a phrase in headx, cbuf1 (can be single
+ PUNC item), do lex lookup, g2p, or copy */
+
+ /* check if cbuf2 is empty as it should be */
+ if (sa->cbuf2Len > 0) {
+ /* enforce emptyness */
+ PICODBG_WARN(("forcing empty cbuf2, discarding buf"));
+ picoos_emRaiseWarning(this->common->em,
+ PICO_WARN_PU_DISCARD_BUF,
+ NULL, NULL);
+ }
+
+ /* cbuf2 overflow avoided in saGrapheme*, saLexInd*,
+ saCopyItem*, phones skipped if needed */
+ for (i = 0; i < sa->headxLen; i++) {
+ switch (sa->headx[i].head.type) {
+ case PICODATA_ITEM_WORDGRAPH:
+ if (PICO_OK != saGraphemeToPhoneme(this, sa,
+ i)) {
+ /* not possible, phones skipped if needed */
+ picoos_emRaiseException(this->common->em,
+ PICO_ERR_OTHER,
+ NULL, NULL);
+ return PICODATA_PU_ERROR;
+ }
+ break;
+ case PICODATA_ITEM_WORDINDEX:
+ if (0 == sa->headx[i].head.info2) {
+ lex = sa->lex;
+ } else {
+ lex = sa->ulex[sa->headx[i].head.info2-1];
+ }
+ if (PICO_OK != saLexIndLookup(this, sa, lex, i)) {
+ /* not possible, phones skipped if needed */
+ picoos_emRaiseException(this->common->em,
+ PICO_ERR_OTHER,
+ NULL, NULL);
+ return PICODATA_PU_ERROR;
+ }
+ break;
+ default:
+ /* copy item unmodified, ie. headx untouched,
+ content from cbuf1 to cbuf2 */
+ if (PICO_OK != saCopyItemContent1to2(this, sa,
+ i)) {
+ /* not possible, phones skipped if needed */
+ picoos_emRaiseException(this->common->em,
+ PICO_ERR_OTHER,
+ NULL, NULL);
+ return PICODATA_PU_ERROR;
+ }
+ break;
+ }
+ }
+ /* set cbuf1 to empty */
+ sa->cbuf1Len = 0;
+ sa->procState = SA_STEPSTATE_PROCESS_TRNS_PARSE;
+
+ } else if (sa->headxLen == 0) { /* no items in inBuf */
+ PICODBG_WARN(("no items in inBuf"));
+ sa->procState = SA_STEPSTATE_COLLECT;
+ return PICODATA_PU_BUSY;
+ }
+
+#if defined (PICO_DEBUG)
+ if (1) {
+ picoos_uint8 i, j, ittype;
+ for (i = 0; i < sa->headxLen; i++) {
+ ittype = sa->headx[i].head.type;
+ PICODBG_INFO_CTX();
+ PICODBG_INFO_MSG(("sa-g: ("));
+ PICODBG_INFO_MSG(("'%c',", ittype));
+ if ((32 <= sa->headx[i].head.info1) &&
+ (sa->headx[i].head.info1 < 127) &&
+ (ittype != PICODATA_ITEM_WORDPHON)) {
+ PICODBG_INFO_MSG(("'%c',",sa->headx[i].head.info1));
+ } else {
+ PICODBG_INFO_MSG(("%3d,", sa->headx[i].head.info1));
+ }
+ if ((32 <= sa->headx[i].head.info2) &&
+ (sa->headx[i].head.info2 < 127)) {
+ PICODBG_INFO_MSG(("'%c',",sa->headx[i].head.info2));
+ } else {
+ PICODBG_INFO_MSG(("%3d,", sa->headx[i].head.info2));
+ }
+ PICODBG_INFO_MSG(("%3d)", sa->headx[i].head.len));
+
+ for (j = 0; j < sa->headx[i].head.len; j++) {
+ if ((ittype == PICODATA_ITEM_CMD)) {
+ PICODBG_INFO_MSG(("%c",
+ sa->cbuf2[sa->headx[i].cind+j]));
+ } else {
+ PICODBG_INFO_MSG(("%4d",
+ sa->cbuf2[sa->headx[i].cind+j]));
+ }
+ }
+ PICODBG_INFO_MSG(("\n"));
+ }
+ }
+#endif
+
+ break;
+
+
+ /* *********************************************************/
+ /* transduction parse state: extract phonemes of item in internal outBuf */
+ case SA_STEPSTATE_PROCESS_TRNS_PARSE:
+
+ PICODBG_DEBUG(("transduce item (bot, remain): (%d, %d)",
+ sa->headxBottom, sa->headxLen));
+
+ /* check for termination condition first */
+ if (0 == sa->headxLen) {
+ /* reset headx, cbuf2 */
+ sa->headxBottom = 0;
+ sa->cbuf2Len = 0;
+ /* reset collect state support variables */
+ sa->inspaceok = TRUE;
+ sa->needsmoreitems = TRUE;
+
+ sa->procState = SA_STEPSTATE_COLLECT;
+ return PICODATA_PU_BUSY;
+ }
+
+ sa->procState = SA_STEPSTATE_FEED;
+ /* copy item unmodified */
+ rv = picodata_put_itemparts(
+ &(sa->headx[sa->headxBottom].head),
+ &(sa->cbuf2[sa->headx[sa->headxBottom].cind]),
+ sa->headx[sa->headxBottom].head.len, sa->tmpbuf,
+ PICOSA_MAXITEMSIZE, &blen);
+
+ if (PICODATA_ITEM_WORDPHON == sa->headx[sa->headxBottom].head.type) {
+ PICODBG_DEBUG(("PARSE found WORDPHON"));
+ rv = saExtractPhonemes(this, sa, 0, &(sa->headx[sa->headxBottom].head),
+ &(sa->cbuf2[sa->headx[sa->headxBottom].cind]));
+ if (PICO_OK == rv) {
+ PICODBG_DEBUG(("PARSE successfully returned from phoneme extraction"));
+ sa->procState = SA_STEPSTATE_PROCESS_TRNS_FST;
+ } else {
+ PICODBG_WARN(("PARSE phone extraction returned exception %i, output WORDPHON untransduced",rv));
+ }
+ } else {
+ PICODBG_DEBUG(("PARSE found other item, just copying"));
+ }
+ if (SA_STEPSTATE_FEED == sa->procState) {
+ PICODATA_INFO_ITEM(this->voice->kbArray[PICOKNOW_KBID_DBG],
+ (picoos_uint8 *)"sa-p: ",
+ sa->tmpbuf, PICOSA_MAXITEMSIZE);
+
+ }
+
+ /* consume item */
+ sa->headxBottom++;
+ sa->headxLen--;
+
+ break;
+
+ /* *********************************************************/
+ /* transduce state: copy item in internal outBuf to tmpBuf and transduce */
+ case SA_STEPSTATE_PROCESS_TRNS_FST:
+
+
+
+
+
+ /* if no word-level FSTs: doing trivial syllabification instead */
+ if (0 == sa->numFsts) {
+ PICODBG_DEBUG(("doing trivial sylabification with %i phones", sa->phonWritePos));
+#if defined(PICO_DEBUG)
+ {
+ PICODBG_INFO_CTX();
+ PICODBG_INFO_MSG(("sa trying to trivially syllabify: "));
+ PICOTRNS_PRINTSYMSEQ(this->voice->kbArray[PICOKNOW_KBID_DBG], sa->phonBuf, sa->phonWritePos);
+ PICODBG_INFO_MSG(("\n"));
+ }
+#endif
+
+ picotrns_trivial_syllabify(sa->tabphones, sa->phonBuf,
+ sa->phonWritePos, sa->phonBufOut,
+ &sa->phonWritePos,PICOTRNS_MAX_NUM_POSSYM);
+ PICODBG_DEBUG(("returned from trivial sylabification with %i phones", sa->phonWritePos));
+#if defined(PICO_DEBUG)
+ {
+ PICODBG_INFO_CTX();
+ PICODBG_INFO_MSG(("sa returned from syllabification: "));
+ PICOTRNS_PRINTSYMSEQ(this->voice->kbArray[PICOKNOW_KBID_DBG], sa->phonBufOut, sa->phonWritePos);
+ PICODBG_INFO_MSG(("\n"));
+ }
+#endif
+
+ /* eliminate deep epsilons */
+ PICODBG_DEBUG(("doing epsilon elimination with %i phones", sa->phonWritePos));
+ picotrns_eliminate_epsilons(sa->phonBufOut,
+ sa->phonWritePos, sa->phonBuf,
+ &sa->phonWritePos,PICOTRNS_MAX_NUM_POSSYM);
+ PICODBG_DEBUG(("returning from epsilon elimination with %i phones", sa->phonWritePos));
+ sa->phonReadPos = 0;
+ sa->phonesTransduced = 1;
+ sa->procState = SA_STEPSTATE_FEED;
+ break;
+ }
+
+ /* there are word-level FSTs */
+ /* termination condition first */
+ if (sa->curFst >= sa->numFsts) {
+ /* reset for next transduction */
+ sa->curFst = 0;
+ sa->phonReadPos = 0;
+ sa->phonesTransduced = 1;
+ sa->procState = SA_STEPSTATE_FEED;
+ break;
+ }
+
+ /* transduce from phonBufIn to PhonBufOut */
+ {
+
+ picoos_uint32 nrSteps;
+#if defined(PICO_DEBUG)
+ {
+ PICODBG_INFO_CTX();
+ PICODBG_INFO_MSG(("sa trying to transduce: "));
+ PICOTRNS_PRINTSYMSEQ(this->voice->kbArray[PICOKNOW_KBID_DBG], sa->phonBuf, sa->phonWritePos);
+ PICODBG_INFO_MSG(("\n"));
+ }
+#endif
+ picotrns_transduce(sa->fst[sa->curFst], FALSE,
+ picotrns_printSolution, sa->phonBuf, sa->phonWritePos, sa->phonBufOut,
+ &sa->phonWritePos,
+ PICOTRNS_MAX_NUM_POSSYM, sa->altDescBuf,
+ sa->maxAltDescLen, &nrSteps);
+#if defined(PICO_DEBUG)
+ {
+ PICODBG_INFO_CTX();
+ PICODBG_INFO_MSG(("sa returned from transduction: "));
+ PICOTRNS_PRINTSYMSEQ(this->voice->kbArray[PICOKNOW_KBID_DBG], sa->phonBufOut, sa->phonWritePos);
+ PICODBG_INFO_MSG(("\n"));
+ }
+#endif
+ }
+
+
+
+ /*
+ The trasduction output will contain equivalent items i.e. (x,y') for each (x,y) plus inserted deep symbols (-1,d).
+ In case of deletions, (x,0) might also be omitted...
+ */
+ /* eliminate deep epsilons */
+ picotrns_eliminate_epsilons(sa->phonBufOut,
+ sa->phonWritePos, sa->phonBuf, &sa->phonWritePos,PICOTRNS_MAX_NUM_POSSYM);
+ sa->phonesTransduced = 1;
+
+ sa->curFst++;
+
+ return PICODATA_PU_ATOMIC;
+ /* break; */
+
+ /* *********************************************************/
+ /* feed state: copy item in internal outBuf to output charBuf */
+
+ case SA_STEPSTATE_FEED:
+
+ PICODBG_DEBUG(("FEED"));
+
+ if (sa->phonesTransduced) {
+ /* replace original phones by transduced */
+ picoos_uint16 phonWritePos = PICODATA_ITEM_HEADSIZE;
+ picoos_uint8 plane;
+ picoos_int16 sym, pos;
+ while (SA_POSSYM_OK == (rv = getNextPosSym(sa,&pos,&sym,sa->nextReadPos))) {
+ PICODBG_TRACE(("FEED inserting phoneme %c into inBuf[%i]",sym,phonWritePos));
+ sym = picotrns_unplane(sym, &plane);
+ PICODBG_ASSERT((PICOKFST_PLANE_PHONEMES == plane));
+ sa->tmpbuf[phonWritePos++] = (picoos_uint8) sym;
+ }
+ PICODBG_DEBUG(("FEED setting item length to %i",phonWritePos - PICODATA_ITEM_HEADSIZE));
+ picodata_set_itemlen(sa->tmpbuf,PICODATA_ITEM_HEADSIZE,phonWritePos - PICODATA_ITEM_HEADSIZE);
+ if (SA_POSSYM_INVALID == rv) {
+ PICODBG_ERROR(("FEED unexpected symbol or unexpected end of phoneme list"));
+ return (picodata_step_result_t)picoos_emRaiseException(this->common->em, PICO_WARN_INCOMPLETE, NULL, NULL);
+ }
+ sa->phonesTransduced = 0;
+
+ } /* if (sa->phonesTransduced) */
+
+
+ rvP = picodata_cbPutItem(this->cbOut, sa->tmpbuf,
+ PICOSA_MAXITEMSIZE, &clen);
+
+ *numBytesOutput += clen;
+
+ PICODBG_DEBUG(("put item, status: %d", rvP));
+
+ if (rvP == PICO_OK) {
+ } else if (rvP == PICO_EXC_BUF_OVERFLOW) {
+ /* try again next time */
+ PICODBG_DEBUG(("feeding overflow"));
+ return PICODATA_PU_OUT_FULL;
+ } else {
+ /* error, should never happen */
+ PICODBG_ERROR(("untreated return value, rvP: %d", rvP));
+ return PICODATA_PU_ERROR;
+ }
+
+ PICODATA_INFO_ITEM(this->voice->kbArray[PICOKNOW_KBID_DBG],
+ (picoos_uint8 *)"sana: ",
+ sa->tmpbuf, PICOSA_MAXITEMSIZE);
+
+ sa->procState = SA_STEPSTATE_PROCESS_TRNS_PARSE;
+ /* return PICODATA_PU_BUSY; */
+ break;
+
+ default:
+ break;
+ } /* switch */
+
+ } /* while */
+
+ /* should be never reached */
+ PICODBG_ERROR(("reached end of function"));
+ picoos_emRaiseException(this->common->em, PICO_ERR_OTHER, NULL, NULL);
+ return PICODATA_PU_ERROR;
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+
+/* end */
diff --git a/lib/picosa.h b/lib/picosa.h
new file mode 100644
index 0000000..c35da2e
--- /dev/null
+++ b/lib/picosa.h
@@ -0,0 +1,216 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picosa.h
+ *
+ * sentence analysis - POS disambiguation - Include file
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+
+/** @addtogroup picosa
+ * ---------------------------------------------------\n
+ * <b> Pico Sentence analysis </b>\n
+ * ---------------------------------------------------\n
+ *
+itemtype, iteminfo1, iteminfo2, content -> TYPE(INFO1,INFO2)content
+in the following
+
+items input
+===========
+
+processed by sa (POS disambiguation):
+- WORDGRAPH(POSes,NA)graph
+- WORDINDEX(POSes,NA)POS|1ind1...POSN|indN
+- CMD(PICODATA_ITEMINFO1_CMD_FLUSH,PICODATA_ITEMINFO2_NA)
+
+processed by sa (Phrasing, Accentuation):
+- PUNC(PUNCtype,PUNCsubtype)
+
+unprocessed:
+- all other item types are forwarded through the PU without modification:
+ CMD
+
+
+minimal input size (before processing starts)
+==================
+
+processing (POS disambiguation, g2p, lexind, phrasing, accentuation)
+is possible with
+
+- one punctuation-phrase, consisting of a sequence (see below for
+ limits) of items terminated by a PUNC item.
+
+(possible but not implemented: as long as the internal buffer is
+empty, non-processed item types can be processed immediately)
+
+Ensuring terminal PUNC item:
+- when reading items from the external buffer a CMD(...FLUSH...) is
+ converted to a PUNC(...FLUSH...) item
+- If needed, a PUNC(PHRASE) is artificially added to ensure a phrase
+ fits in the PUs memory and processing can start.
+
+
+items processed and output
+==========================
+
+precondition:
+CMD(...FLUSH...) already converted to PUNC(...FLUSH...) and trailing
+PUNC item enforced if necessary.
+
+----
+1. PROCESS_POSD: processing input WORDGRAPH or WORDINDEX items, after
+POS disambiguation (POSes -> POS), results in a sequence of:
+
+-> WORDGRAPH(POS,NA)graph
+-> WORDINDEX(POS,NA)POS|ind
+
+----
+2. PROCESS_WPHO: then, after lex-index lookup and G2P in a
+sequence of:
+
+-> WORDPHON(POS,NA)phon
+
+(phon containing primary and secondary word-level stress)
+
+----
+3. PROCESS_PHR: then, after processing these WORDPHON items,
+together with the trailing PUNC item results in:
+
+-> BOUND(BOUNDstrength,BOUNDtype)
+
+being added in the sequence of WORDPHON (respectively inserted instead
+of the PUNC). All PUNC, incl PUNC(...FLUSH...) now gone.
+
+----
+4. PROCESS_ACC: then, after processing the WORDPHON and BOUND items
+results in:
+
+-> WORDPHON(POS,ACC)phon
+
+A postprocessing step of accentuation is hard-coded in the
+accentuation module: In case the whole word does not have any stress
+at all (primary or secondary or both) then do the following mapping:
+
+ ACC0 nostress -> ACC0
+ ACC1 nostress -> ACC3
+ ACC2 nostress -> ACC3
+ ACC3 nostress -> ACC3
+
+----
+- POS
+ a single, unambiguous POS
+
+cf. picodata.h for
+- ACC (sentence-level accent (aka prominence)) %d
+ PICODATA_ACC0
+ PICODATA_ACC1
+ PICODATA_ACC2 (<- maybe mapped to ACC1, ie. no ACC2 in output)
+ PICODATA_ACC3
+
+- BOUNDstrength %d
+ PICODATA_ITEMINFO1_BOUND_SBEG (at sentence start)
+ PICODATA_ITEMINFO1_BOUND_SEND (at sentence end)
+ PICODATA_ITEMINFO1_BOUND_TERM (replaces a flush)
+ PICODATA_ITEMINFO1_BOUND_PHR1 (primary boundary)
+ PICODATA_ITEMINFO1_BOUND_PHR2 (short break)
+ PICODATA_ITEMINFO1_BOUND_PHR3 (secondary phrase boundary, no break)
+ PICODATA_ITEMINFO1_BOUND_PHR0 (no break, not produced by sa, not existing
+ BOUND in item sequence equals PHR0 bound strength)
+
+- BOUNDtype (created in sa base on punctuation, indicates type of phrase
+ following the boundary) %d
+ PICODATA_ITEMINFO2_BOUNDTYPE_P
+ PICODATA_ITEMINFO2_BOUNDTYPE_T
+ PICODATA_ITEMINFO2_BOUNDTYPE_Q
+ PICODATA_ITEMINFO2_BOUNDTYPE_E
+
+
+output sequence (without CMDs):
+
+<output> = { BOUND(BOUND_SBEG,PHRASEtype) <sentence> BOUND(BOUND_SEND,..)} BOUND(BOUND_TERM,..)
+
+
+<sentence> = <phrase> { BOUND(BOUND_PHR1|2|3,BOUNDtype) <phrase> }
+
+<phrase> = WORDPHON(POS,ACC)phon { WORDPHON(POS,ACC)phon }
+
+
+Done in later PU: mapping ACC & word-level stress to syllable accent value
+ ACC0 prim -> 0
+ ACC1 prim -> 1
+ ACC2 prim -> 2
+ ACC3 prim -> 3
+ ACC0 sec -> 0
+ ACC1 sec -> 4
+ ACC2 sec -> 4
+ ACC3 sec -> 4
+
+
+other limitations
+=================
+
+- item size: header plus len=256 (valid for Pico in general)
+- see defines below for max nr of items. Item heads plus ref. to contents
+ buffer are stored in array with fixed size elements. Two restrictions:
+ - MAXNR_HEADX (max nr elements==items in headx array)
+ - CONTENTSSIZE (max size of all contents together
+ */
+
+
+#ifndef PICOSA_H_
+#define PICOSA_H_
+
+#include "picoos.h"
+#include "picodata.h"
+#include "picorsrc.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* nr item restriction: maximum number of extended item heads in headx */
+#define PICOSA_MAXNR_HEADX 60
+/* nr item restriction: maximum size of all item contents together in cont */
+#define PICOSA_MAXSIZE_CBUF 7680
+
+/* maximum length of an item incl. head for input GetItem buffer */
+#define PICOSA_MAXITEMSIZE 260
+
+
+picodata_ProcessingUnit picosa_newSentAnaUnit(
+ picoos_MemoryManager mm,
+ picoos_Common common,
+ picodata_CharBuffer cbIn,
+ picodata_CharBuffer cbOut,
+ picorsrc_Voice voice);
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif /*PICOSA_H_*/
diff --git a/lib/picosig.c b/lib/picosig.c
new file mode 100644
index 0000000..be44b3c
--- /dev/null
+++ b/lib/picosig.c
@@ -0,0 +1,1280 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picosig.c
+ *
+ * Signal Generation PU - Implementation
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#include "picoos.h"
+#include "picodsp.h"
+#include "picosig2.h"
+#include "picodata.h"
+#include "picosig.h"
+#include "picodbg.h"
+#include "picokpdf.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+#define PICOSIG_IN_BUFF_SIZE PICODATA_BUFSIZE_SIG /*input buffer size for SIG */
+#define PICOSIG_OUT_BUFF_SIZE PICODATA_BUFSIZE_SIG /*output buffer size for SIG*/
+
+#define PICOSIG_COLLECT 0
+#define PICOSIG_SCHEDULE 1
+#define PICOSIG_PLAY 2
+#define PICOSIG_PROCESS 3
+#define PICOSIG_FEED 4
+
+/*----------------------------------------------------------
+ // Internal function declarations
+ //---------------------------------------------------------*/
+
+static picodata_step_result_t sigStep(register picodata_ProcessingUnit this,
+ picoos_int16 mode, picoos_uint16 * numBytesOutput);
+
+/*----------------------------------------------------------
+ // Name : sig_subobj
+ // Function: subobject definition for the sig processing
+ // Shortcut: sig
+ //---------------------------------------------------------*/
+typedef struct sig_subobj
+{
+ /*----------------------PU voice management------------------------------*/
+ /* picorsrc_Voice voice; */
+ /*----------------------PU state management------------------------------*/
+ picoos_uint8 procState; /* where to take up work at next processing step */
+ picoos_uint8 retState; /* where to return after next processing step */
+ picoos_uint8 needMoreInput; /* more data necessary to start processing */
+ /*----------------------PU input management------------------------------*/
+ picoos_uint8 inBuf[PICOSIG_IN_BUFF_SIZE]; /* internal input buffer */
+ picoos_uint16 inBufSize;/* actually allocated size */
+ picoos_uint16 inReadPos, inWritePos; /* next pos to read/write from/to inBuf*/
+ /*Input audio file management*/
+ picoos_char sInSDFileName[255];
+ picoos_SDFile sInSDFile;
+ picoos_uint32 sInSDFilePos;
+ /*----------------------PU output management-----------------------------*/
+ picoos_uint8 outBuf[PICOSIG_OUT_BUFF_SIZE]; /* internal output buffer */
+ picoos_uint16 outBufSize; /* actually allocated size */
+ picoos_uint16 outReadPos, outWritePos; /* next pos to read/write from/to outBuf*/
+ picoos_bool outSwitch; /* output destination switch 0:buffer, 1:file*/
+ picoos_char sOutSDFileName[255]; /* output file name */
+ picoos_SDFile sOutSDFile; /* output file handle */
+ picoos_single fSampNorm; /* running normalization factor */
+ picoos_uint32 nNumFrame; /* running count for frame number in output items */
+ /*---------------------- other working variables ---------------------------*/
+ picoos_uint8 innerProcState; /*where to take up work at next processing step*/
+ /*-----------------------Definition of the local storage for this PU--------*/
+ sig_innerobj_t sig_inner;
+ picoos_single pMod; /*pitch modifier*/
+ picoos_single vMod; /*Volume modifier*/
+ picoos_single sMod; /*speaker modifier*/
+ /*knowledge bases */
+ picokpdf_PdfMUL pdflfz, pdfmgc;
+ picoos_uint32 scmeanpowLFZ, scmeanpowMGC;
+ picoos_uint32 scmeanLFZ, scmeanMGC;
+ picokpdf_PdfPHS pdfphs;
+
+} sig_subobj_t;
+
+/* ******************************************************************************
+ * generic PU management
+ ********************************************************************************/
+
+/**
+ * initialization of the PU (processing unit)
+ * @param this : sig PU object
+ * @return PICO_OK : init ok
+ * @return PICO_ERR_OTHER : init failed
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t sigInitialize(register picodata_ProcessingUnit this)
+{
+ sig_subobj_t *sig_subObj;
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ sig_subObj = (sig_subobj_t *) this->subObj;
+ sig_subObj->inBufSize = PICOSIG_IN_BUFF_SIZE;
+ sig_subObj->outBufSize = PICOSIG_OUT_BUFF_SIZE;
+ sig_subObj->inReadPos = 0;
+ sig_subObj->inWritePos = 0;
+ sig_subObj->outReadPos = 0;
+ sig_subObj->outWritePos = 0;
+ sig_subObj->needMoreInput = 0;
+ sig_subObj->procState = PICOSIG_COLLECT;
+ sig_subObj->retState = PICOSIG_COLLECT;
+ sig_subObj->innerProcState = 0;
+ sig_subObj->nNumFrame = 0;
+
+ /*-----------------------------------------------------------------
+ * MANAGE Item I/O control management
+ ------------------------------------------------------------------*/
+ sig_subObj->sInSDFile = NULL;
+ sig_subObj->sInSDFilePos = 0;
+ sig_subObj->sInSDFileName[0] = '\0';
+ sig_subObj->outSwitch = 0; /*PU sends output to buffer (nextPU)*/
+ sig_subObj->sOutSDFile = NULL;
+ sig_subObj->sOutSDFileName[0] = '\0';
+ sig_subObj->nNumFrame = 0;
+
+ /*-----------------------------------------------------------------
+ * MANAGE LINGWARE INITIALIZATION IF NEEDED
+ ------------------------------------------------------------------*/
+ sig_subObj->pdfmgc = picokpdf_getPdfMUL(
+ this->voice->kbArray[PICOKNOW_KBID_PDF_MGC]);
+ sig_subObj->pdflfz = picokpdf_getPdfMUL(
+ this->voice->kbArray[PICOKNOW_KBID_PDF_LFZ]);
+ sig_subObj->pdfphs = picokpdf_getPdfPHS(
+ this->voice->kbArray[PICOKNOW_KBID_PDF_PHS]);
+
+ sig_subObj->scmeanpowLFZ = sig_subObj->pdflfz->bigpow
+ - sig_subObj->pdflfz->meanpow;
+ sig_subObj->scmeanpowMGC = sig_subObj->pdfmgc->bigpow
+ - sig_subObj->pdfmgc->meanpow;
+ sig_subObj->scmeanLFZ = (1 << (picoos_uint32) sig_subObj->scmeanpowLFZ);
+ sig_subObj->scmeanMGC = (1 << (picoos_uint32) sig_subObj->scmeanpowMGC);
+ sig_subObj->fSampNorm = PICOSIG_NORM1 * sig_subObj->pdfmgc->amplif;
+
+ /*-----------------------------------------------------------------
+ * Initialize memory for DSP
+ * ------------------------------------------------------------------*/
+ sigDspInitialize(&(sig_subObj->sig_inner));
+
+ /*-----------------------------------------------------------------
+ * Initialize modifiers
+ * ------------------------------------------------------------------*/
+ /*pitch , volume , speaker modifiers*/
+ sig_subObj->pMod = 1.0f;
+ sig_subObj->vMod = 1.0f;
+ sig_subObj->sMod = 1.0f;
+
+ return PICO_OK;
+}/*sigInitialize*/
+
+/**
+ * terminates the PU (processing unit)
+ * @param this : sig PU object
+ * @return PICO_OK : termination ok
+ * @return PICO_ERR_OTHER : termination failed
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t sigTerminate(register picodata_ProcessingUnit this)
+{
+
+ sig_subobj_t *sig_subObj;
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ sig_subObj = (sig_subobj_t *) this->subObj;
+
+ return PICO_OK;
+}/*sigTerminate*/
+
+/**
+ * deallocates the PU (processing unit) sub object
+ * @param this : sig PU object
+ * @param mm : the engine memory manager
+ * @return PICO_OK : deallocation ok
+ * @return PICO_ERR_OTHER : deallocation failed
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t sigSubObjDeallocate(register picodata_ProcessingUnit this,
+ picoos_MemoryManager mm)
+{
+ sig_subobj_t *sig_subObj;
+ if ((NULL == this) || ((this)->subObj == NULL)) {
+ return PICO_ERR_OTHER;
+ }
+ sig_subObj = (sig_subobj_t *) (this)->subObj;
+
+ if (sig_subObj->sInSDFile != NULL) {
+ picoos_sdfCloseIn(this->common, &(sig_subObj->sInSDFile));
+ sig_subObj->sInSDFile = NULL;
+ sig_subObj->sInSDFileName[0] = '\0';
+ }
+
+ if (sig_subObj->sOutSDFile != NULL) {
+ picoos_sdfCloseOut(this->common, &(sig_subObj->sOutSDFile));
+ sig_subObj->sOutSDFile = NULL;
+ sig_subObj->sOutSDFileName[0] = '\0';
+ }
+
+ sigDeallocate(mm, &(sig_subObj->sig_inner));
+
+ picoos_deallocate(this->common->mm, (void *) &this->subObj);
+
+ return PICO_OK;
+}/*sigSubObjDeallocate*/
+
+/**
+ * creates a new sig processing unit
+ * @param mm : the engine memory manager
+ * @param common : the engine common object
+ * @param cbIn : the PU input buffer
+ * @param cbOut : the PU output buffer
+ * @param voice : the voice descriptor object
+ * @return a valid PU handle if creation is OK
+ * @return NULL if creation is !=OK
+ * @callgraph
+ * @callergraph
+ */
+picodata_ProcessingUnit picosig_newSigUnit(picoos_MemoryManager mm,
+ picoos_Common common, picodata_CharBuffer cbIn,
+ picodata_CharBuffer cbOut, picorsrc_Voice voice)
+{
+ sig_subobj_t *sig_subObj;
+
+ picodata_ProcessingUnit this = picodata_newProcessingUnit(mm, common, cbIn,
+ cbOut, voice);
+ if (NULL == this) {
+ return NULL;
+ }
+ this->initialize = sigInitialize;
+
+ PICODBG_DEBUG(("picosig_newSigUnit -- creating SIG PU"));
+ /*Init function pointers*/
+ this->step = sigStep;
+ this->terminate = sigTerminate;
+ this->subDeallocate = sigSubObjDeallocate;
+ /*sub obj allocation*/
+ this->subObj = picoos_allocate(mm, sizeof(sig_subobj_t));
+
+ if (NULL == this->subObj) {
+ PICODBG_ERROR(("Error in Sig Object allocation"));
+ picoos_deallocate(mm, (void *) &this);
+ return NULL;
+ }
+ sig_subObj = (sig_subobj_t *) this->subObj;
+
+ /*-----------------------------------------------------------------
+ * Allocate memory for DSP inner algorithms
+ * ------------------------------------------------------------------*/
+ if (sigAllocate(mm, &(sig_subObj->sig_inner)) != 0) {
+ PICODBG_ERROR(("Error in Sig Sub Object Allocation"));
+ picoos_deallocate(mm, (void *) &this);
+ return NULL;
+ }
+
+ /*-----------------------------------------------------------------
+ * Initialize memory for DSP (this may be re-used elsewhere, e.g.Reset)
+ * ------------------------------------------------------------------*/
+ if (PICO_OK != sigInitialize(this)) {
+ PICODBG_ERROR(("Error in iSig Sub Object initialization"));
+ sigDeallocate(mm, &(sig_subObj->sig_inner));
+ picoos_deallocate(mm, (void *) &this);
+ return NULL;
+ }PICODBG_DEBUG(("SIG PU creation succeded!!"));
+ return this;
+}/*picosig_newSigUnit*/
+
+/**
+ * pdf access for phase
+ * @param this : sig object pointer
+ * @param phsIndex : index of phase vectot in the pdf
+ * @param phsVect : pointer to base of array where to store the phase values
+ * @param numComponents : pointer to the variable to store the number of components
+ * @return PICO_OK : pdf retrieved
+ * @return PICO_ERR_OTHER : pdf not retrieved
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t getPhsFromPdf(register picodata_ProcessingUnit this,
+ picoos_uint16 phsIndex, picoos_int32 *phsVect,
+ picoos_int16 *numComponents)
+{
+ sig_subobj_t *sig_subObj;
+ picokpdf_PdfPHS pdf;
+ static int nFrame = 0;
+
+ picoos_uint32 nIndexValue;
+ picoos_uint8 *nCurrIndexOffset, *nContent;
+ picoos_uint16 nI;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ sig_subObj = (sig_subobj_t *) this->subObj;
+ pdf = sig_subObj->pdfphs;
+ /*check incoming index*/
+ if (phsIndex >= pdf->numvectors) {
+ return PICODATA_PU_ERROR;
+ }
+ nCurrIndexOffset = ((picoos_uint8*) pdf->indexBase) + phsIndex * sizeof(picoos_uint32);
+ nIndexValue = (0xFF000000 & ((*(nCurrIndexOffset+3)) << 24)) | (0x00FF0000 & ((*(nCurrIndexOffset+2)) << 16)) |
+ (0x0000FF00 & ((*(nCurrIndexOffset+1)) << 8)) | (0x000000FF & ((*nCurrIndexOffset)));
+ nContent = pdf->contentBase;
+ nContent += nIndexValue;
+ *numComponents = (picoos_int16) *nContent++;
+ if (*numComponents>PICODSP_PHASEORDER) {
+ PICODBG_DEBUG(("WARNING : Frame %d -- Phase vector[%d] Components = %d --> too big\n", nFrame, phsIndex, *numComponents));
+ *numComponents = PICODSP_PHASEORDER;
+ }
+ for (nI=0; nI<*numComponents; nI++) {
+ phsVect[nI] = (picoos_int32) *nContent++;
+ }
+ for (nI=*numComponents; nI<PICODSP_PHASEORDER; nI++) {
+ phsVect[nI] = 0;
+ }
+ nFrame++;
+ return PICO_OK;
+}/*getPhsFromPdf*/
+
+/**
+ * processes one item with sig algo
+ * @param this : the PU object pointer
+ * @param inReadPos : read position in input buffer
+ * @param numinb : number of bytes in input buffer (including header)
+ * @param outWritePos : write position in output buffer
+ * @param numoutb : number of bytes produced in output buffer
+ * @return PICO_OK : processing successful and terminated
+ * @return PICO_STEP_BUSY : processing successful but still things to do
+ * @return PICO_ERR_OTHER : errors
+ * @remarks processes a full input item
+ * @remarks input data is one or more item, taken from local storage
+ * @remarks output data is one or more item, written in local storage
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t sigProcess(register picodata_ProcessingUnit this,
+ picoos_uint16 inReadPos, picoos_uint16 numinb,
+ picoos_uint16 outWritePos, picoos_uint16 *numoutb)
+{
+
+ register sig_subobj_t * sig_subObj;
+ picoos_int16 n_i;
+ picoos_int16 n_frames, n_count;
+ picoos_int16 *s_data, offset;
+ picoos_int32 f_data, mlt, *t1, *tmp1, *tmp2;
+ picoos_uint16 tmp_uint16;
+ picopal_int16 tmp_int16;
+ picoos_uint16 i, cnt;
+ picoos_int16 hop_p_half;
+
+ sig_subObj = (sig_subobj_t *) this->subObj;
+
+ numinb = numinb; /* avoid warning "var not used in this function"*/
+
+ /*defaults to 0 for output bytes*/
+ *numoutb = 0;
+
+ /*Input buffer contains an item FRAME_PAR*/
+ switch (sig_subObj->innerProcState) {
+
+ case 0:
+ /*---------------------------------------------
+ Shifting old values
+ ---------------------------------------------*/
+ for (n_count = 0; n_count < CEPST_BUFF_SIZE-1; n_count++) {
+ sig_subObj->sig_inner.F0Buff[n_count]=sig_subObj->sig_inner.F0Buff[n_count+1];
+ sig_subObj->sig_inner.PhIdBuff[n_count]=sig_subObj->sig_inner.PhIdBuff[n_count+1];
+ sig_subObj->sig_inner.VoicingBuff[n_count]=sig_subObj->sig_inner.VoicingBuff[n_count+1];
+ sig_subObj->sig_inner.FuVBuff[n_count]=sig_subObj->sig_inner.FuVBuff[n_count+1];
+ }
+ for (n_count = 0; n_count < PHASE_BUFF_SIZE-1; n_count++) {
+ sig_subObj->sig_inner.VoxBndBuff[n_count]=sig_subObj->sig_inner.VoxBndBuff[n_count+1];
+ }
+
+ tmp1 = sig_subObj->sig_inner.CepBuff[0];
+ for (n_count = 0; n_count < CEPST_BUFF_SIZE-1; n_count++) {
+ sig_subObj->sig_inner.CepBuff[n_count]=sig_subObj->sig_inner.CepBuff[n_count+1];
+ }
+ sig_subObj->sig_inner.CepBuff[CEPST_BUFF_SIZE-1]=tmp1;
+
+ tmp1 = sig_subObj->sig_inner.PhsBuff[0];
+ for (n_count = 0; n_count < PHASE_BUFF_SIZE-1; n_count++) {
+ sig_subObj->sig_inner.PhsBuff[n_count]=sig_subObj->sig_inner.PhsBuff[n_count+1];
+ }
+ sig_subObj->sig_inner.PhsBuff[PHASE_BUFF_SIZE-1]=tmp1;
+
+ /*---------------------------------------------
+ Frame related initializations
+ ---------------------------------------------*/
+ sig_subObj->sig_inner.prevVoiced_p = sig_subObj->sig_inner.voiced_p;
+ /*---------------------------------------------
+ Get input data from PU buffer in internal buffers
+ -------------------------------------------------*/
+ /*load the phonetic id code*/
+ picoos_mem_copy((void *) &sig_subObj->inBuf[inReadPos
+ + sizeof(picodata_itemhead_t)], /*src*/
+ (void *) &tmp_uint16, sizeof(tmp_uint16)); /*dest+size*/
+ sig_subObj->sig_inner.PhIdBuff[CEPST_BUFF_SIZE-1] = (picoos_int16) tmp_uint16; /*store into newest*/
+ tmp_uint16 = (picoos_int16) sig_subObj->sig_inner.PhIdBuff[0]; /*assign oldest*/
+ sig_subObj->sig_inner.phId_p = (picoos_int16) tmp_uint16; /*assign oldest*/
+
+ /*load pitch values*/
+ for (i = 0; i < sig_subObj->pdflfz->ceporder; i++) {
+ picoos_mem_copy((void *) &(sig_subObj->inBuf[inReadPos
+ + sizeof(picodata_itemhead_t) + sizeof(tmp_uint16) + 3
+ * i * sizeof(tmp_uint16)]), /*src*/
+ (void *) &tmp_uint16, sizeof(tmp_uint16)); /*dest+size*/
+
+ sig_subObj->sig_inner.F0Buff[CEPST_BUFF_SIZE-1] = (picoos_int16) tmp_uint16;/*store into newest*/
+ tmp_uint16 = (picoos_int16) sig_subObj->sig_inner.F0Buff[0]; /*assign oldest*/
+
+ /*convert in float*/
+ sig_subObj->sig_inner.F0_p
+ = (tmp_uint16 ? ((picoos_single) tmp_uint16
+ / sig_subObj->scmeanLFZ) : (picoos_single) 0.0);
+
+ if (sig_subObj->sig_inner.F0_p != (picoos_single) 0.0f) {
+ sig_subObj->sig_inner.F0_p = (picoos_single) exp(
+ (picoos_single) sig_subObj->sig_inner.F0_p);
+
+ }
+ /* voicing */
+ picoos_mem_copy((void *) &(sig_subObj->inBuf[inReadPos
+ + sizeof(picodata_itemhead_t) + sizeof(tmp_uint16) + 3
+ * i * sizeof(tmp_uint16) + sizeof(tmp_uint16)]),/*src*/
+ (void *) &tmp_uint16, sizeof(tmp_uint16)); /*dest+size*/
+
+ sig_subObj->sig_inner.VoicingBuff[CEPST_BUFF_SIZE-1] = (picoos_int16) tmp_uint16;/*store into newest*/
+ tmp_uint16 = (picoos_int16) sig_subObj->sig_inner.VoicingBuff[0]; /*assign oldest*/
+
+ sig_subObj->sig_inner.voicing = (picoos_single) ((tmp_uint16
+ & 0x01) * 8 + (tmp_uint16 & 0x0e) / 2)
+ / (picoos_single) 15.0f;
+
+ /* unrectified f0 */
+ picoos_mem_copy((void *) &(sig_subObj->inBuf[inReadPos
+ + sizeof(picodata_itemhead_t) + sizeof(tmp_uint16) + 3
+ * i * sizeof(tmp_uint16) + 2 * sizeof(tmp_uint16)]),/*src*/
+ (void *) &tmp_uint16, sizeof(tmp_uint16)); /*dest+size*/
+
+ sig_subObj->sig_inner.FuVBuff[CEPST_BUFF_SIZE-1] = (picoos_int16) tmp_uint16;/*store into newest*/
+ tmp_uint16 = (picoos_int16) sig_subObj->sig_inner.FuVBuff[0]; /*assign oldest*/
+
+ sig_subObj->sig_inner.Fuv_p = (picoos_single) tmp_uint16
+ / sig_subObj->scmeanLFZ;
+ sig_subObj->sig_inner.Fuv_p = (picoos_single) EXP((double)sig_subObj->sig_inner.Fuv_p);
+ }
+ /*load cep values*/
+ offset = inReadPos + sizeof(picodata_itemhead_t)
+ + sizeof(tmp_uint16) +
+ 3 * sig_subObj->pdflfz->ceporder * sizeof(tmp_int16);
+
+ tmp1 = sig_subObj->sig_inner.CepBuff[CEPST_BUFF_SIZE-1]; /*store into CURR */
+ tmp2 = sig_subObj->sig_inner.CepBuff[0]; /*assign oldest*/
+
+ for (i = 0; i < sig_subObj->pdfmgc->ceporder; i++) {
+ picoos_mem_copy((void *) &(sig_subObj->inBuf[offset + i
+ * sizeof(tmp_int16)]), /*src*/
+ (void *) &tmp_int16, sizeof(tmp_int16)); /*dest+size*/
+
+ tmp1 [i] = (picoos_int32) tmp_int16;
+ sig_subObj->sig_inner.wcep_pI[i] = (picoos_int32) tmp2[i];
+ }
+
+ if (sig_subObj->inBuf[inReadPos+ 3] > sig_subObj->inBuf[inReadPos+ 2]*2 + 8) {
+ /*load phase values*/
+ /*get the index*/
+ picoos_mem_copy((void *) &(sig_subObj->inBuf[offset + sig_subObj->pdfmgc->ceporder
+ * sizeof(tmp_int16)]), /*src*/
+ (void *) &tmp_int16, sizeof(tmp_int16)); /*dest+size*/
+
+ /*store into buffers*/
+ tmp1 = sig_subObj->sig_inner.PhsBuff[PHASE_BUFF_SIZE-1];
+ /*retrieve values from pdf*/
+ getPhsFromPdf(this, tmp_int16, tmp1, &(sig_subObj->sig_inner.VoxBndBuff[PHASE_BUFF_SIZE-1]));
+ } else {
+ /* no support for phase found */
+ sig_subObj->sig_inner.VoxBndBuff[PHASE_BUFF_SIZE-1] = 0;
+ }
+
+ /*pitch modifier*/
+ sig_subObj->sig_inner.F0_p *= sig_subObj->pMod;
+ sig_subObj->sig_inner.Fuv_p *= sig_subObj->pMod;
+ if (sig_subObj->sig_inner.F0_p > 0.0f) {
+ sig_subObj->sig_inner.voiced_p = 1;
+ } else {
+ sig_subObj->sig_inner.voiced_p = 0;
+ }
+ sig_subObj->sig_inner.n_available++;
+ if (sig_subObj->sig_inner.n_available>3) sig_subObj->sig_inner.n_available = 3;
+
+ if (sig_subObj->sig_inner.n_available < 3) {
+ return PICO_STEP_BUSY;
+ }
+
+ sig_subObj->innerProcState = 3;
+ return PICO_STEP_BUSY;
+
+ case 3:
+ /*Convert from mfcc to power spectrum*/
+ save_transition_frame(&(sig_subObj->sig_inner));
+ mel_2_lin_lookup(&(sig_subObj->sig_inner), sig_subObj->scmeanpowMGC);
+ sig_subObj->innerProcState += 1;
+ return PICO_STEP_BUSY;
+
+ case 4:
+ /*Reconstruct PHASE SPECTRUM */
+ phase_spec2(&(sig_subObj->sig_inner));
+ sig_subObj->innerProcState += 1;
+ return PICO_STEP_BUSY;
+
+ case 5:
+ /*Prepare Envelope spectrum for inverse FFT*/
+ env_spec(&(sig_subObj->sig_inner));
+ sig_subObj->innerProcState += 1;
+ return PICO_STEP_BUSY;
+
+ case 6:
+ /*Generate the impulse response of the vocal tract */
+ impulse_response(&(sig_subObj->sig_inner));
+ sig_subObj->innerProcState += 1;
+ return PICO_STEP_BUSY;
+
+ case 7:
+ /*Sum up N impulse responses according to excitation */
+ td_psola2(&(sig_subObj->sig_inner));
+ sig_subObj->innerProcState += 1;
+ return PICO_STEP_BUSY;
+
+ case 8:
+ /*Ovladd */
+ overlap_add(&(sig_subObj->sig_inner));
+ sig_subObj->innerProcState += 1;
+ return PICO_STEP_BUSY;
+
+ case 9:
+ /*-----------------------------------------
+ Save the output FRAME item (0:hop-1)
+ swap remaining buffer
+ ---------------------------------------------*/
+ n_frames = 2;
+ *numoutb = 0;
+ hop_p_half = (sig_subObj->sig_inner.hop_p) / 2;
+ for (n_count = 0; n_count < n_frames; n_count++) {
+ sig_subObj->outBuf[outWritePos]
+ = (picoos_uint8) PICODATA_ITEM_FRAME;
+ sig_subObj->outBuf[outWritePos + 1]
+ = (picoos_uint8) (hop_p_half);
+ sig_subObj->outBuf[outWritePos + 2]
+ = (picoos_uint8) (sig_subObj->nNumFrame % ((hop_p_half)));
+ sig_subObj->outBuf[outWritePos + 3]
+ = (picoos_uint8) sig_subObj->sig_inner.hop_p;
+ s_data = (picoos_int16 *) &(sig_subObj->outBuf[outWritePos + 4]);
+
+ /*range control and clipping*/
+ mlt = (picoos_int32) ((sig_subObj->fSampNorm * sig_subObj->vMod)
+ * PICODSP_END_FLOAT_NORM);
+ t1 = &(sig_subObj->sig_inner.WavBuff_p[n_count * (hop_p_half)]);
+ for (n_i = 0; n_i < hop_p_half; n_i++) { /*Normalization*/
+ f_data = *t1++ * mlt;
+ if (f_data >= 0)
+ f_data >>= 14;
+ else
+ f_data = -(-f_data >> 14);
+ if (f_data > PICOSIG_MAXAMP)
+ f_data = PICOSIG_MAXAMP;
+ if (f_data < PICOSIG_MINAMP)
+ f_data = PICOSIG_MINAMP;
+ *s_data = (picoos_int16) (f_data);
+ s_data++;
+ }
+ sig_subObj->nNumFrame = sig_subObj->nNumFrame + 1;
+ *numoutb += ((picoos_int16) n_i * sizeof(picoos_int16)) + 4;
+ outWritePos += *numoutb;
+ }/*end for n_count*/
+ /*Swap remaining buffer*/
+ cnt = sig_subObj->sig_inner.m2_p - sig_subObj->sig_inner.hop_p;
+ tmp1 = sig_subObj->sig_inner.WavBuff_p;
+ tmp2
+ = &(sig_subObj->sig_inner.WavBuff_p[sig_subObj->sig_inner.hop_p]);
+ FAST_DEVICE(cnt,*(tmp1++)=*(tmp2++);)
+ ;
+ cnt = sig_subObj->sig_inner.m2_p - (sig_subObj->sig_inner.m2_p
+ - sig_subObj->sig_inner.hop_p);
+ FAST_DEVICE(cnt,*(tmp1++)=0;)
+ ;
+ sig_subObj->innerProcState = 0; /*reset to step 0*/
+ sig_subObj->nNumFrame += 2;
+ return PICO_OK;
+ default:
+ return PICO_ERR_OTHER;
+ }
+ return PICO_ERR_OTHER;
+}/*sigProcess*/
+
+/**
+ * selects items to be dealth with by this PU
+ * @param item : pointer to current item head
+ * @return TRUE : item should be managed
+ * @return FALSE : item should be passed on next PU
+ * @remarks item pointed to by *item should be already valid
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t sig_deal_with(const picoos_uint8 *item)
+{
+ picodata_itemhead_t head;
+ pico_status_t s_result;
+ s_result = FALSE;
+ head.type = item[0];
+ head.info1 = item[1];
+ head.info2 = item[2];
+ head.len = item[3];
+ switch (head.type) {
+ case PICODATA_ITEM_FRAME_PAR:
+ /*the only item that is managed by sig, so far, is "FRAME_PAR"*/
+ s_result = TRUE;
+ break;
+ case PICODATA_ITEM_CMD:
+ if ((head.info1 == PICODATA_ITEMINFO1_CMD_PLAY) || (head.info1
+ == PICODATA_ITEMINFO1_CMD_SAVE) || (head.info1
+ == PICODATA_ITEMINFO1_CMD_UNSAVE)) {
+ if (head.info2 == PICODATA_ITEMINFO2_CMD_TO_SIG) {
+ return TRUE;
+ }
+ }
+ if ((head.info1 == PICODATA_ITEMINFO1_CMD_PITCH) || (head.info1
+ == PICODATA_ITEMINFO1_CMD_VOLUME) || (head.info1
+ == PICODATA_ITEMINFO1_CMD_SPEAKER)) {
+ return TRUE;
+ }
+ break;
+ default:
+ break;
+ }
+ return s_result;
+} /*sig_deal_with*/
+
+/**
+ * selects items to be managed as commands by this PU
+ * @param item : pointer to current item head
+ * @return TRUE : item should be managed as a command
+ * @return FALSE : item is not a PU command
+ * @remarks item pointed to by *item should be already valid
+ * @callgraph
+ * @callergraph
+ */
+static pico_status_t sig_is_command(const picoos_uint8 *item)
+{
+ picodata_itemhead_t head;
+ head.type = item[0];
+ head.info1 = item[1];
+ head.info2 = item[2];
+ head.len = item[3];
+ switch (head.type) {
+ case PICODATA_ITEM_CMD:
+ if ((head.info1 == PICODATA_ITEMINFO1_CMD_PLAY) || (head.info1
+ == PICODATA_ITEMINFO1_CMD_SAVE) || (head.info1
+ == PICODATA_ITEMINFO1_CMD_UNSAVE)) {
+ if (head.info2 == PICODATA_ITEMINFO2_CMD_TO_SIG) {
+ return TRUE;
+ }
+ }
+ if ((head.info1 == PICODATA_ITEMINFO1_CMD_PITCH) || (head.info1
+ == PICODATA_ITEMINFO1_CMD_VOLUME) || (head.info1
+ == PICODATA_ITEMINFO1_CMD_SPEAKER)) {
+ return TRUE;
+ }
+ break;
+ default:
+ break;
+ }
+ return FALSE;
+} /*sig_is_command*/
+
+/**
+ * performs a step of the sig processing
+ * @param this : pointer to current PU (Control Unit)
+ * @param mode : mode for the PU
+ * @param numBytesOutput : pointer to number of bytes produced (output)
+ * @param this : pointer to current PU (Control Unit)
+ * @return one of the "picodata_step_result_t" values
+ * @callgraph
+ * @callergraph
+ */
+static picodata_step_result_t sigStep(register picodata_ProcessingUnit this,
+ picoos_int16 mode, picoos_uint16 * numBytesOutput)
+{
+ register sig_subobj_t * sig_subObj;
+ pico_status_t s_result;
+ picoos_bool b_res;
+ pico_status_t s_deal_with;
+ picoos_uint16 blen;
+ picoos_uint16 numinb, numoutb;
+ pico_status_t rv;
+ picoos_int16 *s_data;
+ picoos_int16 hop_p_half;
+ picoos_uint32 n_samp, n_i;
+ picoos_char s_temp_file_name[255];
+ picoos_uint32 n_start, n_fram, n_bytes;
+ picoos_single f_value;
+ picoos_uint16 n_value;
+ picoos_uint32 n_pos;
+ /*wav file play volume control*/
+ picoos_int16 *s_t1;
+ picoos_int32 sf_data, sf_mlt;
+ picoos_uint32 sf;
+ picoos_encoding_t enc;
+ picoos_uint32 numSamples;
+
+ numinb = 0;
+ numoutb = 0;
+ rv = PICO_OK;
+ s_result = PICO_OK;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ sig_subObj = (sig_subobj_t *) this->subObj;
+
+ /*Init number of output bytes*/
+ *numBytesOutput = 0;
+
+ mode = mode; /* avoid warning "var not used in this function" */
+
+ while (1) { /* exit via return */
+
+ PICODBG_DEBUG(("picosig.sigStep -- doing state %i",sig_subObj->procState));
+
+ switch (sig_subObj->procState) {
+
+ case PICOSIG_COLLECT:
+ /* ************** item collector ***********************************/
+ /*collecting items from the PU input buffer*/
+ s_result = picodata_cbGetItem(this->cbIn,
+ &(sig_subObj->inBuf[sig_subObj->inWritePos]),
+ sig_subObj->inBufSize - sig_subObj->inWritePos, &blen);
+
+ PICODBG_DEBUG(("picosig.sigStep -- got item, status: %d",rv));
+
+ if (s_result == PICO_EOF) {
+ /*no items available : remain in state 0 and return idle*/
+ return PICODATA_PU_IDLE;
+ }
+ if ((PICO_OK == s_result) && (blen > 0)) {
+ /* we now have one item : CHECK IT */
+ s_result = picodata_is_valid_item(
+ &(sig_subObj->inBuf[sig_subObj->inWritePos]), blen);
+ if (s_result != TRUE) {
+ PICODBG_DEBUG(("picosig.sigStep -- item is not valid: discard"));
+ /*Item not valid : remain in state PICOSIG_COLLECT*/
+ return PICODATA_PU_BUSY;
+ }
+ /*item ok: it could be sent to schedule state*/
+ sig_subObj->inWritePos += blen;
+ sig_subObj->needMoreInput = FALSE;
+ sig_subObj->procState = PICOSIG_SCHEDULE;
+ /* uncomment next to split into two steps */
+ return PICODATA_PU_ATOMIC;
+ }
+ /*no EOF, no OK --> errors : remain in state PICOSIG_COLLECT and return error*/
+ return PICODATA_PU_ERROR;
+ break;
+
+ case PICOSIG_SCHEDULE:
+ /* *************** item processing ***********************************/
+ numinb = PICODATA_ITEM_HEADSIZE
+ + sig_subObj->inBuf[sig_subObj->inReadPos + 3];
+
+ /*verify that current item has to be dealth with by this PU*/
+ s_deal_with = sig_deal_with(
+ &(sig_subObj->inBuf[sig_subObj->inReadPos]));
+
+ switch (s_deal_with) {
+
+ case TRUE:
+ /* we have to manage this item */
+ if (FALSE == sig_is_command(
+ &(sig_subObj->inBuf[sig_subObj->inReadPos])))
+ {
+ /*no commands, item to deal with : switch to process state*/
+ sig_subObj->procState = PICOSIG_PROCESS;
+ sig_subObj->retState = PICOSIG_COLLECT;
+ return PICODATA_PU_BUSY; /*data still to process or to feed*/
+
+ } else {
+
+ /*we need to manage this item as a SIG command-item*/
+
+ switch (sig_subObj->inBuf[sig_subObj->inReadPos + 1]) {
+
+ case PICODATA_ITEMINFO1_CMD_PLAY:
+ /*CMD recognized : consume the command */
+ sig_subObj->inReadPos += numinb;
+ if (sig_subObj->inReadPos
+ >= sig_subObj->inWritePos) {
+ sig_subObj->inReadPos = 0;
+ sig_subObj->inWritePos = 0;
+ }
+ /*default next state setting*/
+ sig_subObj->procState =
+ sig_subObj->retState = PICOSIG_COLLECT;
+
+ /*--------- wav file play management --------------*/
+ if (sig_subObj->sInSDFile != NULL) {
+ /*input wav file is already open : return*/
+ return PICODATA_PU_BUSY;
+ }
+ /*get temp file name*/
+ picoos_strlcpy(
+ (picoos_char*) s_temp_file_name,
+ (picoos_char*) &(sig_subObj->inBuf[sig_subObj->inReadPos
+ + 4]),
+ sig_subObj->inBuf[sig_subObj->inReadPos
+ + 3] + 1);
+ /*avoid input/output file name clashes*/
+ if (sig_subObj->sOutSDFile != NULL) {
+ if (picoos_strncmp(
+ (picoos_char*) s_temp_file_name,
+ (picoos_char*) sig_subObj->sOutSDFileName,
+ picoos_strlen(
+ (picoos_char*) s_temp_file_name))
+ == 0) {
+ PICODBG_WARN(("input and output files has the same name!\n"));
+ return PICODATA_PU_BUSY;
+ }
+ }
+ /*actual sampled data file open*/
+ b_res = picoos_sdfOpenIn(this->common,
+ &(sig_subObj->sInSDFile),
+ s_temp_file_name, &sf,
+ &enc, &numSamples);
+ if (b_res != TRUE) {
+ PICODBG_DEBUG(("Error on opening file %s\n", s_temp_file_name));
+ sig_subObj->sInSDFile = NULL;
+ sig_subObj->sInSDFileName[0] = '\0';
+ return PICODATA_PU_BUSY;
+ }
+ /*input file handle is now valid : store filename*/
+ picoos_strlcpy(
+ (picoos_char*) sig_subObj->sInSDFileName,
+ (picoos_char*) s_temp_file_name,
+ sig_subObj->inBuf[sig_subObj->inReadPos
+ + 3] + 1);
+ sig_subObj->sInSDFilePos = 0;
+ /*switch to state PLAY and return*/
+ sig_subObj->procState =
+ sig_subObj->retState = PICOSIG_PLAY;
+ return PICODATA_PU_BUSY;
+ break;
+
+ case PICODATA_ITEMINFO1_CMD_SAVE:
+ /*CMD recognized : consume the command */
+ sig_subObj->inReadPos += numinb;
+ if (sig_subObj->inReadPos
+ >= sig_subObj->inWritePos) {
+ sig_subObj->inReadPos = 0;
+ sig_subObj->inWritePos = 0;
+ }
+ /*prepare return state*/
+ sig_subObj->procState = PICOSIG_COLLECT;
+ sig_subObj->retState = PICOSIG_COLLECT;
+ /*check about output file*/
+ if ((sig_subObj->sOutSDFile != NULL)
+ || (sig_subObj->outSwitch == 1)) {
+ /*output sig file is already active : return*/
+ return PICODATA_PU_BUSY;
+ }
+ /*get temp file name*/
+ picoos_strlcpy(
+ (picoos_char*) s_temp_file_name,
+ (picoos_char*) &(sig_subObj->inBuf[sig_subObj->inReadPos
+ + 4]),
+ sig_subObj->inBuf[sig_subObj->inReadPos
+ + 3] + 1);
+ /*check extension*/
+ if (picoos_has_extension(s_temp_file_name,
+ PICODATA_PUTYPE_WAV_OUTPUT_EXTENSION)
+ == FALSE){
+ /*extension unsupported : return*/
+ return PICODATA_PU_BUSY;
+ }
+ /*avoid input/output file name clashes*/
+ if (sig_subObj->sInSDFile != NULL) {
+ if (picoos_strncmp(
+ (picoos_char*) sig_subObj->sInSDFileName,
+ (picoos_char*) s_temp_file_name,
+ picoos_strlen(
+ (picoos_char*) sig_subObj->sInSDFileName))
+ == 0) {
+ /*input and output files has the same name : do not allow opening for writing*/
+ PICODBG_WARN(("input and output files has the same name!\n"));
+ return PICODATA_PU_BUSY;
+ }
+ }
+ /*try to open*/
+ picoos_sdfOpenOut(this->common,
+ &(sig_subObj->sOutSDFile),
+ s_temp_file_name,
+ SAMPLE_FREQ_16KHZ, PICOOS_ENC_LIN);
+ if (sig_subObj->sOutSDFile == NULL) {
+ PICODBG_DEBUG(("Error on opening file %s\n", sig_subObj->sOutSDFileName));
+ sig_subObj->outSwitch = 0;
+ sig_subObj->sOutSDFileName[0] = '\0';
+ } else {
+ /*open OK*/
+ sig_subObj->outSwitch = 1;
+ /*store output filename*/
+ picoos_strlcpy(
+ (picoos_char*) sig_subObj->sOutSDFileName,
+ (picoos_char*) s_temp_file_name,
+ sig_subObj->inBuf[sig_subObj->inReadPos + 3] + 1);
+ }
+ return PICODATA_PU_BUSY;
+ break;
+
+ case PICODATA_ITEMINFO1_CMD_UNSAVE:
+ /*CMD recognized : consume the command */
+ sig_subObj->inReadPos += numinb;
+ if (sig_subObj->inReadPos
+ >= sig_subObj->inWritePos) {
+ sig_subObj->inReadPos = 0;
+ sig_subObj->inWritePos = 0;
+ }
+ /*prepare return state*/
+ sig_subObj->procState = PICOSIG_COLLECT;
+ sig_subObj->retState = PICOSIG_COLLECT;
+ /*close the output file if any*/
+ if ((sig_subObj->sOutSDFile == NULL)
+ || (sig_subObj->outSwitch == 0)) {
+ /*output sig file is not active : return*/
+ PICODBG_DEBUG(("Error on requesting a binary samples file output closing : no file output handle exist\n"));
+ return PICODATA_PU_BUSY;
+ }
+ picoos_sdfCloseOut(this->common,
+ &(sig_subObj->sOutSDFile));
+ sig_subObj->outSwitch = 0;
+ sig_subObj->sOutSDFile = NULL;
+ sig_subObj->sOutSDFileName[0] = '\0';
+ return PICODATA_PU_BUSY;
+ break;
+
+ case PICODATA_ITEMINFO1_CMD_PITCH:
+ case PICODATA_ITEMINFO1_CMD_VOLUME:
+ case PICODATA_ITEMINFO1_CMD_SPEAKER:
+ n_pos = 4;
+ picoos_read_mem_pi_uint16(
+ &(sig_subObj->inBuf[sig_subObj->inReadPos]),
+ &n_pos, &n_value);
+ b_res = FALSE;
+ switch (sig_subObj->inBuf[sig_subObj->inReadPos + 2]) {
+ case 'a' :
+ /*absloute modifier*/
+ f_value = (picoos_single) n_value
+ / (picoos_single) 100.0f;
+ b_res = TRUE;
+ break;
+ case 'r' :
+ /*relative modifier*/
+ f_value = (picoos_single) n_value
+ / (picoos_single) 1000.0f;
+ b_res = TRUE;
+ break;
+ default :
+ f_value = (picoos_single)0; /*avoid warnings*/
+ break;
+ }
+ if (b_res) {
+ switch (sig_subObj->inBuf[sig_subObj->inReadPos + 1]) {
+ case PICODATA_ITEMINFO1_CMD_PITCH :
+ sig_subObj->pMod = f_value;
+ break;
+ case PICODATA_ITEMINFO1_CMD_VOLUME :
+ sig_subObj->vMod = f_value;
+ break;
+ case PICODATA_ITEMINFO1_CMD_SPEAKER :
+ sig_subObj->sMod = f_value;
+ sig_subObj->sig_inner.sMod_p
+ = sig_subObj->sMod;
+ /*call the function needed to initialize the speaker factor*/
+ mel_2_lin_init(
+ &(sig_subObj->sig_inner));
+ break;
+ default :
+ break;
+ }
+ }
+
+ /*CMD recognized : consume the command */
+ sig_subObj->inReadPos += numinb;
+ if (sig_subObj->inReadPos
+ >= sig_subObj->inWritePos) {
+ sig_subObj->inReadPos = 0;
+ sig_subObj->inWritePos = 0;
+ }
+ /*prepare proc state*/
+ sig_subObj->procState = PICOSIG_COLLECT;
+ sig_subObj->retState = PICOSIG_COLLECT;
+ return PICODATA_PU_BUSY;
+ break;
+ default:
+ break;
+ }/*switch command type*/
+ } /*end if is command*/
+ break;
+
+ case FALSE:
+
+ /*we DO NOT have to deal with this item on this PU.
+ * Normally these are still alive boundary or flush items*/
+ /*copy item from PU input to PU output buffer,
+ * i.e. make it ready to FEED*/
+ s_result = picodata_copy_item(
+ &(sig_subObj->inBuf[sig_subObj->inReadPos]),
+ numinb,
+ &(sig_subObj->outBuf[sig_subObj->outWritePos]),
+ sig_subObj->outBufSize - sig_subObj->outWritePos,
+ &numoutb);
+ if (s_result != PICO_OK) {
+ /*item not prepared properly to be sent to next PU :
+ * do not change state and retry next time*/
+ sig_subObj->procState = PICOSIG_SCHEDULE;
+ sig_subObj->retState = PICOSIG_COLLECT;
+ return PICODATA_PU_BUSY; /*data still to process or to feed*/
+ }
+
+ /*if end of sentence reset number of frames(only needed for debugging purposes)*/
+ if ((sig_subObj->inBuf[sig_subObj->inReadPos]
+ == PICODATA_ITEM_BOUND)
+ && ((sig_subObj->inBuf[sig_subObj->inReadPos + 1]
+ == PICODATA_ITEMINFO1_BOUND_SEND)
+ || (sig_subObj->inBuf[sig_subObj->inReadPos
+ + 1]
+ == PICODATA_ITEMINFO1_BOUND_TERM))) {
+ PICODBG_INFO(("End of sentence - Processed frames : %d",
+ sig_subObj->nNumFrame));
+ sig_subObj->nNumFrame = 0;
+ }
+
+ /*item processed and put in oputput buffer : consume the item*/
+ sig_subObj->inReadPos += numinb;
+ sig_subObj->outWritePos += numoutb;
+ if (sig_subObj->inReadPos >= sig_subObj->inWritePos) {
+ /* inBuf exhausted */
+ sig_subObj->inReadPos = 0;
+ sig_subObj->inWritePos = 0;
+ sig_subObj->needMoreInput = FALSE;
+ }
+ sig_subObj->procState = PICOSIG_FEED;
+ sig_subObj->retState = PICOSIG_COLLECT;
+ return PICODATA_PU_BUSY; /*data still to process or to feed*/
+ break;
+
+ default:
+ break;
+ }/*end switch s_deal_with*/
+
+ break; /*end case sig_schedule*/
+
+ case PICOSIG_PROCESS:
+ /* *************** item processing ***********************************/
+ numinb = PICODATA_ITEM_HEADSIZE
+ + sig_subObj->inBuf[sig_subObj->inReadPos + 3];
+
+ /*Process a full item*/
+ s_result = sigProcess(this, sig_subObj->inReadPos, numinb,
+ sig_subObj->outWritePos, &numoutb);
+
+ if (s_result == PICO_OK) {
+ sig_subObj->inReadPos += numinb;
+ if (sig_subObj->inReadPos >= sig_subObj->inWritePos) {
+ sig_subObj->inReadPos = 0;
+ sig_subObj->inWritePos = 0;
+ sig_subObj->needMoreInput = FALSE;
+ }
+ sig_subObj->outWritePos += numoutb;
+ sig_subObj->procState = PICOSIG_FEED;
+ sig_subObj->retState = PICOSIG_COLLECT;
+ PICODBG_DEBUG(("picosig.sigStep -- leaving PICO_PROC, inReadPos = %i, outWritePos = %i",sig_subObj->inReadPos, sig_subObj->outWritePos));
+ return PICODATA_PU_BUSY; /*data to feed*/
+ }
+ return PICODATA_PU_BUSY; /*data still to process : remain in PROCESS STATE*/
+ break;
+
+ case PICOSIG_PLAY:
+
+ /*management of wav file play*/
+ s_data = (picoos_int16 *) &(sig_subObj->outBuf[sig_subObj->outWritePos + 4]);
+ hop_p_half = sig_subObj->sig_inner.hop_p / 2;
+ /*read directly into PU output buffer*/
+ n_samp = hop_p_half;
+ b_res = picoos_sdfGetSamples(sig_subObj->sInSDFile,
+ sig_subObj->sInSDFilePos, &n_samp, s_data);
+ sig_subObj->sInSDFilePos += n_samp;
+
+ if ((FALSE == b_res) || (0 == n_samp)) {
+ /*file play is complete or file read error*/
+ picoos_sdfCloseIn(this->common, &(sig_subObj->sInSDFile));
+ sig_subObj->sInSDFile = NULL;
+ sig_subObj->sInSDFileName[0] = '\0';
+ sig_subObj->procState = PICOSIG_COLLECT;
+ sig_subObj->retState = PICOSIG_COLLECT;
+ return PICODATA_PU_BUSY; /*check if data in input buffer*/
+ }
+ /*-----------------------------------*/
+ /*Volume control of wav file playback*/
+ /* (code borrowed from sigProcess)*/
+ /*Volume mod and clipping control */
+ /* directly into PU output buffer*/
+ /*-----------------------------------*/
+ sf_mlt = (picoos_int32) ((sig_subObj->vMod) * 16.0f);
+ s_t1 = &(s_data[0]);
+
+ for (n_i = 0; n_i < n_samp; n_i++) {
+ if (*s_t1 != 0) {
+ sf_data = (picoos_int32) (*s_t1) * sf_mlt;
+ sf_data >>= 4;
+ if (sf_data > PICOSIG_MAXAMP) {
+ sf_data = PICOSIG_MAXAMP;
+ } else if (sf_data < PICOSIG_MINAMP) {
+ sf_data = PICOSIG_MINAMP;
+ }
+ *s_t1 = (picoos_int16) (sf_data);
+ }
+ s_t1++;
+ }
+ /*Add header info*/
+ sig_subObj->outBuf[sig_subObj->outWritePos]
+ = (picoos_uint8) PICODATA_ITEM_FRAME;
+ sig_subObj->outBuf[sig_subObj->outWritePos + 1]
+ = (picoos_uint8) n_samp;
+ sig_subObj->outBuf[sig_subObj->outWritePos + 2]
+ = (picoos_uint8) (sig_subObj->nNumFrame % (hop_p_half)); /*number of frame % 64*/
+ sig_subObj->outBuf[sig_subObj->outWritePos + 3]
+ = (picoos_uint8) n_samp * 2;
+ /*Item content*/
+ sig_subObj->outWritePos += (n_samp * sizeof(picoos_int16)) + 4; /*including header*/
+ sig_subObj->procState = PICOSIG_FEED;
+ sig_subObj->retState = PICOSIG_PLAY;
+ break;
+
+ case PICOSIG_FEED:
+ /* ************** item output/feeding ***********************************/
+ switch (sig_subObj->outSwitch) {
+ case 0:
+ /*feeding items to PU output buffer*/
+ s_result = picodata_cbPutItem(this->cbOut,
+ &(sig_subObj->outBuf[sig_subObj->outReadPos]),
+ sig_subObj->outWritePos - sig_subObj->outReadPos,
+ &numoutb);
+ break;
+ case 1:
+ /*feeding items to file*/
+ if (sig_subObj->outBuf[sig_subObj->outReadPos]
+ == PICODATA_ITEM_FRAME) {
+ if ((sig_subObj->sOutSDFile) != NULL) {
+ n_start = (picoos_uint32) (sig_subObj->outReadPos)
+ + PICODATA_ITEM_HEADSIZE;
+ n_bytes = (picoos_uint32) sig_subObj->outBuf[(sig_subObj->outReadPos)
+ + PICODATA_ITEMIND_LEN];
+ n_fram = (picoos_uint32) sig_subObj->outBuf[(sig_subObj->outReadPos)
+ + PICODATA_ITEMIND_INFO2];
+ if (picoos_sdfPutSamples(
+ sig_subObj->sOutSDFile,
+ n_bytes / 2,
+ (picoos_int16*) &(sig_subObj->outBuf[n_start]))) {
+ PICODBG_DEBUG(("Nframe:%d - Nbytes %d\n", n_fram, n_bytes));
+ numoutb = n_bytes + 4;
+ s_result = PICO_OK;
+ /* also feed item to next PU */
+ s_result = picodata_cbPutItem(
+ this->cbOut,
+ &(sig_subObj->outBuf[sig_subObj->outReadPos]),
+ sig_subObj->outWritePos
+ - sig_subObj->outReadPos,
+ &numoutb);
+ } else {
+ /*write error : close file + cleanup handles*/
+ if (sig_subObj->sOutSDFile != NULL) {
+ picoos_sdfCloseOut(this->common, &(sig_subObj->sOutSDFile));
+ sig_subObj->sOutSDFile = NULL;
+ }
+ sig_subObj->sOutSDFileName[0] = '\0';
+ sig_subObj->outSwitch = 0;
+ PICODBG_DEBUG(("Error in writing :%d bytes to output file %s\n", numoutb, &(sig_subObj->sOutSDFileName[0])));
+ s_result = PICO_ERR_OTHER;
+ }
+ }
+ } else {
+ /*continue to feed following PU with items != FRAME */
+ s_result = picodata_cbPutItem(
+ this->cbOut,
+ &(sig_subObj->outBuf[sig_subObj->outReadPos]),
+ sig_subObj->outWritePos - sig_subObj->outReadPos,
+ &numoutb);
+ }
+ break;
+ default:
+ s_result = PICO_ERR_OTHER;
+ break;
+ }
+ PICODBG_DEBUG(("picosig.sigStep -- put item, status: %d",s_result));
+
+ if (PICO_OK == s_result) {
+
+ sig_subObj->outReadPos += numoutb;
+ *numBytesOutput = numoutb;
+ /*-------------------------*/
+ /*reset the output pointers*/
+ /*-------------------------*/
+ if (sig_subObj->outReadPos >= sig_subObj->outWritePos) {
+ sig_subObj->outReadPos = 0;
+ sig_subObj->outWritePos = 0;
+ sig_subObj->procState = sig_subObj->retState;
+ }
+ return PICODATA_PU_BUSY;
+
+ } else if (PICO_EXC_BUF_OVERFLOW == s_result) {
+
+ PICODBG_DEBUG(("picosig.sigStep ** feeding, overflow, PICODATA_PU_OUT_FULL"));
+ return PICODATA_PU_OUT_FULL;
+
+ } else if ((PICO_EXC_BUF_UNDERFLOW == s_result)
+ || (PICO_ERR_OTHER == s_result)) {
+
+ PICODBG_DEBUG(("picosig.sigStep ** feeding problem, discarding item"));
+ sig_subObj->outReadPos = 0;
+ sig_subObj->outWritePos = 0;
+ sig_subObj->procState = sig_subObj->retState;
+ return PICODATA_PU_ERROR;
+
+ }
+ break;
+ default:
+ /*NOT feeding items*/
+ s_result = PICO_EXC_BUF_IGNORE;
+ break;
+ }/*end switch*/
+ return PICODATA_PU_BUSY; /*check if there is more data to process after feeding*/
+
+ }/*end while*/
+ return PICODATA_PU_IDLE;
+}/*sigStep*/
+
+#ifdef __cplusplus
+}
+#endif
+
+/* Picosig.c end */
diff --git a/lib/picosig.h b/lib/picosig.h
new file mode 100644
index 0000000..3e29e8f
--- /dev/null
+++ b/lib/picosig.h
@@ -0,0 +1,64 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picosig.h
+ *
+ * Signal Generation PU - Header file
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+/**
+ * @addtogroup picosig
+ *
+ * <b> Pico Signal Generation module </b>\n
+ *
+ * Pico Sig is the PU that makes the parametric representation produce a speech signal.
+ * The PU receives parametric vectors and translates them into signal vectors.
+ * Most of the processing is based on this 1 to 1 relationship between input and output vectors.
+ *
+ * NOTE : "picosig" includes logically "picosig2" module, that is the DSP implementation of the signal generation.
+*/
+
+#ifndef PICOSIG_H_
+#define PICOSIG_H_
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+/* *******************************************************************************
+ * items related to the generic interface
+ ********************************************************************************/
+picodata_ProcessingUnit picosig_newSigUnit(
+ picoos_MemoryManager mm,
+ picoos_Common common,
+ picodata_CharBuffer cbIn,
+ picodata_CharBuffer cbOut,
+ picorsrc_Voice voice);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /*PICOSIG_H_*/
diff --git a/lib/picosig2.c b/lib/picosig2.c
new file mode 100644
index 0000000..e431c90
--- /dev/null
+++ b/lib/picosig2.c
@@ -0,0 +1,4090 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picosig2.c
+ *
+ * Signal Generation PU - Internal functions - Implementation
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+#include "picoos.h"
+#include "picodsp.h"
+#include "picosig2.h"
+#include "picofftsg.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+/*---------------------------------------------------------------------------
+ * INTERNAL FUNCTIONS DECLARATION
+ *---------------------------------------------------------------------------*/
+static void gen_hann2(sig_innerobj_t *sig_inObj);
+static void get_simple_excitation(sig_innerobj_t *sig_inObj,
+ picoos_int16 *nextPeak);
+static void enh_wind_init(sig_innerobj_t *sig_inObj);
+static void init_rand(sig_innerobj_t *sig_inObj);
+static void get_trig(picoos_int32 ang, picoos_int32 *table, picoos_int32 *cs,
+ picoos_int32 *sn);
+
+/*---------------------------------------------------------------------------
+ * PICO SYSTEM FUNCTIONS
+ *---------------------------------------------------------------------------*/
+/**
+ * allocation of DSP memory for SIG PU
+ * @param mm : memory manager
+ * @param sig_inObj : sig PU internal object of the sub-object
+ * @return PICO_OK : allocation successful
+ * @return PICO_ERR_OTHER : allocation NOT successful
+ * @callgraph
+ * @callergraph
+ */
+pico_status_t sigAllocate(picoos_MemoryManager mm, sig_innerobj_t *sig_inObj)
+{
+ picoos_int16 *data_i;
+ picoos_int32 *d32;
+ picoos_int32 nCount;
+
+ sig_inObj->int_vec22 =
+ sig_inObj->int_vec23 =
+ sig_inObj->int_vec24 =
+ sig_inObj->int_vec25 =
+ sig_inObj->int_vec26 =
+ sig_inObj->int_vec28 =
+ sig_inObj->int_vec29 =
+ sig_inObj->int_vec38 =
+ sig_inObj->int_vec30 =
+ sig_inObj->int_vec31 =
+ sig_inObj->int_vec32 =
+ sig_inObj->int_vec33 =
+ sig_inObj->int_vec34 =
+ sig_inObj->int_vec35 =
+ sig_inObj->int_vec36 =
+ sig_inObj->int_vec37 =
+ sig_inObj->int_vec38 =
+ sig_inObj->int_vec39 =
+ sig_inObj->int_vec40 = NULL;
+
+ sig_inObj->sig_vec1 = NULL;
+
+ sig_inObj->idx_vect1 = sig_inObj->idx_vect2 = sig_inObj->idx_vect4 = NULL;
+ sig_inObj->idx_vect5 = sig_inObj->idx_vect6 = sig_inObj->idx_vect7 =
+ sig_inObj->idx_vect8 = sig_inObj->idx_vect9 = NULL;
+ sig_inObj->ivalue17 = sig_inObj->ivalue18 = 0;
+
+ /*-----------------------------------------------------------------
+ * Memory allocations
+ * NOTE : it would be far better to do a single allocation
+ * and to do pointer initialization inside this routine
+ * ------------------------------------------------------------------*/
+ data_i = (picoos_int16 *) picoos_allocate(mm, sizeof(picoos_int16)
+ * PICODSP_FFTSIZE);
+ if (NULL == data_i) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->idx_vect1 = data_i;
+
+ data_i = (picoos_int16 *) picoos_allocate(mm, sizeof(picoos_int16)
+ * PICODSP_HFFTSIZE_P1);
+ if (NULL == data_i) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->idx_vect2 = data_i;
+
+ data_i = (picoos_int16 *) picoos_allocate(mm, sizeof(picoos_int16)
+ * PICODSP_FFTSIZE);
+ if (NULL == data_i) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->idx_vect4 = data_i;
+
+ data_i = (picoos_int16 *) picoos_allocate(mm, sizeof(picoos_int16)
+ * PICODSP_FFTSIZE);
+ if (NULL == data_i) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->idx_vect5 = data_i;
+
+ data_i = (picoos_int16 *) picoos_allocate(mm, sizeof(picoos_int16)
+ * PICODSP_FFTSIZE);
+ if (NULL == data_i) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->idx_vect6 = data_i;
+
+ data_i = (picoos_int16 *) picoos_allocate(mm, sizeof(picoos_int16)
+ * PICODSP_HFFTSIZE_P1);
+ if (NULL == data_i) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->idx_vect7 = data_i;
+
+ data_i = (picoos_int16 *) picoos_allocate(mm, sizeof(picoos_int16)
+ * PICODSP_MAX_EX);
+ if (NULL == data_i) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->idx_vect8 = data_i;
+
+ data_i = (picoos_int16 *) picoos_allocate(mm, sizeof(picoos_int16)
+ * PICODSP_MAX_EX);
+ if (data_i == NULL) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->idx_vect9 = data_i;
+
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32)
+ * PICODSP_FFTSIZE);
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec22 = d32;
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32)
+ * PICODSP_FFTSIZE);
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec23 = d32;
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32)
+ * PICODSP_FFTSIZE);
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec24 = d32;
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32)
+ * PICODSP_FFTSIZE);
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec25 = d32;
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32)
+ * PICODSP_FFTSIZE * 2);
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec26 = d32;
+
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32)
+ * PICODSP_FFTSIZE);
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec28 = d32;
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32)
+ * PICODSP_FFTSIZE);
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec29 = d32;
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32)
+ * PICODSP_FFTSIZE);
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec38 = d32;
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32)
+ * PICODSP_FFTSIZE);
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec30 = d32;
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32)
+ * PICODSP_FFTSIZE);
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec31 = d32;
+
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32)
+ * PICODSP_FFTSIZE);
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec32 = d32;
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32)
+ * PICODSP_FFTSIZE);
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec33 = d32;
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32)
+ * PICODSP_N_RAND_TABLE);
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec34 = d32;
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32)
+ * PICODSP_N_RAND_TABLE);
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec35 = d32;
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32)
+ * PICODSP_N_RAND_TABLE);
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec36 = d32;
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32)
+ * PICODSP_N_RAND_TABLE);
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec37 = d32;
+
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32)
+ * PICODSP_HFFTSIZE_P1);
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec39 = d32;
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32) * (1
+ + PICODSP_COS_TABLE_LEN));
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec40 = d32;
+
+ for (nCount = 0; nCount < CEPST_BUFF_SIZE; nCount++) {
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32) * (PICODSP_CEPORDER));
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec41[nCount] = d32;
+ }
+
+ for (nCount = 0; nCount < PHASE_BUFF_SIZE; nCount++) {
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32) * (PICODSP_PHASEORDER));
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->int_vec42[nCount] = d32;
+ }
+
+ d32 = (picoos_int32 *) picoos_allocate(mm, sizeof(picoos_int32)
+ * PICODSP_FFTSIZE * 2); /* - fixed point */
+ if (NULL == d32) {
+ sigDeallocate(mm, sig_inObj);
+ return PICO_ERR_OTHER;
+ }
+ sig_inObj->sig_vec1 = d32;
+
+ return PICO_OK;
+}/*sigAllocate*/
+
+/**
+ * frees DSP memory for SIG PU
+ * @param mm : memory manager
+ * @param sig_inObj : sig PU internal object of the sub-object
+ * @return void
+ * @callgraph
+ * @callergraph
+ */
+void sigDeallocate(picoos_MemoryManager mm, sig_innerobj_t *sig_inObj)
+{
+ picoos_int32 nCount;
+ /*-----------------------------------------------------------------
+ * Memory de-allocations
+ * ------------------------------------------------------------------*/
+ if (NULL != sig_inObj->idx_vect1)
+ picoos_deallocate(mm, (void *) &(sig_inObj->idx_vect1));
+ if (NULL != sig_inObj->idx_vect2)
+ picoos_deallocate(mm, (void *) &(sig_inObj->idx_vect2));
+ if (NULL != sig_inObj->idx_vect4)
+ picoos_deallocate(mm, (void *) &(sig_inObj->idx_vect4));
+ if (NULL != sig_inObj->idx_vect5)
+ picoos_deallocate(mm, (void *) &(sig_inObj->idx_vect5));
+ if (NULL != sig_inObj->idx_vect6)
+ picoos_deallocate(mm, (void *) &(sig_inObj->idx_vect6));
+ if (NULL != sig_inObj->idx_vect7)
+ picoos_deallocate(mm, (void *) &(sig_inObj->idx_vect7));
+ if (NULL != sig_inObj->idx_vect8)
+ picoos_deallocate(mm, (void *) &(sig_inObj->idx_vect8));
+ if (NULL != sig_inObj->idx_vect9)
+ picoos_deallocate(mm, (void *) &(sig_inObj->idx_vect9));
+
+ if (NULL != sig_inObj->int_vec22)
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec22));
+ if (NULL != sig_inObj->int_vec23)
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec23));
+ if (NULL != sig_inObj->int_vec24)
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec24));
+ if (NULL != sig_inObj->int_vec25)
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec25));
+ if (NULL != sig_inObj->int_vec26)
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec26));
+ if (NULL != sig_inObj->int_vec28)
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec28));
+ if (NULL != sig_inObj->int_vec29)
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec29));
+ if (NULL != sig_inObj->int_vec38)
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec38));
+ if (NULL != sig_inObj->int_vec30)
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec30));
+ if (NULL != sig_inObj->int_vec31)
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec31));
+ if (NULL != sig_inObj->int_vec32)
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec32));
+ if (NULL != sig_inObj->int_vec33)
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec33));
+ if (NULL != sig_inObj->int_vec34)
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec34));
+ if (NULL != sig_inObj->int_vec35)
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec35));
+ if (NULL != sig_inObj->int_vec36)
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec36));
+ if (NULL != sig_inObj->int_vec37)
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec37));
+ if (NULL != sig_inObj->int_vec39)
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec39));
+ if (NULL != sig_inObj->int_vec40)
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec40));
+
+ for (nCount = 0; nCount < CEPST_BUFF_SIZE; nCount++) {
+ if (NULL != sig_inObj->int_vec41[nCount]) {
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec41[nCount]));
+ }
+ }
+
+ for (nCount = 0; nCount < PHASE_BUFF_SIZE; nCount++) {
+ if (NULL != sig_inObj->int_vec42[nCount]) {
+ picoos_deallocate(mm, (void *) &(sig_inObj->int_vec42[nCount]));
+ }
+ }
+
+ if (NULL != sig_inObj->sig_vec1) {
+ picoos_deallocate(mm, (void *) &(sig_inObj->sig_vec1));
+ }
+}/*sigDeAllocate*/
+
+/**
+ * initializes all memory neededed by DSP at instance creation time
+ * @param sig_inObj : sig PU internal object of the sub-object
+ * @return void
+ * @callgraph
+ * @callergraph
+ */
+void sigDspInitialize(sig_innerobj_t *sig_inObj)
+{
+ picoos_int32 i, j;
+ picoos_int32 *pnt;
+
+ /*-----------------------------------------------------------------
+ * Initialization
+ * ------------------------------------------------------------------*/
+ sig_inObj->warp_p = PICODSP_FREQ_WARP_FACT;
+ sig_inObj->VCutoff_p = PICODSP_V_CUTOFF_FREQ; /*voicing cut off frequency in Hz (will be modeled in the future)*/
+ sig_inObj->UVCutoff_p = PICODSP_UV_CUTOFF_FREQ;/*unvoiced frames only (periodize lowest components to mask bad voicing transitions)*/
+ sig_inObj->Fs_p = PICODSP_SAMP_FREQ; /*Sampling freq*/
+
+ sig_inObj->m1_p = PICODSP_CEPORDER;
+ sig_inObj->m2_p = PICODSP_FFTSIZE; /*also initializes windowLen*/
+ sig_inObj->framesz_p = PICODSP_DISPLACE; /*1/4th of the frame size = displacement*/
+ sig_inObj->hfftsize_p = PICODSP_H_FFTSIZE; /*half of the FFT size*/
+ sig_inObj->voxbnd_p = (picoos_int32) ((picoos_single) sig_inObj->hfftsize_p
+ / ((picoos_single) sig_inObj->Fs_p / (picoos_single) 2)
+ * (picoos_single) sig_inObj->VCutoff_p);
+ sig_inObj->voxbnd2_p
+ = (picoos_int32) ((picoos_single) sig_inObj->hfftsize_p
+ / ((picoos_single) sig_inObj->Fs_p / (picoos_single) 2)
+ * (picoos_single) sig_inObj->UVCutoff_p);
+ sig_inObj->hop_p = sig_inObj->framesz_p;
+ sig_inObj->nextPeak_p = (((int) (PICODSP_FFTSIZE))
+ / ((int) PICODSP_DISPLACE) - 1) * sig_inObj->hop_p;
+ sig_inObj->phId_p = 0; /*phonetic id*/
+ sig_inObj->E_p = (picoos_single) 0.0f;
+ sig_inObj->F0_p = (picoos_single) 0.0f;
+ sig_inObj->voiced_p = 0;
+ sig_inObj->nV = sig_inObj->nU = 0;
+ sig_inObj->sMod_p = (picoos_single) 1.0f;
+
+ /*cleanup vectors*/
+ for (i = 0; i < 2 * PICODSP_FFTSIZE; i++) {
+ sig_inObj->sig_vec1[i] = 0;
+ sig_inObj->int_vec26[i] = 0; /*wav buff cleanup */
+ }
+
+ for (i = 0; i < PICODSP_FFTSIZE; i++) {
+ sig_inObj->idx_vect1[i] = sig_inObj->idx_vect4[i]
+ = sig_inObj->idx_vect5[i] = sig_inObj->idx_vect6[i] = 0;
+ sig_inObj->int_vec32[i] = sig_inObj->int_vec33[i] = 0;
+ }
+
+ for (i = 0; i < PICODSP_HFFTSIZE_P1; i++) {
+ sig_inObj->idx_vect2[i] = (picoos_int16) 0;
+ }
+
+ for (i = 0; i < CEPST_BUFF_SIZE; i++) {
+ sig_inObj->F0Buff[i]=0;
+ sig_inObj->PhIdBuff[i]=0;
+ sig_inObj->VoicingBuff[i]=0;
+ sig_inObj->FuVBuff[i]=0;
+ if (NULL != sig_inObj->CepBuff[i]) {
+ pnt = sig_inObj->CepBuff[i];
+ for (j = 0; j < PICODSP_CEPORDER; j++) {
+ pnt[j] = 0;
+ }
+ }
+ }
+
+ for (i = 0; i < PHASE_BUFF_SIZE; i++) {
+ if (NULL != sig_inObj->int_vec42[i]) {
+ pnt = sig_inObj->int_vec42[i];
+ for (j = 0; j < PICODSP_PHASEORDER; j++) {
+ pnt[j] = 0;
+ }
+ }
+ }
+ sig_inObj->n_available=0;
+ /*---------------------------------------------
+ Init formant enhancement window
+ hanning window,
+ Post Filter Hermite's interpolator Matrix
+ Mel-2-Lin lookup tables
+ ---------------------------------------------*/
+ enh_wind_init(sig_inObj); /*creates the formant enhancement window*/
+ init_rand(sig_inObj);
+ gen_hann2(sig_inObj);
+ mel_2_lin_init(sig_inObj);
+
+}/*sigDspInitialize*/
+
+/*-------------------------------------------------------------------------------
+ PROCESSING FUNCTIONS : CALLED WITHIN sigStep (cfr. picosig.c)
+ --------------------------------------------------------------------------------*/
+/**
+ * convert from mel scale to linear scale
+ * @param sig_inObj : sig PU internal object of the sub-object
+ * @param scmeanMGC : mean value of the MGC
+ * @return void
+ * @callgraph
+ * @callergraph
+ * @remarks translated from matlab code to c-code
+ * Input
+ * - c1 : input mfcc vector (ceporder=m1, real)
+ * - m1 : input order
+ * - A,B,D : lookup tables
+ * - m2 : output order
+ * - Xr,Xi (m2=FFT size, real) temporary arrays for FFT
+ * - WNr,WNi (m2=FFT size, real) cos and sin precalculated tables
+ * Output
+ * - Xr (m2=FFT size, real) linear cepstral vector
+ */
+void mel_2_lin_lookup(sig_innerobj_t *sig_inObj, picoos_uint32 scmeanMGC)
+{
+ /*Local vars*/
+ picoos_int16 nI, k;
+ picoos_int32 delta, term1, term2;
+
+ /*Local vars to be linked with sig data object*/
+ picoos_int32 *c1, *XXr;
+ picoos_single K1;
+ picoos_int32 *D, K2, shift;
+ picoos_int16 m1, *A, m2, m4, voiced, i;
+
+ /*Link local variables with sig data object*/
+ c1 = sig_inObj->wcep_pI;
+ m1 = sig_inObj->m1_p;
+ m2 = PICODSP_FFTSIZE;
+ m4 = m2 >> 1;
+
+ A = sig_inObj->A_p;
+ D = sig_inObj->d_p;
+
+ XXr = sig_inObj->wcep_pI;
+ voiced = sig_inObj->voiced_p;
+
+ shift = 27 - scmeanMGC;
+ K2 = 1 << shift;
+ K1 = (picoos_single) PICODSP_START_FLOAT_NORM * K2;
+ XXr[0] = (picoos_int32) ((picoos_single) c1[0] * K1);
+ for (nI = 1; nI < m1; nI++) {
+ XXr[nI] = c1[nI] << shift;
+ }
+ i = sizeof(picoos_int32) * (PICODSP_FFTSIZE + 1 - m1);
+ picoos_mem_set(XXr + m1, 0, i);
+ dfct_nmf(m4, XXr); /* DFCT directly in fixed point */
+
+ /* *****************************************************************************************
+ Linear frequency scale envelope through interpolation.
+ Two additions and one multiplication per entry.
+
+ Optimization of linear interpolation algorithm
+ - Start from 1 and stop at PICODSP_H_FFTSIZE-1 because 0 and PICODSP_H_FFTSIZE are invariant points
+ - B[k]=A[k]+1 except for 0 and PICODSP_H_FFTSIZE
+ - get rid of extra -1 operation by adapting the table A[]
+
+ *******************************************************************************************/
+ for (nI = 1; nI < PICODSP_H_FFTSIZE; nI++) {
+ k = A[nI];
+ term2 = XXr[k];
+ term1 = XXr[k + 1];
+ delta = term1 - term2;
+ XXr[nI] = term2 + ((D[nI] * delta) >> 5); /* ok because nI<=A[nI] <=B[nI] */
+ }
+}/*mel_2_lin_lookup*/
+
+/**
+ * calculate phase
+ * @remarks voiced phase taken from phase codebook and smoothed,
+ * @remarks unvoiced phase - random
+ * @param sig_inObj : sig PU internal object of the sub-object
+ * @return void
+ * @callgraph
+ * @callergraph
+ */
+void phase_spec2(sig_innerobj_t *sig_inObj)
+{
+ picoos_int16 nI, iRand, firstUV;
+ picoos_int32 *tmp1, *tmp2;
+ picoos_int16 VOXBND_M1;
+ picoos_int32 *spect, *ang;
+ picoos_int16 voiced, m2;
+ picoos_int32 *co, *so, *c, *s, voxbnd, voxbnd2;
+ picoos_int16 i,j, k, n_comp;
+ picoos_int16 *Pvoxbnd;
+ picoos_int32 *phs_p2, *phs_p1, *phs_n1, *phs_n2;
+ picoos_int32 *phs;
+
+ /*Link local variables to sig data object*/
+ spect = sig_inObj->wcep_pI; /* spect_p;*/
+ /* current spect scale : times PICODSP_FIX_SCALE1 */
+ ang = sig_inObj->ang_p;
+ voxbnd = (picoos_int32) (sig_inObj->voxbnd_p * sig_inObj->voicing);
+ voxbnd2 = sig_inObj->voxbnd2_p;
+ voiced = sig_inObj->voiced_p;
+ m2 = sig_inObj->m2_p;
+ VOXBND_M1 = voxbnd - 1;
+ firstUV = 1;
+
+ /*code starts here*/
+ if (voiced == 1) {
+ firstUV = voxbnd;
+ Pvoxbnd = sig_inObj->VoxBndBuff;
+ n_comp = Pvoxbnd[2];
+ phs_p2 = sig_inObj->PhsBuff[0];
+ phs_p1 = sig_inObj->PhsBuff[1];
+ phs = sig_inObj->PhsBuff[2];
+ phs_n1 = sig_inObj->PhsBuff[3];
+ phs_n2 = sig_inObj->PhsBuff[4];
+
+ /* find and smooth components which have full context */
+ j = n_comp;
+ for (i=0; i<5; i++) {
+ if (Pvoxbnd[i]<j) j = Pvoxbnd[i];
+ }
+ for (i=0; i<j; i++) {
+ ang[i] = -(((phs_p2[i]+phs_p1[i]+phs[i]+phs_n1[i]+phs_n2[i])<<6) / 5);
+ }
+
+ /* find and smooth components which at least one component on each side */
+ k = n_comp;
+ if (Pvoxbnd[2]<k) k = Pvoxbnd[2];
+ if (Pvoxbnd[4]<k) k = Pvoxbnd[4];
+ for (i=j; i<k; i++) { /* smooth using only two surrounding neighbours */
+ ang[i] = -(((phs_p1[i]+phs[i]+phs_n1[i])<<6) / 3);
+ }
+
+ /* handle rest of components - at least one side does not exist */
+ for (i=k; i<n_comp; i++) {
+ ang[i] = -(phs[i]<<6); /* - simple copy without smoothing */
+ }
+
+ /*Phase unwrap - cumsum */
+ tmp1 = &(ang[1]);
+ tmp2 = &(ang[0]);
+ /* current ang scale : PICODSP_M_PI = PICODSP_FIX_SCALE2 */
+ FAST_DEVICE(VOXBND_M1,*(tmp1++)+=*(tmp2)-PICODSP_FIX_SCALE2;*(tmp2)=(*tmp2>=0)?(*tmp2)>>PICODSP_SHIFT_FACT4:-((-(*tmp2))>>PICODSP_SHIFT_FACT4);tmp2++);
+ *tmp2 = (*tmp2 >= 0) ? (*tmp2) >> PICODSP_SHIFT_FACT4 : -((-(*tmp2))
+ >> PICODSP_SHIFT_FACT4); /*ang[voxbnd-1]/=2;*/
+ }
+
+ /* now for the unvoiced part */
+ iRand = sig_inObj->iRand;
+ c = sig_inObj->randCosTbl + iRand;
+ s = sig_inObj->randSinTbl + iRand;
+ co = sig_inObj->outCosTbl + firstUV;
+ so = sig_inObj->outSinTbl + firstUV;
+ for (nI = firstUV; nI < PICODSP_HFFTSIZE_P1 - 1; nI++) {
+ *co++ = *c++;
+ *so++ = *s++;
+ }
+ *co = 1;
+ *so = 0;
+ sig_inObj->iRand += (PICODSP_HFFTSIZE_P1 - firstUV);
+ if (sig_inObj->iRand > PICODSP_N_RAND_TABLE - PICODSP_HFFTSIZE_P1)
+ sig_inObj->iRand = 1 + sig_inObj->iRand + PICODSP_HFFTSIZE_P1
+ - PICODSP_N_RAND_TABLE;
+}/*phase_spec2*/
+
+/**
+ * Prepare Envelope spectrum for inverse FFT
+ * @param sig_inObj : sig PU internal object of the sub-object
+ * @return void
+ * @remarks make phase bilateral -->> angh (FFT size, real)
+ * @remarks combine in complex input vector for IFFT F = e**(spet/2+j*ang)
+ * @remarks Compute energy -->> E (scalar, real)
+ * @callgraph
+ * @callergraph
+ * Input
+ * - spect (FFT size, real)
+ * - ang (half FFT size -1, real)
+ * - m2 fftsize
+ * - WNr,WNi (FFT size, real) the tabulated sine and cosine values
+ * - brev (FFT size, real) the tabulated bit reversal indexes
+ * Output
+ * - Fr, Fi (FFT size, complex) the envelope spectrum
+ * - E (scalar, real) the energy
+ */
+void env_spec(sig_innerobj_t *sig_inObj)
+{
+
+ picoos_int16 nI;
+ picoos_int32 fcX, fsX, fExp, voxbnd;
+ picoos_int32 *spect, *ang, *ctbl;
+ picoos_int16 voiced, prev_voiced;
+ picoos_int32 *co, *so;
+ picoos_int32 *Fr, *Fi;
+ picoos_single mult;
+
+ /*Link local variables to sig object*/
+ spect = sig_inObj->wcep_pI; /*spect_p*/
+ /* current spect scale : times PICODSP_FIX_SCALE1 */
+ ang = sig_inObj->ang_p;
+ /* current spect scale : PICODSP_M_PI = PICODSP_FIX_SCALE2 */
+ Fr = sig_inObj->F2r_p;
+ Fi = sig_inObj->F2i_p;
+ voiced = sig_inObj->voiced_p;
+ prev_voiced = sig_inObj->prevVoiced_p;
+ voxbnd = (picoos_int32) (sig_inObj->voxbnd_p * sig_inObj->voicing);
+ ctbl = sig_inObj->cos_table;
+ /* ctbl scale : times 4096 */
+ mult = PICODSP_ENVSPEC_K1 / PICODSP_FIX_SCALE1;
+
+ /*remove dc from real part*/
+ if (sig_inObj->F0_p > 120) {
+ spect[0] = spect[1] = 0;
+ spect[2] /= PICODSP_ENVSPEC_K2;
+ } else {
+ spect[0] = 0;
+ }
+
+ /* if using rand table, use sin and cos tables as well */
+ if (voiced || (prev_voiced)) {
+ /*Envelope becomes a complex exponential : F=exp(.5*spect + j*angh);*/
+ for (nI = 0; nI < voxbnd; nI++) {
+ get_trig(ang[nI], ctbl, &fcX, &fsX);
+ fExp = (picoos_int32) EXP((double)spect[nI]*mult);
+ Fr[nI] = fExp * fcX;
+ Fi[nI] = fExp * fsX;
+ }
+ /* ao=sig_inObj->ang_p+(picoos_int32)voxbnd; */
+ co = sig_inObj->outCosTbl + voxbnd;
+ so = sig_inObj->outSinTbl + voxbnd;
+
+ for (nI = voxbnd; nI < PICODSP_HFFTSIZE_P1; nI++) {
+ fcX = *co++;
+ fsX = *so++;
+ fExp = (picoos_int32) EXP((double)spect[nI]*mult);
+ Fr[nI] = fExp * fcX;
+ Fi[nI] = fExp * fsX;
+ }
+ } else {
+ /*ao=sig_inObj->ang_p+1;*/
+ co = sig_inObj->outCosTbl + 1;
+ so = sig_inObj->outSinTbl + 1;
+ for (nI = 1; nI < PICODSP_HFFTSIZE_P1; nI++) {
+ fcX = *co++;
+ fsX = *so++;
+ fExp = (picoos_int32) EXP((double)spect[nI]*mult);
+
+ Fr[nI] = fExp * fcX;
+ Fi[nI] = fExp * fsX;
+ }
+ }
+
+}/*env_spec*/
+
+/**
+ * Calculates the impulse response of the comlpex spectrum through inverse rFFT
+ * @param sig_inObj : sig PU internal object of the sub-object
+ * @return void
+ * @remarks Imp corresponds with the real part of the FFT
+ * @callgraph
+ * @callergraph
+ * Input
+ * - Fr, Fi (FFT size, real & imaginary) the complex envelope spectrum (only first half of spectrum)
+ * Output
+ * - Imp: impulse response (length: m2)
+ * - E (scalar, real) RMS value
+ */
+void impulse_response(sig_innerobj_t *sig_inObj)
+{
+ /*Define local variables*/
+ picoos_single f;
+ picoos_int16 nI, nn, m2, m4, voiced;
+ picoos_single *E;
+ picoos_int32 *norm_window; /* - fixed point */
+ picoos_int32 *fr, *Fr, *Fi, *t1, ff; /* - fixed point */
+
+ /*Link local variables with sig object*/
+ m2 = sig_inObj->m2_p;
+ m4 = m2 >> 1;
+ Fr = sig_inObj->F2r_p;
+ Fi = sig_inObj->F2i_p;
+ norm_window = sig_inObj->norm_window_p;
+ E = &(sig_inObj->E_p); /*as pointer: value will be modified*/
+ voiced = sig_inObj->voiced_p;
+ fr = sig_inObj->imp_p;
+
+ /*Inverse FFT*/
+ for (nI = 0, nn = 0; nI < m4; nI++, nn += 2) {
+ fr[nn] = Fr[nI]; /* - fixed point */
+ }
+
+ fr[1] = (picoos_int32) (Fr[m4]);
+ for (nI = 1, nn = 3; nI < m4; nI++, nn += 2) {
+ fr[nn] = -Fi[nI]; /* - fixed point */
+ }
+
+ rdft(m2, -1, fr);
+ /*window, normalize and differentiate*/
+ *E = norm_result(m2, fr, norm_window);
+
+ if (*E > 0)
+ f = *E * PICODSP_FIXRESP_NORM;
+ else
+ f = 20; /*PICODSP_FIXRESP_NORM*/
+ ff = (picoos_int32) f;
+ if (ff < 1)
+ ff = 1;
+ /*normalize impulse response*/
+ t1 = fr;FAST_DEVICE(PICODSP_FFTSIZE,*(t1++) /= ff;); /* - fixed point */
+
+} /* impulse_response */
+
+/**
+ * time domain pitch synchronous overlap add over two frames (when no voicing transition)
+ * @param sig_inObj : sig PU internal object of the sub-object
+ * @return void
+ * @remarks Special treatment at voicing boundaries
+ * @remarks Introduced to get rid of time-domain aliasing (and additional speed up)
+ * @callgraph
+ * @callergraph
+ */
+void td_psola2(sig_innerobj_t *sig_inObj)
+{
+ picoos_int16 nI;
+ picoos_int16 hop, m2, *nextPeak, voiced;
+ picoos_int32 *t1, *t2;
+ picoos_int16 cnt;
+ picoos_int32 *fr, *v1, ff, f;
+ picoos_int16 a, i;
+ picoos_int32 *window;
+ picoos_int16 s = (picoos_int16) 1;
+ window = sig_inObj->window_p;
+
+ /*Link local variables with sig object*/
+ hop = sig_inObj->hop_p;
+ m2 = sig_inObj->m2_p;
+ nextPeak = &(sig_inObj->nextPeak_p);
+ voiced = sig_inObj->voiced_p;
+ fr = sig_inObj->imp_p;
+ /*toggle the pointers and initialize signal vector */
+ v1 = sig_inObj->sig_vec1;
+
+ t1 = v1;
+ FAST_DEVICE(PICODSP_FFTSIZE-PICODSP_DISPLACE,*(t1++)=0;);
+ t1 = &(v1[PICODSP_FFTSIZE - PICODSP_DISPLACE]);
+ t2 = &(v1[PICODSP_FFTSIZE]);
+ FAST_DEVICE(PICODSP_FFTSIZE, *(t1++)=*(t2++););
+ t1 = &(v1[2 * PICODSP_FFTSIZE - PICODSP_DISPLACE]);FAST_DEVICE(PICODSP_DISPLACE,*(t1++)=0;);
+ /*calculate excitation points*/
+ get_simple_excitation(sig_inObj, nextPeak);
+
+ /*TD-PSOLA based on excitation vector */
+ if ((sig_inObj->nU == 0) && (sig_inObj->voiced_p == 1)) {
+ /* purely voiced */
+ for (nI = 0; nI < sig_inObj->nV; nI++) {
+ f = sig_inObj->EnV[nI];
+ a = 0;
+ cnt = PICODSP_FFTSIZE;
+ ff = (f * window[sig_inObj->LocV[nI]]) >> PICODSP_SHIFT_FACT1;
+ t1 = &(v1[a + sig_inObj->LocV[nI]]);
+ t2 = &(fr[a]);
+ if (cnt > 0)FAST_DEVICE(cnt,*(t1++)+=*(t2++)*ff;);
+ }
+ } else if ((sig_inObj->nV == 0) && (sig_inObj->voiced_p == 0)) {
+ /* PURELY UNVOICED*/
+ for (nI = 0; nI < sig_inObj->nU; nI++) {
+ f = sig_inObj->EnU[nI];
+ s = -s; /*reverse order to reduce the periodicity effect*/
+ if (s == 1) {
+ a = 0;
+ cnt = PICODSP_FFTSIZE;
+ ff = (f * window[sig_inObj->LocU[nI]]) >> PICODSP_SHIFT_FACT1;
+ t1 = &(v1[a + sig_inObj->LocU[nI]]);
+ t2 = &(fr[a]);
+ if (cnt > 0)FAST_DEVICE(cnt,*(t1++)+=*(t2++)*ff; );
+ } else { /*s==-1*/
+ a = 0;
+ cnt = PICODSP_FFTSIZE;
+ ff = (f * window[sig_inObj->LocU[nI]]) >> PICODSP_SHIFT_FACT1;
+ t1 = &(v1[(m2 - 1 - a) + sig_inObj->LocU[nI]]);
+ t2 = &(fr[a]);
+ if (cnt > 0)FAST_DEVICE(cnt,*(t1--)+=*(t2++)*ff; );
+ }
+ }
+ } else if (sig_inObj->VoicTrans == 0) {
+ /*voicing transition from unvoiced to voiced*/
+ for (nI = 0; nI < sig_inObj->nV; nI++) {
+ f = sig_inObj->EnV[nI];
+ a = 0;
+ cnt = PICODSP_FFTSIZE;
+ ff = (f * window[sig_inObj->LocV[nI]]) >> PICODSP_SHIFT_FACT1;
+ t1 = &(v1[a + sig_inObj->LocV[nI]]);
+ t2 = &(fr[a]);
+ if (cnt > 0)FAST_DEVICE(cnt,*(t1++)+=*(t2++)*ff;);
+ }
+ /*add remaining stuff from unvoiced part*/
+ for (nI = 0; nI < sig_inObj->nU; nI++) {
+ f = sig_inObj->EnU[nI];
+ s = -s; /*reverse order to reduce the periodicity effect*/
+ if (s == 1) {
+ a = 0;
+ cnt = PICODSP_FFTSIZE;
+ ff = (f * window[sig_inObj->LocU[nI]]) >> PICODSP_SHIFT_FACT1;
+ t1 = &(v1[a + sig_inObj->LocU[nI]]);
+ t2 = &(sig_inObj->ImpResp_p[a]); /*saved impulse response*/
+ if (cnt > 0)FAST_DEVICE(cnt,*(t1++)+=*(t2++)*ff; );
+ } else {
+ a = 0;
+ cnt = PICODSP_FFTSIZE;
+ ff = (f * window[sig_inObj->LocU[nI]]) >> PICODSP_SHIFT_FACT1;
+ t1 = &(v1[(m2 - 1 - a) + sig_inObj->LocU[nI]]);
+ t2 = &(sig_inObj->ImpResp_p[a]);
+ if (cnt > 0)FAST_DEVICE(cnt,*(t1--)+=*(t2++)*ff; );
+ }
+ }
+ } else {
+ /*voiced to unvoiced*/
+ for (nI = 0; nI < sig_inObj->nU; nI++) {
+ f = sig_inObj->EnU[nI];
+ s = -s; /*reverse order to reduce the periodicity effect*/
+ if (s > 0) {
+ a = 0;
+ cnt = PICODSP_FFTSIZE;
+ ff = (f * window[sig_inObj->LocU[nI]]) >> PICODSP_SHIFT_FACT1;
+ t1 = &(v1[a + sig_inObj->LocU[nI]]);
+ t2 = &(fr[a]);
+ if (cnt > 0)FAST_DEVICE(cnt,*(t1++)+=*(t2++)*ff; );
+ } else {
+ a = 0;
+ cnt = PICODSP_FFTSIZE;
+ ff = (f * window[sig_inObj->LocU[nI]]) >> PICODSP_SHIFT_FACT1;
+ t1 = &(v1[(m2 - 1 - a) + sig_inObj->LocU[nI]]);
+ t2 = &(fr[a]);
+ if (cnt > 0)FAST_DEVICE(cnt,*(t1--)+=*(t2++)*ff; );
+ }
+ }
+ /*add remaining stuff from voiced part*/
+ for (nI = 0; nI < sig_inObj->nV; nI++) {
+ f = sig_inObj->EnV[nI];
+ a = 0;
+ cnt = PICODSP_FFTSIZE;
+ ff = (f * window[sig_inObj->LocV[nI]]) >> PICODSP_SHIFT_FACT1;
+ t1 = &(v1[a + sig_inObj->LocV[nI]]);
+ t2 = &(sig_inObj->ImpResp_p[a]);
+ if (cnt > 0)FAST_DEVICE(cnt,*(t1++)+=*(t2++)*ff;);
+ }
+ }
+
+ t1 = sig_inObj->sig_vec1;
+ for (i = 0; i < PICODSP_FFTSIZE; i++, t1++) {
+ if (*t1 >= 0)
+ *t1 >>= PICODSP_SHIFT_FACT5;
+ else
+ *t1 = -((-*t1) >> PICODSP_SHIFT_FACT5);
+ }
+
+}/*td_psola2*/
+
+/**
+ * overlap + add summing of impulse responses on the final destination sample buffer
+ * @param sig_inObj : sig PU internal object of the sub-object
+ * @return void
+ * @remarks Special treatment at voicing boundaries
+ * @remarks Introduced to get rid of time-domain aliasing (and additional speed up)
+ * Input
+ * - wlet : the generic impulse response (FFT size, real)
+ * - window : the windowing funcion (FFT size, real) fixed
+ * - WavBuff : the destination buffer with past samples (FFT size*2, short)
+ * - m2 : fftsize
+ * Output
+ * - WavBuff : the destination buffer with updated samples (FFT size*2, short)
+ * @callgraph
+ * @callergraph
+ */
+void overlap_add(sig_innerobj_t *sig_inObj)
+{
+ /*Local variables*/
+ picoos_int32 *w, *v;
+
+ /*Link local variables with sig object*/
+ w = sig_inObj->WavBuff_p;
+ v = sig_inObj->sig_vec1;
+
+ FAST_DEVICE(PICODSP_FFTSIZE, *(w++)+=*(v++)<<PICODSP_SHIFT_FACT6;);
+
+}/*overlap_add*/
+
+/*-------------------------------------------------------------------------------
+ INITIALIZATION AND INTERNAL FUNCTIONS
+ --------------------------------------------------------------------------------*/
+/**
+ * Hanning window initialization
+ * @param sig_inObj : sig PU internal object of the sub-object
+ * @return PICO_OK
+ * @callgraph
+ * @callergraph
+ */
+static void gen_hann2(sig_innerobj_t *sig_inObj)
+{
+ picoos_int32 *hann;
+ picoos_int32 *norm;
+ /*link local variables with sig object*/
+ hann = sig_inObj->window_p;
+ norm = sig_inObj->norm_window_p;
+
+ norm[0] = 80224;
+ norm[1] = 320832;
+ norm[2] = 721696;
+ norm[3] = 1282560;
+ norm[4] = 2003104;
+ norm[5] = 2882880;
+ norm[6] = 3921376;
+ norm[7] = 5117984;
+ norm[8] = 6471952;
+ norm[9] = 7982496;
+ norm[10] = 9648720;
+ norm[11] = 11469616;
+ norm[12] = 13444080;
+ norm[13] = 15570960;
+ norm[14] = 17848976;
+ norm[15] = 20276752;
+ norm[16] = 22852864;
+ norm[17] = 25575744;
+ norm[18] = 28443776;
+ norm[19] = 31455264;
+ norm[20] = 34608368;
+ norm[21] = 37901248;
+ norm[22] = 41331904;
+ norm[23] = 44898304;
+ norm[24] = 48598304;
+ norm[25] = 52429696;
+ norm[26] = 56390192;
+ norm[27] = 60477408;
+ norm[28] = 64688944;
+ norm[29] = 69022240;
+ norm[30] = 73474720;
+ norm[31] = 78043744;
+ norm[32] = 82726544;
+ norm[33] = 87520352;
+ norm[34] = 92422272;
+ norm[35] = 97429408;
+ norm[36] = 102538752;
+ norm[37] = 107747248;
+ norm[38] = 113051776;
+ norm[39] = 118449184;
+ norm[40] = 123936224;
+ norm[41] = 129509648;
+ norm[42] = 135166080;
+ norm[43] = 140902192;
+ norm[44] = 146714528;
+ norm[45] = 152599584;
+ norm[46] = 158553904;
+ norm[47] = 164573888;
+ norm[48] = 170655936;
+ norm[49] = 176796448;
+ norm[50] = 182991712;
+ norm[51] = 189238064;
+ norm[52] = 195531744;
+ norm[53] = 201868992;
+ norm[54] = 208246016;
+ norm[55] = 214659040;
+ norm[56] = 221104176;
+ norm[57] = 227577616;
+ norm[58] = 234075488;
+ norm[59] = 240593872;
+ norm[60] = 247128912;
+ norm[61] = 253676688;
+ norm[62] = 260233280;
+ norm[63] = 266794768;
+ norm[64] = 273357248;
+ norm[65] = 279916768;
+ norm[66] = 286469440;
+ norm[67] = 293011360;
+ norm[68] = 299538560;
+ norm[69] = 306047168;
+ norm[70] = 312533312;
+ norm[71] = 318993088;
+ norm[72] = 325422656;
+ norm[73] = 331818144;
+ norm[74] = 338175744;
+ norm[75] = 344491680;
+ norm[76] = 350762176;
+ norm[77] = 356983424;
+ norm[78] = 363151808;
+ norm[79] = 369263520;
+ norm[80] = 375315008;
+ norm[81] = 381302592;
+ norm[82] = 387222720;
+ norm[83] = 393071872;
+ norm[84] = 398846528;
+ norm[85] = 404543232;
+ norm[86] = 410158560;
+ norm[87] = 415689216;
+ norm[88] = 421131840;
+ norm[89] = 426483200;
+ norm[90] = 431740096;
+ norm[91] = 436899392;
+ norm[92] = 441958016;
+ norm[93] = 446912928;
+ norm[94] = 451761152;
+ norm[95] = 456499840;
+ norm[96] = 461126080;
+ norm[97] = 465637152;
+ norm[98] = 470030400;
+ norm[99] = 474303104;
+ norm[100] = 478452800;
+ norm[101] = 482476960;
+ norm[102] = 486373184;
+ norm[103] = 490139200;
+ norm[104] = 493772640;
+ norm[105] = 497271424;
+ norm[106] = 500633440;
+ norm[107] = 503856704;
+ norm[108] = 506939200;
+ norm[109] = 509879168;
+ norm[110] = 512674880;
+ norm[111] = 515324544;
+ norm[112] = 517826688;
+ norm[113] = 520179776;
+ norm[114] = 522382368;
+ norm[115] = 524433184;
+ norm[116] = 526331008;
+ norm[117] = 528074688;
+ norm[118] = 529663200;
+ norm[119] = 531095552;
+ norm[120] = 532370944;
+ norm[121] = 533488576;
+ norm[122] = 534447808;
+ norm[123] = 535248000;
+ norm[124] = 535888768;
+ norm[125] = 536369664;
+ norm[126] = 536690432;
+ norm[127] = 536850880;
+ norm[128] = 536850880;
+ norm[129] = 536690432;
+ norm[130] = 536369664;
+ norm[131] = 535888768;
+ norm[132] = 535248000;
+ norm[133] = 534447808;
+ norm[134] = 533488576;
+ norm[135] = 532370944;
+ norm[136] = 531095552;
+ norm[137] = 529663200;
+ norm[138] = 528074688;
+ norm[139] = 526331008;
+ norm[140] = 524433216;
+ norm[141] = 522382368;
+ norm[142] = 520179776;
+ norm[143] = 517826688;
+ norm[144] = 515324544;
+ norm[145] = 512674880;
+ norm[146] = 509879168;
+ norm[147] = 506939200;
+ norm[148] = 503856704;
+ norm[149] = 500633472;
+ norm[150] = 497271424;
+ norm[151] = 493772672;
+ norm[152] = 490139200;
+ norm[153] = 486373184;
+ norm[154] = 482476992;
+ norm[155] = 478452800;
+ norm[156] = 474303104;
+ norm[157] = 470030400;
+ norm[158] = 465637184;
+ norm[159] = 461126080;
+ norm[160] = 456499840;
+ norm[161] = 451761152;
+ norm[162] = 446912960;
+ norm[163] = 441958016;
+ norm[164] = 436899424;
+ norm[165] = 431740096;
+ norm[166] = 426483200;
+ norm[167] = 421131840;
+ norm[168] = 415689216;
+ norm[169] = 410158560;
+ norm[170] = 404543232;
+ norm[171] = 398846528;
+ norm[172] = 393071872;
+ norm[173] = 387222720;
+ norm[174] = 381302592;
+ norm[175] = 375315008;
+ norm[176] = 369263552;
+ norm[177] = 363151808;
+ norm[178] = 356983456;
+ norm[179] = 350762176;
+ norm[180] = 344491712;
+ norm[181] = 338175776;
+ norm[182] = 331818144;
+ norm[183] = 325422656;
+ norm[184] = 318993088;
+ norm[185] = 312533312;
+ norm[186] = 306047168;
+ norm[187] = 299538560;
+ norm[188] = 293011360;
+ norm[189] = 286469472;
+ norm[190] = 279916800;
+ norm[191] = 273357248;
+ norm[192] = 266794784;
+ norm[193] = 260233280;
+ norm[194] = 253676688;
+ norm[195] = 247128928;
+ norm[196] = 240593888;
+ norm[197] = 234075488;
+ norm[198] = 227577632;
+ norm[199] = 221104192;
+ norm[200] = 214659040;
+ norm[201] = 208246032;
+ norm[202] = 201868992;
+ norm[203] = 195531744;
+ norm[204] = 189238080;
+ norm[205] = 182991728;
+ norm[206] = 176796448;
+ norm[207] = 170655952;
+ norm[208] = 164573888;
+ norm[209] = 158553920;
+ norm[210] = 152599600;
+ norm[211] = 146714528;
+ norm[212] = 140902208;
+ norm[213] = 135166096;
+ norm[214] = 129509648;
+ norm[215] = 123936240;
+ norm[216] = 118449184;
+ norm[217] = 113051776;
+ norm[218] = 107747248;
+ norm[219] = 102538752;
+ norm[220] = 97429424;
+ norm[221] = 92422288;
+ norm[222] = 87520352;
+ norm[223] = 82726544;
+ norm[224] = 78043744;
+ norm[225] = 73474736;
+ norm[226] = 69022240;
+ norm[227] = 64688944;
+ norm[228] = 60477424;
+ norm[229] = 56390192;
+ norm[230] = 52429696;
+ norm[231] = 48598304;
+ norm[232] = 44898304;
+ norm[233] = 41331904;
+ norm[234] = 37901248;
+ norm[235] = 34608384;
+ norm[236] = 31455264;
+ norm[237] = 28443792;
+ norm[238] = 25575744;
+ norm[239] = 22852864;
+ norm[240] = 20276752;
+ norm[241] = 17848976;
+ norm[242] = 15570960;
+ norm[243] = 13444080;
+ norm[244] = 11469616;
+ norm[245] = 9648720;
+ norm[246] = 7982512;
+ norm[247] = 6471952;
+ norm[248] = 5117984;
+ norm[249] = 3921376;
+ norm[250] = 2882880;
+ norm[251] = 2003104;
+ norm[252] = 1282560;
+ norm[253] = 721696;
+ norm[254] = 320832;
+ norm[255] = 80224;
+ hann[0] = 0;
+ hann[1] = 0;
+ hann[2] = 1;
+ hann[3] = 2;
+ hann[4] = 3;
+ hann[5] = 5;
+ hann[6] = 7;
+ hann[7] = 9;
+ hann[8] = 12;
+ hann[9] = 15;
+ hann[10] = 18;
+ hann[11] = 21;
+ hann[12] = 25;
+ hann[13] = 29;
+ hann[14] = 34;
+ hann[15] = 38;
+ hann[16] = 43;
+ hann[17] = 48;
+ hann[18] = 54;
+ hann[19] = 59;
+ hann[20] = 66;
+ hann[21] = 72;
+ hann[22] = 78;
+ hann[23] = 85;
+ hann[24] = 92;
+ hann[25] = 100;
+ hann[26] = 107;
+ hann[27] = 115;
+ hann[28] = 123;
+ hann[29] = 131;
+ hann[30] = 140;
+ hann[31] = 148;
+ hann[32] = 157;
+ hann[33] = 166;
+ hann[34] = 176;
+ hann[35] = 185;
+ hann[36] = 195;
+ hann[37] = 205;
+ hann[38] = 215;
+ hann[39] = 225;
+ hann[40] = 236;
+ hann[41] = 247;
+ hann[42] = 257;
+ hann[43] = 268;
+ hann[44] = 279;
+ hann[45] = 291;
+ hann[46] = 302;
+ hann[47] = 313;
+ hann[48] = 325;
+ hann[49] = 337;
+ hann[50] = 349;
+ hann[51] = 360;
+ hann[52] = 372;
+ hann[53] = 385;
+ hann[54] = 397;
+ hann[55] = 409;
+ hann[56] = 421;
+ hann[57] = 434;
+ hann[58] = 446;
+ hann[59] = 458;
+ hann[60] = 471;
+ hann[61] = 483;
+ hann[62] = 496;
+ hann[63] = 508;
+ hann[64] = 521;
+ hann[65] = 533;
+ hann[66] = 546;
+ hann[67] = 558;
+ hann[68] = 571;
+ hann[69] = 583;
+ hann[70] = 596;
+ hann[71] = 608;
+ hann[72] = 620;
+ hann[73] = 632;
+ hann[74] = 645;
+ hann[75] = 657;
+ hann[76] = 669;
+ hann[77] = 680;
+ hann[78] = 692;
+ hann[79] = 704;
+ hann[80] = 715;
+ hann[81] = 727;
+ hann[82] = 738;
+ hann[83] = 749;
+ hann[84] = 760;
+ hann[85] = 771;
+ hann[86] = 782;
+ hann[87] = 792;
+ hann[88] = 803;
+ hann[89] = 813;
+ hann[90] = 823;
+ hann[91] = 833;
+ hann[92] = 842;
+ hann[93] = 852;
+ hann[94] = 861;
+ hann[95] = 870;
+ hann[96] = 879;
+ hann[97] = 888;
+ hann[98] = 896;
+ hann[99] = 904;
+ hann[100] = 912;
+ hann[101] = 920;
+ hann[102] = 927;
+ hann[103] = 934;
+ hann[104] = 941;
+ hann[105] = 948;
+ hann[106] = 954;
+ hann[107] = 961;
+ hann[108] = 966;
+ hann[109] = 972;
+ hann[110] = 977;
+ hann[111] = 982;
+ hann[112] = 987;
+ hann[113] = 992;
+ hann[114] = 996;
+ hann[115] = 1000;
+ hann[116] = 1003;
+ hann[117] = 1007;
+ hann[118] = 1010;
+ hann[119] = 1012;
+ hann[120] = 1015;
+ hann[121] = 1017;
+ hann[122] = 1019;
+ hann[123] = 1020;
+ hann[124] = 1022;
+ hann[125] = 1023;
+ hann[126] = 1023;
+ hann[127] = 1023;
+ hann[128] = 1023;
+ hann[129] = 1023;
+ hann[130] = 1023;
+ hann[131] = 1022;
+ hann[132] = 1020;
+ hann[133] = 1019;
+ hann[134] = 1017;
+ hann[135] = 1015;
+ hann[136] = 1012;
+ hann[137] = 1010;
+ hann[138] = 1007;
+ hann[139] = 1003;
+ hann[140] = 1000;
+ hann[141] = 996;
+ hann[142] = 992;
+ hann[143] = 987;
+ hann[144] = 982;
+ hann[145] = 977;
+ hann[146] = 972;
+ hann[147] = 966;
+ hann[148] = 961;
+ hann[149] = 954;
+ hann[150] = 948;
+ hann[151] = 941;
+ hann[152] = 934;
+ hann[153] = 927;
+ hann[154] = 920;
+ hann[155] = 912;
+ hann[156] = 904;
+ hann[157] = 896;
+ hann[158] = 888;
+ hann[159] = 879;
+ hann[160] = 870;
+ hann[161] = 861;
+ hann[162] = 852;
+ hann[163] = 842;
+ hann[164] = 833;
+ hann[165] = 823;
+ hann[166] = 813;
+ hann[167] = 803;
+ hann[168] = 792;
+ hann[169] = 782;
+ hann[170] = 771;
+ hann[171] = 760;
+ hann[172] = 749;
+ hann[173] = 738;
+ hann[174] = 727;
+ hann[175] = 715;
+ hann[176] = 704;
+ hann[177] = 692;
+ hann[178] = 680;
+ hann[179] = 669;
+ hann[180] = 657;
+ hann[181] = 645;
+ hann[182] = 632;
+ hann[183] = 620;
+ hann[184] = 608;
+ hann[185] = 596;
+ hann[186] = 583;
+ hann[187] = 571;
+ hann[188] = 558;
+ hann[189] = 546;
+ hann[190] = 533;
+ hann[191] = 521;
+ hann[192] = 508;
+ hann[193] = 496;
+ hann[194] = 483;
+ hann[195] = 471;
+ hann[196] = 458;
+ hann[197] = 446;
+ hann[198] = 434;
+ hann[199] = 421;
+ hann[200] = 409;
+ hann[201] = 397;
+ hann[202] = 385;
+ hann[203] = 372;
+ hann[204] = 360;
+ hann[205] = 349;
+ hann[206] = 337;
+ hann[207] = 325;
+ hann[208] = 313;
+ hann[209] = 302;
+ hann[210] = 291;
+ hann[211] = 279;
+ hann[212] = 268;
+ hann[213] = 257;
+ hann[214] = 247;
+ hann[215] = 236;
+ hann[216] = 225;
+ hann[217] = 215;
+ hann[218] = 205;
+ hann[219] = 195;
+ hann[220] = 185;
+ hann[221] = 176;
+ hann[222] = 166;
+ hann[223] = 157;
+ hann[224] = 148;
+ hann[225] = 140;
+ hann[226] = 131;
+ hann[227] = 123;
+ hann[228] = 115;
+ hann[229] = 107;
+ hann[230] = 100;
+ hann[231] = 92;
+ hann[232] = 85;
+ hann[233] = 78;
+ hann[234] = 72;
+ hann[235] = 66;
+ hann[236] = 59;
+ hann[237] = 54;
+ hann[238] = 48;
+ hann[239] = 43;
+ hann[240] = 38;
+ hann[241] = 34;
+ hann[242] = 29;
+ hann[243] = 25;
+ hann[244] = 21;
+ hann[245] = 18;
+ hann[246] = 15;
+ hann[247] = 12;
+ hann[248] = 9;
+ hann[249] = 7;
+ hann[250] = 5;
+ hann[251] = 3;
+ hann[252] = 2;
+ hann[253] = 1;
+ hann[254] = 0;
+ hann[255] = 0;
+
+} /* gen_hann2 */
+
+/**
+ * Creates the non uniform enhancement window (bilinearly mapped hannning window)
+ * in order to reduce the formant enhancement in the high spectrum
+ * @param mm : memory manager
+ * @param sig_inObj : sig PU internal object of the sub-object
+ * @return void
+ * @callgraph
+ * @callergraph
+ * @remarks the outgput is based on the matlab script below\n
+ matlab script
+ -------------
+ function makeEnhWind(alpha)
+ N=129; % =HFFTSIZEE_P1
+ s=(bilinmap(alpha,N));
+ h=[hann(N)'];
+ x=(1:N)/N;
+ W=interp1(x,h,s); W(1)=0;
+
+ fid=fopen('enhwind.txt','wt');
+ fprintf(fid,'picoos_int16 enh_wind_init(sig_innerobj_t *sig_inObj) {\n');
+ for i=1:N,
+ fprintf(fid,[' sig_inObj->enhwind[' int2str(i) '] = (picoos_single)' num2str(W(i),'%0.7g') ';\n']);
+ end;
+ fprintf(fid,' return PICO_OK;\n');
+ fprintf(fid,' }; \n');
+ fclose(fid);
+ %figure(1); plot(x*8,W);
+ */
+static void enh_wind_init(sig_innerobj_t *sig_inObj)
+{
+ /* picoos_int16 i; */
+ picoos_int32 *c;
+
+ c = sig_inObj->cos_table;
+
+ c[0] = 4096;
+ c[1] = 4095;
+ c[2] = 4095;
+ c[3] = 4095;
+ c[4] = 4095;
+ c[5] = 4095;
+ c[6] = 4095;
+ c[7] = 4095;
+ c[8] = 4094;
+ c[9] = 4094;
+ c[10] = 4094;
+ c[11] = 4093;
+ c[12] = 4093;
+ c[13] = 4092;
+ c[14] = 4092;
+ c[15] = 4091;
+ c[16] = 4091;
+ c[17] = 4090;
+ c[18] = 4089;
+ c[19] = 4089;
+ c[20] = 4088;
+ c[21] = 4087;
+ c[22] = 4086;
+ c[23] = 4085;
+ c[24] = 4084;
+ c[25] = 4083;
+ c[26] = 4082;
+ c[27] = 4081;
+ c[28] = 4080;
+ c[29] = 4079;
+ c[30] = 4078;
+ c[31] = 4077;
+ c[32] = 4076;
+ c[33] = 4075;
+ c[34] = 4073;
+ c[35] = 4072;
+ c[36] = 4071;
+ c[37] = 4069;
+ c[38] = 4068;
+ c[39] = 4066;
+ c[40] = 4065;
+ c[41] = 4063;
+ c[42] = 4062;
+ c[43] = 4060;
+ c[44] = 4058;
+ c[45] = 4057;
+ c[46] = 4055;
+ c[47] = 4053;
+ c[48] = 4051;
+ c[49] = 4049;
+ c[50] = 4047;
+ c[51] = 4045;
+ c[52] = 4043;
+ c[53] = 4041;
+ c[54] = 4039;
+ c[55] = 4037;
+ c[56] = 4035;
+ c[57] = 4033;
+ c[58] = 4031;
+ c[59] = 4029;
+ c[60] = 4026;
+ c[61] = 4024;
+ c[62] = 4022;
+ c[63] = 4019;
+ c[64] = 4017;
+ c[65] = 4014;
+ c[66] = 4012;
+ c[67] = 4009;
+ c[68] = 4007;
+ c[69] = 4004;
+ c[70] = 4001;
+ c[71] = 3999;
+ c[72] = 3996;
+ c[73] = 3993;
+ c[74] = 3990;
+ c[75] = 3988;
+ c[76] = 3985;
+ c[77] = 3982;
+ c[78] = 3979;
+ c[79] = 3976;
+ c[80] = 3973;
+ c[81] = 3970;
+ c[82] = 3967;
+ c[83] = 3963;
+ c[84] = 3960;
+ c[85] = 3957;
+ c[86] = 3954;
+ c[87] = 3950;
+ c[88] = 3947;
+ c[89] = 3944;
+ c[90] = 3940;
+ c[91] = 3937;
+ c[92] = 3933;
+ c[93] = 3930;
+ c[94] = 3926;
+ c[95] = 3923;
+ c[96] = 3919;
+ c[97] = 3915;
+ c[98] = 3912;
+ c[99] = 3908;
+ c[100] = 3904;
+ c[101] = 3900;
+ c[102] = 3897;
+ c[103] = 3893;
+ c[104] = 3889;
+ c[105] = 3885;
+ c[106] = 3881;
+ c[107] = 3877;
+ c[108] = 3873;
+ c[109] = 3869;
+ c[110] = 3864;
+ c[111] = 3860;
+ c[112] = 3856;
+ c[113] = 3852;
+ c[114] = 3848;
+ c[115] = 3843;
+ c[116] = 3839;
+ c[117] = 3834;
+ c[118] = 3830;
+ c[119] = 3826;
+ c[120] = 3821;
+ c[121] = 3816;
+ c[122] = 3812;
+ c[123] = 3807;
+ c[124] = 3803;
+ c[125] = 3798;
+ c[126] = 3793;
+ c[127] = 3789;
+ c[128] = 3784;
+ c[129] = 3779;
+ c[130] = 3774;
+ c[131] = 3769;
+ c[132] = 3764;
+ c[133] = 3759;
+ c[134] = 3754;
+ c[135] = 3749;
+ c[136] = 3744;
+ c[137] = 3739;
+ c[138] = 3734;
+ c[139] = 3729;
+ c[140] = 3723;
+ c[141] = 3718;
+ c[142] = 3713;
+ c[143] = 3708;
+ c[144] = 3702;
+ c[145] = 3697;
+ c[146] = 3691;
+ c[147] = 3686;
+ c[148] = 3680;
+ c[149] = 3675;
+ c[150] = 3669;
+ c[151] = 3664;
+ c[152] = 3658;
+ c[153] = 3652;
+ c[154] = 3647;
+ c[155] = 3641;
+ c[156] = 3635;
+ c[157] = 3629;
+ c[158] = 3624;
+ c[159] = 3618;
+ c[160] = 3612;
+ c[161] = 3606;
+ c[162] = 3600;
+ c[163] = 3594;
+ c[164] = 3588;
+ c[165] = 3582;
+ c[166] = 3576;
+ c[167] = 3570;
+ c[168] = 3563;
+ c[169] = 3557;
+ c[170] = 3551;
+ c[171] = 3545;
+ c[172] = 3538;
+ c[173] = 3532;
+ c[174] = 3526;
+ c[175] = 3519;
+ c[176] = 3513;
+ c[177] = 3506;
+ c[178] = 3500;
+ c[179] = 3493;
+ c[180] = 3487;
+ c[181] = 3480;
+ c[182] = 3473;
+ c[183] = 3467;
+ c[184] = 3460;
+ c[185] = 3453;
+ c[186] = 3447;
+ c[187] = 3440;
+ c[188] = 3433;
+ c[189] = 3426;
+ c[190] = 3419;
+ c[191] = 3412;
+ c[192] = 3405;
+ c[193] = 3398;
+ c[194] = 3391;
+ c[195] = 3384;
+ c[196] = 3377;
+ c[197] = 3370;
+ c[198] = 3363;
+ c[199] = 3356;
+ c[200] = 3348;
+ c[201] = 3341;
+ c[202] = 3334;
+ c[203] = 3326;
+ c[204] = 3319;
+ c[205] = 3312;
+ c[206] = 3304;
+ c[207] = 3297;
+ c[208] = 3289;
+ c[209] = 3282;
+ c[210] = 3274;
+ c[211] = 3267;
+ c[212] = 3259;
+ c[213] = 3252;
+ c[214] = 3244;
+ c[215] = 3236;
+ c[216] = 3229;
+ c[217] = 3221;
+ c[218] = 3213;
+ c[219] = 3205;
+ c[220] = 3197;
+ c[221] = 3190;
+ c[222] = 3182;
+ c[223] = 3174;
+ c[224] = 3166;
+ c[225] = 3158;
+ c[226] = 3150;
+ c[227] = 3142;
+ c[228] = 3134;
+ c[229] = 3126;
+ c[230] = 3117;
+ c[231] = 3109;
+ c[232] = 3101;
+ c[233] = 3093;
+ c[234] = 3085;
+ c[235] = 3076;
+ c[236] = 3068;
+ c[237] = 3060;
+ c[238] = 3051;
+ c[239] = 3043;
+ c[240] = 3034;
+ c[241] = 3026;
+ c[242] = 3018;
+ c[243] = 3009;
+ c[244] = 3000;
+ c[245] = 2992;
+ c[246] = 2983;
+ c[247] = 2975;
+ c[248] = 2966;
+ c[249] = 2957;
+ c[250] = 2949;
+ c[251] = 2940;
+ c[252] = 2931;
+ c[253] = 2922;
+ c[254] = 2914;
+ c[255] = 2905;
+ c[256] = 2896;
+ c[257] = 2887;
+ c[258] = 2878;
+ c[259] = 2869;
+ c[260] = 2860;
+ c[261] = 2851;
+ c[262] = 2842;
+ c[263] = 2833;
+ c[264] = 2824;
+ c[265] = 2815;
+ c[266] = 2806;
+ c[267] = 2796;
+ c[268] = 2787;
+ c[269] = 2778;
+ c[270] = 2769;
+ c[271] = 2760;
+ c[272] = 2750;
+ c[273] = 2741;
+ c[274] = 2732;
+ c[275] = 2722;
+ c[276] = 2713;
+ c[277] = 2703;
+ c[278] = 2694;
+ c[279] = 2684;
+ c[280] = 2675;
+ c[281] = 2665;
+ c[282] = 2656;
+ c[283] = 2646;
+ c[284] = 2637;
+ c[285] = 2627;
+ c[286] = 2617;
+ c[287] = 2608;
+ c[288] = 2598;
+ c[289] = 2588;
+ c[290] = 2578;
+ c[291] = 2569;
+ c[292] = 2559;
+ c[293] = 2549;
+ c[294] = 2539;
+ c[295] = 2529;
+ c[296] = 2519;
+ c[297] = 2510;
+ c[298] = 2500;
+ c[299] = 2490;
+ c[300] = 2480;
+ c[301] = 2470;
+ c[302] = 2460;
+ c[303] = 2450;
+ c[304] = 2439;
+ c[305] = 2429;
+ c[306] = 2419;
+ c[307] = 2409;
+ c[308] = 2399;
+ c[309] = 2389;
+ c[310] = 2379;
+ c[311] = 2368;
+ c[312] = 2358;
+ c[313] = 2348;
+ c[314] = 2337;
+ c[315] = 2327;
+ c[316] = 2317;
+ c[317] = 2306;
+ c[318] = 2296;
+ c[319] = 2286;
+ c[320] = 2275;
+ c[321] = 2265;
+ c[322] = 2254;
+ c[323] = 2244;
+ c[324] = 2233;
+ c[325] = 2223;
+ c[326] = 2212;
+ c[327] = 2201;
+ c[328] = 2191;
+ c[329] = 2180;
+ c[330] = 2170;
+ c[331] = 2159;
+ c[332] = 2148;
+ c[333] = 2138;
+ c[334] = 2127;
+ c[335] = 2116;
+ c[336] = 2105;
+ c[337] = 2094;
+ c[338] = 2084;
+ c[339] = 2073;
+ c[340] = 2062;
+ c[341] = 2051;
+ c[342] = 2040;
+ c[343] = 2029;
+ c[344] = 2018;
+ c[345] = 2007;
+ c[346] = 1997;
+ c[347] = 1986;
+ c[348] = 1975;
+ c[349] = 1964;
+ c[350] = 1952;
+ c[351] = 1941;
+ c[352] = 1930;
+ c[353] = 1919;
+ c[354] = 1908;
+ c[355] = 1897;
+ c[356] = 1886;
+ c[357] = 1875;
+ c[358] = 1864;
+ c[359] = 1852;
+ c[360] = 1841;
+ c[361] = 1830;
+ c[362] = 1819;
+ c[363] = 1807;
+ c[364] = 1796;
+ c[365] = 1785;
+ c[366] = 1773;
+ c[367] = 1762;
+ c[368] = 1751;
+ c[369] = 1739;
+ c[370] = 1728;
+ c[371] = 1717;
+ c[372] = 1705;
+ c[373] = 1694;
+ c[374] = 1682;
+ c[375] = 1671;
+ c[376] = 1659;
+ c[377] = 1648;
+ c[378] = 1636;
+ c[379] = 1625;
+ c[380] = 1613;
+ c[381] = 1602;
+ c[382] = 1590;
+ c[383] = 1579;
+ c[384] = 1567;
+ c[385] = 1555;
+ c[386] = 1544;
+ c[387] = 1532;
+ c[388] = 1520;
+ c[389] = 1509;
+ c[390] = 1497;
+ c[391] = 1485;
+ c[392] = 1474;
+ c[393] = 1462;
+ c[394] = 1450;
+ c[395] = 1438;
+ c[396] = 1427;
+ c[397] = 1415;
+ c[398] = 1403;
+ c[399] = 1391;
+ c[400] = 1379;
+ c[401] = 1368;
+ c[402] = 1356;
+ c[403] = 1344;
+ c[404] = 1332;
+ c[405] = 1320;
+ c[406] = 1308;
+ c[407] = 1296;
+ c[408] = 1284;
+ c[409] = 1272;
+ c[410] = 1260;
+ c[411] = 1248;
+ c[412] = 1237;
+ c[413] = 1225;
+ c[414] = 1213;
+ c[415] = 1201;
+ c[416] = 1189;
+ c[417] = 1176;
+ c[418] = 1164;
+ c[419] = 1152;
+ c[420] = 1140;
+ c[421] = 1128;
+ c[422] = 1116;
+ c[423] = 1104;
+ c[424] = 1092;
+ c[425] = 1080;
+ c[426] = 1068;
+ c[427] = 1056;
+ c[428] = 1043;
+ c[429] = 1031;
+ c[430] = 1019;
+ c[431] = 1007;
+ c[432] = 995;
+ c[433] = 983;
+ c[434] = 970;
+ c[435] = 958;
+ c[436] = 946;
+ c[437] = 934;
+ c[438] = 921;
+ c[439] = 909;
+ c[440] = 897;
+ c[441] = 885;
+ c[442] = 872;
+ c[443] = 860;
+ c[444] = 848;
+ c[445] = 836;
+ c[446] = 823;
+ c[447] = 811;
+ c[448] = 799;
+ c[449] = 786;
+ c[450] = 774;
+ c[451] = 762;
+ c[452] = 749;
+ c[453] = 737;
+ c[454] = 725;
+ c[455] = 712;
+ c[456] = 700;
+ c[457] = 687;
+ c[458] = 675;
+ c[459] = 663;
+ c[460] = 650;
+ c[461] = 638;
+ c[462] = 625;
+ c[463] = 613;
+ c[464] = 601;
+ c[465] = 588;
+ c[466] = 576;
+ c[467] = 563;
+ c[468] = 551;
+ c[469] = 538;
+ c[470] = 526;
+ c[471] = 513;
+ c[472] = 501;
+ c[473] = 488;
+ c[474] = 476;
+ c[475] = 463;
+ c[476] = 451;
+ c[477] = 438;
+ c[478] = 426;
+ c[479] = 413;
+ c[480] = 401;
+ c[481] = 388;
+ c[482] = 376;
+ c[483] = 363;
+ c[484] = 351;
+ c[485] = 338;
+ c[486] = 326;
+ c[487] = 313;
+ c[488] = 301;
+ c[489] = 288;
+ c[490] = 276;
+ c[491] = 263;
+ c[492] = 251;
+ c[493] = 238;
+ c[494] = 226;
+ c[495] = 213;
+ c[496] = 200;
+ c[497] = 188;
+ c[498] = 175;
+ c[499] = 163;
+ c[500] = 150;
+ c[501] = 138;
+ c[502] = 125;
+ c[503] = 113;
+ c[504] = 100;
+ c[505] = 87;
+ c[506] = 75;
+ c[507] = 62;
+ c[508] = 50;
+ c[509] = 37;
+ c[510] = 25;
+ c[511] = 12;
+ c[512] = 0;
+} /*enh_wind_init*/
+
+/**
+ * Initializes a useful large array of random numbers
+ * @param sig_inObj : sig PU internal object of the sub-object
+ * @return void
+ * @callgraph
+ * @callergraph
+ */
+static void init_rand(sig_innerobj_t *sig_inObj)
+{
+ picoos_int32 *q = sig_inObj->int_vec34;
+ picoos_int32 *r = sig_inObj->int_vec35;
+
+ sig_inObj->iRand = 0;
+
+ q[0] = -2198;
+ r[0] = 3455;
+ q[1] = 3226;
+ r[1] = -2522;
+ q[2] = -845;
+ r[2] = 4007;
+ q[3] = -1227;
+ r[3] = 3907;
+ q[4] = -3480;
+ r[4] = 2158;
+ q[5] = -1325;
+ r[5] = -3875;
+ q[6] = 2089;
+ r[6] = -3522;
+ q[7] = -468;
+ r[7] = 4069;
+ q[8] = 711;
+ r[8] = -4033;
+ q[9] = 3862;
+ r[9] = 1362;
+ q[10] = 4054;
+ r[10] = -579;
+ q[11] = 2825;
+ r[11] = 2965;
+ q[12] = 2704;
+ r[12] = -3076;
+ q[13] = 4081;
+ r[13] = 344;
+ q[14] = -3912;
+ r[14] = 1211;
+ q[15] = -3541;
+ r[15] = 2058;
+ q[16] = 2694;
+ r[16] = 3084;
+ q[17] = 835;
+ r[17] = 4009;
+ q[18] = -2578;
+ r[18] = -3182;
+ q[19] = 3205;
+ r[19] = 2550;
+ q[20] = -4074;
+ r[20] = -418;
+ q[21] = -183;
+ r[21] = -4091;
+ q[22] = -2665;
+ r[22] = -3110;
+ q[23] = -1367;
+ r[23] = 3860;
+ q[24] = -2266;
+ r[24] = -3411;
+ q[25] = 3327;
+ r[25] = -2387;
+ q[26] = -2807;
+ r[26] = -2982;
+ q[27] = -3175;
+ r[27] = -2587;
+ q[28] = -4095;
+ r[28] = 27;
+ q[29] = -811;
+ r[29] = -4014;
+ q[30] = 4082;
+ r[30] = 332;
+ q[31] = -2175;
+ r[31] = 3470;
+ q[32] = 3112;
+ r[32] = 2662;
+ q[33] = 1168;
+ r[33] = -3925;
+ q[34] = 2659;
+ r[34] = 3115;
+ q[35] = 4048;
+ r[35] = 622;
+ q[36] = 4092;
+ r[36] = -165;
+ q[37] = -4036;
+ r[37] = 697;
+ q[38] = 1081;
+ r[38] = -3950;
+ q[39] = -548;
+ r[39] = 4059;
+ q[40] = 4038;
+ r[40] = 685;
+ q[41] = -511;
+ r[41] = 4063;
+ q[42] = 3317;
+ r[42] = -2402;
+ q[43] = -3180;
+ r[43] = 2580;
+ q[44] = 851;
+ r[44] = -4006;
+ q[45] = 2458;
+ r[45] = -3276;
+ q[46] = -1453;
+ r[46] = 3829;
+ q[47] = -3577;
+ r[47] = 1995;
+ q[48] = -3708;
+ r[48] = -1738;
+ q[49] = -3890;
+ r[49] = 1282;
+ q[50] = 4041;
+ r[50] = 666;
+ q[51] = -3511;
+ r[51] = -2108;
+ q[52] = -1454;
+ r[52] = -3828;
+ q[53] = 2124;
+ r[53] = 3502;
+ q[54] = -3159;
+ r[54] = 2606;
+ q[55] = 2384;
+ r[55] = -3330;
+ q[56] = -3767;
+ r[56] = -1607;
+ q[57] = -4063;
+ r[57] = -513;
+ q[58] = 3952;
+ r[58] = -1075;
+ q[59] = -3778;
+ r[59] = -1581;
+ q[60] = -301;
+ r[60] = -4084;
+ q[61] = -4026;
+ r[61] = 751;
+ q[62] = -3346;
+ r[62] = 2361;
+ q[63] = -2426;
+ r[63] = 3299;
+ q[64] = 428;
+ r[64] = -4073;
+ q[65] = 3968;
+ r[65] = 1012;
+ q[66] = 2900;
+ r[66] = -2892;
+ q[67] = -263;
+ r[67] = 4087;
+ q[68] = 4083;
+ r[68] = 322;
+ q[69] = 2024;
+ r[69] = 3560;
+ q[70] = 4015;
+ r[70] = 808;
+ q[71] = -3971;
+ r[71] = -1000;
+ q[72] = 3785;
+ r[72] = -1564;
+ q[73] = -3726;
+ r[73] = 1701;
+ q[74] = -3714;
+ r[74] = 1725;
+ q[75] = 743;
+ r[75] = 4027;
+ q[76] = 875;
+ r[76] = -4001;
+ q[77] = 294;
+ r[77] = 4085;
+ q[78] = 2611;
+ r[78] = 3155;
+ q[79] = 2491;
+ r[79] = -3251;
+ q[80] = 1558;
+ r[80] = 3787;
+ q[81] = -2063;
+ r[81] = -3538;
+ q[82] = 3809;
+ r[82] = -1505;
+ q[83] = -2987;
+ r[83] = 2802;
+ q[84] = -1955;
+ r[84] = 3599;
+ q[85] = 1980;
+ r[85] = -3585;
+ q[86] = -539;
+ r[86] = -4060;
+ q[87] = -3210;
+ r[87] = 2543;
+ q[88] = 2415;
+ r[88] = -3308;
+ q[89] = 1587;
+ r[89] = 3775;
+ q[90] = -3943;
+ r[90] = 1106;
+ q[91] = 3476;
+ r[91] = 2165;
+ q[92] = 2253;
+ r[92] = 3420;
+ q[93] = -2584;
+ r[93] = 3177;
+ q[94] = 3804;
+ r[94] = -1518;
+ q[95] = -3637;
+ r[95] = 1883;
+ q[96] = 3289;
+ r[96] = -2440;
+ q[97] = -1621;
+ r[97] = 3761;
+ q[98] = 1645;
+ r[98] = 3751;
+ q[99] = -3471;
+ r[99] = 2173;
+ q[100] = 4071;
+ r[100] = -449;
+ q[101] = -872;
+ r[101] = -4001;
+ q[102] = -3897;
+ r[102] = 1259;
+ q[103] = -3590;
+ r[103] = 1970;
+ q[104] = -2456;
+ r[104] = -3277;
+ q[105] = -3004;
+ r[105] = 2783;
+ q[106] = 2589;
+ r[106] = 3173;
+ q[107] = 3727;
+ r[107] = -1698;
+ q[108] = 2992;
+ r[108] = 2796;
+ q[109] = 794;
+ r[109] = -4018;
+ q[110] = -918;
+ r[110] = 3991;
+ q[111] = 1446;
+ r[111] = -3831;
+ q[112] = 3871;
+ r[112] = -1338;
+ q[113] = -612;
+ r[113] = -4049;
+ q[114] = -1566;
+ r[114] = -3784;
+ q[115] = 672;
+ r[115] = -4040;
+ q[116] = 3841;
+ r[116] = 1422;
+ q[117] = 3545;
+ r[117] = -2051;
+ q[118] = -1982;
+ r[118] = -3584;
+ q[119] = -3413;
+ r[119] = 2263;
+ q[120] = -3265;
+ r[120] = -2473;
+ q[121] = -2876;
+ r[121] = -2915;
+ q[122] = 4094;
+ r[122] = -117;
+ q[123] = -269;
+ r[123] = 4087;
+ q[124] = -4077;
+ r[124] = 391;
+ q[125] = -3759;
+ r[125] = 1626;
+ q[126] = 1639;
+ r[126] = 3753;
+ q[127] = 3041;
+ r[127] = -2743;
+ q[128] = 5;
+ r[128] = 4095;
+ q[129] = 2778;
+ r[129] = -3009;
+ q[130] = 1121;
+ r[130] = -3939;
+ q[131] = -455;
+ r[131] = -4070;
+ q[132] = 3532;
+ r[132] = 2073;
+ q[133] = -143;
+ r[133] = -4093;
+ q[134] = -2357;
+ r[134] = -3349;
+ q[135] = 458;
+ r[135] = 4070;
+ q[136] = -2887;
+ r[136] = -2904;
+ q[137] = -1104;
+ r[137] = 3944;
+ q[138] = -2104;
+ r[138] = -3513;
+ q[139] = 126;
+ r[139] = 4094;
+ q[140] = -3655;
+ r[140] = -1848;
+ q[141] = -3896;
+ r[141] = 1263;
+ q[142] = -3874;
+ r[142] = -1327;
+ q[143] = 4058;
+ r[143] = 553;
+ q[144] = -1831;
+ r[144] = -3663;
+ q[145] = -255;
+ r[145] = -4088;
+ q[146] = -1211;
+ r[146] = 3912;
+ q[147] = 445;
+ r[147] = 4071;
+ q[148] = 2268;
+ r[148] = 3410;
+ q[149] = -4010;
+ r[149] = 833;
+ q[150] = 2621;
+ r[150] = 3147;
+ q[151] = -250;
+ r[151] = 4088;
+ q[152] = -3409;
+ r[152] = -2269;
+ q[153] = -2710;
+ r[153] = -3070;
+ q[154] = 4063;
+ r[154] = 518;
+ q[155] = -3611;
+ r[155] = -1933;
+ q[156] = -3707;
+ r[156] = -1741;
+ q[157] = -1151;
+ r[157] = -3930;
+ q[158] = 3976;
+ r[158] = -983;
+ q[159] = -1736;
+ r[159] = 3709;
+ q[160] = 3669;
+ r[160] = 1820;
+ q[161] = -143;
+ r[161] = 4093;
+ q[162] = -3879;
+ r[162] = -1313;
+ q[163] = -2242;
+ r[163] = -3427;
+ q[164] = -4095;
+ r[164] = 0;
+ q[165] = -1159;
+ r[165] = -3928;
+ q[166] = -3155;
+ r[166] = 2611;
+ q[167] = -2887;
+ r[167] = -2904;
+ q[168] = -4095;
+ r[168] = 56;
+ q[169] = -3861;
+ r[169] = -1364;
+ q[170] = -2814;
+ r[170] = 2976;
+ q[171] = -3680;
+ r[171] = -1798;
+ q[172] = -4094;
+ r[172] = -107;
+ q[173] = -3626;
+ r[173] = 1903;
+ q[174] = 3403;
+ r[174] = 2278;
+ q[175] = -1735;
+ r[175] = -3710;
+ q[176] = -2126;
+ r[176] = -3500;
+ q[177] = 3183;
+ r[177] = -2577;
+ q[178] = -3499;
+ r[178] = 2128;
+ q[179] = -1736;
+ r[179] = 3709;
+ q[180] = 2592;
+ r[180] = -3170;
+ q[181] = 3875;
+ r[181] = 1326;
+ q[182] = 3596;
+ r[182] = 1960;
+ q[183] = 3915;
+ r[183] = -1202;
+ q[184] = 1570;
+ r[184] = 3783;
+ q[185] = -3319;
+ r[185] = -2400;
+ q[186] = 4019;
+ r[186] = 787;
+ q[187] = -187;
+ r[187] = 4091;
+ q[188] = 1370;
+ r[188] = -3859;
+ q[189] = -4091;
+ r[189] = 199;
+ q[190] = 3626;
+ r[190] = 1904;
+ q[191] = -2943;
+ r[191] = 2848;
+ q[192] = 56;
+ r[192] = 4095;
+ q[193] = 2824;
+ r[193] = 2966;
+ q[194] = -3994;
+ r[194] = -904;
+ q[195] = 56;
+ r[195] = 4095;
+ q[196] = -2045;
+ r[196] = 3548;
+ q[197] = -3653;
+ r[197] = 1850;
+ q[198] = -2864;
+ r[198] = 2927;
+ q[199] = -1996;
+ r[199] = 3576;
+ q[200] = -4061;
+ r[200] = 527;
+ q[201] = 159;
+ r[201] = 4092;
+ q[202] = -3363;
+ r[202] = 2336;
+ q[203] = -4074;
+ r[203] = 421;
+ q[204] = 2043;
+ r[204] = 3549;
+ q[205] = 4095;
+ r[205] = -70;
+ q[206] = -2107;
+ r[206] = -3512;
+ q[207] = -1973;
+ r[207] = 3589;
+ q[208] = -3138;
+ r[208] = 2631;
+ q[209] = -3625;
+ r[209] = -1905;
+ q[210] = 2413;
+ r[210] = 3309;
+ q[211] = -50;
+ r[211] = -4095;
+ q[212] = 2813;
+ r[212] = 2976;
+ q[213] = -535;
+ r[213] = -4060;
+ q[214] = 1250;
+ r[214] = 3900;
+ q[215] = 1670;
+ r[215] = -3739;
+ q[216] = 1945;
+ r[216] = -3604;
+ q[217] = -476;
+ r[217] = -4068;
+ q[218] = -3659;
+ r[218] = -1840;
+ q[219] = 2745;
+ r[219] = 3039;
+ q[220] = -674;
+ r[220] = -4040;
+ q[221] = 2383;
+ r[221] = 3330;
+ q[222] = 4086;
+ r[222] = 274;
+ q[223] = -4030;
+ r[223] = 730;
+ q[224] = 768;
+ r[224] = -4023;
+ q[225] = 3925;
+ r[225] = 1170;
+ q[226] = 785;
+ r[226] = 4019;
+ q[227] = -3101;
+ r[227] = -2675;
+ q[228] = 4030;
+ r[228] = -729;
+ q[229] = 3422;
+ r[229] = 2249;
+ q[230] = -3847;
+ r[230] = 1403;
+ q[231] = 3902;
+ r[231] = -1243;
+ q[232] = 2114;
+ r[232] = -3507;
+ q[233] = -2359;
+ r[233] = 3348;
+ q[234] = 3754;
+ r[234] = -1638;
+ q[235] = -4095;
+ r[235] = -83;
+ q[236] = 2301;
+ r[236] = -3388;
+ q[237] = 3336;
+ r[237] = 2375;
+ q[238] = -2045;
+ r[238] = 3548;
+ q[239] = -413;
+ r[239] = -4075;
+ q[240] = 1848;
+ r[240] = 3655;
+ q[241] = 4072;
+ r[241] = -437;
+ q[242] = 4069;
+ r[242] = -463;
+ q[243] = 1386;
+ r[243] = -3854;
+ q[244] = 966;
+ r[244] = 3980;
+ q[245] = -1684;
+ r[245] = -3733;
+ q[246] = 2953;
+ r[246] = 2837;
+ q[247] = -3961;
+ r[247] = -1040;
+ q[248] = 3512;
+ r[248] = -2107;
+ q[249] = 1363;
+ r[249] = 3862;
+ q[250] = 1883;
+ r[250] = 3637;
+ q[251] = 2657;
+ r[251] = 3116;
+ q[252] = 2347;
+ r[252] = -3356;
+ q[253] = -1635;
+ r[253] = -3755;
+ q[254] = 3170;
+ r[254] = 2593;
+ q[255] = 2856;
+ r[255] = 2935;
+ q[256] = 494;
+ r[256] = 4066;
+ q[257] = 1936;
+ r[257] = -3609;
+ q[258] = 245;
+ r[258] = 4088;
+ q[259] = -1211;
+ r[259] = -3912;
+ q[260] = -3600;
+ r[260] = 1952;
+ q[261] = 1632;
+ r[261] = 3756;
+ q[262] = 2341;
+ r[262] = 3360;
+ q[263] = 186;
+ r[263] = -4091;
+ q[264] = 4011;
+ r[264] = 829;
+ q[265] = -3490;
+ r[265] = -2143;
+ q[266] = 269;
+ r[266] = -4087;
+ q[267] = -2939;
+ r[267] = 2852;
+ q[268] = 1600;
+ r[268] = 3770;
+ q[269] = -3405;
+ r[269] = -2275;
+ q[270] = -3134;
+ r[270] = -2636;
+ q[271] = 2642;
+ r[271] = -3129;
+ q[272] = 3629;
+ r[272] = 1898;
+ q[273] = 3413;
+ r[273] = 2264;
+ q[274] = 2050;
+ r[274] = 3545;
+ q[275] = 988;
+ r[275] = -3975;
+ q[276] = -660;
+ r[276] = 4042;
+ q[277] = 978;
+ r[277] = -3977;
+ q[278] = 1965;
+ r[278] = -3593;
+ q[279] = -1513;
+ r[279] = -3806;
+ q[280] = -4076;
+ r[280] = 401;
+ q[281] = -4094;
+ r[281] = -92;
+ q[282] = -1914;
+ r[282] = 3621;
+ q[283] = 2006;
+ r[283] = -3570;
+ q[284] = -1550;
+ r[284] = -3791;
+ q[285] = 3774;
+ r[285] = -1591;
+ q[286] = -3958;
+ r[286] = 1052;
+ q[287] = -3576;
+ r[287] = 1997;
+ q[288] = -382;
+ r[288] = 4078;
+ q[289] = 1288;
+ r[289] = 3888;
+ q[290] = -2965;
+ r[290] = -2825;
+ q[291] = 1608;
+ r[291] = 3767;
+ q[292] = 3052;
+ r[292] = -2731;
+ q[293] = -622;
+ r[293] = 4048;
+ q[294] = -3836;
+ r[294] = 1434;
+ q[295] = -3542;
+ r[295] = 2056;
+ q[296] = -2648;
+ r[296] = 3124;
+ q[297] = -1178;
+ r[297] = -3922;
+ q[298] = -1109;
+ r[298] = 3942;
+ q[299] = 3910;
+ r[299] = 1217;
+ q[300] = 1199;
+ r[300] = -3916;
+ q[301] = -3386;
+ r[301] = 2303;
+ q[302] = -3453;
+ r[302] = 2202;
+ q[303] = -2877;
+ r[303] = 2914;
+ q[304] = 4095;
+ r[304] = -47;
+ q[305] = 3635;
+ r[305] = 1886;
+ q[306] = -2134;
+ r[306] = -3495;
+ q[307] = 613;
+ r[307] = -4049;
+ q[308] = -2700;
+ r[308] = 3079;
+ q[309] = 4091;
+ r[309] = -195;
+ q[310] = 3989;
+ r[310] = -927;
+ q[311] = -2385;
+ r[311] = 3329;
+ q[312] = 4094;
+ r[312] = -103;
+ q[313] = 1044;
+ r[313] = -3960;
+ q[314] = -1734;
+ r[314] = -3710;
+ q[315] = 1646;
+ r[315] = 3750;
+ q[316] = 575;
+ r[316] = 4055;
+ q[317] = -2629;
+ r[317] = -3140;
+ q[318] = 3266;
+ r[318] = 2471;
+ q[319] = 4091;
+ r[319] = -194;
+ q[320] = -2154;
+ r[320] = 3483;
+ q[321] = 659;
+ r[321] = 4042;
+ q[322] = -1785;
+ r[322] = -3686;
+ q[323] = -717;
+ r[323] = -4032;
+ q[324] = 4095;
+ r[324] = -37;
+ q[325] = -2963;
+ r[325] = -2827;
+ q[326] = -2645;
+ r[326] = -3126;
+ q[327] = 2619;
+ r[327] = -3148;
+ q[328] = 1855;
+ r[328] = -3651;
+ q[329] = -3726;
+ r[329] = 1699;
+ q[330] = -3437;
+ r[330] = 2227;
+ q[331] = 2948;
+ r[331] = 2842;
+ q[332] = -2125;
+ r[332] = 3501;
+ q[333] = -1700;
+ r[333] = 3726;
+ q[334] = 4094;
+ r[334] = -101;
+ q[335] = 2084;
+ r[335] = -3525;
+ q[336] = 3225;
+ r[336] = -2524;
+ q[337] = 2220;
+ r[337] = 3442;
+ q[338] = 3174;
+ r[338] = 2588;
+ q[339] = 229;
+ r[339] = -4089;
+ q[340] = -2381;
+ r[340] = -3332;
+ q[341] = -3677;
+ r[341] = -1803;
+ q[342] = -3191;
+ r[342] = -2567;
+ q[343] = 2465;
+ r[343] = 3270;
+ q[344] = 2681;
+ r[344] = -3096;
+ q[345] = 975;
+ r[345] = -3978;
+ q[346] = 2004;
+ r[346] = -3572;
+ q[347] = -3442;
+ r[347] = -2219;
+ q[348] = 3676;
+ r[348] = -1805;
+ q[349] = -3753;
+ r[349] = 1638;
+ q[350] = 3544;
+ r[350] = 2053;
+ q[351] = 397;
+ r[351] = -4076;
+ q[352] = 2221;
+ r[352] = 3440;
+ q[353] = -302;
+ r[353] = 4084;
+ q[354] = 4083;
+ r[354] = -323;
+ q[355] = -2253;
+ r[355] = -3420;
+ q[356] = -3038;
+ r[356] = 2746;
+ q[357] = 2884;
+ r[357] = 2908;
+ q[358] = 4070;
+ r[358] = 454;
+ q[359] = -1072;
+ r[359] = -3953;
+ q[360] = 3831;
+ r[360] = 1449;
+ q[361] = 3663;
+ r[361] = -1831;
+ q[362] = -1971;
+ r[362] = 3590;
+ q[363] = 3226;
+ r[363] = -2522;
+ q[364] = -145;
+ r[364] = -4093;
+ q[365] = 1882;
+ r[365] = -3637;
+ q[366] = 529;
+ r[366] = 4061;
+ q[367] = 2637;
+ r[367] = 3133;
+ q[368] = -4077;
+ r[368] = 389;
+ q[369] = 2156;
+ r[369] = -3482;
+ q[370] = -3276;
+ r[370] = 2458;
+ q[371] = -2687;
+ r[371] = -3090;
+ q[372] = 3469;
+ r[372] = -2177;
+ q[373] = -4093;
+ r[373] = -139;
+ q[374] = -850;
+ r[374] = 4006;
+ q[375] = -625;
+ r[375] = 4048;
+ q[376] = 1110;
+ r[376] = -3942;
+ q[377] = -3078;
+ r[377] = -2702;
+ q[378] = -2719;
+ r[378] = 3063;
+ q[379] = 742;
+ r[379] = 4028;
+ q[380] = -3902;
+ r[380] = -1245;
+ q[381] = 3888;
+ r[381] = -1287;
+ q[382] = -4081;
+ r[382] = 347;
+ q[383] = 1070;
+ r[383] = 3953;
+ q[384] = -996;
+ r[384] = -3972;
+ q[385] = 4041;
+ r[385] = -668;
+ q[386] = -2712;
+ r[386] = 3069;
+ q[387] = -3403;
+ r[387] = -2279;
+ q[388] = -3320;
+ r[388] = -2398;
+ q[389] = 3036;
+ r[389] = -2749;
+ q[390] = 1308;
+ r[390] = -3881;
+ q[391] = 2256;
+ r[391] = 3418;
+ q[392] = -1486;
+ r[392] = 3816;
+ q[393] = -2771;
+ r[393] = -3015;
+ q[394] = -3883;
+ r[394] = -1302;
+ q[395] = -3867;
+ r[395] = -1349;
+ q[396] = 3952;
+ r[396] = -1075;
+ q[397] = -789;
+ r[397] = 4019;
+ q[398] = 1458;
+ r[398] = 3827;
+ q[399] = 3832;
+ r[399] = -1446;
+ q[400] = -3001;
+ r[400] = -2787;
+ q[401] = 3463;
+ r[401] = 2186;
+ q[402] = 3606;
+ r[402] = 1942;
+ q[403] = 4023;
+ r[403] = 764;
+ q[404] = 3387;
+ r[404] = 2303;
+ q[405] = 2648;
+ r[405] = -3124;
+ q[406] = 1370;
+ r[406] = -3860;
+ q[407] = -3134;
+ r[407] = 2636;
+ q[408] = 4051;
+ r[408] = -600;
+ q[409] = -1977;
+ r[409] = -3587;
+ q[410] = 3160;
+ r[410] = 2605;
+ q[411] = 4042;
+ r[411] = 659;
+ q[412] = 3004;
+ r[412] = 2783;
+ q[413] = 3370;
+ r[413] = 2327;
+ q[414] = -419;
+ r[414] = -4074;
+ q[415] = -1968;
+ r[415] = 3591;
+ q[416] = -3705;
+ r[416] = -1746;
+ q[417] = -3331;
+ r[417] = -2383;
+ q[418] = -3634;
+ r[418] = 1888;
+ q[419] = -1981;
+ r[419] = -3584;
+ q[420] = 4069;
+ r[420] = -469;
+ q[421] = -628;
+ r[421] = -4047;
+ q[422] = -1900;
+ r[422] = 3628;
+ q[423] = 1039;
+ r[423] = -3961;
+ q[424] = 2554;
+ r[424] = -3201;
+ q[425] = -2955;
+ r[425] = 2836;
+ q[426] = 2286;
+ r[426] = -3398;
+ q[427] = -1624;
+ r[427] = 3760;
+ q[428] = 2213;
+ r[428] = 3446;
+ q[429] = -3989;
+ r[429] = -926;
+ q[430] = 192;
+ r[430] = -4091;
+ q[431] = -723;
+ r[431] = 4031;
+ q[432] = 2878;
+ r[432] = 2913;
+ q[433] = -2109;
+ r[433] = 3511;
+ q[434] = 1463;
+ r[434] = -3825;
+ q[435] = -741;
+ r[435] = -4028;
+ q[436] = -1314;
+ r[436] = -3879;
+ q[437] = 3115;
+ r[437] = 2659;
+ q[438] = -3160;
+ r[438] = -2605;
+ q[439] = 1868;
+ r[439] = 3644;
+ q[440] = -824;
+ r[440] = 4012;
+ q[441] = 781;
+ r[441] = 4020;
+ q[442] = -1257;
+ r[442] = -3898;
+ q[443] = 3331;
+ r[443] = -2382;
+ q[444] = 1642;
+ r[444] = -3752;
+ q[445] = 3748;
+ r[445] = -1650;
+ q[446] = -487;
+ r[446] = -4066;
+ q[447] = 3085;
+ r[447] = -2694;
+ q[448] = 4009;
+ r[448] = 839;
+ q[449] = -2308;
+ r[449] = -3383;
+ q[450] = 3850;
+ r[450] = 1397;
+ q[451] = -4078;
+ r[451] = -374;
+ q[452] = 2989;
+ r[452] = -2799;
+ q[453] = 3023;
+ r[453] = -2762;
+ q[454] = 1397;
+ r[454] = -3850;
+ q[455] = 323;
+ r[455] = 4083;
+ q[456] = 268;
+ r[456] = -4087;
+ q[457] = 2414;
+ r[457] = 3308;
+ q[458] = 3876;
+ r[458] = 1322;
+ q[459] = -3584;
+ r[459] = 1982;
+ q[460] = 1603;
+ r[460] = 3769;
+ q[461] = -1502;
+ r[461] = 3810;
+ q[462] = 1318;
+ r[462] = 3878;
+ q[463] = 1554;
+ r[463] = -3789;
+ q[464] = 2492;
+ r[464] = 3250;
+ q[465] = -4093;
+ r[465] = -154;
+ q[466] = 4008;
+ r[466] = 842;
+ q[467] = -2279;
+ r[467] = 3403;
+ q[468] = 3013;
+ r[468] = 2774;
+ q[469] = 2557;
+ r[469] = 3199;
+ q[470] = 4068;
+ r[470] = 475;
+ q[471] = 3324;
+ r[471] = -2392;
+ q[472] = 2653;
+ r[472] = -3120;
+ q[473] = 796;
+ r[473] = 4017;
+ q[474] = -1312;
+ r[474] = 3880;
+ q[475] = 1794;
+ r[475] = 3681;
+ q[476] = -2347;
+ r[476] = -3356;
+ q[477] = -4008;
+ r[477] = -840;
+ q[478] = -3773;
+ r[478] = -1592;
+ q[479] = 1609;
+ r[479] = 3766;
+ q[480] = -1564;
+ r[480] = -3785;
+ q[481] = 3004;
+ r[481] = 2784;
+ q[482] = 1258;
+ r[482] = 3897;
+ q[483] = 3729;
+ r[483] = 1693;
+ q[484] = -4095;
+ r[484] = -28;
+ q[485] = -4093;
+ r[485] = -146;
+ q[486] = 1393;
+ r[486] = -3851;
+ q[487] = 297;
+ r[487] = -4085;
+ q[488] = 2294;
+ r[488] = 3393;
+ q[489] = -2562;
+ r[489] = 3195;
+ q[490] = -1716;
+ r[490] = -3718;
+ q[491] = 2224;
+ r[491] = -3439;
+ q[492] = 2032;
+ r[492] = 3555;
+ q[493] = -2968;
+ r[493] = 2822;
+ q[494] = 2338;
+ r[494] = 3363;
+ q[495] = 1584;
+ r[495] = -3776;
+ q[496] = -3072;
+ r[496] = 2708;
+ q[497] = -1596;
+ r[497] = -3771;
+ q[498] = -2256;
+ r[498] = -3418;
+ q[499] = 4095;
+ r[499] = 89;
+ q[500] = -1949;
+ r[500] = 3602;
+ q[501] = 1844;
+ r[501] = 3657;
+ q[502] = -3375;
+ r[502] = 2319;
+ q[503] = -1481;
+ r[503] = -3818;
+ q[504] = 3228;
+ r[504] = -2520;
+ q[505] = 1116;
+ r[505] = 3940;
+ q[506] = -2783;
+ r[506] = 3004;
+ q[507] = 3915;
+ r[507] = 1201;
+ q[508] = 283;
+ r[508] = 4086;
+ q[509] = -3732;
+ r[509] = 1685;
+ q[510] = -433;
+ r[510] = -4072;
+ q[511] = -3667;
+ r[511] = 1823;
+ q[512] = 3883;
+ r[512] = 1300;
+ q[513] = -3742;
+ r[513] = 1663;
+ q[514] = 4093;
+ r[514] = -143;
+ q[515] = 3874;
+ r[515] = 1328;
+ q[516] = -3800;
+ r[516] = 1528;
+ q[517] = -1257;
+ r[517] = 3898;
+ q[518] = -1606;
+ r[518] = 3767;
+ q[519] = 3394;
+ r[519] = 2291;
+ q[520] = 2255;
+ r[520] = 3419;
+ q[521] = -4094;
+ r[521] = 120;
+ q[522] = -3767;
+ r[522] = 1606;
+ q[523] = 1849;
+ r[523] = -3654;
+ q[524] = -2883;
+ r[524] = 2908;
+ q[525] = 3469;
+ r[525] = 2176;
+ q[526] = 2654;
+ r[526] = 3119;
+ q[527] = -239;
+ r[527] = 4088;
+ q[528] = -651;
+ r[528] = 4043;
+ q[529] = -1140;
+ r[529] = 3934;
+ q[530] = 328;
+ r[530] = -4082;
+ q[531] = 3246;
+ r[531] = 2497;
+ q[532] = 4026;
+ r[532] = -753;
+ q[533] = -2041;
+ r[533] = -3550;
+ q[534] = -1154;
+ r[534] = 3929;
+ q[535] = -2710;
+ r[535] = 3070;
+ q[536] = -2860;
+ r[536] = 2932;
+ q[537] = 2097;
+ r[537] = 3517;
+ q[538] = 3492;
+ r[538] = -2140;
+ q[539] = 3123;
+ r[539] = 2649;
+ q[540] = 3360;
+ r[540] = 2342;
+ q[541] = 2498;
+ r[541] = 3245;
+ q[542] = 3976;
+ r[542] = 982;
+ q[543] = -2441;
+ r[543] = -3288;
+ q[544] = 3601;
+ r[544] = 1951;
+ q[545] = -4008;
+ r[545] = -842;
+ q[546] = 1243;
+ r[546] = 3902;
+ q[547] = 4069;
+ r[547] = 466;
+ q[548] = -2031;
+ r[548] = 3556;
+ q[549] = 4077;
+ r[549] = 386;
+ q[550] = -3112;
+ r[550] = -2663;
+ q[551] = 4087;
+ r[551] = -262;
+ q[552] = 4087;
+ r[552] = 266;
+ q[553] = -3907;
+ r[553] = -1228;
+ q[554] = -1611;
+ r[554] = 3765;
+ q[555] = 3066;
+ r[555] = -2715;
+ q[556] = 2657;
+ r[556] = 3117;
+ q[557] = 3912;
+ r[557] = -1213;
+ q[558] = -2531;
+ r[558] = -3220;
+ q[559] = 3500;
+ r[559] = -2127;
+ q[560] = -76;
+ r[560] = -4095;
+ q[561] = 3413;
+ r[561] = -2264;
+ q[562] = -4071;
+ r[562] = -448;
+ q[563] = 828;
+ r[563] = 4011;
+ q[564] = 3664;
+ r[564] = 1830;
+ q[565] = -1578;
+ r[565] = 3779;
+ q[566] = 3555;
+ r[566] = 2033;
+ q[567] = 3868;
+ r[567] = -1345;
+ q[568] = 4054;
+ r[568] = -580;
+ q[569] = -4094;
+ r[569] = 124;
+ q[570] = -3820;
+ r[570] = -1477;
+ q[571] = -3658;
+ r[571] = -1842;
+ q[572] = 2595;
+ r[572] = 3168;
+ q[573] = 3354;
+ r[573] = 2350;
+ q[574] = -701;
+ r[574] = -4035;
+ q[575] = -772;
+ r[575] = -4022;
+ q[576] = 2799;
+ r[576] = 2990;
+ q[577] = -3632;
+ r[577] = 1893;
+ q[578] = 310;
+ r[578] = 4084;
+ q[579] = 3984;
+ r[579] = -947;
+ q[580] = 3794;
+ r[580] = -1542;
+ q[581] = -2419;
+ r[581] = 3304;
+ q[582] = -3916;
+ r[582] = 1200;
+ q[583] = -3886;
+ r[583] = 1292;
+ q[584] = -3299;
+ r[584] = 2426;
+ q[585] = -437;
+ r[585] = 4072;
+ q[586] = 2053;
+ r[586] = -3544;
+ q[587] = 3987;
+ r[587] = 937;
+ q[588] = -789;
+ r[588] = -4019;
+ q[589] = 4055;
+ r[589] = -575;
+ q[590] = -3894;
+ r[590] = 1270;
+ q[591] = 4003;
+ r[591] = -864;
+ q[592] = -3060;
+ r[592] = 2721;
+ q[593] = -4009;
+ r[593] = 836;
+ q[594] = -1655;
+ r[594] = -3746;
+ q[595] = 3954;
+ r[595] = -1067;
+ q[596] = -773;
+ r[596] = 4022;
+ q[597] = -422;
+ r[597] = 4074;
+ q[598] = -3384;
+ r[598] = 2306;
+ q[599] = 195;
+ r[599] = -4091;
+ q[600] = -298;
+ r[600] = 4085;
+ q[601] = -3988;
+ r[601] = 931;
+ q[602] = 2014;
+ r[602] = -3566;
+ q[603] = 3349;
+ r[603] = -2357;
+ q[604] = 3800;
+ r[604] = 1526;
+ q[605] = 3858;
+ r[605] = 1374;
+ q[606] = 2947;
+ r[606] = 2844;
+ q[607] = -1483;
+ r[607] = -3818;
+ q[608] = 4056;
+ r[608] = -565;
+ q[609] = 2612;
+ r[609] = -3154;
+ q[610] = 2326;
+ r[610] = 3371;
+ q[611] = -3545;
+ r[611] = 2051;
+ q[612] = -1001;
+ r[612] = -3971;
+ q[613] = 3211;
+ r[613] = 2541;
+ q[614] = -2717;
+ r[614] = 3065;
+ q[615] = -3159;
+ r[615] = -2606;
+ q[616] = 2869;
+ r[616] = -2922;
+ q[617] = -1290;
+ r[617] = -3887;
+ q[618] = 2479;
+ r[618] = 3260;
+ q[619] = 3420;
+ r[619] = 2252;
+ q[620] = 1823;
+ r[620] = 3667;
+ q[621] = 3368;
+ r[621] = 2330;
+ q[622] = -3819;
+ r[622] = -1480;
+ q[623] = 3800;
+ r[623] = 1528;
+ q[624] = 3773;
+ r[624] = 1594;
+ q[625] = -189;
+ r[625] = -4091;
+ q[626] = -4067;
+ r[626] = -485;
+ q[627] = 2277;
+ r[627] = -3404;
+ q[628] = -4089;
+ r[628] = -233;
+ q[629] = -3634;
+ r[629] = 1889;
+ q[630] = 3292;
+ r[630] = 2437;
+ q[631] = -530;
+ r[631] = 4061;
+ q[632] = -3109;
+ r[632] = 2666;
+ q[633] = -3741;
+ r[633] = 1667;
+ q[634] = -1903;
+ r[634] = -3626;
+ q[635] = 3879;
+ r[635] = -1315;
+ q[636] = 4083;
+ r[636] = -315;
+ q[637] = -1148;
+ r[637] = 3931;
+ q[638] = 2630;
+ r[638] = 3139;
+ q[639] = -4001;
+ r[639] = 876;
+ q[640] = -2295;
+ r[640] = -3392;
+ q[641] = 1090;
+ r[641] = -3948;
+ q[642] = -3024;
+ r[642] = 2762;
+ q[643] = 2728;
+ r[643] = -3054;
+ q[644] = -3305;
+ r[644] = 2419;
+ q[645] = 60;
+ r[645] = 4095;
+ q[646] = 4048;
+ r[646] = -620;
+ q[647] = 589;
+ r[647] = -4053;
+ q[648] = -3867;
+ r[648] = 1347;
+ q[649] = -2944;
+ r[649] = -2847;
+ q[650] = -2721;
+ r[650] = 3060;
+ q[651] = 2928;
+ r[651] = 2863;
+ q[652] = 801;
+ r[652] = 4016;
+ q[653] = -3644;
+ r[653] = 1870;
+ q[654] = -1648;
+ r[654] = -3749;
+ q[655] = 825;
+ r[655] = -4012;
+ q[656] = -2036;
+ r[656] = -3553;
+ q[657] = -1192;
+ r[657] = -3918;
+ q[658] = 2875;
+ r[658] = 2916;
+ q[659] = -1831;
+ r[659] = -3663;
+ q[660] = -2865;
+ r[660] = -2926;
+ q[661] = -575;
+ r[661] = -4055;
+ q[662] = -3870;
+ r[662] = 1340;
+ q[663] = -4080;
+ r[663] = -356;
+ q[664] = -2176;
+ r[664] = -3469;
+ q[665] = -2986;
+ r[665] = -2803;
+ q[666] = 3978;
+ r[666] = -972;
+ q[667] = 2437;
+ r[667] = 3291;
+ q[668] = -3528;
+ r[668] = 2080;
+ q[669] = -3300;
+ r[669] = -2425;
+ q[670] = 3085;
+ r[670] = 2693;
+ q[671] = -3700;
+ r[671] = -1756;
+ q[672] = 3216;
+ r[672] = -2535;
+ q[673] = 4094;
+ r[673] = -91;
+ q[674] = 3775;
+ r[674] = -1589;
+ q[675] = 1097;
+ r[675] = -3946;
+ q[676] = -152;
+ r[676] = -4093;
+ q[677] = -3490;
+ r[677] = 2142;
+ q[678] = 3747;
+ r[678] = 1654;
+ q[679] = -1490;
+ r[679] = -3815;
+ q[680] = -3998;
+ r[680] = -886;
+ q[681] = 3726;
+ r[681] = -1700;
+ q[682] = -1600;
+ r[682] = 3770;
+ q[683] = -87;
+ r[683] = 4095;
+ q[684] = 2538;
+ r[684] = -3214;
+ q[685] = -4095;
+ r[685] = 52;
+ q[686] = -3993;
+ r[686] = -910;
+ q[687] = 4051;
+ r[687] = 603;
+ q[688] = -1242;
+ r[688] = -3902;
+ q[689] = 2155;
+ r[689] = 3482;
+ q[690] = 1270;
+ r[690] = 3893;
+ q[691] = 1919;
+ r[691] = -3618;
+ q[692] = -3145;
+ r[692] = 2623;
+ q[693] = 2475;
+ r[693] = 3263;
+ q[694] = 2226;
+ r[694] = -3437;
+ q[695] = -3894;
+ r[695] = -1269;
+ q[696] = -429;
+ r[696] = 4073;
+ q[697] = -1346;
+ r[697] = 3868;
+ q[698] = 1297;
+ r[698] = 3885;
+ q[699] = 1699;
+ r[699] = 3726;
+ q[700] = -3375;
+ r[700] = 2319;
+ q[701] = 1577;
+ r[701] = -3779;
+ q[702] = -63;
+ r[702] = 4095;
+ q[703] = 1215;
+ r[703] = -3911;
+ q[704] = -1492;
+ r[704] = 3814;
+ q[705] = -1530;
+ r[705] = -3799;
+ q[706] = 3442;
+ r[706] = 2218;
+ q[707] = -3867;
+ r[707] = -1349;
+ q[708] = -3291;
+ r[708] = -2437;
+ q[709] = -2253;
+ r[709] = -3420;
+ q[710] = -150;
+ r[710] = -4093;
+ q[711] = -2686;
+ r[711] = -3092;
+ q[712] = 3470;
+ r[712] = 2175;
+ q[713] = -3826;
+ r[713] = -1461;
+ q[714] = -3148;
+ r[714] = 2619;
+ q[715] = -3858;
+ r[715] = 1375;
+ q[716] = -3844;
+ r[716] = -1412;
+ q[717] = -3652;
+ r[717] = 1854;
+ q[718] = 4018;
+ r[718] = -791;
+ q[719] = 179;
+ r[719] = -4092;
+ q[720] = 3498;
+ r[720] = 2129;
+ q[721] = -1999;
+ r[721] = -3574;
+ q[722] = 3531;
+ r[722] = 2075;
+ q[723] = 4050;
+ r[723] = -606;
+ q[724] = -1639;
+ r[724] = 3753;
+ q[725] = -3661;
+ r[725] = 1835;
+ q[726] = 4039;
+ r[726] = 679;
+ q[727] = 3561;
+ r[727] = 2023;
+ q[728] = 528;
+ r[728] = 4061;
+ q[729] = -634;
+ r[729] = -4046;
+ q[730] = 364;
+ r[730] = -4079;
+ q[731] = 2735;
+ r[731] = 3048;
+ q[732] = 3978;
+ r[732] = 973;
+ q[733] = -4073;
+ r[733] = -427;
+ q[734] = -3722;
+ r[734] = 1708;
+ q[735] = 2356;
+ r[735] = -3350;
+ q[736] = -1125;
+ r[736] = -3938;
+ q[737] = 4054;
+ r[737] = 580;
+ q[738] = 3328;
+ r[738] = -2387;
+ q[739] = 1439;
+ r[739] = -3834;
+ q[740] = 1746;
+ r[740] = 3705;
+ q[741] = 2507;
+ r[741] = 3238;
+ q[742] = 3839;
+ r[742] = -1427;
+ q[743] = 488;
+ r[743] = -4066;
+ q[744] = 1187;
+ r[744] = 3920;
+ q[745] = 2038;
+ r[745] = -3552;
+ q[746] = -905;
+ r[746] = -3994;
+ q[747] = -236;
+ r[747] = 4089;
+ q[748] = 208;
+ r[748] = -4090;
+ q[749] = 1660;
+ r[749] = 3744;
+ q[750] = -4074;
+ r[750] = -415;
+ q[751] = -2304;
+ r[751] = 3385;
+ q[752] = -2457;
+ r[752] = 3276;
+ q[753] = 3302;
+ r[753] = 2423;
+ q[754] = 1778;
+ r[754] = -3689;
+ q[755] = 2019;
+ r[755] = 3563;
+ q[756] = 4037;
+ r[756] = 687;
+ q[757] = -2365;
+ r[757] = 3343;
+ q[758] = 5;
+ r[758] = -4095;
+ q[759] = 160;
+ r[759] = -4092;
+
+} /*initRand*/
+
+/**
+ * initializes the MEL-2_LINEAR LOOKUP TABLE
+ * @param sig_inObj : sig PU internal object of the sub-object
+ * @return void
+ * @remarks translated from matlab code to c-code
+ * @callgraph
+ * @callergraph
+ *
+ * input
+ * - bilinTab : base address of bilinTable destination vector
+ * - alpha : warping factor
+ * - size : size of the vectors to be generated
+ * - A,B : base address of array of indexes for lookup table implementation
+ * - D : base address of delta array for lookup table implementation
+ *
+ * output
+ * - bilinTab, A, B, D : initialized vectors
+ */
+void mel_2_lin_init(sig_innerobj_t *sig_inObj)
+{
+
+ /*Declare variables tied to I/O PARAMS formerly passed by value or reference*/
+ picoos_single alpha;
+ picoos_int32 *D;
+ picoos_int32 size;
+ picoos_int16 *A;
+
+ /*Link local variables with sig data object*/
+
+ alpha = sig_inObj->warp_p;
+ size = (sig_inObj->hfftsize_p) + 1;
+ A = sig_inObj->A_p;
+ D = sig_inObj->d_p;
+ /*
+ fixed point interpolation tables
+ scaling factor: 0x20
+ corresponding bit shift: 5
+ */
+
+ A[0] = 0;
+ D[0] = 0;
+ A[1] = 2;
+ D[1] = 14;
+ A[2] = 4;
+ D[2] = 29;
+ A[3] = 7;
+ D[3] = 11;
+ A[4] = 9;
+ D[4] = 24;
+ A[5] = 12;
+ D[5] = 5;
+ A[6] = 14;
+ D[6] = 18;
+ A[7] = 16;
+ D[7] = 30;
+ A[8] = 19;
+ D[8] = 9;
+ A[9] = 21;
+ D[9] = 19;
+ A[10] = 23;
+ D[10] = 29;
+ A[11] = 26;
+ D[11] = 5;
+ A[12] = 28;
+ D[12] = 12;
+ A[13] = 30;
+ D[13] = 19;
+ A[14] = 32;
+ D[14] = 24;
+ A[15] = 34;
+ D[15] = 27;
+ A[16] = 36;
+ D[16] = 30;
+ A[17] = 38;
+ D[17] = 31;
+ A[18] = 40;
+ D[18] = 31;
+ A[19] = 42;
+ D[19] = 29;
+ A[20] = 44;
+ D[20] = 26;
+ A[21] = 46;
+ D[21] = 22;
+ A[22] = 48;
+ D[22] = 17;
+ A[23] = 50;
+ D[23] = 10;
+ A[24] = 52;
+ D[24] = 2;
+ A[25] = 53;
+ D[25] = 24;
+ A[26] = 55;
+ D[26] = 13;
+ A[27] = 57;
+ D[27] = 1;
+ A[28] = 58;
+ D[28] = 20;
+ A[29] = 60;
+ D[29] = 5;
+ A[30] = 61;
+ D[30] = 21;
+ A[31] = 63;
+ D[31] = 4;
+ A[32] = 64;
+ D[32] = 18;
+ A[33] = 65;
+ D[33] = 31;
+ A[34] = 67;
+ D[34] = 11;
+ A[35] = 68;
+ D[35] = 21;
+ A[36] = 69;
+ D[36] = 31;
+ A[37] = 71;
+ D[37] = 7;
+ A[38] = 72;
+ D[38] = 14;
+ A[39] = 73;
+ D[39] = 21;
+ A[40] = 74;
+ D[40] = 27;
+ A[41] = 75;
+ D[41] = 31;
+ A[42] = 77;
+ D[42] = 3;
+ A[43] = 78;
+ D[43] = 6;
+ A[44] = 79;
+ D[44] = 8;
+ A[45] = 80;
+ D[45] = 10;
+ A[46] = 81;
+ D[46] = 10;
+ A[47] = 82;
+ D[47] = 10;
+ A[48] = 83;
+ D[48] = 9;
+ A[49] = 84;
+ D[49] = 8;
+ A[50] = 85;
+ D[50] = 6;
+ A[51] = 86;
+ D[51] = 3;
+ A[52] = 86;
+ D[52] = 31;
+ A[53] = 87;
+ D[53] = 27;
+ A[54] = 88;
+ D[54] = 23;
+ A[55] = 89;
+ D[55] = 18;
+ A[56] = 90;
+ D[56] = 12;
+ A[57] = 91;
+ D[57] = 6;
+ A[58] = 91;
+ D[58] = 31;
+ A[59] = 92;
+ D[59] = 24;
+ A[60] = 93;
+ D[60] = 16;
+ A[61] = 94;
+ D[61] = 8;
+ A[62] = 94;
+ D[62] = 31;
+ A[63] = 95;
+ D[63] = 22;
+ A[64] = 96;
+ D[64] = 13;
+ A[65] = 97;
+ D[65] = 3;
+ A[66] = 97;
+ D[66] = 25;
+ A[67] = 98;
+ D[67] = 14;
+ A[68] = 99;
+ D[68] = 3;
+ A[69] = 99;
+ D[69] = 24;
+ A[70] = 100;
+ D[70] = 13;
+ A[71] = 101;
+ D[71] = 1;
+ A[72] = 101;
+ D[72] = 21;
+ A[73] = 102;
+ D[73] = 8;
+ A[74] = 102;
+ D[74] = 27;
+ A[75] = 103;
+ D[75] = 14;
+ A[76] = 104;
+ D[76] = 1;
+ A[77] = 104;
+ D[77] = 19;
+ A[78] = 105;
+ D[78] = 6;
+ A[79] = 105;
+ D[79] = 24;
+ A[80] = 106;
+ D[80] = 9;
+ A[81] = 106;
+ D[81] = 27;
+ A[82] = 107;
+ D[82] = 12;
+ A[83] = 107;
+ D[83] = 29;
+ A[84] = 108;
+ D[84] = 14;
+ A[85] = 108;
+ D[85] = 31;
+ A[86] = 109;
+ D[86] = 15;
+ A[87] = 109;
+ D[87] = 31;
+ A[88] = 110;
+ D[88] = 16;
+ A[89] = 110;
+ D[89] = 31;
+ A[90] = 111;
+ D[90] = 15;
+ A[91] = 111;
+ D[91] = 31;
+ A[92] = 112;
+ D[92] = 14;
+ A[93] = 112;
+ D[93] = 30;
+ A[94] = 113;
+ D[94] = 13;
+ A[95] = 113;
+ D[95] = 28;
+ A[96] = 114;
+ D[96] = 11;
+ A[97] = 114;
+ D[97] = 26;
+ A[98] = 115;
+ D[98] = 9;
+ A[99] = 115;
+ D[99] = 23;
+ A[100] = 116;
+ D[100] = 6;
+ A[101] = 116;
+ D[101] = 20;
+ A[102] = 117;
+ D[102] = 2;
+ A[103] = 117;
+ D[103] = 16;
+ A[104] = 117;
+ D[104] = 31;
+ A[105] = 118;
+ D[105] = 13;
+ A[106] = 118;
+ D[106] = 27;
+ A[107] = 119;
+ D[107] = 8;
+ A[108] = 119;
+ D[108] = 22;
+ A[109] = 120;
+ D[109] = 4;
+ A[110] = 120;
+ D[110] = 17;
+ A[111] = 120;
+ D[111] = 31;
+ A[112] = 121;
+ D[112] = 13;
+ A[113] = 121;
+ D[113] = 26;
+ A[114] = 122;
+ D[114] = 8;
+ A[115] = 122;
+ D[115] = 21;
+ A[116] = 123;
+ D[116] = 2;
+ A[117] = 123;
+ D[117] = 15;
+ A[118] = 123;
+ D[118] = 29;
+ A[119] = 124;
+ D[119] = 10;
+ A[120] = 124;
+ D[120] = 23;
+ A[121] = 125;
+ D[121] = 4;
+ A[122] = 125;
+ D[122] = 17;
+ A[123] = 125;
+ D[123] = 31;
+ A[124] = 126;
+ D[124] = 12;
+ A[125] = 126;
+ D[125] = 25;
+ A[126] = 127;
+ D[126] = 6;
+ A[127] = 127;
+ D[127] = 19;
+ A[128] = 128;
+ D[128] = 0;
+
+}/*mel_2_lin_init*/
+
+/**
+ * function to be documented
+ * @param ang : ??
+ * @param table : ??
+ * @param cs : ??
+ * @param sn : ??
+ * @return void
+ * @callgraph
+ * @callergraph
+ */
+static void get_trig(picoos_int32 ang, picoos_int32 *table, picoos_int32 *cs,
+ picoos_int32 *sn)
+{
+ picoos_int32 i, j, k;
+
+ i = k = ang >> PICODSP_PI_SHIFT; /* * PICODSP_COS_TABLE_LEN2/PICODSP_FIX_SCALE2 */
+ if (i < 0)
+ i = -i;
+ j = 1;
+ i &= (PICODSP_COS_TABLE_LEN4 - 1);
+ if (i > PICODSP_COS_TABLE_LEN2)
+ i = PICODSP_COS_TABLE_LEN4 - i;
+ if (i > PICODSP_COS_TABLE_LEN) {
+ j = -1;
+ i = PICODSP_COS_TABLE_LEN2 - i;
+ }
+ if (j == 1)
+ *cs = table[i];
+ else
+ *cs = -table[i];
+
+ i = k - PICODSP_COS_TABLE_LEN;
+ if (i < 0)
+ i = -i;
+ j = 1;
+ i &= (PICODSP_COS_TABLE_LEN4 - 1);
+ if (i > PICODSP_COS_TABLE_LEN2)
+ i = PICODSP_COS_TABLE_LEN4 - i;
+ if (i > PICODSP_COS_TABLE_LEN) {
+ j = -1;
+ i = PICODSP_COS_TABLE_LEN2 - i;
+ }
+ if (j == 1)
+ *sn = table[i];
+ else
+ *sn = -table[i];
+}/*get_trig*/
+
+/**
+ * function to be documented
+ * @param sig_inObj : sig PU internal object of the sub-object
+ * @return void
+ * @callgraph
+ * @callergraph
+ */
+void save_transition_frame(sig_innerobj_t *sig_inObj)
+{
+ picoos_int32 *tmp, *tmp2; /*for loop unrolling*/
+
+ if (sig_inObj->voiced_p != sig_inObj->prevVoiced_p) {
+ sig_inObj->VoicTrans = sig_inObj->prevVoiced_p; /*remember last voicing transition*/
+ tmp = sig_inObj->ImpResp_p;
+ tmp2 = sig_inObj->imp_p;
+ FAST_DEVICE(PICODSP_FFTSIZE,*(tmp++)=*(tmp2++););
+ if (sig_inObj->voiced_p == 1)
+ sig_inObj->nV = 0;
+ else
+ sig_inObj->nU = 0; /*to avoid problems in case of very short voiced or unvoiced parts (less than 4 frames long)*/
+ }
+}/*save_transition_frame*/
+
+/**
+ * calculates an unweighted excitation window
+ * @param sig_inObj : sig PU internal object of the sub-object
+ * @param nextPeak : position of next peak (excitation position)
+ * @return PICO_OK
+ * @callgraph
+ * @callergraph
+ * input
+ * - hop : hop size in samples
+ * - winlen : excitation window length
+ * - E : energy
+ * - F0 : pitch
+ * - nextPeak : state that remembers next excitation index
+ * - Fs - sampling frequency
+ * output
+ * - LocV, LocU : (MAX_EN size) location of excitation points
+ * - EnV, EnU : (MAX_EN size) RMS values of excitation (scaled)
+ * - nV, nU : (integers) number of excitation points
+ * - nextPeak new position of lastPeak to calculate next frame
+ */
+static void get_simple_excitation(sig_innerobj_t *sig_inObj,
+ picoos_int16 *nextPeak)
+{
+ /*Define local variables*/
+ picoos_int16 nI, nJ, k;
+ /* picoos_single InvSqrt3=(picoos_single)2/(picoos_single)sqrt(3.0); *//*constant*/
+ picoos_int32 Ti, sqrtTi;
+ picoos_int16 hop, winlen, Fs;
+ picoos_single E, F0;
+ picoos_int16 voiced;
+ picoos_single fact; /*normalization factor*/
+ picoos_single rounding = 0.5f;
+
+ /*Link local variables to sig object*/
+ hop = sig_inObj->hop_p;
+ winlen = sig_inObj->m2_p;
+ Fs = sig_inObj->Fs_p;
+ E = sig_inObj->E_p;
+ F0 = sig_inObj->F0_p;
+ voiced = sig_inObj->voiced_p;
+
+ /* shift previous excitation window by hop samples*/
+ for (nI = 0; nI < sig_inObj->nV; nI++) {
+ sig_inObj->LocV[nI] = sig_inObj->LocV[nI] - hop;
+ }
+ for (nI = 0; nI < sig_inObj->nU; nI++) {
+ sig_inObj->LocU[nI] = sig_inObj->LocU[nI] - hop;
+ }
+
+ /*get rid of the voiced points that fall out of the interval*/
+ nI = 0;
+ while ((sig_inObj->LocV[nI] < 0) && nI < sig_inObj->nV)
+ nI++;
+
+ for (nJ = nI; nJ < sig_inObj->nV; nJ++) {
+ sig_inObj->LocV[nJ - nI] = sig_inObj->LocV[nJ];
+ sig_inObj->EnV[nJ - nI] = sig_inObj->EnV[nJ];
+ }
+ sig_inObj->nV -= nI;
+ /*get rid of the unvoiced points that fall out of the interval */
+ nI = 0;
+ while ((sig_inObj->LocU[nI] < 0) && nI < sig_inObj->nU)
+ nI++;
+
+ for (nJ = nI; nJ < sig_inObj->nU; nJ++) {
+ sig_inObj->LocU[nJ - nI] = sig_inObj->LocU[nJ];
+ sig_inObj->EnU[nJ - nI] = sig_inObj->EnU[nJ];
+ }
+ sig_inObj->nU -= nI;
+
+ *nextPeak -= hop;
+ k = *nextPeak;
+
+ fact = 3;
+ if (voiced == 0) { /*Unvoiced*/
+
+ Ti = (picoos_int32) (rounding + (picoos_single) Fs
+ / (picoos_single) sig_inObj->Fuv_p); /* round Period*/
+ sqrtTi = (picoos_int32) (E * sqrt((double) Fs
+ / (hop * sig_inObj->Fuv_p)) * fact * PICODSP_GETEXC_K1);
+ while (k < winlen) {
+ if (k < winlen) {
+ sig_inObj->LocU[sig_inObj->nU] = k;
+ sig_inObj->EnU[sig_inObj->nU] = sqrtTi;
+ sig_inObj->nU++;
+ k += (picoos_int16) Ti;
+ }
+ }
+ } else { /*Voiced*/
+ Ti
+ = (picoos_int32) (rounding + (picoos_single) Fs
+ / (picoos_single) F0); /*Period*/
+ sqrtTi = (picoos_int32) (E
+ * sqrt((double) Fs / (hop * sig_inObj->F0_p)) * fact
+ * PICODSP_GETEXC_K1);
+ while (k < winlen) {
+ sig_inObj->LocV[sig_inObj->nV] = k;
+ sig_inObj->EnV[sig_inObj->nV] = sqrtTi;
+ sig_inObj->nV++;
+ k += (picoos_int16) Ti;
+ }
+ }
+ *nextPeak = k;
+
+}/*get_simple_excitation*/
+
+#ifdef __cplusplus
+}
+#endif
+
+/* end picosig2.c */
diff --git a/lib/picosig2.h b/lib/picosig2.h
new file mode 100644
index 0000000..f239ced
--- /dev/null
+++ b/lib/picosig2.h
@@ -0,0 +1,229 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picosig2.h
+ *
+ * Signal Generation PU - Internal functions - header file
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#ifndef PICOSIG2_H_
+#define PICOSIG2_H_
+
+#include "picoos.h"
+#include "picodsp.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+/*----------------------------------------------------------
+ // Name : sig_innerobj
+ // Function: innerobject definition for the sig processing
+ // Shortcut: sig
+ //---------------------------------------------------------*/
+typedef struct sig_innerobj
+{
+
+ /*-----------------------Definition of the local storage for this PU--------*/
+ picoos_int16 *idx_vect1; /*reserved for bit reversal tables*/
+ picoos_int16 *idx_vect2; /*reserved for table lookup "A" vector*/
+ picoos_int16 *idx_vect4; /*reserved for max peak index arrax in pchip*/
+ picoos_int16 *idx_vect5; /*reserved for min index arrax in sigana_singleIMF*/
+ picoos_int16 *idx_vect6; /*reserved for max index arrax in sigana_singleIMF*/
+ picoos_int16 *idx_vect7; /*reserved for dispersed phase */
+ picoos_int16 *idx_vect8; /*reserved for LocV*/
+ picoos_int16 *idx_vect9; /*reserved for LocU*/
+
+ picoos_int32 *int_vec22; /*reserved for normalized hanning window - fixed point */
+ picoos_int32 *int_vec23; /*reserved for impresp - fixed point */
+ picoos_int32 *int_vec24; /*reserved for impresp - fixed point */
+ picoos_int32 *int_vec25; /*reserved for window - fixed point */
+ picoos_int32 *int_vec26; /*reserved for wavBuf - fixed point */
+ picoos_int32 *int_vec28; /*reserved for cepstral vectors input - fixed point */
+ picoos_int32 *int_vec29; /*reserved for cepstral vectors input - fixed point */
+ picoos_int32 *int_vec38; /*reserved for cepstral vectors input - fixed point */
+ picoos_int32 *int_vec30; /*reserved for cepstral vectors input - fixed point */
+ picoos_int32 *int_vec31; /*reserved for cepstral vectors input - fixed point */
+
+ picoos_int32 *int_vec32; /*reserved for cepstral vectors input - fixed point */
+ picoos_int32 *int_vec33; /*reserved for cepstral vectors input - fixed point */
+
+ picoos_int32 *int_vec34; /* reserved for sin table- fixed point */
+ picoos_int32 *int_vec35; /* reserved for cos table - fixed point */
+ picoos_int32 *int_vec36; /* reserved for currently used sin table- fixed point */
+ picoos_int32 *int_vec37; /* reserved for currently used cos table - fixed point */
+
+ picoos_int32 *int_vec39; /* reserved for ang - fixed point */
+ picoos_int32 *int_vec40; /* reserved for cos table - fixed point */
+
+ picoos_int32 *int_vec41[CEPST_BUFF_SIZE]; /*reserved for phase smoothing - cepstrum buffers */
+ picoos_int32 *int_vec42[PHASE_BUFF_SIZE]; /*reserved for phase smoothing - phase buffers */
+
+ picoos_int16 idx_vect10[CEPST_BUFF_SIZE]; /*reserved for pitch value buffering before phase smoothing*/
+ picoos_int16 idx_vect11[CEPST_BUFF_SIZE]; /*reserved for phonetic value bufferingid before phase smoothing*/
+ picoos_int16 idx_vect12[CEPST_BUFF_SIZE]; /*reserved for voicing value bufferingbefore phase smoothing*/
+ picoos_int16 idx_vect13[CEPST_BUFF_SIZE]; /*reserved for unrectified pitch value bufferingbefore phase smoothing*/
+ picoos_int16 idx_vect14[PHASE_BUFF_SIZE]; /*reserved for vox_bnd value buffering before phase smoothing*/
+
+ picoos_int32 *sig_vec1;
+
+ picoos_single bvalue1; /*reserved for warp*/
+ picoos_int32 ibvalue2; /*reserved for voxbnd*/
+ picoos_int32 ibvalue3; /*reserved for voxbnd2*/
+ picoos_single bvalue4; /*reserved for E*/
+ picoos_single bvalue5; /*reserved for F0*/
+ picoos_single bvalue6; /*reserved for sMod*/
+
+ picoos_single bvalue7; /*reserved for voicing*/
+ picoos_single bvalue8; /*reserved for unrectified pitch*/
+
+ picoos_int16 ivalue1; /*reserved for m1,ceporder*/
+ picoos_int16 ivalue2; /*reserved for m2,fftorder,windowlen*/
+ picoos_int16 ivalue3; /*reserved for fftorder/2*/
+ picoos_int16 ivalue4; /*reserved for framelen, displacement*/
+ picoos_int16 ivalue5; /*reserved for voiced*/
+ picoos_int16 ivalue6; /*reserved for generic result code*/
+ picoos_int16 ivalue7; /*reserved for i*/
+ picoos_int16 ivalue8; /*reserved for j*/
+ picoos_int16 ivalue9; /*reserved for hop*/
+ picoos_int16 ivalue10; /*reserved for nextPeak*/
+ picoos_int16 ivalue11; /*reserved for nFrame*/
+ picoos_int16 ivalue12; /*reserved for raw*/
+ picoos_int16 ivalue13; /*reserved for hts engine flag*/
+ picoos_int16 ivalue14; /*reserved for ph id*/
+ picoos_int16 ivalue15; /*reserved for Voiced*/
+ picoos_int16 ivalue16; /*reserved for prevVoiced*/
+ picoos_int16 ivalue17; /*reserved for nV (size of LocV)*/
+ picoos_int16 ivalue18; /*reserved for nU (size of LocU)*/
+
+ picoos_int16 ivalue19; /*reserved for voicTrans*/
+
+ picoos_int16 ivalue20; /*reserved for n_availabe index*/
+
+ picoos_int32 lvalue1; /*reserved for sampling rate*/
+ picoos_int32 lvalue2; /*reserved for VCutoff*/
+ picoos_int32 lvalue3; /*reserved for UVCutoff*/
+ picoos_int32 lvalue4; /*reserved for fMax */
+
+ picoos_int32 iRand; /*reserved for phase random table poointer ())*/
+
+} sig_innerobj_t;
+
+/*------------------------------------------------------------------
+ Exported (to picosig.c) Service routines :
+ routine name and I/O parameters are to be maintained for PICO compatibility!!
+ ------------------------------------------------------------------*/
+extern pico_status_t sigAllocate(picoos_MemoryManager mm,
+ sig_innerobj_t *sig_inObj);
+extern void sigDeallocate(picoos_MemoryManager mm, sig_innerobj_t *sig_inObj);
+extern void sigDspInitialize(sig_innerobj_t *sig_inObj);
+
+/*------------------------------------------------------------------
+ Exported (to picosig.c) Processing routines :
+ routine number, name and content can be changed. unique I/O parameter should be "sig"
+ ------------------------------------------------------------------*/
+extern void mel_2_lin_init(sig_innerobj_t *sig_inObj);
+extern void save_transition_frame(sig_innerobj_t *sig_inObj);
+extern void mel_2_lin_init(sig_innerobj_t *sig_inObj);
+extern void post_filter_init(sig_innerobj_t *sig_inObj);
+extern void mel_2_lin_lookup(sig_innerobj_t *sig_inObj, picoos_uint32 mgc);
+extern void post_filter(sig_innerobj_t *sig_inObj);
+extern void phase_spec2(sig_innerobj_t *sig_inObj);
+extern void env_spec(sig_innerobj_t *sig_inObj);
+extern void save_transition_frame(sig_innerobj_t *sig_inObj);
+extern void td_psola2(sig_innerobj_t *sig_inObj);
+extern void impulse_response(sig_innerobj_t *sig_inObj);
+extern void overlap_add(sig_innerobj_t *sig_inObj);
+
+/* -------------------------------------------------------------------
+ * symbolic vs area assignements
+ * -------------------------------------------------------------------*/
+#define WavBuff_p int_vec26 /*output is Wav buffer (2*FFTSize)*/
+#define window_p int_vec25 /*window function (hanning) */
+#define ImpResp_p int_vec23 /*output step 6*/
+#define imp_p int_vec24 /*output step 6*/
+#define warp_p bvalue1 /*warp factor */
+#define voxbnd_p ibvalue2 /*phase spectra reconstruction noise factor V*/ /* fixed point */
+#define voxbnd2_p ibvalue3 /*phase spectra reconstruction noise factor UV */ /* fixed point */
+#define E_p bvalue4 /*energy after Envelope spectrum calculation */
+#define F0_p bvalue5 /*pitch*/
+#define sMod_p bvalue6 /*speaker modification factor*/
+#define voicing bvalue7 /*voicing*/
+#define Fuv_p bvalue8 /*unrectified pitch (for unvoiced too)*/
+#define m1_p ivalue1 /*ceporder(melorder=ceporder-1) */
+#define m2_p ivalue2 /*fftorder*/
+#define windowLen_p ivalue2 /*same as fftorder*/
+#define hfftsize_p ivalue3 /*fftorder/2*/
+#define framesz_p ivalue4 /*displacement*/
+#define voiced_p ivalue5 /*voicing state*/
+#define nRes_p ivalue6 /*generic result*/
+#define i_p ivalue7 /*generic counter*/
+#define j_p ivalue8 /*generic counter*/
+#define hop_p ivalue9 /*hop */
+#define nextPeak_p ivalue10 /*nextPeak*/
+#define phId_p ivalue14 /*phonetic id*/
+#define prevVoiced_p ivalue16 /*previous voicing state (for frame 1)*/
+#define nV ivalue17
+#define nU ivalue18
+#define VoicTrans ivalue19 /* */
+#define Fs_p lvalue1 /*Sampling frequency*/
+#define VCutoff_p lvalue2 /*voicing cut off frequency in Hz*/
+#define UVCutoff_p lvalue3 /*unvoicing cut off frequency in Hz*/
+/* Reusable area */
+#define wcep_pI int_vec28 /*input step1*/
+#define d_p int_vec38 /*output mel_2_lin_init : table lookup vector D*/
+#define A_p idx_vect2 /*output mel_2_lin_init : table lookup vector A*/
+#define ang_p int_vec39 /*output step4*/
+#define EnV int_vec30
+#define EnU int_vec31
+#define randCosTbl int_vec34
+#define randSinTbl int_vec35
+#define outCosTbl int_vec36
+#define outSinTbl int_vec37
+#define cos_table int_vec40
+#define norm_window_p int_vec22 /*window function (hanning) */
+#define norm_window2_p int_vec27 /*window function (hanning) */
+#define F2r_p int_vec32 /*output step 7*/
+#define F2i_p int_vec33 /*output step 7*/
+#define LocV idx_vect8 /*excitation position voiced pulses*/
+#define LocU idx_vect9 /*excitation position unvoiced pulses*/
+
+#define CepBuff int_vec41 /*Buffer for incoming cepstral vector pointers*/
+#define PhsBuff int_vec42 /*Buffer for incoming phase vector pointers*/
+#define F0Buff idx_vect10 /*Buffer for incoming F0 values*/
+#define PhIdBuff idx_vect11 /*Buffer for incoming PhId values*/
+#define VoicingBuff idx_vect12 /*Buffer for incoming voicing values*/
+#define FuVBuff idx_vect13 /*Buffer for incoming FuV values*/
+#define VoxBndBuff idx_vect14 /*Buffer for incoming VoxBnd values*/
+
+#define n_available ivalue20 /*variable for indexing the incoming buffers*/
+
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
diff --git a/lib/picospho.c b/lib/picospho.c
new file mode 100644
index 0000000..0d0cdf8
--- /dev/null
+++ b/lib/picospho.c
@@ -0,0 +1,1694 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picospho.c
+ *
+ * sentence phonemic/phonetic FSTs PU
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#include "picoos.h"
+#include "picodbg.h"
+#include "picodata.h"
+
+#include "picoknow.h"
+#include "picokfst.h"
+#include "picoktab.h"
+#include "picotrns.h"
+
+#include "picospho.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+#define SPHO_BUFSIZE (3 * PICODATA_BUFSIZE_DEFAULT)
+
+
+
+#define SPHO_MAX_ALTDESC_SIZE (60 * PICOTRNS_MAX_NUM_POSSYM)
+
+
+#define SPHO_SMALLEST_SIL_DUR 1
+
+
+/** @addtogroup picospho
+ *
+ * Algorithmic description
+ * =======================
+ * The main function, sphoStep, is divided into the subprocesses (processing states) described further down.
+ *
+ * Flow control:
+ * ------------
+ * The processing flow is controlled by setting
+ * - 'procState' : the next state to be processed
+ * - 'feedFollowState' : the state to be processed after the feed state (the feed state is treated like a primitive "subroutine")
+ * - some other flags
+ *
+ * Buffering:
+ * ---------
+ * - The input items are mainly stored and processed in two buffers, collectively called 'inBuf'
+ * - cbuf : unstructured buffer containing item contents
+ * - headx : structured buffer containing item heads, each expanded by a pointer to the item contents
+ * and space for a boundary potentially to be inserted (to the left of the original item)
+ * - For transduction, phonemes and their position are extracted from inBuf into
+ * - phonBuf,
+ * processed there, and the resulting phonemes realigned with inBuf.
+ * - Word items are split into syllables, stored in
+ * - sylBuf
+ * - Items to be output are stored in outBuf
+ *
+ * Windowing:
+ * ---------
+ * Optimal solutions are achieved if a whole sentence is processed at once. However, if any of the buffers are too small,
+ * only sentence parts are processed. To improve the quality of such sub-optimal solutions, a moving-window-with-overlap is applied:
+ * - [0,headxReadPos[ : the window considered for transduction
+ * - [activeStartPos,activeEndPos[ : the "active" subrange of the window actually used for output
+ * - penultima : the position (within the active range) that should be used as new window start when shifting the window
+ *
+ * After PROCESS_PARSE:
+ * 0 activeStartPos penultima activeEndPos headxReadPos headxWritePos
+ * | | | | | |
+ * |-------------=================================---------------| ['----': context '====' : active subrange)
+ *
+ * After PROCESS_SHIFT:
+ * 0 activeStartPos headWritePos
+ * | | | |
+ * |------------... (only left context is known; new active range, penultima, and right context to be established at next parse)
+ *
+ * Processing states:
+ * -----------------
+ * - INIT : initialize state variables
+ * - COLLECT : collect items into internal buffers ("inBuf")
+ * - PROCESS_PARSE : go through inBuf items and extract position/phoneme pairs into phoneme buffer 'phonBuf'
+ * word boundary phonemes are inserted between words
+ * - PROCESS_TRANSDUCE : transduce phonBuf
+ * - PROCESS_BOUNDS : go through inBuf items again and match against transduced pos/phoneme
+ * this is the first round of alignment, only inserting/deleting/modifying bounds, according to
+ * - existing BOUND items
+ * - newly produced word bounds separating WORDPHON items
+ * - bound upgrades/downgrades from transduction
+ * - bound upgrades/downgrades/insertions from SIL command items (originating e.g. from <break> text commands)
+ * all relevant bounds are placed in the corresponding headx extention; original bound items become invalid.
+ * - PROCESS_RECOMB : go through inBuf items again and match against transduced pos/phoneme
+ * this is the second round of alignment, treating non-BOUND items
+ * - WORDPHONs are broken into syllables by "calling" PROCESS_SYL
+ * - "side-bounds" (in the headx extension) are output by "calling" FEED
+ * - BOUND items are consumed with no effect
+ * - other items are output unchanged "calling" FEED
+ * - PROCESS_SYL : the WORDPHON coming from RECOMB is matched against the phonBuf and (new) SYLLPHON items
+ * are created. (the original wordphon is consumed)
+ * - FEED : feeds one item and returns to spho->feedFollowState
+ * - SHIFT : items in inBuf are shifted left to make room for new items. If a sentence doesn't fit
+ * inBuf in its entirety, left and/or right contexts are kept so they can be considered in
+ * the next transduction.
+ */
+
+
+
+/* PU sphoStep states */
+#define SPHO_STEPSTATE_INIT 0
+#define SPHO_STEPSTATE_COLLECT 1
+#define SPHO_STEPSTATE_PROCESS_PARSE 2
+#define SPHO_STEPSTATE_PROCESS_TRANSDUCE 3
+#define SPHO_STEPSTATE_PROCESS_BOUNDS 4
+#define SPHO_STEPSTATE_PROCESS_RECOMB 5
+#define SPHO_STEPSTATE_PROCESS_SYL 6
+#define SPHO_STEPSTATE_FEED 7
+#define SPHO_STEPSTATE_SHIFT 8
+
+#define SPHO_POS_INVALID (PICOTRNS_POS_INVALID) /* indicates that no position was set yet */
+
+/* nr item restriction: maximum number of extended item heads in headx */
+#define SPHO_MAXNR_HEADX 60
+
+/* nr item restriction: maximum size of all item contents together in cont */
+#define SPHO_MAXSIZE_CBUF (30 * 255)
+
+/* "expanded head": item head expanded by a content position and a by boundary information
+ * potentially inserted "to the left" of the item */
+typedef struct {
+ picodata_itemhead_t head;
+ picoos_uint16 cind;
+ picoos_uint8 boundstrength; /* bstrength to the left, 0 if not set */
+ picoos_uint8 phrasetype; /* btype for following phrase, 0 if not set */
+ picoos_int16 sildur; /* silence duration for boundary, -1 if not set */
+} picospho_headx_t;
+
+
+
+#define SPHO_MSGSTR_SIZE 32
+
+/** object : SentPhoUnit
+ * shortcut : spho
+ * derived from : picodata_ProcessingUnit
+ */
+typedef struct spho_subobj {
+ picoos_Common common;
+
+ /* we use int16 for buffer positions so we can indicate exceptional positions (invalid etc.) with negative
+ * integers */
+ picoos_uint8 procState; /* for next processing step decision */
+
+ /* buffer for item headers */
+ picoos_uint8 tmpbuf[PICODATA_MAX_ITEMSIZE]; /* tmp. location for an item */
+
+ picospho_headx_t headx[SPHO_MAXNR_HEADX]; /* "expanded head" buffer */
+ picoos_uint16 headxBufSize; /* actually allocated size (if one day headxBuf is allocated dynamically) */
+ picoos_uint16 headxReadPos, headxWritePos;
+
+ picoos_uint8 cbuf[SPHO_MAXSIZE_CBUF];
+ picoos_uint16 cbufBufSize; /* actually allocated size */
+ picoos_uint16 cbufWritePos; /* next position to write to, 0 if buffer empty */
+
+ picoos_uint8 outBuf[PICODATA_BUFSIZE_DEFAULT]; /* internal output buffer to hold just one item */
+ picoos_uint16 outBufSize; /* actually allocated size (if one day outBuf is allocated dynamically) */
+ picoos_uint16 outReadPos; /* next pos to read from inBuf for output */
+
+ /* picoos_int16 outWritePos; */ /* next pos to output from in buf */
+
+ picoos_uint8 sylBuf[255]; /* internal buffer to hold contents of syl item to be output */
+ picoos_uint8 sylReadPos, sylWritePos; /* next pos to read from sylBuf, next pos to write to sylBuf */
+
+ /* buffer for internal calculation of transducer */
+ picotrns_AltDesc altDescBuf;
+ /* the number of AltDesc in the buffer */
+ picoos_uint16 maxAltDescLen;
+
+ /* the input to a transducer should not be larger than PICOTRNS_MAX_NUM_POSSYM
+ * so the output may expand (up to 4*PICOTRNS_MAX_NUM_POSSYM) */
+
+ picotrns_possym_t phonBufA[4 * PICOTRNS_MAX_NUM_POSSYM + 1];
+ picotrns_possym_t phonBufB[4 * PICOTRNS_MAX_NUM_POSSYM + 1];
+ picotrns_possym_t * phonBuf;
+ picotrns_possym_t * phonBufOut;
+ picoos_uint16 phonReadPos, phonWritePos; /* next pos to read from phonBufIn, next pos to write to phonBufIn */
+
+ picoos_int16 activeStartPos; /* start position of items to be treated (at end of left context) */
+ picoos_int16 penultima, activeEndPos; /* positions of last two bounds/words; SPHO_POS_INVALID means uninitialized */
+ picoos_int16 lastPhraseBoundPos; /* position of the last bound encountered (<0 if inexistent or not reachable */
+ picoos_uint8 lastPhraseType; /* phrase type of the last phrase boundary, 0 if not set */
+
+ picoos_uint8 needMoreInput, /* more data necessary to decide on token */
+ suppressParseWordBound, /* dont produce word boundary */
+ suppressRecombWordBound, /* dont produce word boundary */
+ breakPending, /* received a break but didn't interpret it yet */
+ /* sentEnd, */ /* sentence end detected */
+ force, /* in forced state */
+ wordStarted, /* is it the first syl in the word: expect POS */
+ sentenceStarted;
+
+ picoos_uint16 breakTime; /* time argument of the pending break command */
+
+ picoos_uint8 feedFollowState; /* where to return after feed */
+
+ /* fst knowledge bases */
+ picoos_uint8 numFsts;
+ picokfst_FST fst[PICOKNOW_MAX_NUM_SPHO_FSTS];
+ picoos_uint8 curFst; /* the fst to be applied next */
+
+ /* fixed ids knowledge base */
+ picoktab_FixedIds fixedIds;
+
+ /* phones kb */
+ picoktab_Phones phones;
+
+ /* some soecial ids from phones */
+ picoos_uint8 primStressId, secondStressId, syllSepId;
+
+} spho_subobj_t;
+
+
+static pico_status_t sphoReset(register picodata_ProcessingUnit this)
+{
+
+ spho_subobj_t * spho;
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(this->common->em,
+ PICO_ERR_NULLPTR_ACCESS, NULL, NULL);
+ }
+ spho = (spho_subobj_t *) this->subObj;
+
+ spho->curFst = 0;
+
+/* processing state */
+ spho->procState = SPHO_STEPSTATE_INIT;
+ spho->needMoreInput = TRUE;
+ spho->suppressParseWordBound = FALSE;
+ spho->suppressRecombWordBound = FALSE;
+ spho->breakPending = FALSE;
+ spho->force = 0;
+ spho->sentenceStarted = 0;
+
+
+ /* item buffer headx/cbuf */
+ spho->headxBufSize = SPHO_MAXNR_HEADX;
+ spho->headxReadPos = 0;
+ spho->headxWritePos = 0;
+
+ spho->cbufWritePos = 0;
+ spho->cbufBufSize = SPHO_MAXSIZE_CBUF;
+
+ /* possym buffer */
+ spho->phonBuf = spho->phonBufA;
+ spho->phonBufOut = spho->phonBufB;
+ spho->phonReadPos = 0;
+
+ /* overlapping */
+ spho->activeStartPos = 0;
+ spho->penultima = SPHO_POS_INVALID;
+ spho->activeEndPos = SPHO_POS_INVALID;
+
+ return PICO_OK;
+}
+
+
+static pico_status_t sphoInitialize(register picodata_ProcessingUnit this)
+{
+ picoos_uint8 i;
+ spho_subobj_t * spho;
+ picokfst_FST fst;
+
+ picoknow_kb_id_t myKbIds[PICOKNOW_MAX_NUM_SPHO_FSTS] = PICOKNOW_KBID_SPHO_ARRAY;
+
+ PICODBG_DEBUG(("init"));
+
+ if (NULL == this || NULL == this->subObj) {
+ return picoos_emRaiseException(this->common->em,
+ PICO_ERR_NULLPTR_ACCESS, NULL, NULL);
+ }
+
+ spho = (spho_subobj_t *) this->subObj;
+
+ spho->numFsts = 0;
+
+ spho->curFst = 0;
+
+ for (i = 0; i<PICOKNOW_MAX_NUM_SPHO_FSTS; i++) {
+ fst = picokfst_getFST(this->voice->kbArray[myKbIds[i]]);
+ if (NULL != fst) {
+ spho->fst[spho->numFsts++] = fst;
+ }
+ }
+ spho->fixedIds = picoktab_getFixedIds(this->voice->kbArray[PICOKNOW_KBID_FIXED_IDS]);
+ spho->phones = picoktab_getPhones(this->voice->kbArray[PICOKNOW_KBID_TAB_PHONES]);
+
+ spho->syllSepId = picoktab_getSyllboundID(spho->phones);
+ spho->primStressId = picoktab_getPrimstressID(spho->phones);
+ spho->secondStressId = picoktab_getSecstressID(spho->phones);
+
+ PICODBG_DEBUG(("got %i fsts", spho->numFsts));
+
+
+ return sphoReset(this);
+
+}
+
+static picodata_step_result_t sphoStep(register picodata_ProcessingUnit this,
+ picoos_int16 mode, picoos_uint16 *numBytesOutput);
+
+
+
+
+static pico_status_t sphoTerminate(register picodata_ProcessingUnit this)
+{
+ return PICO_OK;
+}
+
+
+static pico_status_t sphoSubObjDeallocate(register picodata_ProcessingUnit this,
+ picoos_MemoryManager mm)
+{
+ spho_subobj_t * spho;
+
+ spho = (spho_subobj_t *) this->subObj;
+
+ if (NULL != this) {
+ if (NULL != this->subObj) {
+ spho = (spho_subobj_t *) (this->subObj);
+ picotrns_deallocate_alt_desc_buf(spho->common->mm,&spho->altDescBuf);
+ picoos_deallocate(mm, (void *) &this->subObj);
+ }
+ }
+ return PICO_OK;
+}
+
+picodata_ProcessingUnit picospho_newSentPhoUnit(picoos_MemoryManager mm,
+ picoos_Common common, picodata_CharBuffer cbIn,
+ picodata_CharBuffer cbOut, picorsrc_Voice voice)
+{
+ spho_subobj_t * spho;
+
+ picodata_ProcessingUnit this = picodata_newProcessingUnit(mm, common, cbIn, cbOut, voice);
+ if (this == NULL) {
+ return NULL;
+ }
+
+ this->initialize = sphoInitialize;
+ this->step = sphoStep;
+ this->terminate = sphoTerminate;
+ this->subDeallocate = sphoSubObjDeallocate;
+
+ this->subObj = picoos_allocate(mm, sizeof(spho_subobj_t));
+ if (this->subObj == NULL) {
+ picoos_deallocate(mm, (void **)(void*)&this);
+ return NULL;
+ }
+ spho = (spho_subobj_t *) this->subObj;
+
+ spho->common = this->common;
+
+ /* these are given by the pre-allocated array sizes */
+ spho->outBufSize = PICODATA_BUFSIZE_DEFAULT;
+
+
+ spho->altDescBuf = picotrns_allocate_alt_desc_buf(spho->common->mm, SPHO_MAX_ALTDESC_SIZE, &spho->maxAltDescLen);
+ if (NULL == spho->altDescBuf) {
+ picotrns_deallocate_alt_desc_buf(spho->common->mm,&spho->altDescBuf);
+ picoos_emRaiseException(spho->common->em,PICO_EXC_OUT_OF_MEM, NULL,NULL);
+ return NULL;
+ }
+
+ sphoInitialize(this);
+ return this;
+}
+
+
+/* ***********************************************************************/
+/* process buffered item list */
+/* ***********************************************************************/
+
+
+/* shift relevant data in headx/'cbuf' (between 'readPos' incl and writePos non-incl) to 'start'.
+ * modify read/writePos accordingly */
+static picoos_int16 shift_range_left_1(spho_subobj_t *spho, picoos_int16 * from, picoos_int16 to)
+{
+
+ /* remember shift parameters for cbuf */
+ picoos_uint16
+ c_i,
+ c_j,
+ c_diff,
+ c_writePos,
+ i,
+ j,
+ diff,
+ writePos;
+ i = to;
+ j = *from;
+ diff = j-i;
+ writePos = spho->headxWritePos;
+ c_i = spho->headx[to].cind;
+ if (j < writePos) {
+ c_j = spho->headx[j].cind;
+ } else {
+ c_j = spho->cbufWritePos;
+ }
+ c_diff = c_j - c_i;
+ c_writePos = spho->cbufWritePos;
+
+ PICODBG_DEBUG((
+ "shifting buffer region [%i,%i[ down to %i",*from, writePos, to
+ ));
+
+
+ /* PICODBG_ASSERT((i<j)); */
+ if (i > j) {
+ return -1;
+ }
+ /* shift cbuf */
+ while (c_j < c_writePos) {
+ spho->cbuf[c_i++] = spho->cbuf[c_j++];
+ }
+ /* shift headx */
+ while (j < writePos) {
+ spho->headx[j].cind -= c_diff;
+ spho->headx[i++] = spho->headx[j++];
+ }
+ spho->headxWritePos -= diff;
+ *from = to;
+ spho->cbufWritePos -= c_diff;
+ /* */
+ PICODBG_DEBUG((
+ "readPos,WritePos are now [%i,%i[, returning shift amount %i",*from, spho->headxWritePos, diff
+ ));
+ return diff;
+}
+
+static pico_status_t sphoAddPhoneme(register spho_subobj_t *spho, picoos_int16 pos, picoos_int16 sym) {
+ picoos_uint8 plane, unshifted;
+ /* just for debuging */
+ unshifted = picotrns_unplane(sym,&plane);
+ PICODBG_TRACE(("adding %i/%i (%c on plane %i) at phonBuf[%i]",pos,sym,unshifted,plane,spho->phonWritePos));
+ if (2* PICOTRNS_MAX_NUM_POSSYM <= spho->phonWritePos) {
+ /* not an error! */
+ PICODBG_DEBUG(("couldn't add because phon buffer full"));
+ return PICO_EXC_BUF_OVERFLOW;
+ } else {
+ spho->phonBuf[spho->phonWritePos].pos = pos;
+ spho->phonBuf[spho->phonWritePos].sym = sym;
+ spho->phonWritePos++;
+ return PICO_OK;
+ }
+}
+
+static pico_status_t sphoAddStartPhoneme(register spho_subobj_t *spho) {
+ return sphoAddPhoneme(spho, PICOTRNS_POS_IGNORE,
+ (PICOKFST_PLANE_INTERN << 8) + spho->fixedIds->phonStartId);
+}
+
+static pico_status_t sphoAddTermPhonemes(register spho_subobj_t *spho, picoos_uint16 pos) {
+ return sphoAddPhoneme(spho, pos,
+ (PICOKFST_PLANE_PB_STRENGTHS << 8) + PICODATA_ITEMINFO1_BOUND_SEND)
+ && sphoAddPhoneme(spho, PICOTRNS_POS_IGNORE,
+ (PICOKFST_PLANE_INTERN << 8) + spho->fixedIds->phonTermId);
+}
+
+/* return "syllable accent" (or prominence) symbol, given "word accent" symbol 'wacc' and stress value (no=0, primary=1, secondary=2) */
+static picoos_uint16 sphoGetSylAccent(register spho_subobj_t *spho,
+ picoos_uint8 wacc, picoos_uint8 sylStress)
+{
+ PICODBG_ASSERT(sylStress <= 2);
+
+ spho = spho; /* avoid warning "var not used in this function"*/
+
+ switch (sylStress) {
+ case 0: /* non-stressed syllable gets no prominence */
+ /* return spho->fixedIds->accId[0]; */
+ return PICODATA_ACC0;
+ break;
+ case 1: /* primary-stressed syllable gets word prominence */
+ return wacc;
+ break;
+ case 2: /* secondary-stressed syllable gets no prominence or secondary stress prom. (4) */
+ return (PICODATA_ACC0 == wacc) ? PICODATA_ACC0
+ : PICODATA_ACC4;
+ /*return (spho->fixedIds->accId[0] == wacc) ? spho->fixedIds->accId[0]
+ : spho->fixedIds->accId[4]; */
+ break;
+ default:
+ /* never occurs :-) */
+ return PICODATA_ACC0;
+ break;
+ }
+}
+
+
+/* ***********************************************************************/
+/* extract phonemes of an item into a phonBuf */
+/* ***********************************************************************/
+static pico_status_t sphoExtractPhonemes(register picodata_ProcessingUnit this,
+ register spho_subobj_t *spho, picoos_uint16 pos,
+ picoos_uint8 convertAccents, picoos_uint8 * suppressWB)
+{
+ pico_status_t rv = PICO_OK;
+ picoos_uint16 i, j;
+ picoos_int16 fstSymbol;
+ picoos_uint8 curStress;
+ picotrns_possym_t tmpPosSym;
+ picoos_uint16 oldPos, curPos;
+ picodata_itemhead_t * head;
+ picoos_uint8* content;
+
+#if defined(PICO_DEBUG)
+ picoos_char msgstr[SPHO_MSGSTR_SIZE];
+#endif
+
+
+ /*
+ Items considered in a transduction are a BOUND or a WORDPHON item. its starting offset within the
+ headxBuf is given as 'pos'.
+ Elements that go into the transduction receive "their" position in the buffer.
+ */
+
+ oldPos = spho->phonWritePos;
+
+ head = &(spho->headx[pos].head);
+ content = spho->cbuf + spho->headx[pos].cind;
+
+ PICODBG_TRACE(("doing item %s\n",
+ picodata_head_to_string(head,msgstr,SPHO_MSGSTR_SIZE)));
+
+ switch (head->type) {
+ case PICODATA_ITEM_BOUND:
+ /* map SBEG, SEND and TERM (as sentence closing) to SEND */
+ fstSymbol = (PICODATA_ITEMINFO1_BOUND_SBEG == head->info1 || PICODATA_ITEMINFO1_BOUND_TERM == head->info1) ? PICODATA_ITEMINFO1_BOUND_SEND : head->info1;
+ PICODBG_TRACE(("found bound of type %c\n",head->info1));
+ /* BOUND(<bound strength><phrase type>) */
+ /* insert bound strength */
+ PICODBG_TRACE(("inserting phrase bound phoneme %c and setting suppresWB=1\n",fstSymbol));
+ fstSymbol += (PICOKFST_PLANE_PB_STRENGTHS << 8);
+ rv = sphoAddPhoneme(spho,pos,fstSymbol);
+ /* phrase type not used */
+ /* suppress next word boundary */
+ (*suppressWB) = 1;
+ break;
+
+ case PICODATA_ITEM_WORDPHON:
+ /* WORDPHON(POS,WACC)phon */
+ PICODBG_TRACE(("found WORDPHON"));
+ /* insert word boundary if not suppressed */
+ if (!(*suppressWB)) {
+ fstSymbol = (PICOKFST_PLANE_PB_STRENGTHS << 8) + PICODATA_ITEMINFO1_BOUND_PHR0;
+ PICODBG_TRACE(("adding word boundary phone"));
+ rv = sphoAddPhoneme(spho,pos,fstSymbol);
+ }
+ (*suppressWB) = 0;
+ /* for the time being, we force to use POS so we can transduce all fsts in a row without reconsulting the items */
+
+
+ /* If 'convertAccents' then the accentuation is not directly encoded. It rather influences the mapping of
+ * the word accent symbol to the actual accent phoneme which is put after the syllable separator. */
+ if (convertAccents) {
+ PICODBG_TRACE(("converting accents"));
+ /* extracting phonemes IN REVERSE order replacing syllable symbols with prominence symbols */
+ curPos = spho->phonWritePos;
+ curStress = 0; /* no stress */
+ for (i = head->len; i > 0 ;) {
+ i--;
+ if (spho->primStressId == content[i]) {
+ curStress = 1;
+ PICODBG_DEBUG(("skipping primary stress at pos %i (in 1 .. %i)",i, head->len));
+ continue; /* skip primary stress symbol */
+ } else if (spho->secondStressId == content[i]) {
+ curStress = 2;
+ PICODBG_DEBUG(("skipping secondary stress at pos %i (in 1 .. %i)",i, head->len));
+ continue; /* skip secundary stress symbol */
+ } else if (spho->syllSepId == content[i]) {
+ fstSymbol = (PICOKFST_PLANE_POS << 8) + head->info1;
+ rv = sphoAddPhoneme(spho, pos, fstSymbol);
+ /* replace syllSepId by combination of syllable stress and word prominence */
+ fstSymbol = sphoGetSylAccent(spho,head->info2,curStress);
+ curStress = 0;
+ /* add accent */
+ fstSymbol += (PICOKFST_PLANE_ACCENTS << 8);
+ rv = sphoAddPhoneme(spho,pos,fstSymbol);
+ if (PICO_OK != rv) {
+ break;
+ }
+ /* and keep syllable boundary */
+ fstSymbol = (PICOKFST_PLANE_PHONEMES << 8) + content[i];
+ } else {
+ /* normal phoneme */
+ fstSymbol = (PICOKFST_PLANE_PHONEMES << 8) + content[i];
+ }
+ if (PICO_OK == rv) {
+ rv = sphoAddPhoneme(spho,pos,fstSymbol);
+ }
+ }
+ if (PICO_OK == rv) {
+ /* bug 366: we position the "head" into the item header and not on the first phoneme
+ * because there might be no phonemes at all */
+ /* insert head of the first syllable of a word */
+ fstSymbol = (PICOKFST_PLANE_POS << 8) + head->info1;
+ rv = sphoAddPhoneme(spho,pos,fstSymbol);
+ fstSymbol = sphoGetSylAccent(spho,head->info2,curStress);
+ curStress = 0;
+ fstSymbol += (PICOKFST_PLANE_ACCENTS << 8);
+ rv = sphoAddPhoneme(spho,pos,fstSymbol);
+ }
+ if (PICO_OK == rv) {
+ /* invert sympos portion */
+ i = curPos;
+ j=spho->phonWritePos-1;
+ while (i < j) {
+ tmpPosSym.pos = spho->phonBuf[i].pos;
+ tmpPosSym.sym = spho->phonBuf[i].sym;
+ spho->phonBuf[i].pos = spho->phonBuf[j].pos;
+ spho->phonBuf[i].sym = spho->phonBuf[j].sym;
+ spho->phonBuf[j].pos = tmpPosSym.pos;
+ spho->phonBuf[j].sym = tmpPosSym.sym;
+ i++;
+ j--;
+ }
+ }
+ } else { /* convertAccents */
+ for (i = 0; i <head->len; i++) {
+ fstSymbol = (PICOKFST_PLANE_PHONEMES << 8) + content[i];
+ rv = sphoAddPhoneme(spho,pos,fstSymbol);
+ }
+ }
+ break;
+ default:
+ picoos_emRaiseException(this->common->em,rv,NULL,NULL);
+ break;
+ } /* switch(head->type) */
+ if (PICO_OK != rv) {
+ spho->phonWritePos = oldPos;
+ }
+ return rv;
+}
+
+
+
+
+
+#define SPHO_POSSYM_OK 0
+#define SPHO_POSSYM_OUT_OF_RANGE 1
+#define SPHO_POSSYM_END 2
+#define SPHO_POSSYM_INVALID -3
+/* *readPos is the next position in phonBuf to be read, and *writePos is the first position not to be read (may be outside
+ * buf).
+ * 'rangeEnd' is the first possym position outside the desired range.
+ * Possible return values:
+ * SPHO_POSSYM_OK : 'pos' and 'sym' are set to the read possym, *readPos is advanced
+ * SPHO_POSSYM_OUT_OF_RANGE : pos is out of range. 'pos' is set to that of the read possym, 'sym' is undefined
+ * SPHO_POSSYM_UNDERFLOW : no more data in buf. 'pos' is set to PICOTRNS_POS_INVALID, 'sym' is undefined
+ * SPHO_POSSYM_INVALID : "strange" pos. 'pos' is set to PICOTRNS_POS_INVALID, 'sym' is undefined
+ */
+static pico_status_t getNextPosSym(spho_subobj_t * spho, picoos_int16 * pos, picoos_int16 * sym,
+ picoos_int16 rangeEnd) {
+ /* skip POS_IGNORE */
+ while ((spho->phonReadPos < spho->phonWritePos) && (PICOTRNS_POS_IGNORE == spho->phonBuf[spho->phonReadPos].pos)) {
+ PICODBG_DEBUG(("ignoring phone at spho->phonBuf[%i] because it has pos==IGNORE",spho->phonReadPos));
+ spho->phonReadPos++;
+ }
+ if ((spho->phonReadPos < spho->phonWritePos)) {
+ *pos = spho->phonBuf[spho->phonReadPos].pos;
+ if ((PICOTRNS_POS_INSERT == *pos) || ((0 <= *pos) && (*pos < rangeEnd))) {
+ *sym = spho->phonBuf[spho->phonReadPos++].sym;
+ return SPHO_POSSYM_OK;
+ } else if (*pos < 0){ /* *pos is "strange" (e.g. POS_INVALID) */
+ return SPHO_POSSYM_INVALID;
+ } else {
+ return SPHO_POSSYM_OUT_OF_RANGE;
+ }
+ } else {
+ /* no more possyms to read */
+ *pos = PICOTRNS_POS_INVALID;
+ return SPHO_POSSYM_END;
+ }
+}
+
+
+
+/** Calculate bound strength modified by transduction
+ *
+ * Given the original bound strength 'orig' and the desired target strength 'target' (suggested by fst),
+ * calculate the modified bound strength.
+ *
+ * @param orig original bound strength
+ * @param target target bound strength
+ * @return resulting bound strength
+ */
+static picoos_uint8 fstModifiedBoundStrength(picoos_uint8 orig, picoos_uint8 target)
+{
+ switch (orig) {
+ case PICODATA_ITEMINFO1_BOUND_PHR1:
+ case PICODATA_ITEMINFO1_BOUND_PHR2:
+ /* don't allow primary phrase bounds to be demoted to word bound */
+ if (PICODATA_ITEMINFO1_BOUND_PHR0 == target) {
+ return PICODATA_ITEMINFO1_BOUND_PHR3;
+ }
+ case PICODATA_ITEMINFO1_BOUND_PHR0:
+ case PICODATA_ITEMINFO1_BOUND_PHR3:
+ return target;
+ break;
+ default:
+ /* don't allow bounds other than phrase or word bounds to be changed */
+ return orig;
+ break;
+ }
+}
+
+/** Calculate bound strength modified by a \<break> command
+ *
+ * Given the original (predicted and possibly fst-modified) bound strength, and a time value from an
+ * overwriding \<break> command, calculate the modified bound strength.
+ *
+ * @param orig original bound strength
+ * @param time time given as property of \<break> command
+ * @param wasPrimary
+ * @return modified bound strength
+ */
+static picoos_uint8 breakModifiedBoundStrength(picoos_uint8 orig, picoos_uint16 time, picoos_bool wasPrimary)
+{
+ picoos_uint8 modified = (0 == time) ? PICODATA_ITEMINFO1_BOUND_PHR3 :
+ (50 < time) ? PICODATA_ITEMINFO1_BOUND_PHR1 : PICODATA_ITEMINFO1_BOUND_PHR2;
+ switch (orig) {
+ /* for word and phrase breaks, return 'modified', unless a non-silence gets time==0, in which
+ * case return no break (word break) */
+ case PICODATA_ITEMINFO1_BOUND_PHR0:
+ if (0 == time) {
+ return PICODATA_ITEMINFO1_BOUND_PHR0;
+ }
+ case PICODATA_ITEMINFO1_BOUND_PHR3:
+ if (!wasPrimary && (0 == time)) {
+ return PICODATA_ITEMINFO1_BOUND_PHR0;
+ }
+ case PICODATA_ITEMINFO1_BOUND_PHR1:
+ case PICODATA_ITEMINFO1_BOUND_PHR2:
+ return modified;
+ break;
+ default:
+ return orig;
+ break;
+ }
+}
+
+static picoos_bool breakStateInterrupting(picodata_itemhead_t * head,
+ picoos_bool * breakBefore, picoos_bool * breakAfter) {
+
+ picoos_bool result = 1;
+
+ *breakBefore = 0;
+ *breakAfter = 0;
+
+ if (PICODATA_ITEM_WORDPHON == head->type) {
+
+ } else if (PICODATA_ITEM_CMD == head->type) {
+ if ((PICODATA_ITEMINFO1_CMD_PLAY == head->info1)
+ || (PICODATA_ITEMINFO1_CMD_SAVE == head->info1)
+ || (PICODATA_ITEMINFO1_CMD_UNSAVE == head->info1)) {
+ *breakBefore = 1;
+ *breakAfter = 1;
+ } else if (PICODATA_ITEMINFO1_CMD_SAVE == head->info1) {
+ *breakBefore = 1;
+ } else if (PICODATA_ITEMINFO1_CMD_UNSAVE == head->info1) {
+ *breakAfter = 1;
+ } else if (PICODATA_ITEMINFO1_CMD_IGNSIG == head->info1) {
+ if (PICODATA_ITEMINFO2_CMD_START == head->info2) {
+ *breakBefore = 1;
+ } else {
+ *breakAfter = 1;
+ }
+ }
+ } else {
+ result = 0;
+ }
+ return result;
+}
+
+
+static void putSideBoundToOutput(spho_subobj_t * spho)
+{
+
+ picodata_itemhead_t ohead;
+ picoos_uint8 ocontent[2*sizeof(picoos_uint16)];
+ picoos_int16 sildur;
+ picoos_uint16 clen;
+
+ /* create boundary */
+ ohead.type = PICODATA_ITEM_BOUND;
+ ohead.info1 = spho->headx[spho->outReadPos].boundstrength;
+ ohead.info2 = spho->headx[spho->outReadPos].phrasetype;
+ sildur = spho->headx[spho->outReadPos].sildur;
+ if ((sildur < 0)
+ || (PICODATA_ITEMINFO1_BOUND_PHR0 == ohead.info1)
+ || (PICODATA_ITEMINFO1_BOUND_PHR3 == ohead.info1)) {
+ PICODBG_DEBUG(("outputting a bound of strength '%c' and type '%c' without duration constraints",ohead.info1, ohead.info2));
+ ohead.len = 0;
+ } else {
+ picoos_uint32 pos = 0;
+ picoos_write_mem_pi_uint16(ocontent,&pos,sildur);
+ picoos_write_mem_pi_uint16(ocontent,&pos,sildur);
+ PICODBG_DEBUG(("outputting a bound of strength '%c' and type '%c' with duration constraints [%i,%i]",ohead.info1, ohead.info2,sildur, sildur));
+ ohead.len = pos;
+ }
+ picodata_put_itemparts(&ohead, ocontent, ohead.len,
+ spho->outBuf, spho->outBufSize, &clen);
+ /* disable side bound */
+ spho->headx[spho->outReadPos].boundstrength = 0;
+}
+
+/** Set bound strength and sil dur.
+ *
+ * given the original bound strength 'orig_strength' and the fst-suggested bound strength 'fst_strength'
+ * and possibly being in a pending break state, calculate the resulting bound strength and set boundstrength
+ * and sildur of the current item (spho->headx[spho->outReadPos]) accordingly.
+ * if a boundstrength was set, also calculate the phrasetype and if necessary (and reachable), modify the phrase type
+ * of the previous phrase boundary.
+ *
+ * @param spho
+ * @param orig_strength
+ * @param orig_type
+ * @param fst_strength
+ */
+static void setSideBound(spho_subobj_t * spho, picoos_uint8 orig_strength, picoos_uint8 orig_type, picoos_uint8 fst_strength) {
+ picoos_uint8 strength;
+
+ /* insert modified bound according to transduction symbol, if any */
+ if (PICODATA_ITEMINFO1_NA == orig_strength) {
+ /* no original/fst strength given */
+ orig_strength = PICODATA_ITEMINFO1_BOUND_PHR0;
+ strength = PICODATA_ITEMINFO1_BOUND_PHR0;
+ } else {
+ strength = fstModifiedBoundStrength(orig_strength,fst_strength);
+ spho->headx[spho->outReadPos].boundstrength = strength;
+ spho->headx[spho->outReadPos].sildur = -1;
+ PICODBG_DEBUG(("setting bound strength to fst-suggested value %c (was %c)",strength, spho->headx[spho->outReadPos].boundstrength, spho->breakTime));
+ }
+
+ /* insert modified bound according to pending break, if any */
+ if (spho->breakPending) {
+ /* the calculation is based on the fst-modified value (because this is what the customer wants to
+ * override)
+ */
+ strength = breakModifiedBoundStrength(strength, spho->breakTime, (PICODATA_ITEMINFO1_BOUND_PHR1 == orig_strength));
+ PICODBG_DEBUG(("setting bound strength to break-imposed value %c (was %c) and time to %i",strength, spho->headx[spho->outReadPos].boundstrength, spho->breakTime));
+ spho->headx[spho->outReadPos].boundstrength = strength;
+ spho->headx[spho->outReadPos].sildur = spho->breakTime;
+ spho->breakPending = FALSE;
+ }
+ if (spho->headx[spho->outReadPos].boundstrength) {
+ /* we did set a bound strength, possibly promoting or demoting a boundary; now set the phrase type
+ * possibly also changing the phrase type of the previous phrase bound
+ */
+ picoos_uint8 fromPhrase = ((PICODATA_ITEMINFO1_BOUND_PHR0 != orig_strength));
+ picoos_uint8 toPhrase = ((PICODATA_ITEMINFO1_BOUND_PHR0 != strength));
+
+ PICODBG_DEBUG(("setting phrase type (wasPhrase=%i, isPhrase=%i)",fromPhrase,toPhrase));
+ if (toPhrase) {
+ if (fromPhrase) {
+ spho->lastPhraseType = orig_type;
+ } else { /*promote */
+ if (spho->activeStartPos <= spho->lastPhraseBoundPos) {
+ /* we still can change prev phrase bound */
+ /* since a new phrase boundary is introduced, we have to 'invent'
+ * an additional phrase type here. For that, we have to use some of the
+ * knowledge that otherwise is handled in picoacph.
+ */
+ spho->headx[spho->lastPhraseBoundPos].phrasetype
+ = PICODATA_ITEMINFO2_BOUNDTYPE_P;
+ }
+ }
+ spho->lastPhraseBoundPos = spho->outReadPos;
+ spho->headx[spho->lastPhraseBoundPos].phrasetype
+ = spho->lastPhraseType;
+
+ } else {
+ spho->headx[spho->outReadPos].phrasetype = PICODATA_ITEMINFO2_NA;
+ if (fromPhrase) { /* demote */
+ spho->lastPhraseType = orig_type;
+ if (spho->activeStartPos <= spho->lastPhraseBoundPos) {
+ /* we still can change prev phrase bound */
+ spho->headx[spho->lastPhraseBoundPos].phrasetype
+ = spho->lastPhraseType;
+ }
+ }
+ }
+ }
+}
+
+
+/* ***********************************************************************/
+/* sphoStep function */
+/* ***********************************************************************/
+
+
+static picodata_step_result_t sphoStep(register picodata_ProcessingUnit this,
+ picoos_int16 mode, picoos_uint16 * numBytesOutput)
+{
+
+ register spho_subobj_t *spho;
+ pico_status_t rv= PICO_OK;
+ picoos_uint16 blen;
+ picodata_itemhead_t ihead, ohead;
+ picoos_uint8 *icontent;
+ picoos_uint16 nextInPos;
+#if defined(PICO_DEBUG)
+ picoos_char msgstr[SPHO_MSGSTR_SIZE];
+#endif
+
+ /* used in FEED and FEED_SYM */
+ picoos_uint16 clen;
+ picoos_int16 pos, sym, sylsym;
+ picoos_uint8 plane;
+
+ /* used in BOUNDS */
+ picoos_bool breakBefore, breakAfter;
+
+ /* pico_status_t rvP= PICO_OK; */
+
+ picoos_uint16 curPos /*, nextPos */;
+ picoos_uint16 remHeadxSize, remCbufSize;
+
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ spho = (spho_subobj_t *) this->subObj;
+
+ mode = mode; /* avoid warning "var not used in this function"*/
+
+ *numBytesOutput = 0;
+ while (1) { /* exit via return */
+ PICODBG_INFO(("doing state %i, headxReadPos: %d, headxWritePos: %d",
+ spho->procState, spho->headxReadPos, spho->headxWritePos));
+
+ switch (spho->procState) {
+
+ case SPHO_STEPSTATE_INIT:
+ /* **********************************************************************/
+ /* INIT */
+ /* **********************************************************************/
+ PICODBG_DEBUG(("INIT"));
+ /* (re)set values for PARSE */
+ spho->penultima = SPHO_POS_INVALID;
+ spho->activeEndPos = SPHO_POS_INVALID;
+ spho->headxReadPos = 0;
+ spho->phonReadPos = 0;
+ spho->phonWritePos = 0;
+ spho->lastPhraseType = PICODATA_ITEMINFO2_NA;
+ spho->lastPhraseBoundPos = -1;
+
+ spho->procState = SPHO_STEPSTATE_COLLECT;
+ break;
+
+
+ case SPHO_STEPSTATE_COLLECT:
+ /* **********************************************************************/
+ /* COLLECT */
+ /* **********************************************************************/
+ /* collect state: get items from charBuf and store in
+ * internal inBuf
+ */
+ PICODBG_TRACE(("COLLECT"));
+ rv = PICO_OK;
+ remHeadxSize = spho->headxBufSize - spho->headxWritePos;
+ remCbufSize = spho->cbufBufSize - spho->cbufWritePos;
+ curPos = spho->headxWritePos;
+ while ((PICO_OK == rv) && (remHeadxSize > 0) && (remCbufSize > 0)) {
+ PICODBG_DEBUG(("COLLECT getting item at headxWritePos %i (remaining %i)",spho->headxWritePos, remHeadxSize));
+ rv = picodata_cbGetItem(this->cbIn, spho->tmpbuf, PICODATA_MAX_ITEMSIZE, &blen);
+ if (PICO_OK == rv) {
+ rv = picodata_get_itemparts(spho->tmpbuf,
+ PICODATA_MAX_ITEMSIZE, &(spho->headx[spho->headxWritePos].head),
+ &(spho->cbuf[spho->cbufWritePos]), remCbufSize, &blen);
+ if (PICO_OK == rv) {
+ spho->headx[spho->headxWritePos].cind = spho->cbufWritePos;
+ spho->headx[spho->headxWritePos].boundstrength = 0;
+ spho->headxWritePos++;
+ remHeadxSize--;
+ spho->cbufWritePos += blen;
+ remCbufSize -= blen;
+ }
+ }
+ }
+ if ((PICO_OK == rv) && ((remHeadxSize <= 0) || (remCbufSize <= 0))) {
+ rv = PICO_EXC_BUF_OVERFLOW;
+ }
+
+ /* in normal circumstances, rv is either PICO_EOF (no more items in cbIn) or PICO_BUF_OVERFLOW
+ * (if no more items fit into headx) */
+ if ((PICO_EOF != rv) && (PICO_EXC_BUF_OVERFLOW != rv)) {
+ PICODBG_DEBUG(("COLLECT ** problem getting item, unhandled, rv: %i", rv));
+ picoos_emRaiseException(this->common->em, rv,
+ NULL, NULL);
+ return PICODATA_PU_ERROR;
+ }
+ if (PICO_EOF == rv) { /* there are no more items available */
+ if (curPos < spho->headxWritePos) { /* we did get some new items */
+ PICODBG_DEBUG(("COLLECT read %i items",
+ spho->headxWritePos - curPos));
+ spho->needMoreInput = FALSE;
+ }
+ if (spho->needMoreInput) { /* not enough items to proceed */
+ PICODBG_DEBUG(("COLLECT need more data, returning IDLE"));
+ return PICODATA_PU_IDLE;
+ } else {
+ spho->procState = SPHO_STEPSTATE_PROCESS_PARSE;
+ /* uncomment next to split into two steps */
+ /* return PICODATA_PU_ATOMIC; */
+ }
+ } else { /* input buffer full */
+ PICODBG_DEBUG(("COLLECT input buffer full"));
+ if (spho->needMoreInput) { /* forced output because we can't get more data */
+ spho->needMoreInput = FALSE;
+ spho->force = TRUE;
+ }
+ spho->procState = SPHO_STEPSTATE_PROCESS_PARSE;
+ }
+ break;
+
+ case SPHO_STEPSTATE_PROCESS_PARSE:
+
+ /* **********************************************************************/
+ /* PARSE: items -> input pos/phon pairs */
+ /* **********************************************************************/
+
+ /* parse one item at a time */
+ /* If
+ * - the item is a sentence end or
+ * - it is the last item and force=1 or
+ * - the phon buffer is full
+ * then set inReadPos to 0 and go to TRANSDUCE
+ * else advance by one item */
+
+ /* look at the current item */
+ PICODBG_TRACE(("PARSE"));
+ if (spho->headxReadPos >= spho->headxWritePos) {
+ /* no more items in headx */
+ if (spho->force) {
+ PICODBG_INFO(("no more items in headx but we are forced to transduce"));
+
+ /* headx is full; we are forced to transduce before reaching the sentence end */
+ spho->force = FALSE;
+ if (SPHO_POS_INVALID == spho->activeEndPos) {
+ spho->activeEndPos = spho->headxReadPos;
+ }
+ spho->procState = SPHO_STEPSTATE_PROCESS_TRANSDUCE;
+ } else {
+ /* we try to get more data */
+ PICODBG_INFO(("no more items in headx, try to collect more"));
+ spho->needMoreInput = TRUE;
+ spho->procState = SPHO_STEPSTATE_COLLECT;
+ }
+ break;
+ }
+
+ ihead = spho->headx[spho->headxReadPos].head;
+ icontent = spho->cbuf + spho->headx[spho->headxReadPos].cind;
+
+ PICODBG_DEBUG(("PARSE looking at item %s",picodata_head_to_string(&ihead,msgstr,SPHO_MSGSTR_SIZE)));
+ /* treat header */
+ if (PICODATA_ITEM_BOUND == ihead.type) {
+ /* see if it is a sentence end or termination boundary (flush) */
+ if ((PICODATA_ITEMINFO1_BOUND_SEND == ihead.info1)
+ || (PICODATA_ITEMINFO1_BOUND_TERM == ihead.info1)) {
+ PICODBG_INFO(("PARSE found sentence end or term BOUND"));
+
+ if (spho->sentenceStarted) {
+ /* its the end of the sentence */
+ PICODBG_INFO(("PARSE found sentence end"));
+ spho->sentenceStarted = 0;
+ /* there is no need for a right context; move the active end to the end */
+ /* add sentence termination phonemes */
+ sphoAddTermPhonemes(spho, spho->headxReadPos);
+ spho->headxReadPos++;
+ spho->activeEndPos = spho->headxReadPos;
+ /* we may discard all information up to activeEndPos, after processing of last
+ * sentence part
+ */
+ spho->penultima = spho->activeEndPos;
+
+ /* transduce */
+ spho->procState = SPHO_STEPSTATE_PROCESS_TRANSDUCE;
+ /* uncomment to split */
+ /* return PICODATA_PU_BUSY; */
+ break;
+ } else {
+ if (PICODATA_ITEMINFO1_BOUND_TERM == ihead.info1) {
+ /* its the end of input (flush) */
+ PICODBG_INFO(("PARSE forwarding input end (flush)"));
+ /* copy item unmodified */
+ picodata_put_itemparts(&ihead,
+ icontent,
+ ihead.len,
+ spho->outBuf, spho->outBufSize,
+ &clen);
+
+ spho->headxReadPos++;
+ spho->activeEndPos = spho->headxReadPos;
+ spho->penultima = SPHO_POS_INVALID;
+ spho->feedFollowState = SPHO_STEPSTATE_SHIFT;
+ spho->procState = SPHO_STEPSTATE_FEED;
+ break;
+ } else {
+ /* this should never happen */
+ /* eliminate bound */
+ spho->headxReadPos++;
+ spho->activeEndPos = spho->headxReadPos;
+ spho->penultima = SPHO_POS_INVALID;
+ PICODBG_ERROR(("PARSE found a sentence end without a sentence start; eliminated"));
+ }
+ }
+ } else if (PICODATA_ITEMINFO1_BOUND_SBEG == ihead.info1) {
+ /* its the start of the sentence */
+ PICODBG_INFO(("PARSE found sentence start"));
+ /* add sentence starting phoneme */
+ sphoAddStartPhoneme(spho);
+
+ spho->sentenceStarted = 1;
+ }
+ }
+
+ if ((PICODATA_ITEM_WORDPHON == ihead.type)
+ || (PICODATA_ITEM_BOUND == ihead.type)) {
+ /* if it is a word or a bound try to extract phonemes */
+ PICODBG_INFO(("PARSE found WORD phon or phrase BOUND"));
+ rv = sphoExtractPhonemes(this, spho, spho->headxReadPos,
+ TRUE /* convertAccents */,
+ &spho->suppressParseWordBound);
+ if (PICO_OK == rv) {
+ PICODBG_INFO(("PARSE successfully returned from phoneme extraction"));
+ /* replace activeEndPos if the new item is a word, or activeEndPos was not set yet, or
+ * activeEndPos was a bound */
+ if ((spho->activeStartPos <= spho->headxReadPos) && ((PICODATA_ITEM_WORDPHON == ihead.type)
+ || (SPHO_POS_INVALID == spho->activeEndPos)
+ || (PICODATA_ITEM_BOUND == spho->headx[spho->activeEndPos].head.type))) {
+ PICODBG_INFO(("PARSE found new activeEndPos: %i,%i -> %i,%i",
+ spho->penultima,spho->activeEndPos,spho->activeEndPos,spho->headxReadPos));
+ spho->penultima = spho->activeEndPos;
+ spho->activeEndPos = spho->headxReadPos;
+ }
+
+ } else if (PICO_EXC_BUF_OVERFLOW == rv) {
+ /* phoneme buffer cannot take this item anymore;
+ if the phoneme buffer has some contents, we are forced to transduce before reaching the sentence end
+ else we skip the (too long word) */
+ PICODBG_INFO(("PARSE returned from phoneme extraction with overflow, number of phonemes in phonBuf: %i; forced to TRANSDUCE", spho->phonWritePos));
+ if ((SPHO_POS_INVALID == spho->activeEndPos) || (spho->activeStartPos == spho->activeEndPos)) {
+ spho->activeEndPos = spho->headxReadPos;
+ }
+ spho->procState = SPHO_STEPSTATE_PROCESS_TRANSDUCE;
+ break;
+ } else {
+ PICODBG_ERROR(("PARSE returned from phoneme extraction with exception %i",rv));
+ return (picodata_step_result_t)picoos_emRaiseException(this->common->em,
+ PICO_ERR_OTHER, NULL, NULL);
+ }
+ } else {
+ PICODBG_INFO(("PARSE found other item, passing over"));
+ /* it is "other" item, ignore */
+ }
+ /* set pos at next item */
+ PICODBG_INFO(("PARSE going to next item: %i -> %i",spho->headxReadPos, spho->headxReadPos + 1));
+ spho->headxReadPos++;
+ break;
+
+ case SPHO_STEPSTATE_PROCESS_TRANSDUCE:
+
+ /* **********************************************************************/
+ /* TRANSDUCE: transduction input pos/phon pairs to output pos/phon pairs */
+ /* **********************************************************************/
+ PICODBG_DEBUG(("TRANSDUCE (%i-th of %i fsts",spho->curFst+1, spho->numFsts));
+
+ /* termination condition first */
+ if (spho->curFst >= spho->numFsts) {
+
+#if defined(PICO_DEBUG)
+ {
+ PICODBG_INFO_CTX();
+ PICODBG_INFO_MSG(("result of all transductions: "));
+ PICOTRNS_PRINTSYMSEQ(this->voice->kbArray[PICOKNOW_KBID_DBG], spho->phonBufOut, spho->phonWritePos);
+ PICODBG_INFO_MSG(("\n"));
+ }
+#endif
+
+ /* reset for next transduction */
+ spho->curFst = 0;
+ /* prepare BOUNDS */
+ spho->outReadPos = 0;
+ spho->phonReadPos = 0;
+
+ spho->procState = SPHO_STEPSTATE_PROCESS_BOUNDS;
+ break;
+ }
+
+ /* transduce from phonBufIn to PhonBufOut */
+ {
+
+ picoos_uint32 nrSteps;
+#if defined(PICO_DEBUG)
+ {
+ PICODBG_INFO_CTX();
+ PICODBG_INFO_MSG(("spho trying to transduce: "));
+ PICOTRNS_PRINTSYMSEQ(this->voice->kbArray[PICOKNOW_KBID_DBG], spho->phonBuf, spho->phonWritePos);
+ PICODBG_INFO_MSG(("\n"));
+ }
+#endif
+ rv = picotrns_transduce(spho->fst[spho->curFst], FALSE,
+ picotrns_printSolution, spho->phonBuf, spho->phonWritePos, spho->phonBufOut,
+ &spho->phonWritePos,
+ 4*PICOTRNS_MAX_NUM_POSSYM, spho->altDescBuf,
+ spho->maxAltDescLen, &nrSteps);
+ if (PICO_OK == rv) {
+#if defined(PICO_DEBUG)
+ {
+ PICODBG_INFO_CTX();
+ PICODBG_INFO_MSG(("result of transduction: (output symbols: %i)", spho->phonWritePos));
+ PICOTRNS_PRINTSYMSEQ(this->voice->kbArray[PICOKNOW_KBID_DBG], spho->phonBufOut, spho->phonWritePos);
+ PICODBG_INFO_MSG(("\n"));
+ }
+#endif
+ PICODBG_TRACE(("number of steps done in tranduction: %i", nrSteps));
+ } else {
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_FALLBACK,NULL,(picoos_char *)"phon buffer full");
+ }
+ }
+ /* eliminate deep epsilons */
+ picotrns_eliminate_epsilons(spho->phonBufOut, spho->phonWritePos, spho->phonBuf,
+ &spho->phonWritePos,4*PICOTRNS_MAX_NUM_POSSYM);
+
+ spho->curFst++;
+
+ /* return PICODATA_PU_ATOMIC */
+ break;
+
+
+ case SPHO_STEPSTATE_PROCESS_BOUNDS:
+ /* ************************************************************************/
+ /* BOUNDS: combine input item with pos/phon pairs to insert/modify bounds */
+ /* ************************************************************************/
+
+ PICODBG_INFO(("BOUNDS"));
+
+ /* get the suppressRecombWordBound in the left context */
+ spho->suppressRecombWordBound = FALSE;
+ while (spho->outReadPos < spho->activeStartPos) {
+ /* look at the current item */
+ ihead = spho->headx[spho->outReadPos].head;
+ /* icontent = spho->cbuf + spho->headx[spho->outReadPos].cind; */
+ PICODBG_INFO(("in position %i, looking at item %s",spho->outReadPos,picodata_head_to_string(&ihead,msgstr,SPHO_MSGSTR_SIZE)));
+ if (PICODATA_ITEM_BOUND == ihead.type) {
+ spho->suppressRecombWordBound = TRUE;
+ } else if (PICODATA_ITEM_WORDPHON == ihead.type) {
+ spho->suppressRecombWordBound = FALSE;
+ }
+ spho->outReadPos++;
+ }
+ /* spho->outReadPos point now to the active region */
+
+ /* advance the phone reading pos to the active range */
+ spho->phonReadPos = 0;
+ while (SPHO_POSSYM_OK == (rv = getNextPosSym(spho, &pos, &sym,
+ spho->activeStartPos))) {
+ /* ignore */
+ }
+ PICODBG_INFO(("skipping left context phones results in %s", (SPHO_POSSYM_OUT_OF_RANGE==rv) ? "OUT_OF_RANGE" : (SPHO_POSSYM_END ==rv) ? "END" : "OTHER"));
+
+ /*
+ * Align input items with transduced phones and note bound stregth changes and break commands
+ */
+
+ while (spho->outReadPos < spho->activeEndPos) {
+
+ /* look at the current item */
+ ihead = spho->headx[spho->outReadPos].head;
+ icontent = spho->cbuf + spho->headx[spho->outReadPos].cind;
+ nextInPos = spho->outReadPos + 1;
+ /* */
+ PICODBG_INFO(("in position %i, looking at item %s",spho->outReadPos,picodata_head_to_string(&ihead,msgstr,SPHO_MSGSTR_SIZE)));
+
+ if ((PICODATA_ITEM_BOUND == ihead.type)
+ || ((PICODATA_ITEM_WORDPHON == ihead.type)
+ && (!spho->suppressRecombWordBound))) {
+ /* there was a boundary originally */
+ picoos_uint8 orig_strength, orig_type;
+ if (PICODATA_ITEM_BOUND == ihead.type) {
+ orig_strength = ihead.info1;
+ orig_type = ihead.info2;
+ spho->suppressRecombWordBound = TRUE;
+ } else {
+ orig_strength = PICODATA_ITEMINFO1_BOUND_PHR0;
+ orig_type = PICODATA_ITEMINFO2_NA;
+ }
+ /* i expect a boundary phone here */
+ /* consume FST bound phones, consider pending break and set the side-bound */
+ PICODBG_INFO(("got BOUND or WORDPHON item and expects corresponding phone"));
+ rv = getNextPosSym(spho, &pos, &sym, nextInPos);
+ if (SPHO_POSSYM_OK != rv) {
+ PICODBG_ERROR(("unexpected symbol or unexpected end of phoneme list (%s)", (SPHO_POSSYM_OUT_OF_RANGE==rv) ? "OUT_OF_RANGE" : (SPHO_POSSYM_END ==rv) ? "END" :"OTHER"));
+ return (picodata_step_result_t)picoos_emRaiseException(this->common->em,
+ PICO_ERR_OTHER, NULL, NULL);
+ }
+ sym = picotrns_unplane(sym, &plane);
+ /* */
+ PICODBG_ASSERT((PICOKFST_PLANE_PB_STRENGTHS == plane));
+
+ /* insert modified bound according to transduction and possibly pending break */
+ setSideBound(spho, orig_strength, orig_type,
+ (picoos_uint8) sym);
+ } else if ((PICODATA_ITEM_CMD == ihead.type)
+ && (PICODATA_ITEMINFO1_CMD_SIL == ihead.info1)) {
+ /* it's a SIL (break) command */
+ picoos_uint16 time;
+ picoos_uint32 pos = 0;
+ picoos_read_mem_pi_uint16(icontent, &pos, &time);
+ if (spho->breakPending) {
+ spho->breakTime += time;
+ } else {
+ spho->breakTime = time;
+ spho->breakPending = TRUE;
+ }
+ } else if ((PICODATA_ITEM_CMD == ihead.type) && (PICODATA_ITEMINFO1_CMD_PLAY == ihead.info1)) {
+ /* insert break of at least one ms */
+ if (!spho->breakPending || (spho->breakTime <= 0)) {
+ spho->breakTime = SPHO_SMALLEST_SIL_DUR;
+ spho->breakPending = TRUE;
+ }
+ setSideBound(spho, PICODATA_ITEMINFO1_NA,
+ PICODATA_ITEMINFO2_NA, PICODATA_ITEMINFO1_NA);
+ /* force following break to be at least one ms */
+ spho->breakTime = SPHO_SMALLEST_SIL_DUR;
+ spho->breakPending = TRUE;
+ } else if (breakStateInterrupting(&ihead, &breakBefore, &breakAfter)) {
+
+ if (breakBefore &&(!spho->breakPending || (spho->breakTime <= 0))) {
+ spho->breakTime = SPHO_SMALLEST_SIL_DUR;
+ spho->breakPending = TRUE;
+ }
+ setSideBound(spho, PICODATA_ITEMINFO1_NA,
+ PICODATA_ITEMINFO2_NA, PICODATA_ITEMINFO1_NA);
+
+ if (breakAfter) {
+ spho->breakTime = SPHO_SMALLEST_SIL_DUR;
+ spho->breakPending = TRUE;
+ }
+ if (PICODATA_ITEM_WORDPHON == ihead.type) {
+ spho->suppressRecombWordBound = FALSE;
+ }
+ }
+
+ /* skip phones of that item */
+ while (SPHO_POSSYM_OK == (rv = getNextPosSym(spho, &pos,
+ &sym, nextInPos))) {
+ /* ignore */
+ }
+ spho->outReadPos++;
+ }
+
+ /* reset for RECOMB */
+ spho->outReadPos = 0;
+ spho->phonReadPos = 0;
+ spho->suppressRecombWordBound = FALSE;
+
+ spho->procState = SPHO_STEPSTATE_PROCESS_RECOMB;
+ return PICODATA_PU_ATOMIC;
+
+ break;
+
+ case SPHO_STEPSTATE_PROCESS_RECOMB:
+ /* **********************************************************************/
+ /* RECOMB: combine input item with pos/phon pairs to output item */
+ /* **********************************************************************/
+
+ PICODBG_TRACE(("RECOMB"));
+
+ /* default place to come after feed: here */
+ spho->feedFollowState = SPHO_STEPSTATE_PROCESS_RECOMB;
+
+ /* check termination condition first */
+ if (spho->outReadPos >= spho->activeEndPos) {
+ PICODBG_DEBUG(("RECOMB reached active region's end at %i",spho->outReadPos));
+ spho->procState = SPHO_STEPSTATE_SHIFT;
+ break;
+ }
+
+ /* look at the current item */
+ ihead = spho->headx[spho->outReadPos].head;
+ icontent = spho->cbuf + spho->headx[spho->outReadPos].cind;
+
+ PICODBG_DEBUG(("RECOMB looking at item %s",picodata_head_to_string(&ihead,msgstr,SPHO_MSGSTR_SIZE)));
+
+ nextInPos = spho->outReadPos + 1;
+
+ PICODBG_DEBUG(("RECOMB treating item in headx at pos %i",spho->outReadPos));
+ if (nextInPos <= spho->activeStartPos) { /* we're in the (passive) left context. Just skip it */
+ PICODBG_DEBUG(("RECOMB skipping item in the left context (%i <= %i)",nextInPos, spho->activeStartPos));
+ if (PICODATA_ITEM_BOUND == ihead.type) {
+ spho->suppressRecombWordBound = 1;
+ } else if (PICODATA_ITEM_WORDPHON == ihead.type) {
+ spho->suppressRecombWordBound = 0;
+ }
+
+ /* consume possyms */
+ while (SPHO_POSSYM_OK == (rv = getNextPosSym(spho,&pos,&sym,nextInPos))) {
+ /* ignore */
+ }
+ if (rv == SPHO_POSSYM_INVALID) {
+ return (picodata_step_result_t)picoos_emRaiseException(this->common->em,
+ PICO_ERR_OTHER, NULL, NULL);
+ }
+ spho->outReadPos = nextInPos;
+ } else { /* active region */
+ if (spho->headx[spho->outReadPos].boundstrength) {
+/* ***************** "side-bound" *********************/
+ /* copy to outbuf */
+ putSideBoundToOutput(spho);
+ /* mark as processed */
+ spho->headx[spho->outReadPos].boundstrength = 0;
+ /* output it */
+ spho->procState = SPHO_STEPSTATE_FEED;
+ } else if (PICODATA_ITEM_BOUND == ihead.type) {
+/* ***************** BOUND *********************/
+ /* expect a boundary phone here */
+ PICODBG_DEBUG(("RECOMB got BOUND item and expects corresponding phone"));
+ rv = getNextPosSym(spho, &pos, &sym, nextInPos);
+ if (SPHO_POSSYM_OK != rv) {
+ PICODBG_ERROR(("unexpected symbol or unexpected end of phoneme list"));
+ return (picodata_step_result_t)picoos_emRaiseException(
+ this->common->em, PICO_ERR_OTHER, NULL,
+ NULL);
+ }
+ sym = picotrns_unplane(sym, &plane);
+ /* */
+ PICODBG_ASSERT((PICOKFST_PLANE_PB_STRENGTHS == plane));
+
+ spho->suppressRecombWordBound = TRUE; /* if word following, don't need word boundary */
+ /* just consume item and come back here*/
+ spho->outReadPos = nextInPos;
+
+ } else if (PICODATA_ITEM_WORDPHON == ihead.type) {
+/* ***************** WORDPHON *********************/
+ spho->wordStarted = TRUE;
+ /* i expect a word boundary symbol in this range unless a phrase boundary was encountered before */
+ if (spho->suppressRecombWordBound) {
+ PICODBG_DEBUG(("RECOMB got WORDPHON item but skips expecting BOUND"));
+ spho->suppressRecombWordBound = FALSE;
+ } else {
+ PICODBG_DEBUG(("RECOMB got WORDPHON item and expects corresponding bound phone"));
+ rv = getNextPosSym(spho, &pos, &sym, nextInPos);
+ if (SPHO_POSSYM_OK != rv) {
+ PICODBG_ERROR(("unexpected symbol or unexpected end of phoneme list"));
+ return (picodata_step_result_t)picoos_emRaiseException(this->common->em,
+ PICO_ERR_OTHER, NULL, NULL);
+ }
+ }
+ spho->procState = SPHO_STEPSTATE_PROCESS_SYL;
+ } else if ((PICODATA_ITEM_CMD == ihead.type) && (PICODATA_ITEMINFO1_CMD_SIL == ihead.info1)) {
+/* ***************** BREAK COMMAND *********************/
+ /* just consume and come back here */
+ PICODBG_DEBUG(("RECOMB consuming item from inBuf %i -> %i",spho->outReadPos, nextInPos));
+ spho->outReadPos = nextInPos;
+ } else {
+/* ***************** OTHER *********************/
+ /* just copy item */
+ PICODBG_DEBUG(("RECOMB found other item, just copying"));
+ picodata_put_itemparts(&ihead, icontent, ihead.len,
+ spho->outBuf, spho->outBufSize, &clen);
+ PICODBG_DEBUG(("RECOMB consuming item from inBuf %i -> %i",spho->outReadPos, nextInPos));
+ spho->outReadPos = nextInPos;
+ /* and output it */
+ spho->procState = SPHO_STEPSTATE_FEED;
+ } /* if (ihead.type) */
+
+ }
+
+ /* return PICODATA_PU_BUSY; */
+ break;
+
+ case SPHO_STEPSTATE_PROCESS_SYL:
+ /* **********************************************************************/
+ /* SYL: combine input word item with pos/phon pairs to syl output item */
+ /* **********************************************************************/
+
+ /* consume all transduced phonemes with pos in in the range [spho->outReadPos,nextInPos[ */
+ PICODBG_DEBUG(("SYL"));
+
+ spho->feedFollowState = SPHO_STEPSTATE_PROCESS_SYL;
+
+ /* look at the current item */
+ ihead = spho->headx[spho->outReadPos].head;
+ icontent = spho->cbuf + spho->headx[spho->outReadPos].cind;
+ nextInPos = spho->outReadPos + 1;
+ PICODBG_DEBUG(("SYL (1) treating item in headx at pos %i",spho->outReadPos));
+ /* create syllable item in ohead (head) and sylBuf (contents) */
+ ohead.type = PICODATA_ITEM_SYLLPHON;
+
+ PICODBG_TRACE(("SYL expects accent at phonBuf[%i] = (%i,%i) (outReadPos=%i)", spho->phonReadPos, spho->phonBuf[spho->phonReadPos].pos, spho->phonBuf[spho->phonReadPos].sym,spho->outReadPos));
+ rv = getNextPosSym(spho,&pos,&sym,nextInPos);
+ if (SPHO_POSSYM_OK != rv) {
+ PICODBG_ERROR(("unexpected symbol or unexpected end of phoneme list (%i)",rv));
+ return (picodata_step_result_t)picoos_emRaiseException(this->common->em, PICO_ERR_OTHER, NULL, NULL);
+ }
+ ohead.info2 = picotrns_unplane(sym, &plane);
+ PICODBG_ASSERT((PICOKFST_PLANE_ACCENTS == plane));
+ PICODBG_DEBUG(("SYL sets accent to %c", sym));
+
+ /* for the time being, we force to use POS so we can transduce all fsts in a row without reconsulting the items */
+ PICODBG_TRACE(("SYL expects POS"));
+ PICODBG_DEBUG(("SYL (2) treating item in inBuf range [%i,%i[",spho->outReadPos,nextInPos));
+ rv = getNextPosSym(spho,&pos,&sym,nextInPos);
+ if (SPHO_POSSYM_OK != rv) {
+ PICODBG_ERROR(("unexpected symbol or unexpected end of phoneme list"));
+ return (picodata_step_result_t)picoos_emRaiseException(this->common->em, PICO_ERR_OTHER, NULL, NULL);
+ }
+ if (spho->wordStarted) {
+ spho->wordStarted = FALSE;
+ ohead.info1 = picotrns_unplane(sym, &plane);
+ /* */
+ PICODBG_ASSERT(PICOKFST_PLANE_POS == plane);
+ /* */
+ PICODBG_DEBUG(("SYL setting POS to %c", ohead.info1));
+ } else {
+ ohead.info1 = PICODATA_ITEMINFO1_NA;
+ }
+
+ PICODBG_DEBUG(("SYL (3) treating item in inBuf range [%i,%i[",spho->outReadPos,nextInPos));
+ /* get phonemes of that syllable; stop if syllable boundary or outside word */
+ sylsym = (PICOKFST_PLANE_PHONEMES << 8)
+ + spho->syllSepId;
+ PICODBG_DEBUG(("collecting syllable phonemes before headx position %i",nextInPos));
+ spho->sylWritePos = 0;
+ while (SPHO_POSSYM_OK == (rv = getNextPosSym(spho,&pos,&sym,nextInPos)) && (sym != sylsym)) {
+ spho->sylBuf[spho->sylWritePos++] = picotrns_unplane(sym, &plane);
+ /* */
+ PICODBG_TRACE(("SYL adding phoneme to syllable: (pos %i,sym %i)[plane %i,sym %c]",pos,sym,plane,sym & 0xFF));
+ PICODBG_ASSERT((PICOKFST_PLANE_PHONEMES == plane));
+ }
+ PICODBG_DEBUG(("SYL (4) treating item in inBuf range [%i,%i[",spho->outReadPos,nextInPos));
+ ohead.len = spho->sylWritePos;
+ if (SPHO_POS_INVALID == rv) {
+ PICODBG_ERROR(("unexpected symbol or unexpected end of phoneme list"));
+ return (picodata_step_result_t)picoos_emRaiseException(this->common->em, PICO_WARN_INCOMPLETE, NULL, NULL);
+ } else if ((SPHO_POSSYM_OUT_OF_RANGE == rv) || (SPHO_POSSYM_END == rv)) {
+ PICODBG_DEBUG(("SYL arrived at end of word and/or end of phon buffer, go to next word"));
+ spho->outReadPos = nextInPos; /* advance to next item */
+ spho->feedFollowState = SPHO_STEPSTATE_PROCESS_RECOMB; /* go to RECOMB after feed */
+ } else {
+ PICODBG_ASSERT((sym == sylsym));
+ }
+ PICODBG_DEBUG(("SYL (5) treating item in inBuf range [%i,%i[",spho->outReadPos,nextInPos));
+
+ if (ohead.len > 0) {
+ /* prepare syllable output */
+ picodata_put_itemparts(&ohead, spho->sylBuf,
+ PICODATA_BUFSIZE_DEFAULT, spho->outBuf,
+ spho->outBufSize, &clen);
+
+ spho->procState = SPHO_STEPSTATE_FEED;
+ } else { /* skip feeding output of empty syllable */
+ spho->procState = spho->feedFollowState;
+ }
+ break;
+
+ case SPHO_STEPSTATE_FEED:
+ /* **********************************************************************/
+ /* FEED: output output item and proceed to feedFollowState */
+ /* **********************************************************************/
+
+ PICODBG_DEBUG(("FEED"));
+
+ PICODBG_DEBUG(("FEED putting outBuf item into cb"));
+
+ /*feeding items to PU output buffer*/
+ rv = picodata_cbPutItem(this->cbOut, spho->outBuf,
+ spho->outBufSize, &clen);
+
+ PICODATA_INFO_ITEM(this->voice->kbArray[PICOKNOW_KBID_DBG],
+ (picoos_uint8 *)"spho: ",
+ spho->outBuf, spho->outBufSize);
+
+ if (PICO_EXC_BUF_OVERFLOW == rv) {
+ /* we have to redo this item */
+ PICODBG_DEBUG(("FEED got overflow, returning ICODATA_PU_OUT_FULL"));
+ return PICODATA_PU_OUT_FULL;
+ } else if (PICO_OK == rv) {
+ *numBytesOutput += clen;
+ spho->procState = spho->feedFollowState;
+ PICODBG_DEBUG(("FEED ok, going back to procState %i", spho->procState));
+ return PICODATA_PU_BUSY;
+ } else {
+ PICODBG_DEBUG(("FEED got exception %i when trying to output item",rv));
+ spho->procState = spho->feedFollowState;
+ return (picodata_step_result_t)rv;
+ }
+ break;
+
+ case SPHO_STEPSTATE_SHIFT:
+ /* **********************************************************************/
+ /* SHIFT */
+ /* **********************************************************************/
+ /* If there exists a valid penultima, it should replace any left context (from 0 to activeStartPos)
+ * else discard the current active range (from activeStartPos to activeEndPos), leaving the current
+ * left context intact. Often, PARSE would move activeStartPos to 0, so that there is no left context
+ * after the shift.
+ */
+
+ PICODBG_DEBUG(("SHIFT"));
+
+ if (spho->penultima != SPHO_POS_INVALID) {
+ picoos_int16 shift;
+ /* set penultima as new left context and set activeStartPos to the shifted activeEndPos */
+ PICODBG_DEBUG((
+ "SHIFT shifting penultima from %i to 0",
+ spho->penultima));
+ shift = shift_range_left_1(spho, &spho->penultima, 0);
+ if (shift < 0) {
+ picoos_emRaiseException(this->common->em,PICO_ERR_OTHER,NULL,NULL);
+ return PICODATA_PU_ERROR;
+ }
+ spho->activeStartPos = spho->activeEndPos
+ - shift;
+ spho->lastPhraseBoundPos -= shift;
+ spho->suppressParseWordBound = FALSE;
+ spho->suppressRecombWordBound = FALSE;
+
+ } else {
+ picoos_int16 shift;
+ picoos_bool lastPhraseBoundActive;
+ if (spho->activeStartPos == spho->activeEndPos) {
+ /* no items consumed; we have to abandon left context */
+ spho->activeStartPos = 0;
+ }
+ lastPhraseBoundActive = (spho->lastPhraseBoundPos >= spho->activeStartPos);
+ /* dummy comment */
+ PICODBG_DEBUG(("SHIFT shift active end from %i to %i",
+ spho->activeEndPos, spho->activeStartPos));
+ shift = shift_range_left_1(spho, &spho->activeEndPos, spho->activeStartPos);
+ if (shift < 0) {
+ picoos_emRaiseException(this->common->em,PICO_ERR_OTHER,NULL,NULL);
+ return PICODATA_PU_ERROR;
+ }
+ if (lastPhraseBoundActive) {
+ spho->lastPhraseBoundPos -= shift;
+ }
+ }
+
+ spho->procState = SPHO_STEPSTATE_INIT;
+ break;
+
+ default:
+ picoos_emRaiseException(this->common->em, PICO_ERR_OTHER, NULL, NULL);
+ return PICODATA_PU_ERROR;
+ break;
+
+ } /* switch (spho->procState) */
+
+ } /* while (1) */
+
+ /* should be never reached */
+ picoos_emRaiseException(this->common->em, PICO_ERR_OTHER, NULL, NULL);
+ return PICODATA_PU_ERROR;
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+/* end picospho.c */
diff --git a/lib/picospho.h b/lib/picospho.h
new file mode 100644
index 0000000..0c4a0d8
--- /dev/null
+++ b/lib/picospho.h
@@ -0,0 +1,175 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picospho.h
+ *
+ * sentence phonemic/phonetic FSTs PU
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+/** @addtogroup picospho
+itemtype, iteminfo1, iteminfo2, content -> TYPE(INFO1,INFO2)content
+in the following
+
+items input
+===========
+
+processed:
+
+- WORDPHON(POS,WACC)phon
+
+- BOUND(BOUNDstrength,BOUNDtype)
+
+
+unprocessed:
+- all other item types are forwared through the PU without modification
+
+
+
+- POS
+ a the single, unambiguous POS
+
+cf. picodata.h for
+- WACC (sentence-level accent (aka prominence))
+ PICODATA_ACC0
+ PICODATA_ACC1
+ PICODATA_ACC2 (<- maybe mapped to ACC1, ie. no ACC2 in output)
+ PICODATA_ACC3
+
+
+- BOUNDstrength
+ PICODATA_ITEMINFO1_BOUND_SBEG (sentence start)
+ PICODATA_ITEMINFO1_BOUND_SEND (at sentence end)
+ PICODATA_ITEMINFO1_BOUND_TERM (replaces a flush)
+ PICODATA_ITEMINFO1_BOUND_PHR0 (no break)
+ PICODATA_ITEMINFO1_BOUND_PHR1 (primary boundary)
+ PICODATA_ITEMINFO1_BOUND_PHR2 (short break)
+ PICODATA_ITEMINFO1_BOUND_PHR3 (secondary phrase boundary, no break)
+
+- BOUNDtype (actually phrase type of the following phrase)
+ PICODATA_ITEMINFO2_BOUNDTYPE_P (non-terminal phrase)
+ PICODATA_ITEMINFO2_BOUNDTYPE_T (terminal phrase)
+ PICODATA_ITEMINFO2_BOUNDTYPE_Q (question terminal phrase)
+ PICODATA_ITEMINFO2_BOUNDTYPE_E (exclamation terminal phrase)
+
+
+output sequence (without CMDs):
+
+<output> = { BOUND(BOUND_SBEG,PHRASEtype) <sentence> BOUND(BOUND_SEND,..)} BOUND(BOUND_TERM,..)
+
+<sentence> = <phrase> { BOUND(BOUND_PHR1|2|3,PHRASEtype) <phrase> }
+
+<phrase> = WORDPHON(POS,ACC)phon { WORDPHON(POS,ACC)phon }
+
+
+
+mapping ACC & word-level stress to syllable accent value
+
+ ACC0 prim -> 0
+ ACC1 prim -> 1
+ ACC2 prim -> 2
+ ACC3 prim -> 3
+
+ ACC0 sec -> 0
+ ACC1 sec -> 4
+ ACC2 sec -> 4
+ ACC3 sec -> 4
+
+Mapping of values to FST symbol id (has to identical to the symbol table used when compiling the FST)
+
+Value FST symbol id
+phoneme_id -> phoneme_id + 256 * PICOKFST_PLANE_PHONEMES
+POS_id -> POS_id + 256 * PICOKFST_PLANE_POS
+phrasetype_id -> phrasetype_id + 256 * PICOKFST_PLANE_PHRASETYPES
+accentlevel_id -> accentlevel_id + 256 * PICOKFST_PLANE_ACCENTS
+
+
+
+
+
+
+
+minimal input size (before processing starts)
+==================
+
+processing (ie. sequencially applying spho transducers to phoneme sequence composed of
+ - phonemes inside WORDPHON items and
+ - pseudo-phonemes derived from boundaries and POS) is possible with
+
+- one phrase, consisting of a sequence of maximal 30 non-PUNC items
+ terminated by a PUNC item. A PUNC is artificially enforced if
+ needed to start processing.
+
+- as long as the internal buffer is empty, non-processed item types
+ can be processed immediately
+
+
+
+items output
+============
+- BOUND(BOUNDstrength,BOUNDtype)
+
+ bound strength may be changed by the fsts
+
+ in addition, BOUNDs of BOUNDstrength = PHR0 are inserted to mark word boundaries
+
+- SYLLPHON(POS,ACC)phon
+ where POS is only set for the first syllable of a word, otherwise NA
+
+
+
+
+
+
+other limitations
+=================
+
+
+ */
+#ifndef PICOSPHO_H_
+#define PICOSPHO_H_
+
+#include "picoos.h"
+#include "picodata.h"
+#include "picorsrc.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+picodata_ProcessingUnit picospho_newSentPhoUnit(
+ picoos_MemoryManager mm,
+ picoos_Common common,
+ picodata_CharBuffer cbIn,
+ picodata_CharBuffer cbOut,
+ picorsrc_Voice voice);
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif /*PICOSPHO_H_*/
diff --git a/lib/picotok.c b/lib/picotok.c
new file mode 100644
index 0000000..4488609
--- /dev/null
+++ b/lib/picotok.c
@@ -0,0 +1,1563 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picotok.c
+ *
+ * tokenizer
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+
+/* ************************************************************/
+/* tokenisation and markup handling */
+/* ************************************************************/
+
+/** @addtogroup picotok
+ @b tokenisation_overview
+
+ markup handling overview:
+
+ The following markups are recognized
+ - ignore
+ - speed
+ - pitch
+ - volume
+ - voice
+ - preproccontext
+ - mark
+ - play
+ - usesig
+ - genfile
+ - sentence
+ - s
+ - paragraph
+ - p
+ - break
+ - spell (pauses between letter)
+ - phoneme
+
+ All markups which are recognized but are not yet implemented in pico
+ system have the mark.
+*/
+
+
+#include "picodefs.h"
+#include "picoos.h"
+#include "picobase.h"
+#include "picodbg.h"
+#include "picodata.h"
+#include "picotok.h"
+#include "picoktab.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+/* *****************************************************************************/
+
+#define IN_BUF_SIZE 255
+#define OUT_BUF_SIZE IN_BUF_SIZE + 3 * PICODATA_ITEM_HEADSIZE + 3
+
+#define MARKUP_STRING_BUF_SIZE (IN_BUF_SIZE*5)
+#define MAX_NR_MARKUP_PARAMS 6
+#define MARKUP_HANDLING_DISABLED 0
+#define MARKUP_HANDLING_ENABLED 1
+#define EOL '\n'
+
+
+typedef picoos_int8 pico_tokenSubType;
+typedef picoos_uint8 pico_tokenType;
+
+/** @todo : consider adding these specialized exception codes: */
+
+#define PICO_ERR_MARKUP_VALUE_OUT_OF_RANGE PICO_ERR_OTHER
+#define PICO_ERR_INVALID_MARKUP_TAG PICO_ERR_OTHER
+#define PICO_ERR_INTERNAL_LIMIT PICO_ERR_OTHER
+
+typedef enum {MIDummyStart, MIIgnore,
+ MIPitch, MISpeed, MIVolume,
+ MIVoice, MIPreprocContext, MIMarker,
+ MIPlay, MIUseSig, MIGenFile, MIParagraph,
+ MISentence, MIBreak, MISpell, MIPhoneme, MIItem, MISpeaker, MIDummyEnd
+ } MarkupId;
+typedef enum {MSNotInMarkup, MSGotStart, MSExpectingmarkupTagName, MSInmarkupTagName,
+ MSGotmarkupTagName, MSInAttrName, MSGotAttrName, MSGotEqual, MSInAttrValue,
+ MSInAttrValueEscaped, MSGotAttrValue, MSGotEndSlash, MSGotEnd,
+ MSError, MSErrorTooLong, MSErrorSyntax
+ } MarkupState;
+typedef enum {MENone, MEMissingStart, MEUnknownTag, MEIdent, MEMissingEqual,
+ MEMissingQuote, MEMissingEnd, MEUnexpectedChar, MEInterprete
+ } MarkupParseError;
+
+typedef enum {MTNone, MTStart, MTEnd, MTEmpty} MarkupTagType;
+
+#define UTF_CHAR_COMPLETE 2
+#define UTF_CHAR_INCOMPLETE 1
+#define UTF_CHAR_MALFORMED 0
+
+#define TOK_MARKUP_KW_IGNORE (picoos_uchar*)"ignore"
+#define TOK_MARKUP_KW_SPEED (picoos_uchar*)"speed"
+#define TOK_MARKUP_KW_PITCH (picoos_uchar*)"pitch"
+#define TOK_MARKUP_KW_VOLUME (picoos_uchar*)"volume"
+#define TOK_MARKUP_KW_VOICE (picoos_uchar*)"voice"
+#define TOK_MARKUP_KW_CONTEXT (picoos_uchar*)"preproccontext"
+#define TOK_MARKUP_KW_MARK (picoos_uchar*)"mark"
+#define TOK_MARKUP_KW_PLAY (picoos_uchar*)"play"
+#define TOK_MARKUP_KW_USESIG (picoos_uchar*)"usesig"
+#define TOK_MARKUP_KW_GENFILE (picoos_uchar*)"genfile"
+#define TOK_MARKUP_KW_SENTENCE (picoos_uchar*)"sentence"
+#define TOK_MARKUP_KW_S (picoos_uchar*)"s"
+#define TOK_MARKUP_KW_PARAGRAPH (picoos_uchar*)"paragraph"
+#define TOK_MARKUP_KW_P (picoos_uchar*)"p"
+#define TOK_MARKUP_KW_BREAK (picoos_uchar*)"break"
+#define TOK_MARKUP_KW_SPELL (picoos_uchar*)"spell"
+#define TOK_MARKUP_KW_PHONEME (picoos_uchar*)"phoneme"
+#define TOK_MARKUP_KW_ITEM (picoos_uchar*)"item"
+#define TOK_MARKUP_KW_SPEAKER (picoos_uchar*)"speaker"
+
+#define KWLevel (picoos_uchar *)"level"
+#define KWName (picoos_uchar *)"name"
+#define KWProsDomain (picoos_uchar *)"prosodydomain"
+#define KWTime (picoos_uchar *)"time"
+#define KWMode (picoos_uchar *)"mode"
+#define KWSB (picoos_uchar *)"sb"
+#define KWPB (picoos_uchar *)"pb"
+#define KWFile (picoos_uchar *)"file"
+#define KWType (picoos_uchar *)"type"
+#define KWF0Beg (picoos_uchar *)"f0beg"
+#define KWF0End (picoos_uchar *)"f0end"
+#define KWXFadeBeg (picoos_uchar *)"xfadebeg"
+#define KWXFadeEnd (picoos_uchar *)"xfadeend"
+#define KWAlphabet (picoos_uchar *)"alphabet"
+#define KWPH (picoos_uchar *)"ph"
+#define KWOrthMode (picoos_uchar *)"orthmode"
+#define KWIgnorePunct (picoos_uchar *)"ignorepunct"
+#define KWInfo1 (picoos_uchar *)"info1"
+#define KWInfo2 (picoos_uchar *)"info2"
+#define KWDATA (picoos_uchar *)"data"
+
+#define PICO_SPEED_MIN 20
+#define PICO_SPEED_MAX 500
+#define PICO_SPEED_DEFAULT 100
+#define PICO_SPEED_FACTOR_MIN 500
+#define PICO_SPEED_FACTOR_MAX 2000
+
+#define PICO_PITCH_MIN 50
+#define PICO_PITCH_MAX 200
+#define PICO_PITCH_DEFAULT 100
+#define PICO_PITCH_FACTOR_MIN 500
+#define PICO_PITCH_FACTOR_MAX 2000
+#define PICO_PITCH_ADD_MIN -100
+#define PICO_PITCH_ADD_MAX 100
+#define PICO_PITCH_ADD_DEFAULT 0
+
+#define PICO_VOLUME_MIN 0
+#define PICO_VOLUME_MAX 500
+#define PICO_VOLUME_DEFAULT 100
+#define PICO_VOLUME_FACTOR_MIN 500
+#define PICO_VOLUME_FACTOR_MAX 2000
+
+#define PICO_SPEAKER_MIN 20
+#define PICO_SPEAKER_MAX 180
+#define PICO_SPEAKER_DEFAULT 100
+#define PICO_SPEAKER_FACTOR_MIN 500
+#define PICO_SPEAKER_FACTOR_MAX 2000
+
+#define PICO_CONTEXT_DEFAULT (picoos_uchar*)"DEFAULT"
+
+#define PARAGRAPH_PAUSE_DUR 500
+#define SPELL_WITH_PHRASE_BREAK 1
+#define SPELL_WITH_SENTENCE_BREAK 2
+
+/* *****************************************************************************/
+
+#define TOK_PUNC_FLUSH (picoos_char) '\0'
+
+typedef picoos_uchar Word[MARKUP_STRING_BUF_SIZE];
+
+
+struct MarkupParam {
+ Word paramId;
+ Word paramVal;
+};
+
+typedef struct MarkupParam MarkupParams[MAX_NR_MARKUP_PARAMS];
+
+
+/** subobject : TokenizeUnit
+ * shortcut : tok
+ */
+typedef struct tok_subobj
+{
+ picoos_int32 ignLevel;
+
+ picoos_uchar utf[5];
+ picoos_int32 utfpos;
+ picoos_int32 utflen;
+
+ MarkupParams markupParams;
+ picoos_int32 nrMarkupParams;
+ MarkupState markupState;
+ picoos_uchar markupStr[MARKUP_STRING_BUF_SIZE];
+ picoos_int32 markupPos;
+ picoos_int32 markupLevel[MIDummyEnd+1];
+ picoos_uchar markupTagName[IN_BUF_SIZE];
+ MarkupTagType markupTagType;
+ MarkupParseError markupTagErr;
+
+ picoos_int32 strPos;
+ picoos_uchar strDelim;
+ picoos_bool isFileAttr;
+
+ pico_tokenType tokenType;
+ pico_tokenSubType tokenSubType;
+
+ picoos_int32 tokenPos;
+ picoos_uchar tokenStr[IN_BUF_SIZE];
+
+ picoos_int32 nrEOL;
+
+ picoos_bool markupHandlingMode; /* to be moved ??? */
+ picoos_bool aborted; /* to be moved ??? */
+
+ picoos_bool start;
+
+ picoos_uint8 outBuf[OUT_BUF_SIZE]; /* internal output buffer */
+ picoos_uint16 outReadPos; /* next pos to read from outBuf */
+ picoos_uint16 outWritePos; /* next pos to write to outBuf */
+
+ picoos_uchar saveFile[IN_BUF_SIZE];
+ Word phonemes;
+
+ picotrns_SimpleTransducer transducer;
+
+ /* kbs */
+
+ picoktab_Graphs graphTab;
+ picokfst_FST xsampa_parser;
+ picokfst_FST svoxpa_parser;
+ picokfst_FST xsampa2svoxpa_mapper;
+
+
+
+} tok_subobj_t;
+
+/* *****************************************************************************/
+
+static void tok_treatMarkupAsSimpleToken (picodata_ProcessingUnit this, tok_subobj_t * tok);
+static void tok_treatChar (picodata_ProcessingUnit this, tok_subobj_t * tok, picoos_uchar ch, picoos_bool markupHandling);
+static void tok_treatMarkup (picodata_ProcessingUnit this, tok_subobj_t * tok);
+static void tok_putToMarkup (picodata_ProcessingUnit this, tok_subobj_t * tok, picoos_uchar str[]);
+static void tok_treatSimpleToken (picodata_ProcessingUnit this, tok_subobj_t * tok);
+static MarkupId tok_markupTagId (picoos_uchar tagId[]);
+
+/* *****************************************************************************/
+
+static picoos_bool tok_strEqual(picoos_uchar * str1, picoos_uchar * str2)
+{
+ return (picoos_strcmp((picoos_char*)str1, (picoos_char*)str2) == 0);
+}
+
+static void tok_reduceBlanks(picoos_uchar * str)
+ /* Remove leading and trailing blanks of 'str' and reduce
+ groups of blanks within string to exactly one blank. */
+
+{
+ int i = 0;
+ int j = 0;
+
+ while (str[j] != 0) {
+ if (str[j] == (picoos_uchar)' ') {
+ /* note one blank except at the beginning of string */
+ if (i > 0) {
+ str[i] = (picoos_uchar)' ';
+ i++;
+ }
+ j++;
+ while (str[j] == (picoos_uchar)' ') {
+ j++;
+ }
+ } else {
+ str[i] = str[j];
+ j++;
+ i++;
+ }
+ }
+
+ /* remove blanks at end of string */
+ if ((i > 0) && (str[i - 1] == ' ')) {
+ i--;
+ }
+ str[i] = 0;
+}
+
+
+static void tok_startIgnore (tok_subobj_t * tok)
+{
+ tok->ignLevel++;
+}
+
+
+static void tok_endIgnore (tok_subobj_t * tok)
+{
+ if (tok->ignLevel > 0) {
+ tok->ignLevel--;
+ }
+}
+
+
+static void tok_getParamIntVal (MarkupParams params, picoos_uchar paramId[], picoos_int32 * paramVal, picoos_bool * paramFound)
+{
+ int i=0;
+
+ while ((i < MAX_NR_MARKUP_PARAMS) && !tok_strEqual(paramId,params[i].paramId)) {
+ i++;
+ }
+ if ((i < MAX_NR_MARKUP_PARAMS)) {
+ (*paramVal) = picoos_atoi((picoos_char*)params[i].paramVal);
+ (*paramFound) = TRUE;
+ } else {
+ (*paramVal) = -1;
+ (*paramFound) = FALSE;
+ }
+}
+
+
+
+static void tok_getParamStrVal (MarkupParams params, picoos_uchar paramId[], picoos_uchar paramStrVal[], picoos_bool * paramFound)
+{
+ int i=0;
+
+ while ((i < MAX_NR_MARKUP_PARAMS) && !tok_strEqual(paramId,params[i].paramId)) {
+ i++;
+ }
+ if (i < MAX_NR_MARKUP_PARAMS) {
+ picoos_strcpy((picoos_char*)paramStrVal, (picoos_char*)params[i].paramVal);
+ (*paramFound) = TRUE;
+ } else {
+ paramStrVal[0] = 0;
+ (*paramFound) = FALSE;
+ }
+}
+
+
+static void tok_getParamPhonesStr (MarkupParams params, picoos_uchar paramId[], picoos_uchar alphabet[], picoos_uchar phones[], picoos_int32 phoneslen, picoos_bool * paramFound)
+{
+
+ int i;
+ picoos_bool done;
+
+ i = 0;
+ while ((i < MAX_NR_MARKUP_PARAMS) && !tok_strEqual(paramId, params[i].paramId)) {
+ i++;
+ }
+ if (i < MAX_NR_MARKUP_PARAMS) {
+ if (tok_strEqual(alphabet, PICODATA_XSAMPA) || tok_strEqual(alphabet, (picoos_uchar*)"")) {
+ picoos_strlcpy((picoos_char*)phones, (picoos_char*)params[i].paramVal, phoneslen);
+ done = TRUE;
+ } else {
+ done = FALSE;
+ }
+ (*paramFound) = TRUE;
+ } else {
+ done = FALSE;
+ (*paramFound) = FALSE;
+ }
+ if (!done) {
+ phones[0] = 0;
+ }
+}
+
+
+static void tok_clearMarkupParams (MarkupParams params)
+{
+ int i;
+
+ for (i = 0; i<MAX_NR_MARKUP_PARAMS; i++) {
+ params[i].paramId[0] = 0;
+ params[i].paramVal[0] = 0;
+ }
+}
+
+
+static void tok_getDur (picoos_uchar durStr[], picoos_uint32 * dur, picoos_bool * done)
+{
+
+ int num=0;
+ int i=0;
+ picoos_uchar tmpWord[IN_BUF_SIZE];
+
+ picoos_strlcpy((picoos_char*)tmpWord, (picoos_char*)durStr, sizeof(tmpWord));
+ tok_reduceBlanks(tmpWord);
+ while ((durStr[i] >= '0') && (durStr[i] <= '9')) {
+ num = 10 * num + (int)durStr[i] - (int)'0';
+ tmpWord[i] = ' ';
+ i++;
+ }
+ tok_reduceBlanks(tmpWord);
+ if (tok_strEqual(tmpWord, (picoos_uchar*)"s")) {
+ (*dur) = (1000 * num);
+ (*done) = TRUE;
+ } else if (tok_strEqual(tmpWord,(picoos_uchar*)"ms")) {
+ (*dur) = num;
+ (*done) = TRUE;
+ } else {
+ (*dur) = 0;
+ (*done) = FALSE;
+ }
+}
+
+
+static picoos_int32 tok_putToUtf (tok_subobj_t * tok, picoos_uchar ch)
+{
+ if (tok->utfpos < PICOBASE_UTF8_MAXLEN) {
+ tok->utf[tok->utfpos] = ch;
+ if (tok->utfpos == 0) {
+ tok->utflen = picobase_det_utf8_length(ch);
+ } else if (((ch < (picoos_uchar)'\200') || (ch >= (picoos_uchar)'\300'))) {
+ tok->utflen = 0;
+ }
+ (tok->utfpos)++;
+ if ((tok->utfpos == tok->utflen)) {
+ if ((tok->utfpos < PICOBASE_UTF8_MAXLEN)) {
+ tok->utf[tok->utfpos] = 0;
+ }
+ return UTF_CHAR_COMPLETE;
+ } else if (tok->utfpos < tok->utflen) {
+ return UTF_CHAR_INCOMPLETE;
+ } else {
+ return UTF_CHAR_MALFORMED;
+ }
+ } else {
+ return UTF_CHAR_MALFORMED;
+ }
+}
+
+
+static picoos_bool tok_isRelative (picoos_uchar strval[], picoos_uint32 * val)
+{
+ picoos_int32 len;
+ picoos_bool rel;
+
+ rel = FALSE;
+ len = picoos_strlen((picoos_char*)strval);
+ if (len > 0) {
+ if (strval[len - 1] == '%') {
+ strval[len - 1] = 0;
+ if ((strval[0] == '+') || (strval[0] == '-')) {
+ (*val) = 1000 + (picoos_atoi((picoos_char*)strval) * 10);
+ } else {
+ (*val) = picoos_atoi((picoos_char*)strval) * 10;
+ }
+ rel = TRUE;
+ }
+ }
+ return rel;
+}
+
+
+static void tok_putItem (picodata_ProcessingUnit this, tok_subobj_t * tok,
+ picoos_uint8 itemType, picoos_uint8 info1, picoos_uint8 info2,
+ picoos_uint16 val,
+ picoos_uchar str[])
+{
+ picoos_int32 len, i;
+
+ if ((itemType == PICODATA_ITEM_CMD) && (info1 == PICODATA_ITEMINFO1_CMD_FLUSH)) {
+ tok->outBuf[tok->outWritePos++] = itemType;
+ tok->outBuf[tok->outWritePos++] = info1;
+ tok->outBuf[tok->outWritePos++] = info2;
+ tok->outBuf[tok->outWritePos++] = 0;
+ }
+ else if (tok->ignLevel <= 0) {
+ switch (itemType) {
+ case PICODATA_ITEM_CMD:
+ switch (info1) {
+ case PICODATA_ITEMINFO1_CMD_CONTEXT:
+ case PICODATA_ITEMINFO1_CMD_VOICE:
+ case PICODATA_ITEMINFO1_CMD_MARKER:
+ case PICODATA_ITEMINFO1_CMD_PLAY:
+ case PICODATA_ITEMINFO1_CMD_SAVE:
+ case PICODATA_ITEMINFO1_CMD_UNSAVE:
+ case PICODATA_ITEMINFO1_CMD_PROSDOMAIN:
+ case PICODATA_ITEMINFO1_CMD_PHONEME:
+ len = picoos_strlen((picoos_char*)str);
+ if (tok->outWritePos + 4 + len < OUT_BUF_SIZE) {
+ tok->outBuf[tok->outWritePos++] = itemType;
+ tok->outBuf[tok->outWritePos++] = info1;
+ tok->outBuf[tok->outWritePos++] = info2;
+ tok->outBuf[tok->outWritePos++] = len;
+ for (i=0; i<len; i++) {
+ tok->outBuf[tok->outWritePos++] = str[i];
+ }
+ }
+ else {
+ PICODBG_WARN(("tok_putItem: output buffer too small"));
+ }
+ break;
+ case PICODATA_ITEMINFO1_CMD_IGNSIG:
+ case PICODATA_ITEMINFO1_CMD_IGNORE:
+ if (tok->outWritePos + 4 < OUT_BUF_SIZE) {
+ tok->outBuf[tok->outWritePos++] = itemType;
+ tok->outBuf[tok->outWritePos++] = info1;
+ tok->outBuf[tok->outWritePos++] = info2;
+ tok->outBuf[tok->outWritePos++] = 0;
+ }
+ else {
+ PICODBG_WARN(("tok_putItem: output buffer too small"));
+ }
+ break;
+ case PICODATA_ITEMINFO1_CMD_SPEED:
+ case PICODATA_ITEMINFO1_CMD_PITCH:
+ case PICODATA_ITEMINFO1_CMD_VOLUME:
+ case PICODATA_ITEMINFO1_CMD_SPELL:
+ case PICODATA_ITEMINFO1_CMD_SIL:
+ case PICODATA_ITEMINFO1_CMD_SPEAKER:
+ if (tok->outWritePos + 4 + 2 < OUT_BUF_SIZE) {
+ tok->outBuf[tok->outWritePos++] = itemType;
+ tok->outBuf[tok->outWritePos++] = info1;
+ tok->outBuf[tok->outWritePos++] = info2;
+ tok->outBuf[tok->outWritePos++] = 2;
+ tok->outBuf[tok->outWritePos++] = val % 256;
+ tok->outBuf[tok->outWritePos++] = val / 256;
+ }
+ else {
+ PICODBG_WARN(("tok_putItem: output buffer too small"));
+ }
+ break;
+ default:
+ PICODBG_WARN(("tok_putItem: unknown command type"));
+ }
+ break;
+ case PICODATA_ITEM_TOKEN:
+ len = picoos_strlen((picoos_char*)str);
+ if (tok->outWritePos + 4 + len < OUT_BUF_SIZE) {
+ tok->outBuf[tok->outWritePos++] = itemType;
+ tok->outBuf[tok->outWritePos++] = info1;
+ tok->outBuf[tok->outWritePos++] = info2;
+ tok->outBuf[tok->outWritePos++] = len;
+ for (i=0; i<len; i++) {
+ tok->outBuf[tok->outWritePos++] = str[i];
+ }
+ }
+ else {
+ PICODBG_WARN(("tok_putItem: output buffer too small"));
+ }
+ break;
+ default:
+ PICODBG_WARN(("tok_putItem: unknown item type"));
+ }
+ }
+}
+
+
+static void tok_putItem2 (picodata_ProcessingUnit this, tok_subobj_t * tok,
+ picoos_uint8 type,
+ picoos_uint8 info1, picoos_uint8 info2,
+ picoos_uint8 len,
+ picoos_uint8 data[])
+{
+ picoos_int32 i;
+
+ if (is_valid_itemtype(type)) {
+ tok->outBuf[tok->outWritePos++] = type;
+ tok->outBuf[tok->outWritePos++] = info1;
+ tok->outBuf[tok->outWritePos++] = info2;
+ tok->outBuf[tok->outWritePos++] = len;
+ for (i=0; i<len; i++) {
+ tok->outBuf[tok->outWritePos++] = data[i];
+ }
+ }
+}
+
+
+static MarkupId tok_markupTagId (picoos_uchar tagId[])
+{
+ if (picoos_strstr(tagId,(picoos_char *)"svox:") == (picoos_char *)tagId) {
+ tagId+=5;
+ }
+ if (tok_strEqual(tagId, TOK_MARKUP_KW_IGNORE)) {
+ return MIIgnore;
+ } else if (tok_strEqual(tagId, TOK_MARKUP_KW_SPEED)) {
+ return MISpeed;
+ } else if (tok_strEqual(tagId, TOK_MARKUP_KW_PITCH)) {
+ return MIPitch;
+ } else if (tok_strEqual(tagId, TOK_MARKUP_KW_VOLUME)) {
+ return MIVolume;
+ } else if (tok_strEqual(tagId, TOK_MARKUP_KW_SPEAKER)) {
+ return MISpeaker;
+ } else if (tok_strEqual(tagId, TOK_MARKUP_KW_VOICE)) {
+ return MIVoice;
+ } else if (tok_strEqual(tagId, TOK_MARKUP_KW_CONTEXT)) {
+ return MIPreprocContext;
+ } else if (tok_strEqual(tagId, TOK_MARKUP_KW_MARK)) {
+ return MIMarker;
+ } else if (tok_strEqual(tagId, TOK_MARKUP_KW_PLAY)) {
+ return MIPlay;
+ } else if (tok_strEqual(tagId, TOK_MARKUP_KW_USESIG)) {
+ return MIUseSig;
+ } else if (tok_strEqual(tagId, TOK_MARKUP_KW_GENFILE)) {
+ return MIGenFile;
+ } else if (tok_strEqual(tagId, TOK_MARKUP_KW_SENTENCE) || tok_strEqual(tagId, TOK_MARKUP_KW_S)) {
+ return MISentence;
+ } else if (tok_strEqual(tagId, TOK_MARKUP_KW_PARAGRAPH) || tok_strEqual(tagId, TOK_MARKUP_KW_P)) {
+ return MIParagraph;
+ } else if (tok_strEqual(tagId, TOK_MARKUP_KW_BREAK)) {
+ return MIBreak;
+ } else if (tok_strEqual(tagId, TOK_MARKUP_KW_SPELL)) {
+ return MISpell;
+ } else if (tok_strEqual(tagId, TOK_MARKUP_KW_PHONEME)) {
+ return MIPhoneme;
+ } else if (tok_strEqual(tagId, TOK_MARKUP_KW_ITEM)) {
+ return MIItem;
+ } else {
+ return MIDummyEnd;
+ }
+}
+
+
+extern void tok_checkLimits (picodata_ProcessingUnit this, picoos_uint32 * value, picoos_uint32 min, picoos_uint32 max, picoos_uchar valueType[])
+{
+ if ((((*value) < min) || ((*value) > max))) {
+ picoos_emRaiseWarning(this->common->em, PICO_ERR_MARKUP_VALUE_OUT_OF_RANGE, (picoos_char*)"", (picoos_char*)"attempt to set illegal value %i for %s", *value, valueType);
+ if (((*value) < min)) {
+ (*value) = min;
+ } else if (((*value) > max)) {
+ (*value) = max;
+ }
+ }
+}
+
+
+
+extern void tok_checkRealLimits (picodata_ProcessingUnit this, picoos_single * value, picoos_single min, picoos_single max, picoos_uchar valueType[])
+{
+ if ((((*value) < min) || ((*value) > max))) {
+ picoos_emRaiseWarning(this->common->em, PICO_ERR_MARKUP_VALUE_OUT_OF_RANGE, (picoos_char*)"", (picoos_char*)"attempt to set illegal value %f for %s", *value, valueType);
+ if (((*value) < min)) {
+ (*value) = min;
+ } else if (((*value) > max)) {
+ (*value) = max;
+ }
+ }
+}
+
+
+#define VAL_STR_LEN 21
+
+static void tok_interpretMarkup (picodata_ProcessingUnit this, tok_subobj_t * tok, picoos_bool isStartTag, MarkupId mId)
+{
+ picoos_bool done;
+ picoos_int32 ival;
+ picoos_uint32 uval;
+ picoos_int32 ival2;
+ picoos_uchar valStr[VAL_STR_LEN];
+ picoos_uchar valStr2[VAL_STR_LEN];
+ picoos_uchar valStr3[VAL_STR_LEN];
+ picoos_int32 i2;
+ picoos_uint32 dur;
+ picoos_bool done1;
+ picoos_bool paramFound;
+ picoos_uint8 type, info1, info2;
+ picoos_uint8 data[256];
+ picoos_int32 pos, n, len;
+ picoos_uchar part[10];
+
+ done = FALSE;
+ switch (mId) {
+ case MIIgnore:
+ if (isStartTag && tok_strEqual(tok->markupParams[0].paramId,(picoos_uchar*)"")) {
+ tok_startIgnore(tok);
+ done = TRUE;
+ } else if (!isStartTag && tok_strEqual(tok->markupParams[0].paramId, (picoos_uchar*)"")) {
+ tok_endIgnore(tok);
+ done = TRUE;
+ }
+ break;
+ case MISpeed:
+ if (isStartTag && tok_strEqual(tok->markupParams[0].paramId, KWLevel)) {
+ if (tok_isRelative(tok->markupParams[0].paramVal, & uval)) {
+ tok_checkLimits(this, & uval, PICO_SPEED_FACTOR_MIN, PICO_SPEED_FACTOR_MAX,(picoos_uchar*)"relative speed factor");
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_SPEED, PICODATA_ITEMINFO2_CMD_RELATIVE, uval, (picoos_uchar*)"");
+ } else {
+ uval = picoos_atoi((picoos_char*)tok->markupParams[0].paramVal);
+ tok_checkLimits(this, & uval, PICO_SPEED_MIN, PICO_SPEED_MAX,(picoos_uchar*)"speed");
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_SPEED, PICODATA_ITEMINFO2_CMD_ABSOLUTE, uval, (picoos_uchar*)"");
+ }
+ done = TRUE;
+ } else if (!isStartTag && tok_strEqual(tok->markupParams[0].paramId, (picoos_uchar*)"")) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_SPEED, PICODATA_ITEMINFO2_CMD_ABSOLUTE, PICO_SPEED_DEFAULT, (picoos_uchar*)"");
+ done = TRUE;
+ }
+ break;
+ case MIPitch:
+ if (isStartTag && tok_strEqual(tok->markupParams[0].paramId, KWLevel)) {
+ if (tok_isRelative(tok->markupParams[0].paramVal, & uval)) {
+ tok_checkLimits(this, & uval,PICO_PITCH_FACTOR_MIN,PICO_PITCH_FACTOR_MAX, (picoos_uchar*)"relative pitch factor");
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_PITCH, PICODATA_ITEMINFO2_CMD_RELATIVE, uval, (picoos_uchar*)"");
+ } else {
+ uval = picoos_atoi((picoos_char*)tok->markupParams[0].paramVal);
+ tok_checkLimits(this, & uval,PICO_PITCH_MIN,PICO_PITCH_MAX, (picoos_uchar*)"pitch");
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_PITCH,PICODATA_ITEMINFO2_CMD_ABSOLUTE, uval, (picoos_uchar*)"");
+ }
+ done = TRUE;
+ } else if (!isStartTag && tok_strEqual(tok->markupParams[0].paramId, (picoos_uchar*)"")) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_PITCH,PICODATA_ITEMINFO2_CMD_ABSOLUTE, PICO_PITCH_DEFAULT, (picoos_uchar*)"");
+ done = TRUE;
+ }
+ break;
+ case MIVolume:
+ if (isStartTag && tok_strEqual(tok->markupParams[0].paramId, KWLevel)) {
+ if (tok_isRelative(tok->markupParams[0].paramVal, & uval)) {
+ tok_checkLimits(this, & uval, PICO_VOLUME_FACTOR_MIN, PICO_VOLUME_FACTOR_MAX, (picoos_uchar*)"relative volume factor");
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_VOLUME, PICODATA_ITEMINFO2_CMD_RELATIVE, uval, (picoos_uchar*)"");
+ } else {
+ uval = picoos_atoi((picoos_char*)tok->markupParams[0].paramVal);
+ tok_checkLimits(this, & uval, PICO_VOLUME_MIN, PICO_VOLUME_MAX, (picoos_uchar*)"volume");
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_VOLUME, PICODATA_ITEMINFO2_CMD_ABSOLUTE, uval, (picoos_uchar*)"");
+ }
+ done = TRUE;
+ } else if (!isStartTag && tok_strEqual(tok->markupParams[0].paramId, (picoos_uchar*)"")) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_VOLUME, PICODATA_ITEMINFO2_CMD_ABSOLUTE, PICO_VOLUME_DEFAULT, (picoos_uchar*)"");
+ done = TRUE;
+ }
+ break;
+ case MISpeaker:
+ if (isStartTag && tok_strEqual(tok->markupParams[0].paramId, KWLevel)) {
+ if (tok_isRelative(tok->markupParams[0].paramVal, & uval)) {
+ tok_checkLimits(this, & uval, PICO_SPEAKER_FACTOR_MIN, PICO_SPEAKER_FACTOR_MAX, (picoos_uchar*)"relative speaker factor");
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_SPEAKER, PICODATA_ITEMINFO2_CMD_RELATIVE, uval, (picoos_uchar*)"");
+ } else {
+ uval = picoos_atoi((picoos_char*)tok->markupParams[0].paramVal);
+ tok_checkLimits(this, & uval, PICO_SPEAKER_MIN, PICO_SPEAKER_MAX, (picoos_uchar*)"volume");
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_SPEAKER, PICODATA_ITEMINFO2_CMD_ABSOLUTE, uval, (picoos_uchar*)"");
+ }
+ done = TRUE;
+ } else if (!isStartTag && tok_strEqual(tok->markupParams[0].paramId, (picoos_uchar*)"")) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_SPEAKER, PICODATA_ITEMINFO2_CMD_ABSOLUTE, PICO_SPEAKER_DEFAULT, (picoos_uchar*)"");
+ done = TRUE;
+ }
+ break;
+
+ case MIVoice:
+ if (isStartTag && tok_strEqual(tok->markupParams[0].paramId, KWName)) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_VOICE, PICODATA_ITEMINFO2_NA, 0, tok->markupParams[0].paramVal);
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_FLUSH, PICODATA_ITEMINFO2_NA, 0, (picoos_uchar*)"");
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_PROSDOMAIN, 0, 0, (picoos_uchar*)"");
+ done = TRUE;
+ } else if (!isStartTag && tok_strEqual(tok->markupParams[0].paramId,(picoos_uchar*)"")) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_VOICE, PICODATA_ITEMINFO2_NA, 0, (picoos_uchar*)"");
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_FLUSH, PICODATA_ITEMINFO2_NA, 0, (picoos_uchar*)"");
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_PROSDOMAIN, 0, 0, (picoos_uchar*)"");
+ done = TRUE;
+ }
+ break;
+ case MIPreprocContext:
+ if (isStartTag && tok_strEqual(tok->markupParams[0].paramId, KWName)) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_CONTEXT, PICODATA_ITEMINFO2_NA, 0, tok->markupParams[0].paramVal);
+ done = TRUE;
+ } else if (!isStartTag && tok_strEqual(tok->markupParams[0].paramId,(picoos_uchar*)"")) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_CONTEXT, PICODATA_ITEMINFO2_NA, 0, PICO_CONTEXT_DEFAULT);
+ done = TRUE;
+ }
+ break;
+ case MIMarker:
+ if (isStartTag && tok_strEqual(tok->markupParams[0].paramId, KWName)) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_MARKER, PICODATA_ITEMINFO2_NA, 0, tok->markupParams[0].paramVal);
+ done = TRUE;
+ } else if (!isStartTag && tok_strEqual(tok->markupParams[0].paramId,(picoos_uchar*)"")) {
+ done = TRUE;
+ }
+ break;
+ case MISentence:
+ if (isStartTag) {
+ tok_getParamStrVal(tok->markupParams, KWProsDomain, (picoos_uchar*)valStr, & paramFound);
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_FLUSH, PICODATA_ITEMINFO2_NA, 0, (picoos_uchar*)"");
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_PROSDOMAIN, 2, 0, valStr);
+ done = TRUE;
+ } else if (!isStartTag && tok_strEqual(tok->markupParams[0].paramId, (picoos_uchar*)"")) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_FLUSH, PICODATA_ITEMINFO2_NA, 0, (picoos_uchar*)"");
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_PROSDOMAIN, 2, 0, (picoos_uchar*)"");
+ done = TRUE;
+ }
+ break;
+ case MIParagraph:
+ if (isStartTag) {
+ tok_getParamStrVal(tok->markupParams, KWProsDomain, (picoos_uchar*)valStr, & paramFound);
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_FLUSH, PICODATA_ITEMINFO2_NA, 0, (picoos_uchar*)"");
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_PROSDOMAIN, 1, 0, valStr);
+ done = TRUE;
+ } else if (!isStartTag && tok_strEqual(tok->markupParams[0].paramId, (picoos_uchar*)"")) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_FLUSH, PICODATA_ITEMINFO2_NA, 0, (picoos_uchar*)"");
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_SIL, PICODATA_ITEMINFO2_NA, PARAGRAPH_PAUSE_DUR, (picoos_uchar*)"");
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_PROSDOMAIN, 1, 0, (picoos_uchar*)"");
+ done = TRUE;
+ }
+ break;
+ case MIBreak:
+ if (isStartTag && tok_strEqual(tok->markupParams[0].paramId, KWTime)) {
+ tok_getDur(tok->markupParams[0].paramVal, & dur, & done1);
+ tok_checkLimits (this, &dur, 0, 65535, (picoos_uchar*)"time");
+ if (done1) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_SIL, PICODATA_ITEMINFO2_NA, dur, (picoos_uchar*)"");
+ done = TRUE;
+ }
+ } else if (!isStartTag && tok_strEqual(tok->markupParams[0].paramId, (picoos_uchar*)"")) {
+ done = TRUE;
+ }
+ break;
+ case MISpell:
+ if (isStartTag) {
+ if (tok_strEqual(tok->markupParams[0].paramId, KWMode)) {
+ if (tok_strEqual(tok->markupParams[0].paramVal, KWPB)) {
+ uval = SPELL_WITH_PHRASE_BREAK;
+ } else if (tok_strEqual(tok->markupParams[0].paramVal, KWSB)) {
+ uval = SPELL_WITH_SENTENCE_BREAK;
+ } else {
+ tok_getDur(tok->markupParams[0].paramVal, & uval, & done1);
+ tok_checkLimits (this, & uval, 0, 65535, (picoos_uchar*)"time");
+ if (done1) {
+ done = TRUE;
+ }
+ }
+ } else {
+ uval = SPELL_WITH_PHRASE_BREAK;
+ }
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_SPELL, PICODATA_ITEMINFO2_CMD_START, uval, (picoos_uchar*)"");
+ done = TRUE;
+ } else if (!isStartTag && tok_strEqual(tok->markupParams[0].paramId, (picoos_uchar*)"")) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_SPELL, PICODATA_ITEMINFO2_CMD_END, 0, (picoos_uchar*)"");
+ done = TRUE;
+ }
+ break;
+ case MIGenFile:
+ if (isStartTag && tok_strEqual(tok->markupParams[0].paramId, KWFile)) {
+ if (tok->saveFile[0] != 0) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_UNSAVE,
+ picodata_getPuTypeFromExtension(tok->saveFile, /*input*/FALSE), 0, tok->saveFile);
+ tok->saveFile[0] = 0;
+ }
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_SAVE,
+ picodata_getPuTypeFromExtension(tok->markupParams[0].paramVal, /*input*/FALSE), 0, tok->markupParams[0].paramVal);
+ picoos_strcpy((picoos_char*)tok->saveFile, (picoos_char*)tok->markupParams[0].paramVal);
+ done = TRUE;
+ } else if (!isStartTag && tok_strEqual(tok->markupParams[0].paramId, (picoos_uchar*)"")) {
+ if (tok->saveFile[0] != 0) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_UNSAVE,
+ picodata_getPuTypeFromExtension(tok->saveFile, /*input*/FALSE), 0, (picoos_uchar*)"");
+ tok->saveFile[0] = 0;
+ }
+ done = TRUE;
+ }
+ break;
+ case MIPlay:
+ if (isStartTag && tok_strEqual(tok->markupParams[0].paramId, KWFile)) {
+ if (picoos_FileExists(this->common, (picoos_char*)tok->markupParams[0].paramVal)) {
+ tok_getParamIntVal(tok->markupParams,KWF0Beg,& ival,& paramFound);
+ tok_getParamIntVal(tok->markupParams,KWF0End,& ival2,& paramFound);
+ tok_getParamStrVal(tok->markupParams,KWAlphabet,valStr3,& paramFound);
+ tok_getParamPhonesStr(tok->markupParams,KWXFadeBeg,valStr3,valStr,VAL_STR_LEN,& paramFound);
+ tok_getParamPhonesStr(tok->markupParams,KWXFadeEnd,valStr3,valStr2,VAL_STR_LEN,& paramFound);
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_PLAY,
+ picodata_getPuTypeFromExtension(tok->markupParams[0].paramVal, /*input*/TRUE), 0, tok->markupParams[0].paramVal);
+ tok_startIgnore(tok);
+ } else {
+ if (tok->ignLevel > 0) {
+ tok_startIgnore(tok);
+ } else {
+ picoos_emRaiseWarning(this->common->em, PICO_EXC_CANT_OPEN_FILE, (picoos_char*)"", (picoos_char*)"file '%s' not found; synthesizing enclosed text instead\n", tok->markupParams[0].paramVal);
+ }
+ }
+ done = TRUE;
+ } else if (!isStartTag && tok_strEqual(tok->markupParams[0].paramId, (picoos_uchar*)"")) {
+ tok_endIgnore(tok);
+ done = TRUE;
+ }
+ break;
+ case MIUseSig:
+ if (isStartTag && tok_strEqual(tok->markupParams[0].paramId, KWFile)) {
+ if (picoos_FileExists(this->common, (picoos_char*)tok->markupParams[0].paramVal)) {
+ tok_getParamIntVal(tok->markupParams,KWF0Beg,& ival,& paramFound);
+ tok_getParamIntVal(tok->markupParams,KWF0End,& ival2,& paramFound);
+ tok_getParamStrVal(tok->markupParams,KWAlphabet,valStr3, & paramFound);
+ tok_getParamPhonesStr(tok->markupParams,KWXFadeBeg,valStr3,valStr,VAL_STR_LEN,& paramFound);
+ tok_getParamPhonesStr(tok->markupParams,KWXFadeEnd,valStr3,valStr2,VAL_STR_LEN,& paramFound);
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_PLAY,
+ picodata_getPuTypeFromExtension(tok->markupParams[0].paramVal, /*input*/TRUE), 0, tok->markupParams[0].paramVal);
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_IGNSIG, PICODATA_ITEMINFO2_CMD_START, 0, (picoos_uchar*)"");
+ } else {
+ if (tok->ignLevel <= 0) {
+ picoos_emRaiseWarning(this->common->em, PICO_EXC_CANT_OPEN_FILE, (picoos_char*)"", (picoos_char*)"file '%s' not found; synthesizing enclosed text instead", tok->markupParams[0].paramVal);
+ }
+ }
+ done = TRUE;
+ } else if (!isStartTag && tok_strEqual(tok->markupParams[0].paramId, (picoos_uchar*)"")) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_IGNSIG, PICODATA_ITEMINFO2_CMD_END, 0, (picoos_uchar*)"");
+ done = TRUE;
+ }
+ break;
+ case MIPhoneme:
+ i2 = 0;
+ if (isStartTag) {
+ if (tok_strEqual(tok->markupParams[0].paramId, KWAlphabet) && tok_strEqual(tok->markupParams[1].paramId, KWPH)) {
+ if (tok_strEqual(tok->markupParams[2].paramId, KWOrthMode)
+ && tok_strEqual(tok->markupParams[2].paramVal, KWIgnorePunct)) {
+ i2 = 1;
+ }
+ if (picodata_mapPAStrToPAIds(tok->transducer, this->common, tok->xsampa_parser, tok->svoxpa_parser, tok->xsampa2svoxpa_mapper, tok->markupParams[1].paramVal, tok->markupParams[0].paramVal, tok->phonemes, sizeof(tok->phonemes)-1) == PICO_OK) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_PHONEME,
+ PICODATA_ITEMINFO2_CMD_START, i2, tok->phonemes);
+ done = TRUE;
+ } else {
+ PICODBG_WARN(("cannot map phonetic string '%s'; synthesizeing text instead", tok->markupParams[1].paramVal));
+ picoos_emRaiseWarning(this->common->em, PICO_ERR_MARKUP_VALUE_OUT_OF_RANGE,(picoos_char*)"", (picoos_char*)"cannot map phonetic string '%s'; synthesizeing text instead", tok->markupParams[1].paramVal);
+ done = TRUE;
+ }
+ } else if (tok_strEqual(tok->markupParams[0].paramId, KWPH)) {
+ if (tok_strEqual(tok->markupParams[1].paramId, KWOrthMode)
+ && tok_strEqual(tok->markupParams[1].paramVal, KWIgnorePunct)) {
+ i2 = 1;
+ }
+ if (picodata_mapPAStrToPAIds(tok->transducer, this->common, tok->xsampa_parser, tok->svoxpa_parser, tok->xsampa2svoxpa_mapper, tok->markupParams[0].paramVal, PICODATA_XSAMPA, tok->phonemes, sizeof(tok->phonemes)) == PICO_OK) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_PHONEME,
+ PICODATA_ITEMINFO2_CMD_START, i2, tok->phonemes);
+ done = TRUE;
+ }
+ else {
+ PICODBG_WARN(("cannot map phonetic string '%s'; synthesizeing text instead", tok->markupParams[1].paramVal));
+ picoos_emRaiseWarning(this->common->em, PICO_ERR_MARKUP_VALUE_OUT_OF_RANGE,(picoos_char*)"", (picoos_char*)"cannot map phonetic string '%s'; synthesizing text instead", tok->markupParams[0].paramVal);
+ done = TRUE;
+ }
+ }
+ } else if (!isStartTag && tok_strEqual(tok->markupParams[0].paramId, (picoos_uchar*)"")) {
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_PHONEME,
+ PICODATA_ITEMINFO2_CMD_END, i2, (picoos_uchar*)"");
+ done = TRUE;
+ }
+ break;
+ case MIItem:
+ if (isStartTag && tok_strEqual(tok->markupParams[0].paramId, KWType) &&
+ tok_strEqual(tok->markupParams[1].paramId, KWInfo1)&&
+ tok_strEqual(tok->markupParams[2].paramId, KWInfo2)&&
+ tok_strEqual(tok->markupParams[3].paramId, KWDATA)) {
+ picoos_int32 len2, n2;
+ type = picoos_atoi(tok->markupParams[0].paramVal);
+ info1 = picoos_atoi(tok->markupParams[1].paramVal);
+ info2 = picoos_atoi(tok->markupParams[2].paramVal);
+ n = 0; n2 = 0;
+ len2 = (picoos_int32)picoos_strlen(tok->markupParams[3].paramVal);
+ while (n<len2) {
+ while ((tok->markupParams[3].paramVal[n] != 0) && (tok->markupParams[3].paramVal[n] <= 32)) {
+ n++;
+ }
+ tok->markupParams[3].paramVal[n2] = tok->markupParams[3].paramVal[n];
+ n++;
+ n2++;
+ }
+ if (is_valid_itemtype(type)) {
+ done = TRUE;
+ len = 0;
+ pos = 0;
+ picoos_get_sep_part_str(tok->markupParams[3].paramVal, picoos_strlen(tok->markupParams[3].paramVal),
+ &pos, ',', part, 10, &done1);
+ while (done && done1) {
+ n = picoos_atoi(part);
+ if ((n>=0) && (n<256) && (len<256)) {
+ data[len++] = n;
+ }
+ else {
+ done = FALSE;
+ }
+ picoos_get_sep_part_str(tok->markupParams[3].paramVal, picoos_strlen(tok->markupParams[3].paramVal),
+ &pos, ',', part, 10, &done1);
+ }
+ if (done) {
+ tok_putItem2(this, tok, type, info1, info2, len, data);
+ }
+ }
+ else {
+ done = FALSE;
+ }
+ } else if (!isStartTag && tok_strEqual(tok->markupParams[0].paramId,(picoos_uchar*)"")) {
+ done = TRUE;
+ }
+ break;
+ default:
+ break;
+ }
+ if (!done) {
+ tok->markupTagErr = MEInterprete;
+ }
+ if (isStartTag) {
+ tok->markupLevel[mId]++;
+ } else if ((tok->markupLevel[mId] > 0)) {
+ tok->markupLevel[mId]--;
+ }
+}
+
+
+static picoos_bool tok_attrChar (picoos_uchar ch, picoos_bool first)
+{
+ return ((((ch >= (picoos_uchar)'A') && (ch <= (picoos_uchar)'Z')) ||
+ ((ch >= (picoos_uchar)'a') && (ch <= (picoos_uchar)'z'))) ||
+ ( !(first) && ((ch >= (picoos_uchar)'0') && (ch <= (picoos_uchar)'9'))));
+}
+
+
+
+static picoos_bool tok_idChar (picoos_uchar ch, picoos_bool first)
+{
+ return tok_attrChar(ch, first) || ( !(first) && (ch == (picoos_uchar)':'));
+}
+
+
+static void tok_setIsFileAttr (picoos_uchar name[], picoos_bool * isFile)
+{
+ (*isFile) = tok_strEqual(name, KWFile);
+}
+
+/* *****************************************************************************/
+
+static void tok_putToSimpleToken (picodata_ProcessingUnit this, tok_subobj_t * tok, picoos_uchar str[], pico_tokenType type, pico_tokenSubType subtype)
+{
+ int i, len;
+
+ if (str[0] != 0) {
+ len = picoos_strlen((picoos_char*)str);
+ for (i = 0; i < len; i++) {
+ if (tok->tokenPos >= IN_BUF_SIZE) {
+ picoos_emRaiseWarning(this->common->em, PICO_ERR_INTERNAL_LIMIT, (picoos_char*)"", (picoos_char*)"simple token too long; forced treatment");
+ tok_treatSimpleToken(this, tok);
+ }
+ tok->tokenStr[tok->tokenPos] = str[i];
+ tok->tokenPos++;
+ }
+ }
+ tok->tokenType = type;
+ tok->tokenSubType = subtype;
+}
+
+
+static void tok_putToMarkup (picodata_ProcessingUnit this, tok_subobj_t * tok, picoos_uchar str[])
+{
+ picoos_int32 i, len;
+ picoos_uint8 ok;
+
+ tok->markupTagErr = MENone;
+ len = picoos_strlen((picoos_char*)str);
+ for (i = 0; i< len; i++) {
+ if (tok->markupPos >= (MARKUP_STRING_BUF_SIZE - 1)) {
+ if ((tok->markupPos == (MARKUP_STRING_BUF_SIZE - 1)) && (tok_markupTagId(tok->markupTagName) != MIDummyEnd)) {
+ picoos_emRaiseWarning(this->common->em, PICO_ERR_INTERNAL_LIMIT ,(picoos_char*)"", (picoos_char*)"markup tag too long");
+ }
+ tok->markupState = MSErrorTooLong;
+ } else if ((str[i] == (picoos_uchar)' ') && ((tok->markupState == MSExpectingmarkupTagName) || (tok->markupState == MSGotmarkupTagName) || (tok->markupState == MSGotAttrName) || (tok->markupState == MSGotEqual) || (tok->markupState == MSGotAttrValue))) {
+ } else if ((str[i] == (picoos_uchar)'>') && ((tok->markupState == MSGotmarkupTagName) || (tok->markupState == MSInmarkupTagName) || (tok->markupState == MSGotAttrValue))) {
+ tok->markupState = MSGotEnd;
+ } else if ((str[i] == (picoos_uchar)'/') && ((tok->markupState == MSGotmarkupTagName) || (tok->markupState == MSInmarkupTagName) || (tok->markupState == MSGotAttrValue))) {
+ if (tok->markupTagType == MTEnd) {
+ tok->markupTagErr = MEUnexpectedChar;
+ tok->markupState = MSError;
+ } else {
+ tok->markupTagType = MTEmpty;
+ tok->markupState = MSGotEndSlash;
+ }
+ } else {
+ switch (tok->markupState) {
+ case MSNotInMarkup:
+ if (str[i] == (picoos_uchar)'<') {
+ tok_clearMarkupParams(tok->markupParams);
+ tok->nrMarkupParams = 0;
+ tok->strPos = 0;
+ tok->markupTagType = MTStart;
+ tok->markupState = MSGotStart;
+ } else {
+ tok->markupTagErr = MEMissingStart;
+ tok->markupState = MSError;
+ }
+ break;
+ case MSGotStart:
+ if (str[i] == (picoos_uchar)'/') {
+ tok->markupTagType = MTEnd;
+ tok->markupState = MSExpectingmarkupTagName;
+ } else if (str[i] == (picoos_uchar)' ') {
+ tok->markupState = MSExpectingmarkupTagName;
+ } else if (tok_idChar(str[i],TRUE)) {
+ tok->markupTagType = MTStart;
+ tok->markupTagName[tok->strPos] = str[i];
+ tok->strPos++;
+ tok->markupTagName[tok->strPos] = 0;
+ tok->markupState = MSInmarkupTagName;
+ } else {
+ tok->markupTagErr = MEUnexpectedChar;
+ tok->markupState = MSError;
+ }
+ break;
+ case MSInmarkupTagName: case MSExpectingmarkupTagName:
+ if (tok_idChar(str[i],tok->markupState == MSExpectingmarkupTagName)) {
+ tok->markupTagName[tok->strPos] = str[i];
+ tok->strPos++;
+ tok->markupTagName[(tok->strPos)] = 0;
+ tok->markupState = MSInmarkupTagName;
+ } else if ((tok->markupState == MSInmarkupTagName) && (str[i] == (picoos_uchar)' ')) {
+ tok->markupState = MSGotmarkupTagName;
+ picobase_lowercase_utf8_str(tok->markupTagName, (picoos_char*)tok->markupTagName, IN_BUF_SIZE, &ok);
+ tok->strPos = 0;
+ } else {
+ tok->markupTagErr = MEIdent;
+ tok->markupState = MSError;
+ }
+ break;
+ case MSGotmarkupTagName: case MSGotAttrValue:
+ if (tok_attrChar(str[i], TRUE)) {
+ if (tok->markupTagType == MTEnd) {
+ tok->markupTagErr = MEUnexpectedChar;
+ tok->markupState = MSError;
+ } else {
+ if (tok->nrMarkupParams < MAX_NR_MARKUP_PARAMS) {
+ tok->markupParams[tok->nrMarkupParams].paramId[tok->strPos] = str[i];
+ tok->strPos++;
+ tok->markupParams[tok->nrMarkupParams].paramId[tok->strPos] = 0;
+ } else {
+ picoos_emRaiseWarning(this->common->em, PICO_ERR_INTERNAL_LIMIT ,(picoos_char*)"", (picoos_char*)"too many attributes in markup; ignoring");
+ }
+ tok->markupState = MSInAttrName;
+ }
+ } else {
+ tok->markupTagErr = MEUnexpectedChar;
+ tok->markupState = MSError;
+ }
+ break;
+ case MSInAttrName:
+ if (tok_attrChar(str[i], FALSE)) {
+ if (tok->nrMarkupParams < MAX_NR_MARKUP_PARAMS) {
+ tok->markupParams[tok->nrMarkupParams].paramId[tok->strPos] = str[i];
+ tok->strPos++;
+ tok->markupParams[tok->nrMarkupParams].paramId[tok->strPos] = 0;
+ }
+ tok->markupState = MSInAttrName;
+ } else if (str[i] == (picoos_uchar)' ') {
+ picobase_lowercase_utf8_str(tok->markupParams[tok->nrMarkupParams].paramId, (picoos_char*)tok->markupParams[tok->nrMarkupParams].paramId, IN_BUF_SIZE, &ok);
+ tok_setIsFileAttr(tok->markupParams[tok->nrMarkupParams].paramId, & tok->isFileAttr);
+ tok->markupState = MSGotAttrName;
+ } else if (str[i] == (picoos_uchar)'=') {
+ picobase_lowercase_utf8_str(tok->markupParams[tok->nrMarkupParams].paramId, (picoos_char*)tok->markupParams[tok->nrMarkupParams].paramId, IN_BUF_SIZE, &ok);
+ tok_setIsFileAttr(tok->markupParams[tok->nrMarkupParams].paramId, & tok->isFileAttr);
+ tok->markupState = MSGotEqual;
+ } else {
+ tok->markupTagErr = MEMissingEqual;
+ tok->markupState = MSError;
+ }
+ break;
+ case MSGotAttrName:
+ if (str[i] == (picoos_uchar)'=') {
+ tok->markupState = MSGotEqual;
+ } else {
+ tok->markupTagErr = MEMissingEqual;
+ tok->markupState = MSError;
+ }
+ break;
+ case MSGotEqual:
+ if ((str[i] == (picoos_uchar)'"') || (str[i] == (picoos_uchar)'\'')) {
+ tok->strDelim = str[i];
+ tok->strPos = 0;
+ tok->markupState = MSInAttrValue;
+ } else {
+ tok->markupTagErr = MEMissingQuote;
+ tok->markupState = MSError;
+ }
+ break;
+ case MSInAttrValue:
+ if (!(tok->isFileAttr) && (str[i] == (picoos_uchar)'\\')) {
+ tok->markupState = MSInAttrValueEscaped;
+ } else if (str[i] == tok->strDelim) {
+ if (tok->nrMarkupParams < MAX_NR_MARKUP_PARAMS) {
+ tok->nrMarkupParams++;
+ }
+ tok->strPos = 0;
+ tok->markupState = MSGotAttrValue;
+ } else {
+ if (tok->nrMarkupParams < MAX_NR_MARKUP_PARAMS) {
+ tok->markupParams[tok->nrMarkupParams].paramVal[tok->strPos] = str[i];
+ tok->strPos++;
+ tok->markupParams[tok->nrMarkupParams].paramVal[tok->strPos] = 0;
+ }
+ tok->markupState = MSInAttrValue;
+ }
+ break;
+ case MSInAttrValueEscaped:
+ if (tok->nrMarkupParams < MAX_NR_MARKUP_PARAMS) {
+ tok->markupParams[tok->nrMarkupParams].paramVal[tok->strPos] = str[i];
+ tok->strPos++;
+ tok->markupParams[tok->nrMarkupParams].paramVal[tok->strPos] = 0;
+ }
+ tok->markupState = MSInAttrValue;
+ break;
+ case MSGotEndSlash:
+ if (str[i] == (picoos_uchar)'>') {
+ tok->markupState = MSGotEnd;
+ } else {
+ tok->markupTagErr = MEUnexpectedChar;
+ tok->markupState = MSError;
+ }
+ break;
+ default:
+ tok->markupTagErr = MEUnexpectedChar;
+ tok->markupState = MSError;
+ break;
+ }
+ }
+ tok->markupStr[tok->markupPos] = str[i];
+ tok->markupPos++;
+ tok->markupStr[tok->markupPos] = 0;
+ }
+ /*
+ PICODBG_DEBUG(("putToMarkup %s", tok->markupStr));
+ */
+}
+
+/* *****************************************************************************/
+
+static void tok_treatMarkupAsSimpleToken (picodata_ProcessingUnit this, tok_subobj_t * tok)
+{
+ picoos_int32 i;
+
+ tok->utfpos = 0;
+ tok->utflen = 0;
+ tok->markupState = MSNotInMarkup;
+ for (i = 0; i < tok->markupPos; i++) {
+ tok_treatChar(this, tok, tok->markupStr[i], FALSE);
+ }
+ tok->markupPos = 0;
+ tok->strPos = 0;
+}
+
+
+static void tok_treatMarkup (picodata_ProcessingUnit this, tok_subobj_t * tok)
+{
+ MarkupId mId;
+
+ if (tok_markupTagId(tok->markupTagName) != MIDummyEnd) {
+ if (tok->markupTagErr == MENone) {
+ tok->markupState = MSNotInMarkup;
+ if ((tok->tokenType != PICODATA_ITEMINFO1_TOKTYPE_SPACE) && (tok->tokenType != PICODATA_ITEMINFO1_TOKTYPE_UNDEFINED)) {
+ tok_treatSimpleToken(this, tok);
+ }
+ tok_putToSimpleToken(this, tok, (picoos_uchar*)" ", PICODATA_ITEMINFO1_TOKTYPE_SPACE, -1);
+ mId = tok_markupTagId(tok->markupTagName);
+ if ((tok->markupTagType == MTStart) || (tok->markupTagType == MTEmpty)) {
+ tok_interpretMarkup(this, tok, TRUE, mId);
+ }
+ if (((tok->markupTagType == MTEnd) || (tok->markupTagType == MTEmpty))) {
+ tok_clearMarkupParams(tok->markupParams);
+ tok->nrMarkupParams = 0;
+ tok_interpretMarkup(this, tok, FALSE,mId);
+ }
+ }
+ if (tok->markupTagErr != MENone) {
+ if (!tok->aborted) {
+ picoos_emRaiseWarning(this->common->em, PICO_ERR_INVALID_MARKUP_TAG, (picoos_char*)"", (picoos_char*)"syntax error in markup token '%s'",tok->markupStr);
+ }
+ tok_treatMarkupAsSimpleToken(this, tok);
+ }
+ } else {
+ tok_treatMarkupAsSimpleToken(this, tok);
+ }
+ tok->markupState = MSNotInMarkup;
+ tok->markupPos = 0;
+ tok->strPos = 0;
+}
+
+
+
+static void tok_treatChar (picodata_ProcessingUnit this, tok_subobj_t * tok, picoos_uchar ch, picoos_bool markupHandling)
+{
+ picoos_int32 id;
+ picoos_uint8 uval8;
+ pico_tokenType type = PICODATA_ITEMINFO1_TOKTYPE_UNDEFINED;
+ pico_tokenSubType subtype = -1;
+ picoos_bool dummy;
+
+ if (ch == NULLC) {
+ tok_treatSimpleToken(this, tok);
+ tok_putItem(this, tok, PICODATA_ITEM_CMD, PICODATA_ITEMINFO1_CMD_FLUSH, PICODATA_ITEMINFO2_NA, 0, (picoos_uchar*)"");
+ }
+ else {
+ switch (tok_putToUtf(tok, ch)) {
+ case UTF_CHAR_MALFORMED:
+ tok->utfpos = 0;
+ tok->utflen = 0;
+ break;
+ case UTF_CHAR_INCOMPLETE:
+ break;
+ case UTF_CHAR_COMPLETE:
+ markupHandling = (markupHandling && (tok->markupHandlingMode == MARKUP_HANDLING_ENABLED));
+ id = picoktab_graphOffset(tok->graphTab, tok->utf);
+ if (id > 0) {
+ if (picoktab_getIntPropTokenType(tok->graphTab, id, &uval8)) {
+ type = (pico_tokenType)uval8;
+ if (type == PICODATA_ITEMINFO1_TOKTYPE_LETTERV) {
+ type = PICODATA_ITEMINFO1_TOKTYPE_LETTER;
+ }
+ }
+ dummy = picoktab_getIntPropTokenSubType(tok->graphTab, id, &subtype);
+ } else if (ch <= (picoos_uchar)' ') {
+ type = PICODATA_ITEMINFO1_TOKTYPE_SPACE;
+ subtype = -1;
+ } else {
+ type = PICODATA_ITEMINFO1_TOKTYPE_UNDEFINED;
+ subtype = -1;
+ }
+ if ((ch > (picoos_uchar)' ')) {
+ tok->nrEOL = 0;
+ } else if ((ch == EOL)) {
+ tok->nrEOL++;
+ }
+ if (markupHandling && (tok->markupState != MSNotInMarkup)) {
+ tok_putToMarkup(this, tok, tok->utf);
+ if (tok->markupState >= MSError) {
+ tok_treatMarkupAsSimpleToken(this, tok);
+ } else if (tok->markupState == MSGotEnd) {
+ tok_treatMarkup(this, tok);
+ }
+ } else if ((markupHandling && (ch == (picoos_uchar)'<'))) {
+ tok_putToMarkup(this, tok, tok->utf);
+ } else if (type != PICODATA_ITEMINFO1_TOKTYPE_UNDEFINED) {
+ if ((type != tok->tokenType) || (type == PICODATA_ITEMINFO1_TOKTYPE_CHAR) || (subtype != tok->tokenSubType)) {
+ tok_treatSimpleToken(this, tok);
+ } else if ((ch == EOL) && (tok->nrEOL == 2)) {
+ tok_treatSimpleToken(this, tok);
+ tok_putToSimpleToken(this, tok, (picoos_uchar*)".", PICODATA_ITEMINFO1_TOKTYPE_CHAR, -1);
+ tok_treatSimpleToken(this, tok);
+ }
+ tok_putToSimpleToken(this, tok, tok->utf, type, subtype);
+ } else {
+ tok_treatSimpleToken(this, tok);
+ }
+ tok->utfpos = 0;
+ tok->utflen = 0;
+ break;
+ }
+ }
+}
+
+
+static void tok_treatSimpleToken (picodata_ProcessingUnit this, tok_subobj_t * tok)
+{
+ if (tok->tokenPos < IN_BUF_SIZE) {
+ tok->tokenStr[tok->tokenPos] = 0;
+ }
+ if (tok->markupState != MSNotInMarkup) {
+ if (!(tok->aborted) && (tok->markupState >= MSGotmarkupTagName) && (tok_markupTagId(tok->markupTagName) != MIDummyEnd)) {
+ picoos_emRaiseWarning(this->common->em, PICO_ERR_INVALID_MARKUP_TAG, (picoos_char*)"", (picoos_char*)"unfinished markup tag '%s'",tok->markupStr);
+ }
+ tok_treatMarkupAsSimpleToken(this, tok);
+ tok_treatSimpleToken(this, tok);
+ } else if ((tok->tokenPos > 0) && ((tok->ignLevel <= 0) || (tok->tokenType == PICODATA_ITEMINFO1_TOKTYPE_SPACE))) {
+ tok_putItem(this, tok, PICODATA_ITEM_TOKEN, tok->tokenType, (picoos_uint8)tok->tokenSubType, 0, tok->tokenStr);
+ }
+ tok->tokenPos = 0;
+ tok->tokenType = PICODATA_ITEMINFO1_TOKTYPE_UNDEFINED;
+ tok->tokenSubType = -1;
+}
+
+/* *****************************************************************************/
+
+static pico_status_t tokReset(register picodata_ProcessingUnit this)
+{
+ tok_subobj_t * tok;
+ MarkupId mId;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ tok = (tok_subobj_t *) this->subObj;
+
+ tok->ignLevel = 0;
+
+ tok->utfpos = 0;
+ tok->utflen = 0;
+
+ tok_clearMarkupParams(tok->markupParams);
+ tok->nrMarkupParams = 0;
+ tok->markupState = MSNotInMarkup;
+ tok->markupPos = 0;
+ for (mId = MIDummyStart; mId <= MIDummyEnd; mId++) {
+ tok->markupLevel[mId] = 0;
+ }
+ tok->markupTagName[0] = 0;
+ tok->markupTagType = MTNone;
+ tok->markupTagErr = MENone;
+
+ tok->strPos = 0;
+ tok->strDelim = 0;
+ tok->isFileAttr = FALSE;
+
+ tok->tokenType = PICODATA_ITEMINFO1_TOKTYPE_UNDEFINED;
+ tok->tokenSubType = -1;
+ tok->tokenPos = 0;
+
+ tok->nrEOL = 0;
+
+
+ tok->markupHandlingMode = TRUE;
+ tok->aborted = FALSE;
+
+ tok->start = TRUE;
+
+ tok->outReadPos = 0;
+ tok->outWritePos = 0;
+
+ tok->saveFile[0] = 0;
+
+
+ tok->graphTab = picoktab_getGraphs(this->voice->kbArray[PICOKNOW_KBID_TAB_GRAPHS]);
+
+ tok->xsampa_parser = picokfst_getFST(this->voice->kbArray[PICOKNOW_KBID_FST_XSAMPA_PARSE]);
+ PICODBG_TRACE(("got xsampa_parser @ %i",tok->xsampa_parser));
+
+ tok->svoxpa_parser = picokfst_getFST(this->voice->kbArray[PICOKNOW_KBID_FST_SVOXPA_PARSE]);
+ PICODBG_TRACE(("got svoxpa_parser @ %i",tok->svoxpa_parser));
+
+ tok->xsampa2svoxpa_mapper = picokfst_getFST(this->voice->kbArray[PICOKNOW_KBID_FST_XSAMPA2SVOXPA]);
+ PICODBG_TRACE(("got xsampa2svoxpa_mapper @ %i",tok->xsampa2svoxpa_mapper));
+
+
+
+ return PICO_OK;
+}
+
+static pico_status_t tokInitialize(register picodata_ProcessingUnit this)
+{
+/*
+
+ tok_subobj_t * tok;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICO_ERR_OTHER;
+ }
+ tok = (tok_subobj_t *) this->subObj;
+*/
+ return tokReset(this);
+}
+
+
+static pico_status_t tokTerminate(register picodata_ProcessingUnit this)
+{
+ return PICO_OK;
+}
+
+static picodata_step_result_t tokStep(register picodata_ProcessingUnit this, picoos_int16 mode, picoos_uint16 * numBytesOutput);
+
+static pico_status_t tokSubObjDeallocate(register picodata_ProcessingUnit this,
+ picoos_MemoryManager mm)
+{
+
+ if (NULL != this) {
+ picoos_deallocate(this->common->mm, (void *) &this->subObj);
+ }
+ mm = mm; /* avoid warning "var not used in this function"*/
+ return PICO_OK;
+}
+
+picodata_ProcessingUnit picotok_newTokenizeUnit(picoos_MemoryManager mm, picoos_Common common,
+ picodata_CharBuffer cbIn, picodata_CharBuffer cbOut,
+ picorsrc_Voice voice)
+{
+ tok_subobj_t * tok;
+ picodata_ProcessingUnit this = picodata_newProcessingUnit(mm, common, cbIn, cbOut, voice);
+ if (this == NULL) {
+ return NULL;
+ }
+ this->initialize = tokInitialize;
+ PICODBG_DEBUG(("set this->step to tokStep"));
+ this->step = tokStep;
+ this->terminate = tokTerminate;
+ this->subDeallocate = tokSubObjDeallocate;
+ this->subObj = picoos_allocate(mm, sizeof(tok_subobj_t));
+ if (this->subObj == NULL) {
+ picoos_deallocate(mm, (void *)&this);
+ return NULL;
+ }
+ tok = (tok_subobj_t *) this->subObj;
+ tok->transducer = picotrns_newSimpleTransducer(mm, common, 10*(PICOTRNS_MAX_NUM_POSSYM+2));
+ if (NULL == tok->transducer) {
+ tokSubObjDeallocate(this,mm);
+ picoos_deallocate(mm, (void *)&this);
+ return NULL;
+ }
+ tokInitialize(this);
+ return this;
+}
+
+/**
+ * fill up internal buffer, try to locate token, write token to output
+ */
+picodata_step_result_t tokStep(register picodata_ProcessingUnit this,
+ picoos_int16 mode, picoos_uint16 * numBytesOutput)
+{
+ register tok_subobj_t * tok;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ tok = (tok_subobj_t *) this->subObj;
+
+ mode = mode; /* avoid warning "var not used in this function"*/
+
+ *numBytesOutput = 0;
+ while (1) { /* exit via return */
+ picoos_int16 ch;
+
+ if ((tok->outWritePos - tok->outReadPos) > 0) {
+ if (picodata_cbPutItem(this->cbOut, &tok->outBuf[tok->outReadPos], tok->outWritePos - tok->outReadPos, numBytesOutput) == PICO_OK) {
+ PICODATA_INFO_ITEM(this->voice->kbArray[PICOKNOW_KBID_DBG],
+ (picoos_uint8 *)"tok:", &tok->outBuf[tok->outReadPos], tok->outWritePos - tok->outReadPos);
+ tok->outReadPos += *numBytesOutput;
+ if (tok->outWritePos == tok->outReadPos) {
+ tok->outWritePos = 0;
+ tok->outReadPos = 0;
+ }
+ }
+ else {
+ return PICODATA_PU_OUT_FULL;
+ }
+
+ }
+ else if (PICO_EOF != (ch = picodata_cbGetCh(this->cbIn))) {
+ PICODBG_DEBUG(("read in %c", (picoos_char) ch));
+ tok_treatChar(this, tok, (picoos_uchar) ch, /*markupHandling*/TRUE);
+ }
+ else {
+ return PICODATA_PU_IDLE;
+ }
+ }
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+/* end */
diff --git a/lib/picotok.h b/lib/picotok.h
new file mode 100644
index 0000000..b602408
--- /dev/null
+++ b/lib/picotok.h
@@ -0,0 +1,115 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picotok.h
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+
+/** @addtogroup picotok
+itemtype, iteminfo1, iteminfo2, content -> TYPE(INFO1,INFO2)content
+in the following
+
+input
+=====
+
+- UTF8 text
+
+limitations: currently only german umlauts in addition to ASCII
+
+
+minimal input size (before processing starts)
+==================
+
+processing (ie. tokenization) starts when
+- 'PICO_EOF' char received (which happens whenever the cbIn buffer is empty)
+- tok-internal buffer is full
+
+
+items output
+============
+
+processing the character stream can result in one of the
+following items:
+-> WORDGRAPH(NA,NA)graph <- mapped to lower case; incl. 1-2 digit nrs (0-99)
+-> OTHER(NA,NA)string <- skip or spell
+-> PUNC(PUNCtype,PUNCsubtype)
+-> CMD(CMDtype,CMDsubtype)args
+
+with
+- PUNCtype %d
+ PICODATA_ITEMINFO1_PUNC_SENTEND
+ PICODATA_ITEMINFO1_PUNC_PHRASEEND
+- PUNCsubtype %d
+ PICODATA_ITEMINFO2_PUNC_SENT_T
+ PICODATA_ITEMINFO2_PUNC_SENT_Q
+ PICODATA_ITEMINFO2_PUNC_SENT_E
+ PICODATA_ITEMINFO2_PUNC_PHRASE
+ (used later: PICODATA_ITEMINFO2_PUNC_PHRASE_FORCED)
+- CMDtype %d
+ PICODATA_ITEMINFO1_CMD_FLUSH (no args)
+ ? PICODATA_ITEMINFO1_CMD_PLAY ? (not yet)
+- CMDsubtype %d
+ PICODATA_ITEMINFO2_NA
+ ? PICODATA_ITEMINFO2_CMD_PLAY_G2P ? (not yet)
+- graph, len>0, utf8 graphemes, %s
+- string, len>0, can be any string with printable ascii characters, %s
+
+
+other limitations
+=================
+
+- item size: header plus len=256 (valid for Pico in general)
+ */
+
+
+#ifndef PICOTOK_H_
+#define PICOTOK_H_
+
+#include "picoos.h"
+#include "picodata.h"
+#include "picorsrc.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+
+picodata_ProcessingUnit picotok_newTokenizeUnit(
+ picoos_MemoryManager mm,
+ picoos_Common common,
+ picodata_CharBuffer cbIn,
+ picodata_CharBuffer cbOut,
+ picorsrc_Voice voice);
+
+#define PICOTOK_OUTBUF_SIZE 256
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif /*PICOTOK_H_*/
diff --git a/lib/picotrns.c b/lib/picotrns.c
new file mode 100644
index 0000000..2d16340
--- /dev/null
+++ b/lib/picotrns.c
@@ -0,0 +1,745 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picotrns.c
+ *
+ * fst processing
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#include "picoos.h"
+#include "picodbg.h"
+/* #include "picodata.h" */
+/* #include "picoknow.h" */
+#include "picoktab.h"
+#include "picokfst.h"
+#include "picotrns.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+
+picoos_uint8 picotrns_unplane(picoos_int16 symIn, picoos_uint8 * plane) {
+ if (symIn < 0) {
+ (*plane) = 0;
+ return (picoos_uint8) symIn;
+ } else {
+ (*plane) = symIn >> 8;
+ return (picoos_uint8) (symIn & 0xFF);
+ }
+}
+
+#if defined(PICO_DEBUG)
+
+void PICOTRNS_PRINTSYM1(picoknow_KnowledgeBase kbdbg, picoos_int16 insym, picoos_uint8 phonemic)
+{
+#include "picokdbg.h"
+ picoos_int16 sym;
+ picoos_uint8 plane;
+ picokdbg_Dbg dbg = (NULL == kbdbg) ? NULL : picokdbg_getDbg(kbdbg);
+ sym = picotrns_unplane(insym, &plane);
+ switch (plane) {
+ case PICOKFST_PLANE_PHONEMES: /* phones */
+ if ((NULL == dbg) || !phonemic) {
+ PICODBG_INFO_MSG((" %c", sym));
+ } else {
+ PICODBG_INFO_MSG((" %s", picokdbg_getPhoneSym(dbg, (picoos_uint8) sym)));
+ }
+ break;
+ case PICOKFST_PLANE_ACCENTS: /* accents */
+ PICODBG_INFO_MSG((" {A%c}", sym));
+ break;
+ case PICOKFST_PLANE_XSAMPA: /* xsampa symbols */
+ PICODBG_INFO_MSG((" {XS:(%i)}", sym));
+ break;
+ case PICOKFST_PLANE_POS: /* part of speech */
+ PICODBG_INFO_MSG((" {P:%d}", sym));
+ break;
+ case PICOKFST_PLANE_PB_STRENGTHS: /* phrases */
+ if (sym == 48) {
+ PICODBG_INFO_MSG((" {WB}", sym));
+ } else if (sym == 115) {
+ PICODBG_INFO_MSG((" {P0}", sym));
+ } else {
+ PICODBG_INFO_MSG((" {P%c}", sym));
+ }
+ break;
+ case PICOKFST_PLANE_INTERN: /* intern */
+ PICODBG_INFO_MSG((" [%c]", sym));
+ break;
+ }
+}
+
+void PICOTRNS_PRINTSYM(picoknow_KnowledgeBase kbdbg, picoos_int16 insym)
+{
+ PICOTRNS_PRINTSYM1(kbdbg,insym,1);
+}
+
+void PICOTRNS_PRINTSYMSEQ1(picoknow_KnowledgeBase kbdbg, const picotrns_possym_t seq[], const picoos_uint16 seqLen,
+ picoos_uint8 phonemic) {
+ picoos_uint16 i;
+ for (i=0; i<seqLen; i++) {
+ PICOTRNS_PRINTSYM1(kbdbg, seq[i].sym, phonemic);
+ }
+}
+
+void PICOTRNS_PRINTSYMSEQ(picoknow_KnowledgeBase kbdbg, const picotrns_possym_t seq[], const picoos_uint16 seqLen) {
+ PICOTRNS_PRINTSYMSEQ1(kbdbg,seq, seqLen, 1);
+}
+
+void picotrns_printSolution(const picotrns_possym_t outSeq[], const picoos_uint16 outSeqLen)
+{
+ PICODBG_INFO_CTX();
+ PICODBG_INFO_MSG(("solution: "));
+ PICOTRNS_PRINTSYMSEQ(NULL, outSeq, outSeqLen);
+ PICODBG_INFO_MSG(("\n"));
+}
+
+void picotrns_printSolutionAscii(const picotrns_possym_t outSeq[], const picoos_uint16 outSeqLen)
+{
+ PICODBG_INFO_CTX();
+ PICODBG_INFO_MSG(("solution: "));
+ PICOTRNS_PRINTSYMSEQ1(NULL, outSeq, outSeqLen,0);
+ PICODBG_INFO_MSG(("\n"));
+}
+
+#endif
+
+
+
+
+/* * +CT+ ***/
+struct picotrns_transductionState {
+ picoos_uint16 phase; /* transduction phase:
+ 0 = before start
+ 1 = before regular recursion step
+ 2 = before finish
+ 3 = after finish */
+ picoos_uint32 nrSol; /* nr of solutions so far */
+ picoos_int16 recPos; /* recursion position; must be signed! */
+};
+
+typedef struct picotrns_altDesc {
+ picokfst_state_t startFSTState; /**< starting FST state in current recursion position */
+ picoos_int32 inPos; /**< corresponding position in input string */
+ picokfst_state_t altState; /**< state of alternatives search;
+ - 0 = before pair search
+ - 1 = search state is a valid pair search state
+ - 2 = before inEps search
+ - 3 = search state is a valid inEps trans search state
+ - 4 = no more alternatives */
+ picoos_int32 searchState; /**< pair search state or inEps trans search state */
+ picokfst_symid_t altOutSym; /**< current output symbol at this recursion position */
+ picoos_int32 altOutRefPos; /**< output reference position at this recursion position */
+} picotrns_altDesc_t;
+
+
+picotrns_AltDesc picotrns_allocate_alt_desc_buf(picoos_MemoryManager mm, picoos_uint32 maxByteSize, picoos_uint16 * numAltDescs)
+{
+ picotrns_AltDesc buf;
+ (*numAltDescs) = (picoos_uint32) (maxByteSize / sizeof(picotrns_altDesc_t));
+ buf = (picotrns_AltDesc) picoos_allocate(mm, (*numAltDescs) * sizeof(picotrns_altDesc_t));
+ if (NULL == buf) {
+ (*numAltDescs) = 0;
+ return NULL;
+ } else {
+ return buf;
+ }
+}
+
+ void picotrns_deallocate_alt_desc_buf(picoos_MemoryManager mm, picotrns_AltDesc * altDescBuf)
+{
+ picoos_deallocate(mm, (void *) altDescBuf);
+}
+
+/* copy elements from inSeq to outSeq, ignoring elements with epsilon symbol */
+pico_status_t picotrns_eliminate_epsilons(const picotrns_possym_t inSeq[], picoos_uint16 inSeqLen,
+ picotrns_possym_t outSeq[], picoos_uint16 * outSeqLen, picoos_uint16 maxOutSeqLen)
+{
+ picoos_uint16 i, j = 0;
+
+ for (i=0; i < inSeqLen; i++) {
+ /* it is assumed that PICOKFST_SYMID_EPS is a hardwired value and not shifted */
+ if (PICOKFST_SYMID_EPS != inSeq[i].sym) {
+ if (j < maxOutSeqLen) {
+ outSeq[j].pos = inSeq[i].pos;
+ outSeq[j].sym = inSeq[i].sym;
+ j++;
+ }
+ }
+ *outSeqLen = j;
+ }
+ return PICO_OK;
+}
+
+
+static void insertSym(picotrns_possym_t inSeq[], picoos_uint16 pos, picoos_int16 sym) {
+ inSeq[pos].sym = sym;
+ inSeq[pos].pos = PICOTRNS_POS_INSERT;
+}
+
+/* copy elements from inSeq to outSeq, inserting syllable separators in some trivial way.
+ * inSeq is assumed to be at most PICOTRNS_MAX_NUM_POSSYM, outSeq at least of size PICOTRNS_MAX_NUM_POSSYM */
+pico_status_t picotrns_trivial_syllabify(picoktab_Phones phones,
+ const picotrns_possym_t inSeq[], const picoos_uint16 inSeqLen,
+ picotrns_possym_t outSeq[], picoos_uint16 * outSeqLen, picoos_uint16 maxOutSeqLen)
+{
+ picoos_uint16 i = 0, j = 0, out = 0, numInserted = 0;
+ picoos_uint8 vowelFound = FALSE;
+ picoos_uint16 accentpos = 0;
+ picoos_int16 accent = 0;
+
+ PICODBG_TRACE(("start"));
+
+
+ while (i < inSeqLen) {
+ /* make sure that at least one more sylSep can be inserted */
+ if (inSeqLen+numInserted+1 >= maxOutSeqLen) {
+ return PICO_EXC_BUF_OVERFLOW;
+ }
+ /* let j skip consonant cluster */
+ accent = 0;
+ accentpos = 0;
+ while ((j < inSeqLen) && !picoktab_isSyllCarrier(phones,(picoos_uint8)inSeq[j].sym)) {
+ if ((inSeq[j].sym == picoktab_getPrimstressID(phones))
+ || (inSeq[j].sym == picoktab_getPrimstressID(phones))) {
+ PICODBG_TRACE(("j skipping stress symbol inSeq[%i].sym = %c", j, inSeq[j].sym));
+ accent = inSeq[j].sym;
+ accentpos = j;
+ } else {
+ PICODBG_TRACE(("j skipping consonant inSeq[%i].sym = %c", j, inSeq[j].sym));
+ }
+ j++;
+ }
+ if (j < inSeqLen) { /* j is at the start of a new vowel */
+ /* copy consonant cluster (moving i) to output, insert syll separator if between vowels */
+ while (i < j-1) {
+ if ((accent > 0) && (i == accentpos)) {
+ PICODBG_TRACE(("skipping inSeq[%i].sym = %c (stress)", i, inSeq[i].sym));
+ i++;
+ } else {
+ PICODBG_TRACE(("copying inSeq[%i].sym = %c (consonant) into output buffer", i, inSeq[i].sym));
+ outSeq[out++] = inSeq[i++];
+ }
+ }
+ if (vowelFound) { /* we're between vowels */
+ PICODBG_TRACE(("inserting syllable separator into output buffer"));
+ insertSym(outSeq,out++,picoktab_getSyllboundID(phones));
+ if (accent > 0) {
+ insertSym(outSeq,out++,accent);
+ }
+ numInserted++;
+ }
+ if ((accent > 0) && (i == accentpos)) {
+ PICODBG_TRACE(("skipping inSeq[%i].sym = %c (stress)", i, inSeq[i].sym));
+ i++;
+ } else {
+ PICODBG_TRACE(("copying inSeq[%i].sym = %c (consonant) into output buffer", i, inSeq[i].sym));
+ outSeq[out++] = inSeq[i++];
+ }
+ vowelFound = TRUE;
+ /* now copy vowel cluster */
+ while ((i < inSeqLen) && picoktab_isSyllCarrier(phones,(picoos_uint8)inSeq[i].sym)) {
+ PICODBG_TRACE(("copying inSeq[%i].sym = %c (vowel) into output buffer", i, inSeq[i].sym));
+ outSeq[out++] = inSeq[i++];
+ }
+ j = i;
+ } else { /* j is at end of word or end of input */
+ while (i < j) {
+ PICODBG_TRACE(("copying inSeq[%i].sym = %c (consonant or stress) into output buffer", i, inSeq[i].sym));
+ outSeq[out++] = inSeq[i++];
+ }
+ }
+ *outSeqLen = out;
+ }
+ PICODBG_ASSERT((out == inSeqLen + numInserted));
+
+ return PICO_OK;
+}
+
+
+/* ******** +CT+: full transduction procedure **********/
+
+
+/* Gets next acceptable alternative for output symbol '*outSym' at current recursion position
+ starting from previous alternative in 'altDesc'; possibly uses input symbol
+ given by 'inSeq'/'inSeq'; returns whether alterative was found in '*found';
+ if '*found', the other output values ('*outRefPos', '*endFSTstate', '*nextInPos'*)
+ return the characteristics for next recursion step;
+ if '*found' is false, the output values are undefined. */
+
+static void GetNextAlternative (picokfst_FST fst, picotrns_AltDesc altDesc,
+ const picotrns_possym_t inSeq[], picoos_uint16 inSeqLen,
+ picokfst_symid_t * outSym, picoos_int32 * outRefPos,
+ picokfst_state_t * endFSTState, picoos_int32 * nextInPos, picoos_bool * found)
+{
+
+ picoos_bool inSymFound;
+ picoos_bool pairFound;
+ picokfst_class_t pairClass;
+ picoos_bool inEpsTransFound;
+ picokfst_symid_t inSym;
+
+ (*found) = 0;
+ do {
+ switch (altDesc->altState) {
+ case 0: /* before pair search */
+ if (altDesc->inPos < inSeqLen) {
+ inSym = inSeq[altDesc->inPos].sym;
+ if (inSym == PICOKFST_SYMID_EPS) {
+ /* very special case: input epsilon simply produces eps in output
+ without fst state change */
+ (*found) = 1;
+ (*outSym) = PICOKFST_SYMID_EPS;
+ (*outRefPos) = inSeq[altDesc->inPos].pos;
+ (*endFSTState) = altDesc->startFSTState;
+ (*nextInPos) = altDesc->inPos + 1;
+ altDesc->altState = 2;
+ } else {
+ /* start search for alternatives using input symbol */
+ picokfst_kfstStartPairSearch(fst,inSeq[altDesc->inPos].sym,& inSymFound,& altDesc->searchState);
+ if (!inSymFound) {
+ altDesc->altState = 2;
+ PICODBG_INFO_CTX();
+ PICODBG_INFO_MSG((" didnt find symbol "));
+ PICOTRNS_PRINTSYM(NULL, inSeq[altDesc->inPos].sym);
+ PICODBG_INFO_MSG(("\n"));
+
+ } else {
+ altDesc->altState = 1;
+ }
+ }
+ } else {
+ altDesc->altState = 2;
+ }
+ break;
+ case 1: /* within pair search */
+ picokfst_kfstGetNextPair(fst,& altDesc->searchState,& pairFound,& (*outSym),& pairClass);
+ if (pairFound) {
+ picokfst_kfstGetTrans(fst,altDesc->startFSTState,pairClass,& (*endFSTState));
+ if ((*endFSTState) > 0) {
+ (*found) = 1;
+ (*outRefPos) = inSeq[altDesc->inPos].pos;
+ (*nextInPos) = altDesc->inPos + 1;
+ }
+ } else {
+ /* no more pair found */
+ altDesc->altState = 2;
+ }
+ break;
+ case 2: /* before inEps trans search */
+ picokfst_kfstStartInEpsTransSearch(fst,altDesc->startFSTState,& inEpsTransFound,& altDesc->searchState);
+ if (inEpsTransFound) {
+ altDesc->altState = 3;
+ } else {
+ altDesc->altState = 4;
+ }
+ break;
+ case 3: /* within inEps trans search */
+ picokfst_kfstGetNextInEpsTrans(fst,& altDesc->searchState,& inEpsTransFound,& (*outSym),& (*endFSTState));
+ if (inEpsTransFound) {
+ (*found) = 1;
+ (*outRefPos) = PICOTRNS_POS_INSERT;
+ (*nextInPos) = altDesc->inPos;
+ } else {
+ altDesc->altState = 4;
+ }
+ break;
+ case 4: /* no more alternatives */
+ break;
+ }
+ } while (! ((*found) || (altDesc->altState == 4)) ); /* i.e., until (*found) || (altState == 4) */
+}
+
+
+
+/* Transfers current alternatives path stored in 'altDesc' with current path length 'pathLen'
+ into 'outSeq'/'outSeqLen'. The number of solutions is incremented. */
+
+static void NoteSolution (picoos_uint32 * nrSol, picotrns_printSolutionFct printSolution,
+ picotrns_altDesc_t altDesc[], picoos_uint16 pathLen,
+ picotrns_possym_t outSeq[], picoos_uint16 * outSeqLen, picoos_uint16 maxOutSeqLen)
+{
+ register picotrns_AltDesc ap;
+ picoos_uint32 i;
+
+ (*nrSol)++;
+ (*outSeqLen) = 0;
+ for (i = 0; i < pathLen; i++) {
+ if (i < maxOutSeqLen) {
+ ap = &altDesc[i];
+ outSeq[i].sym = ap->altOutSym;
+ outSeq[i].pos = ap->altOutRefPos;
+ (*outSeqLen)++;
+ }
+ }
+ if (pathLen > maxOutSeqLen) {
+ PICODBG_WARN(("**** output symbol array too small to hold full solution\n"));
+ }
+ if (printSolution != NULL) {
+ printSolution(outSeq,(*outSeqLen));
+ }
+}
+
+
+
+/* *
+ general scheme to get all solutions ("position" refers to abstract backtracking recursion depth,
+ which in the current solution is equal to the output symbol position):
+
+ "set position to first position";
+ "initialize alternatives in first position";
+ REPEAT
+ IF "current state in current position is a solution" THEN
+ "note solution";
+ END;
+ "get first or next acceptable alternative in current position";
+ IF "acceptable alternative found" THEN
+ "note alternative";
+ "go to next position";
+ "initialize alternatives in that position";
+ ELSE
+ "step back to previous position";
+ END;
+ UNTIL "current position is before first position"
+***/
+
+
+/* Initializes transduction state for further use in repeated application
+ of 'TransductionStep'. */
+
+static void StartTransduction (struct picotrns_transductionState * transductionState)
+{
+ (*transductionState).phase = 0;
+}
+
+
+
+/* Performs one step in the transduction of 'inSeqLen' input symbols with corresponding
+ reference positions in 'inSeq'. '*transductionState' must have been
+ initialized by 'StartTransduction'. Repeat calls to this procedure until '*finished' returns true.
+ The output is returned in 'outSeqLen' symbols and reference positions in 'outSeq'.
+ The output reference positions refer to the corresponding input reference positions.
+ Inserted output symbols receive the reference position -1. If several solutions are possible,
+ only the last found solution is returned.
+ 'altDesc' is a temporary workspace which should be at least one cell longer than 'outSeq'.
+ 'firstSolOnly' determines whether only the first solution should be found or if
+ the search should go on to find all solutions (mainly for testing purposes).
+
+ NOTE: current version written for use in single repetitive steps;
+ could be simplified if full transduction can be done as an atomic operation */
+
+static void TransductionStep (picokfst_FST fst, struct picotrns_transductionState * transductionState,
+ picotrns_altDesc_t altDesc[], picoos_uint16 maxAltDescLen,
+ picoos_bool firstSolOnly, picotrns_printSolutionFct printSolution,
+ const picotrns_possym_t inSeq[], picoos_uint16 inSeqLen,
+ picotrns_possym_t outSeq[], picoos_uint16 * outSeqLen, picoos_uint16 maxOutSeqLen,
+ picoos_bool * finished)
+{
+ register picotrns_AltDesc ap;
+ picoos_int32 i;
+ picokfst_state_t endFSTState;
+ picoos_int32 nextInPos;
+ picoos_bool found;
+ picokfst_symid_t outSym;
+ picoos_int32 outRefPos;
+ picoos_int32 tmpRecPos;
+
+ (*finished) = 0;
+ tmpRecPos = (*transductionState).recPos;
+ switch ((*transductionState).phase) {
+ case 0: /* before initialization */
+ (*transductionState).nrSol = 0;
+
+ /* check for initial solution (empty strings are always accepted) */
+ if (inSeqLen == 0) {
+ NoteSolution(& (*transductionState).nrSol,printSolution,altDesc,0,outSeq,outSeqLen,maxOutSeqLen);
+ }
+
+ /* initialize first recursion position */
+ tmpRecPos = 0;
+ ap = & altDesc[0];
+ ap->startFSTState = 1;
+ ap->inPos = 0;
+ ap->altState = 0;
+ (*transductionState).phase = 1;
+ break;
+
+ case 1: /* before regular recursion step */
+ if ((tmpRecPos < 0) || (firstSolOnly && ((*transductionState).nrSol > 0))) {
+ /* end reached */
+ (*transductionState).phase = 2;
+ } else {
+ /* not finished; do regular step */
+
+ /* get first or next acceptable alternative in current position */
+ GetNextAlternative(fst,& altDesc[tmpRecPos],inSeq,inSeqLen,& outSym,& outRefPos,& endFSTState,& nextInPos,& found);
+ if (found) {
+ /* note alternative in current position */
+ ap = & altDesc[tmpRecPos];
+ ap->altOutSym = outSym;
+ ap->altOutRefPos = outRefPos;
+
+ /* check for solution after found alternative */
+ if ((nextInPos == inSeqLen) && picokfst_kfstIsAcceptingState(fst,endFSTState)) {
+ NoteSolution(& (*transductionState).nrSol,printSolution,altDesc,tmpRecPos+1,
+ outSeq,outSeqLen,maxOutSeqLen);
+ }
+
+ /* go to next position if possible, start search for follower alternative symbols */
+ if (tmpRecPos < maxAltDescLen-1) {
+ /* got to next position */
+ tmpRecPos = tmpRecPos + 1;
+
+ /* initialize alternatives in new position */
+ ap = & altDesc[tmpRecPos];
+ ap->startFSTState = endFSTState;
+ ap->inPos = nextInPos;
+ ap->altState = 0;
+
+ } else {
+ /* do not go on due to limited path but still treat alternatives in current position */
+ PICODBG_WARN(("--- transduction path too long; may fail to find solution\n"));
+ }
+ } else { /* no more acceptable alternative found in current position */
+ /* backtrack to previous recursion */
+ tmpRecPos = tmpRecPos - 1;
+ }
+ }
+ break;
+
+ case 2: /* before finish */
+ if ((*transductionState).nrSol == 0) {
+ PICODBG_WARN(("--- no transduction solution found, using input as output\n"));
+ i = 0;
+ while ((i < inSeqLen) && (i < maxOutSeqLen)) {
+ outSeq[i].sym = inSeq[i].sym;
+ outSeq[i].pos = inSeq[i].pos;
+ i++;
+ }
+ (*outSeqLen) = i;
+ } else if ((*transductionState).nrSol > 1) {
+ PICODBG_WARN(("--- more than one transducer solutions found\n"));
+ }
+ (*transductionState).phase = 3;
+ break;
+
+ case 3: /* after finish */
+ (*finished) = 1;
+ break;
+ }
+ (*transductionState).recPos = tmpRecPos;
+}
+
+
+
+/* see description in header */
+pico_status_t picotrns_transduce (picokfst_FST fst, picoos_bool firstSolOnly,
+ picotrns_printSolutionFct printSolution,
+ const picotrns_possym_t inSeq[], picoos_uint16 inSeqLen,
+ picotrns_possym_t outSeq[], picoos_uint16 * outSeqLen, picoos_uint16 maxOutSeqLen,
+ picotrns_AltDesc altDescBuf, picoos_uint16 maxAltDescLen,
+ picoos_uint32 *nrSteps)
+{
+ struct picotrns_transductionState transductionState;
+ picoos_bool finished;
+
+#if defined(PICO_DEBUG)
+ {
+ picoos_uint16 i;
+
+ PICODBG_INFO_CTX();
+ PICODBG_INFO_MSG(("got input: "));
+ for (i=0; i<inSeqLen; i++) {
+ PICODBG_INFO_MSG((" %d", inSeq[i].sym));
+ }
+ PICODBG_INFO_MSG((" ("));
+ PICOTRNS_PRINTSYMSEQ(NULL,inSeq,inSeqLen);
+ PICODBG_INFO_MSG((")\n"));
+ }
+#endif
+ StartTransduction(&transductionState);
+ finished = 0;
+ *nrSteps = 0;
+ while (!finished) {
+ TransductionStep(fst,&transductionState,altDescBuf,maxAltDescLen,firstSolOnly,printSolution,
+ inSeq,inSeqLen,outSeq,outSeqLen,maxOutSeqLen,&finished);
+ (*nrSteps)++;
+ }
+
+ return PICO_OK;
+}
+
+
+/**
+ * Data structure for picotrns_SimpleTransducer object.
+ */
+typedef struct picotrns_simple_transducer {
+ picoos_Common common;
+ picotrns_possym_t possymBufA[PICOTRNS_MAX_NUM_POSSYM+1];
+ picotrns_possym_t possymBufB[PICOTRNS_MAX_NUM_POSSYM+1];
+ picotrns_possym_t * possymBuf; /**< the buffer of the pos/sym pairs */
+ picotrns_possym_t * possymBufTmp;
+ picoos_uint16 possymReadPos, possymWritePos; /* next pos to read from phonBufIn, next pos to write to phonBufIn */
+
+ /* buffer for internal calculation of transducer */
+ picotrns_AltDesc altDescBuf;
+ /* the number of AltDesc in the buffer */
+ picoos_uint16 maxAltDescLen;
+} picotrns_simple_transducer_t;
+
+
+pico_status_t picotrns_stInitialize(picotrns_SimpleTransducer transducer)
+{
+ transducer->possymBuf = transducer->possymBufA;
+ transducer->possymBufTmp = transducer->possymBufB;
+ transducer->possymReadPos = 0;
+ transducer->possymWritePos = 0;
+ return PICO_OK;
+}
+/** creates a SimpleTranducer with a working buffer of given size
+ *
+ * @param mm MemoryManager handle
+ * @param common Common handle
+ * @param maxAltDescLen maximal size for working buffer (in bytes)
+ * @return handle to new SimpleTransducer or NULL if error
+ */
+picotrns_SimpleTransducer picotrns_newSimpleTransducer(picoos_MemoryManager mm,
+ picoos_Common common,
+ picoos_uint16 maxAltDescLen)
+{
+ picotrns_SimpleTransducer this;
+ this = picoos_allocate(mm, sizeof(picotrns_simple_transducer_t));
+ if (this == NULL) {
+ picoos_deallocate(mm, (void *)&this);
+ picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM, NULL, NULL);
+ return NULL;
+ }
+
+ /* allocate working buffer */
+ this->altDescBuf = picotrns_allocate_alt_desc_buf(mm, maxAltDescLen, &this->maxAltDescLen);
+ if (this->altDescBuf == NULL) {
+ picoos_deallocate(mm, (void *)&this);
+ picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM, NULL, NULL);
+ return NULL;
+ }
+ this->common = common;
+ picotrns_stInitialize(this);
+ return this;
+}
+/** disposes a SimpleTransducer
+ *
+ * @param this
+ * @param mm
+ * @return PICO_OK
+ */
+pico_status_t picotrns_disposeSimpleTransducer(picotrns_SimpleTransducer * this,
+ picoos_MemoryManager mm)
+{
+ if (NULL != (*this)) {
+ picotrns_deallocate_alt_desc_buf(mm,&(*this)->altDescBuf);
+ picoos_deallocate(mm, (void *) this);
+ (*this) = NULL;
+ }
+ return PICO_OK;
+}
+
+/** transduces the contents previously inserted via @ref picotrns_newSimpleTransducer and @ref
+ * picotrns_disposeSimpleTransducer.
+ *
+ * @param this
+ * @param fst
+ * @return
+ */
+pico_status_t picotrns_stTransduce(picotrns_SimpleTransducer this, picokfst_FST fst)
+{
+ picoos_uint16 outSeqLen;
+ picoos_uint32 nrSteps;
+ pico_status_t status;
+
+ status = picotrns_transduce(fst,TRUE,NULL,
+ this->possymBuf, this->possymWritePos,
+ this->possymBufTmp,&outSeqLen, PICOTRNS_MAX_NUM_POSSYM,
+ this->altDescBuf,this->maxAltDescLen,&nrSteps);
+ if (PICO_OK != status) {
+ return status;
+ }
+ return picotrns_eliminate_epsilons(this->possymBufTmp,outSeqLen,this->possymBuf,&this->possymWritePos,PICOTRNS_MAX_NUM_POSSYM);
+}
+
+/**
+ * Add chars from NULLC-terminated string \c inStr, shifted to plane \c plane, to internal input buffer of
+ * \c transducer.
+ *
+ * @param this is an initialized picotrns_SimpleTransducer
+ * @param inStr NULLC-terminated byte sequence
+ * @param plane
+ * @return PICO_OK, if all bytes fit into buffer, or PICO_EXC_BUF_OVERFLOW otherwise
+ */
+pico_status_t picotrns_stAddWithPlane(picotrns_SimpleTransducer this, picoos_char * inStr, picoos_uint8 plane)
+{
+ while ((*inStr) && (this->possymWritePos < PICOTRNS_MAX_NUM_POSSYM)) {
+ this->possymBuf[this->possymWritePos].pos = PICOTRNS_POS_INSERT;
+ this->possymBuf[this->possymWritePos].sym = (plane << 8) + (*inStr);
+ PICODBG_DEBUG(("inserting pos/sym = %i/'%c' at pos %i",
+ this->possymBuf[this->possymWritePos].pos,
+ this->possymBuf[this->possymWritePos].sym,
+ this->possymWritePos));
+ this->possymWritePos++;
+ inStr++;
+ }
+ if (!(*inStr)) {
+ return PICO_OK;
+ } else {
+ return PICO_EXC_BUF_OVERFLOW;
+ }
+}
+
+pico_status_t picotrns_stGetSymSequence(
+ picotrns_SimpleTransducer this,
+ picoos_uint8 * outputSymIds,
+ picoos_uint32 maxOutputSymIds)
+{
+ picoos_uint8 plane;
+ picoos_uint32 outputCount = 0;
+ while ((this->possymReadPos < this->possymWritePos) && (outputCount < maxOutputSymIds)) {
+ *outputSymIds++ = picotrns_unplane(this->possymBuf[this->possymReadPos++].sym, &plane);
+ outputCount++;
+ }
+ *outputSymIds = NULLC;
+ if (outputCount <= maxOutputSymIds) {
+ return PICO_OK;
+ } else {
+ return PICO_EXC_BUF_OVERFLOW;
+ }
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+/* end picotrns.c */
diff --git a/lib/picotrns.h b/lib/picotrns.h
new file mode 100644
index 0000000..f535ce0
--- /dev/null
+++ b/lib/picotrns.h
@@ -0,0 +1,191 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picotrns.h
+ *
+ * fst processing
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+/** @addtogroup picotrns
+ *
+ * Conventions:
+ *
+ * - The input to the transducer is a list of pos/sym pairs, where pos are arbitrary position markers
+ * - All positions are allowed on input (in particular all those coming as an output of a previous transduction)
+ * - A phone sequence to be transduced has to begin with PICOKNOW_PHON_START_ID and end with PICOKNOW_PHON_TERM_ID
+ * These special symbols are kept in the transduction output (as first and last symbol)
+ * - Symbols inserted by the transduction process allways get their position marker pos=PICOTRNS_POS_INSERT
+ * - The order of positions on output must be the same as that on input, i.e. apart from inserted pairs, the
+ * output position sequence must be a sub-sequence of the input position sequence.
+ * - Inserted symbols are allways preceded by a positioned pos/sym pair, e.g.
+ * if the sequence pos1/sym1, pos2/sym2 should be tranduced to x/sym3, y/sym4, z/sym5, then x must be pos1 or pos2
+ * and not PICOTRNS_POS_INSERT
+ *
+ * For lingware developers: Insertions are always interpreted "to the right"
+ * - E.g.: The original sequence is phon1 , command , phon2
+ * - The input to the transducer is then pos1/phon1 , pos2/phon2
+ * - The output is pos1/phon1' -1/phon_ins pos2/phon2' [assuming -1 is the special insertion pos]
+ * - Then the new sequence will be recomposed as phon1' , phon_ins , command , phon2' [note position of command!]
+ * - To overwrite this behaviour, rules must be formulated such that the transduction output is
+ * pos1/phon1' pos2/phon_ins -1/phon2'
+ */
+#ifndef PICOTRNS_H_
+#define PICOTRNS_H_
+
+#include "picoos.h"
+#include "picokfst.h"
+#include "picoktab.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+#define PICOTRNS_MAX_NUM_POSSYM 255
+
+#define PICOTRNS_POS_INSERT (picoos_int16) -1 /* position returned by transducer to mark symbols inserted by the transducer */
+#define PICOTRNS_POS_INVALID (picoos_int16) -2 /* value to mark an invalid (e.g. uninitiated) position */
+#define PICOTRNS_POS_IGNORE (picoos_int16) -3 /* value to mark a pos/sym pair to be ignored (e.g. start/term symbols only used by the transducer) */
+
+
+typedef struct picotrns_possym {
+ picoos_int16 pos;
+ picoos_int16 sym;
+} picotrns_possym_t;
+
+picoos_uint8 picotrns_unplane(picoos_int16 symIn, picoos_uint8 * plane);
+
+
+#if defined(PICO_DEBUG)
+
+void PICOTRNS_PRINTSYM(picoknow_KnowledgeBase kbdbg, picoos_int16 insym);
+
+void PICOTRNS_PRINTSYMSEQ(picoknow_KnowledgeBase kbdbg, const picotrns_possym_t seq[], const picoos_uint16 seqLen);
+
+void picotrns_printSolution(const picotrns_possym_t outSeq[], const picoos_uint16 outSeqLen);
+
+#else
+#define PICOTRNS_PRINTSYM(x,y)
+#define PICOTRNS_PRINTSYMSEQ(x,y,z)
+#define picotrns_printSolution NULL
+#endif
+
+
+typedef struct picotrns_altDesc * picotrns_AltDesc;
+
+
+picotrns_AltDesc picotrns_allocate_alt_desc_buf(picoos_MemoryManager mm, picoos_uint32 maxByteSize, picoos_uint16 * numAltDescs);
+
+void picotrns_deallocate_alt_desc_buf(picoos_MemoryManager mm, picotrns_AltDesc * altDescBuf);
+
+
+/* type of function for printing transduction solutions;
+ only for testing purposes in transduction mode where all solutions
+ are produced */
+typedef void picotrns_printSolutionFct(const picotrns_possym_t outSeq[], const picoos_uint16 outSeqLen);
+
+
+
+/** overall transduction; transduces 'inSeq' with 'inSeqLen' elements
+ to '*outSeqLen' elements in 'outSeq';
+ *
+ * @param fst the finite-state transducer used for transduction
+ * @param firstSolOnly determines whether only the first solution (usually)
+ or all solutions should be produced (for testing); only the last found
+ solution is returned in 'outSeq';
+ * @param printSolution if not NULL, every found solution is displayed using
+ the given function
+ * @param inSeq the input sequence
+ * @param inSeqLen the input sequence length
+ * @retval outSeq the output sequence
+ * @retval outSeqLen the output sequence length
+ * @param maxOutSeqLen must provide the maximum length of 'outSeq'
+ * @param altDescBuf must provide a working array of length 'maxAltDescLen'
+ * @param maxAltDescLen should be chosen at least 'maxOutSeqLen' + 1
+ * @retval nrSteps returns the overall internal number of iterative steps done
+ * @return status of the transduction: PICO_OK, if transduction successful
+ @note if 'outSeq' or 'altDesc' are too small to hold a solution,
+ an error occurs and the input is simply transfered to the output
+ (up to maximum possible length)
+ */
+extern pico_status_t picotrns_transduce (picokfst_FST fst, picoos_bool firstSolOnly,
+ picotrns_printSolutionFct printSolution,
+ const picotrns_possym_t inSeq[], picoos_uint16 inSeqLen,
+ picotrns_possym_t outSeq[], picoos_uint16 * outSeqLen, picoos_uint16 maxOutSeqLen,
+ picotrns_AltDesc altDescBuf, picoos_uint16 maxAltDescLen,
+ picoos_uint32 *nrSteps);
+
+
+
+/* transduce 'inSeq' into 'outSeq' 'inSeq' has to be terminated with the id for symbol '#'. 'outSeq' is terminated in the same way. */
+/*
+pico_status_t picotrns_transduce_sequence(picokfst_FST fst, const picotrns_possym_t inSeq[], picoos_uint16 inSeqLen,
+ picotrns_possym_t outSeq[], picoos_uint16 * outSeqLen);
+*/
+
+/* copy elements from inSeq to outSeq, ignoring elements with epsilon symbol */
+pico_status_t picotrns_eliminate_epsilons(const picotrns_possym_t inSeq[], picoos_uint16 inSeqLen,
+ picotrns_possym_t outSeq[], picoos_uint16 * outSeqLen, picoos_uint16 maxOutSeqLen);
+
+/* copy elements from inSeq to outSeq, inserting syllable separators in some trivial way.
+ * inSeq is assumed to be at most, outSeq at least of size PICOTRNS_MAX_NUM_POSSYM */
+pico_status_t picotrns_trivial_syllabify(picoktab_Phones phones,
+ const picotrns_possym_t inSeq[], const picoos_uint16 inSeqLen,
+ picotrns_possym_t outSeq[], picoos_uint16 * outSeqLen, picoos_uint16 maxOutSeqLen);
+
+
+/** object : SimpleTransducer
+ * shortcut : st
+ *
+ */
+typedef struct picotrns_simple_transducer * picotrns_SimpleTransducer;
+
+picotrns_SimpleTransducer picotrns_newSimpleTransducer(picoos_MemoryManager mm,
+ picoos_Common common,
+ picoos_uint16 maxAltDescLen);
+
+pico_status_t picotrns_disposeSimpleTransducer(picotrns_SimpleTransducer * this,
+ picoos_MemoryManager mm);
+
+pico_status_t picotrns_stInitialize(picotrns_SimpleTransducer transducer);
+
+pico_status_t picotrns_stAddWithPlane(picotrns_SimpleTransducer this, picoos_char * inStr, picoos_uint8 plane);
+
+pico_status_t picotrns_stTransduce(picotrns_SimpleTransducer this, picokfst_FST fst);
+
+pico_status_t picotrns_stGetSymSequence(
+ picotrns_SimpleTransducer this,
+ picoos_uint8 * outputSymIds,
+ picoos_uint32 maxOutputSymIds);
+
+
+
+
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /*PICOTRNS_H_*/
diff --git a/lib/picowa.c b/lib/picowa.c
new file mode 100644
index 0000000..3e793c9
--- /dev/null
+++ b/lib/picowa.c
@@ -0,0 +1,578 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picowa.c
+ *
+ * word analysis PU - lexicon lookup and POS prediction
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+#include "picoos.h"
+#include "picodbg.h"
+#include "picodata.h"
+#include "picowa.h"
+#include "picoklex.h"
+#include "picokdt.h"
+#include "picoktab.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+/* PU waStep states */
+#define WA_STEPSTATE_COLLECT 0
+#define WA_STEPSTATE_PROCESS 1
+#define WA_STEPSTATE_FEED 2
+
+
+/* subobject : WordAnaUnit
+ * shortcut : wa
+ * context size : one item
+ */
+typedef struct wa_subobj {
+ picoos_uint8 procState; /* for next processing step decision */
+
+ /* one item only */
+ picoos_uint8 inBuf[PICOWA_MAXITEMSIZE]; /* internal input buffer */
+ picoos_uint16 inBufSize; /* actually allocated size */
+ picoos_uint16 inLen; /* length of item in inBuf, 0 for empty buf */
+
+ picoos_uint8 outBuf[PICOWA_MAXITEMSIZE]; /* internal output buffer */
+ picoos_uint16 outBufSize; /* actually allocated size */
+ picoos_uint16 outLen; /* length of item in outBuf, 0 for empty buf */
+
+ /* lex knowledge base */
+ picoklex_Lex lex;
+
+ /* ulex knowledge bases */
+ picoos_uint8 numUlex;
+ picoklex_Lex ulex[PICOKNOW_MAX_NUM_ULEX];
+
+ /* tab knowledge base */
+ picoktab_Pos tabpos;
+
+ /* dtposp knowledge base */
+ picokdt_DtPosP dtposp;
+} wa_subobj_t;
+
+
+static pico_status_t waInitialize(register picodata_ProcessingUnit this) {
+ picoos_uint8 i;
+ picoklex_Lex ulex;
+ wa_subobj_t * wa;
+
+ picoknow_kb_id_t ulexKbIds[PICOKNOW_MAX_NUM_ULEX] = PICOKNOW_KBID_ULEX_ARRAY;
+
+ PICODBG_DEBUG(("calling"));
+
+ if (NULL == this || NULL == this->subObj) {
+ return (picodata_step_result_t) picoos_emRaiseException(this->common->em,
+ PICO_ERR_NULLPTR_ACCESS, NULL, NULL);
+ }
+ wa = (wa_subobj_t *) this->subObj;
+ wa->procState = WA_STEPSTATE_COLLECT;
+ wa->inBufSize = PICOWA_MAXITEMSIZE;
+ wa->inLen = 0;
+ wa->outBufSize = PICOWA_MAXITEMSIZE;
+ wa->outLen = 0;
+
+ /* kb lex */
+ wa->lex = picoklex_getLex(this->voice->kbArray[PICOKNOW_KBID_LEX_MAIN]);
+ if (wa->lex == NULL) {
+ return picoos_emRaiseException(this->common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ PICODBG_DEBUG(("got lex"));
+
+ /* kb ulex[] */
+ wa->numUlex = 0;
+ for (i = 0; i<PICOKNOW_MAX_NUM_ULEX; i++) {
+ ulex = picoklex_getLex(this->voice->kbArray[ulexKbIds[i]]);
+ if (NULL != ulex) {
+ wa->ulex[wa->numUlex++] = ulex;
+ }
+ }
+ PICODBG_DEBUG(("got %i user lexica", wa->numUlex));
+
+ /* kb tabpos */
+ wa->tabpos =
+ picoktab_getPos(this->voice->kbArray[PICOKNOW_KBID_TAB_POS]);
+ if (wa->tabpos == NULL) {
+ return picoos_emRaiseException(this->common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ PICODBG_DEBUG(("got tabpos"));
+
+ /* kb dtposp */
+ wa->dtposp = picokdt_getDtPosP(this->voice->kbArray[PICOKNOW_KBID_DT_POSP]);
+ if (wa->dtposp == NULL) {
+ return picoos_emRaiseException(this->common->em, PICO_EXC_KB_MISSING,
+ NULL, NULL);
+ }
+ PICODBG_DEBUG(("got dtposp"));
+ return PICO_OK;
+}
+
+static picodata_step_result_t waStep(register picodata_ProcessingUnit this,
+ picoos_int16 mode,
+ picoos_uint16 *numBytesOutput);
+
+static pico_status_t waTerminate(register picodata_ProcessingUnit this) {
+ return PICO_OK;
+}
+
+static pico_status_t waSubObjDeallocate(register picodata_ProcessingUnit this,
+ picoos_MemoryManager mm) {
+ if (NULL != this) {
+ picoos_deallocate(this->common->mm, (void *) &this->subObj);
+ }
+ mm = mm; /* avoid warning "var not used in this function"*/
+ return PICO_OK;
+}
+
+
+picodata_ProcessingUnit picowa_newWordAnaUnit(picoos_MemoryManager mm,
+ picoos_Common common,
+ picodata_CharBuffer cbIn,
+ picodata_CharBuffer cbOut,
+ picorsrc_Voice voice) {
+ picodata_ProcessingUnit this;
+
+ this = picodata_newProcessingUnit(mm, common, cbIn, cbOut, voice);
+ if (this == NULL) {
+ return NULL;
+ }
+
+ this->initialize = waInitialize;
+ PICODBG_DEBUG(("set this->step to waStep"));
+ this->step = waStep;
+ this->terminate = waTerminate;
+ this->subDeallocate = waSubObjDeallocate;
+ this->subObj = picoos_allocate(mm, sizeof(wa_subobj_t));
+ if (this->subObj == NULL) {
+ picoos_deallocate(mm, (void *)&this);
+ picoos_emRaiseException(common->em, PICO_EXC_OUT_OF_MEM, NULL, NULL);
+ return NULL;
+ }
+
+ waInitialize(this);
+ return this;
+}
+
+/* ***********************************************************************/
+/* WORDGRAPH proc functions */
+/* ***********************************************************************/
+
+static picoos_uint8 waClassifyPos(register picodata_ProcessingUnit this,
+ register wa_subobj_t *wa,
+ const picoos_uint8 *graph,
+ const picoos_uint16 graphlen) {
+ picokdt_classify_result_t dtres;
+ picoos_uint8 specchar;
+ picoos_uint16 i;
+
+ PICODBG_DEBUG(("graphlen %d", graphlen));
+
+ /* check existence of special char (e.g. hyphen) in graph:
+ for now, check existence of hard-coded ascii hyphen,
+ ie. preproc needs to match all UTF8 hyphens to the ascii
+ hyphen. */
+ /* @todo : consider specifying special char(s) in lingware. */
+ specchar = FALSE;
+ i = 0;
+ while ((i < graphlen) && (!specchar)) {
+ if (graph[i++] == '-') {
+ specchar = TRUE;
+ }
+ }
+
+ /* construct input vector, which is set in dtposp */
+ if (!picokdt_dtPosPconstructInVec(wa->dtposp, graph, graphlen, specchar)) {
+ /* error constructing invec */
+ PICODBG_WARN(("problem with invec"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_INVECTOR, NULL, NULL);
+ return PICODATA_ITEMINFO1_ERR;
+ }
+
+ /* classify */
+ if (!picokdt_dtPosPclassify(wa->dtposp)) {
+ /* error doing classification */
+ PICODBG_WARN(("problem classifying"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_CLASSIFICATION,
+ NULL, NULL);
+ return PICODATA_ITEMINFO1_ERR;
+ }
+
+ /* decompose */
+ if (!picokdt_dtPosPdecomposeOutClass(wa->dtposp, &dtres)) {
+ /* error decomposing */
+ PICODBG_WARN(("problem decomposing"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_OUTVECTOR,
+ NULL, NULL);
+ return PICODATA_ITEMINFO1_ERR;
+ }
+
+ if (dtres.set) {
+ PICODBG_DEBUG(("class %d", dtres.class));
+ return (picoos_uint8)dtres.class;
+ } else {
+ PICODBG_WARN(("result not set"));
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_CLASSIFICATION,
+ NULL, NULL);
+ return PICODATA_ITEMINFO1_ERR;
+ }
+}
+
+
+static pico_status_t waProcessWordgraph(register picodata_ProcessingUnit this,
+ register wa_subobj_t *wa /*inout*/,
+ picodata_itemhead_t *head /*inout*/,
+ const picoos_uint8 *content) {
+ pico_status_t status;
+ picoklex_lexl_result_t lexres;
+ picoos_uint8 posbuf[PICOKTAB_MAXNRPOS_IN_COMB];
+ picoos_uint8 i;
+ picoos_uint8 foundIndex;
+ picoos_bool found;
+
+
+ PICODBG_DEBUG(("type %c, len %d", head->type, head->len));
+
+ /* do lookup
+ if no entry found:
+ do POS prediction: -> WORDGRAPH(POSes,NA)graph
+ else:
+ if incl-phone:
+ N entries possible -> WORDINDEX(POSes,NA)POS1|ind1...POSN|indN
+ (N in {1,...,PICOKLEX_MAX_NRRES}, now up to 4)
+ else:
+ no phone, one entry -> WORDGRAPH(POS,NA)graph
+ */
+
+ found = FALSE;
+ i = 0;
+ while (!found && (i < wa->numUlex)) {
+ found = picoklex_lexLookup(wa->ulex[i], content, head->len, &lexres);
+ i++;
+ }
+ /* note that if found, i will be incremented nevertheless, so i >= 1 */
+ if (found) {
+ foundIndex = i;
+ } else {
+ foundIndex = 0;
+ }
+ if (!found && !picoklex_lexLookup(wa->lex, content, head->len, &lexres)) {
+ /* no lex entry found, WORDGRAPH(POS,NA)graph */
+ if (PICO_OK == picodata_copy_item(wa->inBuf, wa->inLen,
+ wa->outBuf, wa->outBufSize,
+ &wa->outLen)) {
+ wa->inLen = 0;
+ /* predict and modify pos in info1 */
+ if (PICO_OK != picodata_set_iteminfo1(wa->outBuf, wa->outLen,
+ waClassifyPos(this, wa, content, head->len))) {
+ return picoos_emRaiseException(this->common->em,
+ PICO_EXC_BUF_OVERFLOW,NULL,NULL);
+ }
+ }
+
+ } else { /* at least one entry found */
+ PICODBG_DEBUG(("at least one entry found in lexicon %i",foundIndex));
+ if (lexres.phonfound) { /* incl. ind-phone and possibly multi-ent. */
+ if (lexres.nrres > PICOKLEX_MAX_NRRES) {
+ /* not possible with system lexicon, needs to be
+ ensured for user lex too */
+ picoos_emRaiseWarning(this->common->em, PICO_WARN_FALLBACK,NULL,
+ (picoos_char *)"using %d lexicon lookup results",
+ PICOKLEX_MAX_NRRES);
+ lexres.nrres = PICOKLEX_MAX_NRRES;
+ }
+ head->type = PICODATA_ITEM_WORDINDEX;
+ if (lexres.nrres == 1) {
+ head->info1 = lexres.posind[0];
+ } else {
+ /* more than one result, POSgroup info needs to be
+ determined for later POS disambiguation */
+ for (i = 0; i < lexres.nrres; i++) {
+ posbuf[i] = lexres.posind[i * PICOKLEX_POSIND_SIZE];
+ }
+ head->info1 = picoktab_getPosGroup(wa->tabpos, posbuf,
+ lexres.nrres);
+ }
+ head->info2 = foundIndex;
+ head->len = lexres.posindlen;
+ if ((status = picodata_put_itemparts(head, lexres.posind,
+ lexres.posindlen,
+ wa->outBuf, wa->outBufSize,
+ &wa->outLen)) == PICO_OK) {
+ wa->inLen = 0;
+ } else {
+ return picoos_emRaiseException(this->common->em, status,
+ NULL, NULL);
+ }
+
+ } else { /* no phone, :G2P, one entry: WORDGRAPH(POS,NA)graph */
+ if (PICO_OK == picodata_copy_item(wa->inBuf, wa->inLen,
+ wa->outBuf, wa->outBufSize,
+ &wa->outLen)) {
+ wa->inLen = 0;
+ /* set lex pos in info1 */
+ if (PICO_OK != picodata_set_iteminfo1(wa->outBuf, wa->outLen,
+ lexres.posind[0])) {
+ return picoos_emRaiseException(this->common->em,
+ PICO_EXC_BUF_OVERFLOW,
+ NULL, NULL);
+ }
+ }
+ }
+ }
+ return PICO_OK;
+}
+
+
+/* ***********************************************************************/
+/* waStep function */
+/* ***********************************************************************/
+
+/*
+ collect into internal buffer, process, and then feed to output buffer
+
+ init state: COLLECT ext ext
+ state transitions: in IN OUTout
+ COLLECT | getOneItem ->-1 +1 0 0 | (ATOMIC) -> PROCESS (got item)
+ COLLECT | getOneItem -> 0 0 0 0 | IDLE (got no item)
+
+ PROCESS | procOneItem -> 0 -1 +1 0 | (ATOMIC) -> FEED (proc'ed item)
+ PROCESS | procOneItem -> 0 -1 0 0 | BUSY -> COLLECT (item skipped)
+
+ FEED | putOneItem -> 0 0 -1 +1 | BUSY -> COLLECT (put item)
+ FEED | putOneItem -> 0 0 1 0 | OUT_FULL (put no item)
+*/
+
+static picodata_step_result_t waStep(register picodata_ProcessingUnit this,
+ picoos_int16 mode,
+ picoos_uint16 * numBytesOutput) {
+ register wa_subobj_t *wa;
+ pico_status_t rv = PICO_OK;
+
+ if (NULL == this || NULL == this->subObj) {
+ return PICODATA_PU_ERROR;
+ }
+ wa = (wa_subobj_t *) this->subObj;
+ mode = mode; /* avoid warning "var not used in this function"*/
+ *numBytesOutput = 0;
+ while (1) { /* exit via return */
+ PICODBG_DEBUG(("doing state %i, inLen: %d, outLen: %d",
+ wa->procState, wa->inLen, wa->outLen));
+
+ switch (wa->procState) {
+ /* collect state: get item from charBuf and store in
+ * internal inBuf
+ */
+ case WA_STEPSTATE_COLLECT:
+ if (wa->inLen == 0) { /* is input buffer empty? */
+ picoos_uint16 blen;
+ /* try to get one item */
+ rv = picodata_cbGetItem(this->cbIn, wa->inBuf,
+ wa->inBufSize, &blen);
+ PICODBG_DEBUG(("after getting item, status: %d", rv));
+ if (PICO_OK == rv) {
+ /* we now have one item */
+ wa->inLen = blen;
+ wa->procState = WA_STEPSTATE_PROCESS;
+ /* uncomment next line to split into two steps */
+ /* return PICODATA_PU_ATOMIC; */
+ } else if (PICO_EOF == rv) {
+ /* there was no item in the char buffer */
+ return PICODATA_PU_IDLE;
+ } else if ((PICO_EXC_BUF_UNDERFLOW == rv)
+ || (PICO_EXC_BUF_OVERFLOW == rv)) {
+ PICODBG_ERROR(("problem getting item"));
+ picoos_emRaiseException(this->common->em, rv,
+ NULL, NULL);
+ return PICODATA_PU_ERROR;
+ } else {
+ PICODBG_ERROR(("problem getting item, unhandled"));
+ picoos_emRaiseException(this->common->em, rv,
+ NULL, NULL);
+ return PICODATA_PU_ERROR;
+ }
+ } else { /* there already is an item in the input buffer */
+ PICODBG_WARN(("item already in input buffer"));
+ picoos_emRaiseWarning(this->common->em,
+ PICO_WARN_PU_IRREG_ITEM, NULL, NULL);
+ wa->procState = WA_STEPSTATE_PROCESS;
+ /* uncomment next to split into two steps */
+ /* return PICODATA_PU_ATOMIC; */
+ }
+ break;
+
+
+ /* process state: process item in internal inBuf and put
+ * result in internal outBuf
+ */
+ case WA_STEPSTATE_PROCESS:
+
+ /* ensure there is an item in inBuf and it is valid */
+ if ((wa->inLen > 0) && picodata_is_valid_item(wa->inBuf,
+ wa->inLen)) {
+ picodata_itemhead_t ihead;
+ picoos_uint8 *icontent;
+ pico_status_t rvP = PICO_OK;
+
+ rv = picodata_get_iteminfo(wa->inBuf, wa->inLen, &ihead,
+ &icontent);
+ if (PICO_OK == rv) {
+
+ switch (ihead.type) {
+ case PICODATA_ITEM_WORDGRAPH:
+
+ if (0 < ihead.len) {
+ rvP = waProcessWordgraph(this, wa, &ihead,
+ icontent);
+ } else {
+ /* else ignore empty WORDGRAPH */
+ wa->inLen = 0;
+ wa->procState = WA_STEPSTATE_COLLECT;
+ return PICODATA_PU_BUSY;
+ }
+ break;
+ case PICODATA_ITEM_OTHER:
+ /* skip item */
+ rvP = PICO_WARN_PU_DISCARD_BUF;
+ break;
+ default:
+ /* copy item unmodified */
+ rvP = picodata_copy_item(wa->inBuf,
+ wa->inLen, wa->outBuf,
+ wa->outBufSize, &wa->outLen);
+ break;
+ }
+
+ if (PICO_OK == rvP) {
+ wa->inLen = 0;
+ wa->procState = WA_STEPSTATE_FEED;
+ /* uncomment next to split into two steps */
+ /* return PICODATA_PU_ATOMIC; */
+ } else if (PICO_WARN_PU_DISCARD_BUF == rvP) {
+ /* discard input buffer and get a new item */
+ PICODBG_INFO(("skipping OTHER item"));
+/* picoos_emRaiseWarning(this->common->em,
+ PICO_WARN_PU_DISCARD_BUF, NULL, NULL);
+*/
+ wa->inLen = 0;
+ wa->procState = WA_STEPSTATE_COLLECT;
+ return PICODATA_PU_BUSY;
+ } else {
+ /* PICO_EXC_BUF_OVERFLOW <- overflow in outbuf
+ PICO_ERR_OTHER <- no valid item in inbuf
+ or return from processWordgraph
+ */
+ PICODBG_ERROR(("problem processing item", rvP));
+ picoos_emRaiseException(this->common->em, rvP,
+ NULL, NULL);
+ return PICODATA_PU_ERROR;
+ }
+
+ } else { /* could not get iteminfo */
+ /* PICO_EXC_BUF_OVERFLOW <- overflow in outbuf
+ PICO_ERR_OTHER <- no valid item in inbuf
+ */
+ PICODBG_ERROR(("problem getting item info, "
+ "discard buffer content"));
+ wa->inLen = 0;
+ wa->procState = WA_STEPSTATE_COLLECT;
+ picoos_emRaiseException(this->common->em, rv,
+ NULL, NULL);
+ return PICODATA_PU_ERROR;
+ }
+
+ } else if (wa->inLen == 0) { /* no item in inBuf */
+ PICODBG_INFO(("no item in inBuf"));
+ /* wa->inLen = 0;*/
+ wa->procState = WA_STEPSTATE_COLLECT;
+ return PICODATA_PU_BUSY;
+
+ } else { /* no valid item in inBuf */
+ /* bad state/item, discard buffer content */
+ PICODBG_WARN(("no valid item, discard buffer content"));
+ picoos_emRaiseWarning(this->common->em,
+ PICO_WARN_PU_IRREG_ITEM, NULL, NULL);
+ picoos_emRaiseWarning(this->common->em,
+ PICO_WARN_PU_DISCARD_BUF, NULL, NULL);
+ wa->inLen = 0;
+ wa->procState = WA_STEPSTATE_COLLECT;
+ return PICODATA_PU_BUSY;
+ }
+ break;
+
+
+ /* feed state: copy item in internal outBuf to output charBuf */
+ case WA_STEPSTATE_FEED:
+
+ /* check that item fits in cb should not be needed */
+ rv = picodata_cbPutItem(this->cbOut, wa->outBuf,
+ wa->outLen, numBytesOutput);
+
+ PICODATA_INFO_ITEM(this->voice->kbArray[PICOKNOW_KBID_DBG],
+ (picoos_uint8 *)"wana: ", wa->outBuf,
+ wa->outLen);
+
+ PICODBG_DEBUG(("put item, status: %d", rv));
+ if (PICO_OK == rv) {
+ wa->outLen = 0;
+ wa->procState = WA_STEPSTATE_COLLECT;
+ return PICODATA_PU_BUSY;
+ } else if (PICO_EXC_BUF_OVERFLOW == rv) {
+ PICODBG_INFO(("feeding, overflow, PICODATA_PU_OUT_FULL"));
+ return PICODATA_PU_OUT_FULL;
+ } else if ((PICO_EXC_BUF_UNDERFLOW == rv)
+ || (PICO_ERR_OTHER == rv)) {
+ PICODBG_WARN(("feeding problem, discarding item"));
+ wa->outLen = 0;
+ wa->procState = WA_STEPSTATE_COLLECT;
+ picoos_emRaiseWarning(this->common->em, rv, NULL,NULL);
+ return PICODATA_PU_BUSY;
+ }
+ break;
+
+ default:
+ break;
+
+ } /* switch */
+
+ } /* while */
+
+ /* should be never reached */
+ PICODBG_ERROR(("reached end of function"));
+ picoos_emRaiseException(this->common->em, PICO_ERR_OTHER, NULL, NULL);
+ return PICODATA_PU_ERROR;
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+
+/* end */
diff --git a/lib/picowa.h b/lib/picowa.h
new file mode 100644
index 0000000..d7fbcd3
--- /dev/null
+++ b/lib/picowa.h
@@ -0,0 +1,135 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+/**
+ * @file picowa.h
+ *
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ * All rights reserved.
+ *
+ * History:
+ * - 2009-04-20 -- initial version
+ *
+ */
+
+
+/**
+ * @addtogroup picowa
+ * ---------------------------------------------------\n
+ * <b> Pico Word Analysis </b>\n
+ * ---------------------------------------------------\n
+itemtype, iteminfo1, iteminfo2, content -> TYPE(INFO1,INFO2)content
+in the following
+
+items input\n
+===========
+
+processed by wa:
+- WORDGRAPH(NA,NA)graph
+- OTHER(NA,NA)string
+
+unprocessed:
+- all other item types are forwarded through the PU without modification:
+ - PUNC
+ - CMD
+
+
+minimal input size (before processing starts)\n
+==================
+
+processing (ie. lex lookup and POS prediction) is possible with
+- one item
+
+
+items processed and output\n
+==========================
+
+processing an input WORDGRAPH results in one of the following items:
+- WORDGRAPH(POSes,NA)graph
+ - graph not in lex, POSes determined with dtree, or
+ - graph in lex - single entry without phone (:G2P), POSes from lex
+- WORDINDEX(POSes,NA)pos1|ind1...posN|indN
+ - graph in lex - {1,4} entries with phone, pos1...posN from lex,
+ {1,4} lexentries indices in content, POSes combined with map table
+ in klex
+
+processing an input OTHER results in the item being skipped (in the
+future this can be extended to e.g. spelling)
+
+see picotok.h for PUNC and CMD
+
+- POSes %d
+ - is the superset of all single POS and POS combinations defined
+ in the lingware as unique symbol
+- graph, len>0, utf8 graphemes, %s
+- pos1|ind1, pos2|ind2, ..., posN|indN
+ - pos? are the single, unambiguous POS only, one byte %d
+ - ind? are the lexentry indices, three bytes %d %d %d
+
+
+lexicon (system lexicon, but must also be ensured for user lexica)\n
+=======
+
+- POS GRAPH PHON, all mandatory, but
+ - * PHON can be an empty string -> no pronunciation in the resulting TTS output
+ - * PHON can be :G2P -> use G2P later to add pronunciation
+- (POS,GRAPH) is a uniq key (only one entry allowed)
+- (GRAPH) is almost a uniq key (2-4 entries with the same GRAPH, and
+ differing POS and differing PHON possible)
+ - for one graph we can have 2-4 solutions from the lex which all
+ need to be passed on the the next PU
+ - in this case GRAPH, POS, and PHON all must be available in lex
+ - in this case for each entry only a non-ambiguous, unique POS ID
+ is possible)
+
+other limitations\n
+=================
+
+- item size: header plus len=256 (valid for Pico in general)
+- wa uses one item context only -> internal buffer set to 256+4
+ */
+
+
+#ifndef PICOWA_H_
+#define PICOWA_H_
+
+#include "picoos.h"
+#include "picodata.h"
+#include "picorsrc.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+#if 0
+}
+#endif
+
+
+/* maximum length of an item incl. head for input and output buffers */
+#define PICOWA_MAXITEMSIZE 260
+
+
+picodata_ProcessingUnit picowa_newWordAnaUnit(
+ picoos_MemoryManager mm,
+ picoos_Common common,
+ picodata_CharBuffer cbIn,
+ picodata_CharBuffer cbOut,
+ picorsrc_Voice voice);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /*PICOWA_H_*/
diff --git a/tts/com_svox_picottsengine.cpp b/tts/com_svox_picottsengine.cpp
new file mode 100644
index 0000000..880edff
--- /dev/null
+++ b/tts/com_svox_picottsengine.cpp
@@ -0,0 +1,845 @@
+/*
+ * Copyright (C) 2008-2009 SVOX AG, Baslerstr. 30, 8048 Zuerich, Switzerland
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ *
+ * History:
+ * 2009-05-18 -- initial version
+ * 2009-06-04 -- updated for new TtsEngine interface
+ *
+ */
+
+#include <stdio.h>
+#include <unistd.h>
+
+#define LOG_TAG "SVOX Pico Engine"
+
+#include <utils/Log.h>
+#include <android_runtime/AndroidRuntime.h>
+#include <tts/TtsEngine.h>
+#include <picoapi.h>
+#include <picodefs.h>
+
+using namespace android;
+
+/* adaptation layer defines */
+#define PICO_MEM_SIZE 2500000
+#define PICO_MIN_RATE 20
+#define PICO_DEF_RATE 100
+#define PICO_MAX_RATE 500
+#define PICO_MIN_PITCH 50
+#define PICO_DEF_PITCH 100
+#define PICO_MAX_PITCH 200
+#define PICO_MIN_VOLUME 0
+#define PICO_DEF_VOLUME 100
+#define PICO_MAX_VOLUME 500
+#define MAX_OUTBUF_SIZE 128
+const char* PICO_LINGWARE_PATH = "/sdcard/svox/";
+const char* PICO_VOICE_NAME = "PicoVoice";
+const char* PICO_SPEED_OPEN_TAG = "<speed level='%d'>";
+const char* PICO_SPEED_CLOSE_TAG = "</speed>";
+const char* PICO_PITCH_OPEN_TAG = "<pitch level='%d'>";
+const char* PICO_PITCH_CLOSE_TAG = "</pitch>";
+const char* PICO_VOLUME_OPEN_TAG = "<volume level='%d'>";
+const char* PICO_VOLUME_CLOSE_TAG = "</volume>";
+
+const char* picoSupportedLang[] = { "en-rUS", "en-rGB", "de-rDE", "es-rES", "fr-rFR", "it-rIT" };
+const char* picoInternalLang[] = { "en-US", "en-GB", "de-DE", "es-ES", "fr-FR", "it-IT" };
+const char* picoInternalTaLingware[] = { "en-US_ta.bin", "en-GB_ta.bin", "de-DE_ta.bin", "es-ES_ta.bin", "fr-FR_ta.bin", "it-IT_ta.bin" };
+const char* picoInternalSgLingware[] = { "en-US_lh0_sg.bin", "en-GB_kh0_sg.bin", "de-DE_gl0_sg.bin", "es-ES_zl0_sg.bin", "fr-FR_nk0_sg.bin", "it-IT_cm0_sg.bin" };
+const char* picoInternalUtppLingware[] = { "en-US_utpp.bin", "en-GB_utpp.bin", "de-DE_utpp.bin", "es-ES_utpp.bin", "fr-FR_utpp.bin", "it-IT_utpp.bin" };
+const int picoNumSupportedLang = 6;
+
+const char* picoSupportedProperties[] = { "language", "rate", "pitch", "volume" };
+const int picoNumSupportedProperties = 4;
+
+/* adapation layer globals */
+synthDoneCB_t* picoSynthDoneCBPtr;
+void* picoMemArea = NULL;
+pico_System picoSystem = NULL;
+pico_Resource picoTaResource = NULL;
+pico_Resource picoSgResource = NULL;
+pico_Resource picoUtppResource = NULL;
+pico_Engine picoEngine = NULL;
+pico_Char* picoTaFileName = NULL;
+pico_Char* picoSgFileName = NULL;
+pico_Char* picoUtppFileName = NULL;
+pico_Char* picoTaResourceName = NULL;
+pico_Char* picoSgResourceName = NULL;
+pico_Char* picoUtppResourceName = NULL;
+int picoSynthAbort = 0;
+char* picoProp_currLang = NULL;
+int picoProp_currRate = PICO_DEF_RATE;
+int picoProp_currPitch = PICO_DEF_PITCH;
+int picoProp_currVolume = PICO_DEF_VOLUME;
+
+
+/* internal helper functions */
+
+/** checkForLanguage
+ * Checks if the requested language is among the supported languages
+ * @language - the language to check, either in xx or xx-rYY format
+ * return index of the language, or -1 if not supported
+*/
+static int checkForLanguage(const char* language)
+{
+ // verify that it's a language we support
+ int found = -1;
+ for (int i = 0; i < picoNumSupportedLang; i++)
+ {
+ if (strcmp(language, picoSupportedLang[i]) == 0)
+ {
+ found = i;
+ break;
+ }
+ }
+ if (found < 0)
+ {
+ // didn't find an exact match, may have been specified with only the first 2 characters
+ for (int i = 0; i < picoNumSupportedLang; i++)
+ {
+ if (strncmp(language, picoSupportedLang[i], 2) == 0)
+ {
+ found = i;
+ break;
+ }
+ }
+ if (found < 0)
+ {
+ LOGE("TtsEngine::set language called with unsupported language");
+ }
+ }
+ return found;
+}
+
+/** cleanResources
+ * Unloads any loaded pico resources
+*/
+static void cleanResources()
+{
+ if (picoEngine)
+ {
+ pico_disposeEngine(picoSystem, &picoEngine);
+ pico_releaseVoiceDefinition(picoSystem, (pico_Char*)PICO_VOICE_NAME);
+ picoEngine = NULL;
+ }
+ if (picoUtppResource)
+ {
+ pico_unloadResource(picoSystem, &picoUtppResource);
+ picoUtppResource = NULL;
+ }
+ if (picoTaResource)
+ {
+ pico_unloadResource(picoSystem, &picoTaResource);
+ picoTaResource = NULL;
+ }
+ if (picoSgResource)
+ {
+ pico_unloadResource(picoSystem, &picoSgResource);
+ picoSgResource = NULL;
+ }
+}
+
+/** cleanFiles
+ * Frees any memory allocated for file and resource strings
+*/
+static void cleanFiles()
+{
+ if (picoProp_currLang)
+ {
+ free(picoProp_currLang);
+ picoProp_currLang = NULL;
+ }
+
+ if (picoTaFileName)
+ {
+ free(picoTaFileName);
+ picoTaFileName = NULL;
+ }
+
+ if (picoSgFileName)
+ {
+ free(picoSgFileName);
+ picoSgFileName = NULL;
+ }
+
+ if (picoUtppFileName)
+ {
+ free(picoUtppFileName);
+ picoUtppFileName = NULL;
+ }
+
+ if (picoTaResourceName)
+ {
+ free(picoTaResourceName);
+ picoTaResourceName = NULL;
+ }
+
+ if (picoSgResourceName)
+ {
+ free(picoSgResourceName);
+ picoSgResourceName = NULL;
+ }
+
+ if (picoUtppResourceName)
+ {
+ free(picoUtppResourceName);
+ picoUtppResourceName = NULL;
+ }
+}
+
+/** doLanguageSwitch
+ * Switch to requested language. If language is already loaded it returns
+ * immediately, if another language is loaded this will first be unloaded
+ * and the new one then loaded. If no language is loaded the requested will be loaded.
+ * @language - the language to check, either in xx or xx-rYY format (i.e "en" or "en-rUS")
+ * return TTS_SUCCESS or TTS_FAILURE
+*/
+static int doLanguageSwitch(const char* language)
+{
+ // load new language
+ int langIndex = checkForLanguage(language);
+ if (langIndex < 0)
+ {
+ LOGE("Tried to swith to non-supported language %s", language);
+ return TTS_FAILURE;
+ }
+ LOGI("Found supported language %s", picoSupportedLang[langIndex]);
+
+ // if we already have a loaded language, check if it's the same one as requested
+ if (picoProp_currLang && (strcmp(picoProp_currLang, picoSupportedLang[langIndex]) == 0))
+ {
+ LOGI("Language %s already loaded (%s == %s)", language, picoProp_currLang, picoSupportedLang[langIndex]);
+ return TTS_SUCCESS;
+ }
+
+ // not the same language, unload the current one first
+ cleanResources();
+
+ // allocate memory for file and resource names
+ cleanFiles();
+ picoProp_currLang = (char*)malloc(10);
+ picoTaFileName = (pico_Char*)malloc(PICO_MAX_DATAPATH_NAME_SIZE + PICO_MAX_FILE_NAME_SIZE);
+ picoSgFileName = (pico_Char*)malloc(PICO_MAX_DATAPATH_NAME_SIZE + PICO_MAX_FILE_NAME_SIZE);
+ picoUtppFileName = (pico_Char*)malloc(PICO_MAX_DATAPATH_NAME_SIZE + PICO_MAX_FILE_NAME_SIZE);
+ picoTaResourceName = (pico_Char*)malloc(PICO_MAX_RESOURCE_NAME_SIZE);
+ picoSgResourceName = (pico_Char*)malloc(PICO_MAX_RESOURCE_NAME_SIZE);
+ picoUtppResourceName = (pico_Char*)malloc(PICO_MAX_RESOURCE_NAME_SIZE);
+
+ // set path and file names for resource files
+ strcpy((char*)picoTaFileName, PICO_LINGWARE_PATH);
+ strcat((char*)picoTaFileName, (const char*)picoInternalTaLingware[langIndex]);
+ strcpy((char*)picoSgFileName, PICO_LINGWARE_PATH);
+ strcat((char*)picoSgFileName, (const char*)picoInternalSgLingware[langIndex]);
+ strcpy((char*)picoUtppFileName, PICO_LINGWARE_PATH);
+ strcat((char*)picoUtppFileName, (const char*)picoInternalUtppLingware[langIndex]);
+
+ // load text analysis Lingware resource file
+ int ret = pico_loadResource(picoSystem, picoTaFileName, &picoTaResource);
+ if (PICO_OK != ret)
+ {
+ LOGE("Failed to load textana resource for %s [%d]", language, ret);
+ cleanResources();
+ cleanFiles();
+ return TTS_FAILURE;
+ }
+
+ // load signal generation Lingware resource file
+ ret = pico_loadResource(picoSystem, picoSgFileName, &picoSgResource);
+ if (PICO_OK != ret)
+ {
+ LOGE("Failed to load siggen resource for %s [%d]", language, ret);
+ cleanResources();
+ cleanFiles();
+ return TTS_FAILURE;
+ }
+
+ // Load utpp Lingware resource file if exists - NOTE: this file is optional
+ // and is currently not used. Loading is only attempted for future compatibility.
+ // If this file is not present the loading will still succeed.
+ ret = pico_loadResource(picoSystem, picoUtppFileName, &picoUtppResource);
+ if (PICO_OK != ret && ret != PICO_EXC_CANT_OPEN_FILE)
+ {
+ LOGE("Failed to load utpp resource for %s [%d]", language, ret);
+ cleanResources();
+ cleanFiles();
+ return TTS_FAILURE;
+ }
+
+ // Get text analysis resource name
+ ret = pico_getResourceName(picoSystem, picoTaResource, (char*)picoTaResourceName);
+ if (PICO_OK != ret)
+ {
+ LOGE("Failed to get textana resource name for %s [%d]", language, ret);
+ cleanResources();
+ cleanFiles();
+ return TTS_FAILURE;
+ }
+
+ // Get signal generation resource name
+ ret = pico_getResourceName(picoSystem, picoSgResource, (char*)picoSgResourceName);
+ if (PICO_OK == ret && picoUtppResource != NULL)
+ {
+ // Get utpp resource name - optional: see note above
+ ret = pico_getResourceName(picoSystem, picoUtppResource, (char*)picoUtppResourceName);
+ if (PICO_OK != ret)
+ {
+ LOGE("Failed to get utpp resource name for %s [%d]", language, ret);
+ cleanResources();
+ cleanFiles();
+ return TTS_FAILURE;
+ }
+ }
+ if (PICO_OK != ret)
+ {
+ LOGE("Failed to get siggen resource name for %s [%d]", language, ret);
+ cleanResources();
+ cleanFiles();
+ return TTS_FAILURE;
+ }
+
+ // create a voice definition
+ ret = pico_createVoiceDefinition(picoSystem, (const pico_Char*)PICO_VOICE_NAME);
+ if (PICO_OK != ret)
+ {
+ LOGE("Failed to create voice for %s [%d]", language, ret);
+ cleanResources();
+ cleanFiles();
+ return TTS_FAILURE;
+ }
+
+ // add text analysis resource to voice
+ ret = pico_addResourceToVoiceDefinition(picoSystem, (const pico_Char*)PICO_VOICE_NAME, picoTaResourceName);
+ if (PICO_OK != ret)
+ {
+ LOGE("Failed to add textana resource to voice for %s [%d]", language, ret);
+ cleanResources();
+ cleanFiles();
+ return TTS_FAILURE;
+ }
+
+ // add signal generation resource to voice
+ ret = pico_addResourceToVoiceDefinition(picoSystem, (const pico_Char*)PICO_VOICE_NAME, picoSgResourceName);
+ if (PICO_OK == ret && picoUtppResource != NULL)
+ {
+ // add utpp resource to voice - optional: see note above
+ ret = pico_addResourceToVoiceDefinition(picoSystem, (const pico_Char*)PICO_VOICE_NAME, picoUtppResourceName);
+ if (PICO_OK != ret)
+ {
+ LOGE("Failed to add utpp resource to voice for %s [%d]", language, ret);
+ cleanResources();
+ cleanFiles();
+ return TTS_FAILURE;
+ }
+ }
+
+ if (PICO_OK != ret)
+ {
+ LOGE("Failed to add siggen resource to voice for %s [%d]", language, ret);
+ cleanResources();
+ cleanFiles();
+ return TTS_FAILURE;
+ }
+
+ ret = pico_newEngine(picoSystem, (const pico_Char*)PICO_VOICE_NAME, &picoEngine);
+ if (PICO_OK != ret)
+ {
+ LOGE("Failed to create engine for %s [%d]", language, ret);
+ cleanResources();
+ cleanFiles();
+ return TTS_FAILURE;
+ }
+
+ strcpy(picoProp_currLang, picoSupportedLang[langIndex]);
+
+ LOGI("loaded %s successfully", picoProp_currLang);
+
+ return TTS_SUCCESS;
+}
+
+/** doAddProperties
+ * add <speed>, <pitch> and <volume> tags to text if properties have been set to non-default values
+ * and returns the new string. Calling function is responsible for freeing returned string
+ * @str - text to apply tags to
+ * return new string with tags applied
+*/
+static char* doAddProperties(const char* str)
+{
+ char* data = NULL;
+ int haspitch = 0, hasspeed = 0, hasvol = 0;
+ int textlen = strlen(str) + 1;
+
+ if (picoProp_currPitch != PICO_DEF_PITCH)
+ {
+ textlen += strlen(PICO_PITCH_OPEN_TAG) + 5;
+ textlen += strlen(PICO_PITCH_CLOSE_TAG);
+ haspitch = 1;
+ }
+ if (picoProp_currRate != PICO_DEF_RATE)
+ {
+ textlen += strlen(PICO_SPEED_OPEN_TAG) + 5;
+ textlen += strlen(PICO_SPEED_CLOSE_TAG);
+ hasspeed = 1;
+ }
+ if (picoProp_currVolume != PICO_DEF_VOLUME)
+ {
+ textlen += strlen(PICO_VOLUME_OPEN_TAG) + 5;
+ textlen += strlen(PICO_VOLUME_CLOSE_TAG);
+ hasvol = 1;
+ }
+
+ data = (char*)malloc(textlen);
+ if (!data)
+ {
+ return NULL;
+ }
+ memset(data, 0, textlen);
+ if (haspitch)
+ {
+ char* tmp = (char*)malloc(strlen(PICO_PITCH_OPEN_TAG) + strlen(PICO_PITCH_CLOSE_TAG) + 5);
+ sprintf(tmp, PICO_PITCH_OPEN_TAG, picoProp_currPitch);
+ strcat(data, tmp);
+ free(tmp);
+ }
+
+ if (hasspeed)
+ {
+ char* tmp = (char*)malloc(strlen(PICO_SPEED_OPEN_TAG) + strlen(PICO_SPEED_CLOSE_TAG) + 5);
+ sprintf(tmp, PICO_SPEED_OPEN_TAG, picoProp_currRate);
+ strcat(data, tmp);
+ free(tmp);
+ }
+
+ if (hasvol)
+ {
+ char* tmp = (char*)malloc(strlen(PICO_VOLUME_OPEN_TAG) + strlen(PICO_VOLUME_CLOSE_TAG) + 5);
+ sprintf(tmp, PICO_VOLUME_OPEN_TAG, picoProp_currVolume);
+ strcat(data, tmp);
+ free(tmp);
+ }
+
+ strcat(data, str);
+
+ if (hasvol)
+ {
+ strcat(data, PICO_VOLUME_CLOSE_TAG);
+ }
+
+ if (hasspeed)
+ {
+ strcat(data, PICO_SPEED_CLOSE_TAG);
+ }
+
+ if (haspitch)
+ {
+ strcat(data, PICO_PITCH_CLOSE_TAG);
+ }
+
+ return data;
+}
+
+/* API function implementations */
+
+/** init
+ * allocates pico memory block and initializes pico system
+ * synthDoneCBPtr - Pointer to callback function which will receive generated samples
+ * return tts_result
+*/
+tts_result TtsEngine::init(synthDoneCB_t synthDoneCBPtr)
+{
+ if (synthDoneCBPtr == NULL)
+ {
+ LOGE("Callback pointer is NULL");
+ return TTS_FAILURE;
+ }
+
+ picoMemArea = malloc(PICO_MEM_SIZE);
+ if (!picoMemArea)
+ {
+ LOGE("Failed to allocate memory for Pico system");
+ return TTS_FAILURE;
+ }
+
+ pico_Status ret = pico_initialize(picoMemArea, PICO_MEM_SIZE, &picoSystem);
+ if (PICO_OK != ret)
+ {
+ LOGE("Failed to initialize Pico system");
+ free(picoMemArea);
+ picoMemArea = NULL;
+ return TTS_FAILURE;
+ }
+
+ picoSynthDoneCBPtr = synthDoneCBPtr;
+ return TTS_SUCCESS;
+}
+
+/** shutdown
+ * unloads all pico resources, terminates pico system and frees pico memory block
+ * return tts_result
+*/
+tts_result TtsEngine::shutdown()
+{
+ cleanResources();
+
+ if (picoSystem)
+ {
+ pico_terminate(&picoSystem);
+ picoSystem = NULL;
+ }
+ if (picoMemArea)
+ {
+ free(picoMemArea);
+ picoMemArea = NULL;
+ }
+
+ cleanFiles();
+
+ return TTS_SUCCESS;
+}
+
+/** loadLanguage
+ * Load a new language
+ * @value - language string in xx or xx-rYY format (i.e. "en" or "en-rUS")
+ * @size - size of value
+ * return tts_result
+*/
+tts_result TtsEngine::loadLanguage(const char *value, const size_t size)
+{
+ return setProperty("language", value, size);
+}
+
+/** setLanguage
+ * Load a new language
+ * @value - language string in xx or xx-rYY format (i.e. "en" or "en-rUS")
+ * @size - size of value
+ * return tts_result
+*/
+tts_result TtsEngine::setLanguage(const char *value, const size_t size)
+{
+ return setProperty("language", value, size);
+}
+
+/** getLanguage
+ * Get currently loaded language - if any
+ * @value - buffer which will receive value
+ * @iosize - size of value - if value is too small to contain the return string, this will contain the actual size needed
+ * return tts_result
+*/
+tts_result TtsEngine::getLanguage(char *value, size_t *iosize)
+{
+ return getProperty("language", value, iosize);
+}
+
+/** setProperty
+ * set property, supported properties are language, rate, pitch and volume
+ * @property - name of property to set
+ * @value - value to set
+ * @size - size of value
+ * return tts_result
+*/
+tts_result TtsEngine::setProperty(const char *property, const char *value, const size_t size)
+{
+ // sanity check
+ if (property == NULL)
+ {
+ LOGE("setProperty called with property NULL");
+ return TTS_PROPERTY_UNSUPPORTED;
+ }
+
+ if (value == NULL)
+ {
+ LOGE("setProperty called with value NULL");
+ return TTS_VALUE_INVALID;
+ }
+
+ if (strncmp(property, "language", 8) == 0)
+ {
+ // verify it's in correct format
+ if (strlen(value) != 2 && strlen(value) != 6)
+ {
+ LOGE("change language called with incorrect format");
+ return TTS_VALUE_INVALID;
+ }
+
+ // try to switch to specified language
+ if (doLanguageSwitch(value) == TTS_FAILURE)
+ {
+ LOGE("failed to load language");
+ return TTS_FAILURE;
+ }
+ else
+ {
+ return TTS_SUCCESS;
+ }
+ }
+ else if (strncmp(property, "rate", 4) == 0)
+ {
+ int rate = atoi(value);
+ if (rate < PICO_MIN_RATE) rate = PICO_MIN_RATE;
+ if (rate > PICO_MAX_RATE) rate = PICO_MAX_RATE;
+ picoProp_currRate = rate;
+ return TTS_SUCCESS;
+ }
+ else if (strncmp(property, "pitch", 5) == 0)
+ {
+ int pitch = atoi(value);
+ if (pitch < PICO_MIN_PITCH) pitch = PICO_MIN_PITCH;
+ if (pitch > PICO_MAX_PITCH) pitch = PICO_MAX_PITCH;
+ picoProp_currPitch = pitch;
+ return TTS_SUCCESS;
+ }
+ else if (strncmp(property, "volume", 6) == 0)
+ {
+ int volume = atoi(value);
+ if (volume < PICO_MIN_VOLUME) volume = PICO_MIN_VOLUME;
+ if (volume > PICO_MAX_VOLUME) volume = PICO_MAX_VOLUME;
+ picoProp_currVolume = volume;
+ return TTS_SUCCESS;
+ }
+
+ return TTS_PROPERTY_UNSUPPORTED;
+}
+
+/** getProperty
+ * get property, supported properties are language, rate, pitch and volume
+ * @property - name of property to get
+ * @value - buffer which will receive value of property
+ * @iosize - size of value - if size is too small on return this will contain actual size needed
+ * return tts_result
+*/
+tts_result TtsEngine::getProperty(const char *property, char *value, size_t* iosize)
+{
+ // sanity check
+ if (property == NULL)
+ {
+ LOGE("getProperty called with property NULL");
+ return TTS_PROPERTY_UNSUPPORTED;
+ }
+
+ if (value == NULL)
+ {
+ LOGE("getProperty called with value NULL");
+ return TTS_VALUE_INVALID;
+ }
+
+ if (strncmp(property, "language", 8) == 0)
+ {
+ if (picoProp_currLang == NULL)
+ {
+ strcpy(value, "");
+ }
+ else
+ {
+ if (*iosize < strlen(picoProp_currLang)+1)
+ {
+ *iosize = strlen(picoProp_currLang) + 1;
+ return TTS_PROPERTY_SIZE_TOO_SMALL;
+ }
+ strcpy(value, picoProp_currLang);
+ }
+ return TTS_SUCCESS;
+ }
+ else if (strncmp(property, "rate", 4) == 0)
+ {
+ char tmprate[4];
+ sprintf(tmprate, "%d", picoProp_currRate);
+ if (*iosize < strlen(tmprate)+1)
+ {
+ *iosize = strlen(tmprate) + 1;
+ return TTS_PROPERTY_SIZE_TOO_SMALL;
+ }
+ strcpy(value, tmprate);
+ return TTS_SUCCESS;
+ }
+ else if (strncmp(property, "pitch", 5) == 0)
+ {
+ char tmppitch[4];
+ sprintf(tmppitch, "%d", picoProp_currPitch);
+ if (*iosize < strlen(tmppitch)+1)
+ {
+ *iosize = strlen(tmppitch) + 1;
+ return TTS_PROPERTY_SIZE_TOO_SMALL;
+ }
+ strcpy(value, tmppitch);
+ return TTS_SUCCESS;
+ }
+ else if (strncmp(property, "volume", 6) == 0)
+ {
+ char tmpvol[4];
+ sprintf(tmpvol, "%d", picoProp_currVolume);
+ if (*iosize < strlen(tmpvol)+1)
+ {
+ *iosize = strlen(tmpvol) + 1;
+ return TTS_PROPERTY_SIZE_TOO_SMALL;
+ }
+ strcpy(value, tmpvol);
+ return TTS_SUCCESS;
+ }
+ else
+ {
+ LOGE("Unsupported property");
+ return TTS_PROPERTY_UNSUPPORTED;
+ }
+}
+
+/** synthesizeText
+ * synthesizes a text string
+ * @text - text to synthesize
+ * @buffer - buffer which will receive generated samples
+ * @bufferSize - size of buffer
+ * @userdata - pointer to user data which will be passed back to callback function
+ * return tts_result
+*/
+tts_result TtsEngine::synthesizeText(const char *text, int8_t *buffer, size_t bufferSize, void *userdata)
+{
+ pico_Char* inp = NULL;
+ pico_Char* local_text = NULL;
+ short outbuf[MAX_OUTBUF_SIZE/2];
+ pico_Int16 bytes_sent, bytes_recv, text_remaining, out_data_type;
+ pico_Status ret;
+ picoSynthAbort = 0;
+
+ if (text == NULL)
+ {
+ LOGE("synthesizeText called with NULL string");
+ return TTS_FAILURE;
+ }
+
+ if (buffer == NULL)
+ {
+ LOGE("synthesizeText called with NULL buffer");
+ return TTS_FAILURE;
+ }
+
+ // add property tags to string - if any
+ local_text = (pico_Char*)doAddProperties(text);
+ if (!local_text)
+ {
+ LOGE("Failed to allocate memory for text string");
+ return TTS_FAILURE;
+ }
+
+ text_remaining = strlen((const char*)local_text) + 1;
+
+ inp = (pico_Char*)local_text;
+
+ size_t bufused = 0;
+
+ // synthesis loop
+ while (text_remaining)
+ {
+ if (picoSynthAbort)
+ {
+ ret = pico_resetEngine(picoEngine);
+ break;
+ }
+
+ // feed text into engine
+ ret = pico_putTextUtf8(picoEngine, inp, text_remaining, &bytes_sent);
+ if (ret != PICO_OK)
+ {
+ LOGE("Error synthesizing string '%s': [%d]", text, ret);
+ if (local_text) free(local_text);
+ return TTS_FAILURE;
+ }
+
+ text_remaining -= bytes_sent;
+ inp += bytes_sent;
+ do
+ {
+ if (picoSynthAbort)
+ {
+ break;
+ }
+ // retrieve samples and add to buffer
+ ret = pico_getData(picoEngine, (void*)outbuf, MAX_OUTBUF_SIZE, &bytes_recv, &out_data_type);
+ if (bytes_recv)
+ {
+ if (bufused + bytes_recv <= bufferSize)
+ {
+ memcpy(buffer+bufused, (int8_t*)outbuf, bytes_recv);
+ bufused += bytes_recv;
+ }
+ else
+ {
+ // buffer filled, pass on to callback function
+ int cbret = picoSynthDoneCBPtr(userdata, 16000, AudioSystem::PCM_16_BIT, 1, buffer, bufused, TTS_SYNTH_PENDING);
+ if (cbret == TTS_CALLBACK_HALT)
+ {
+ LOGI("Halt requested by caller. Halting.");
+ picoSynthAbort = 1;
+ break;
+ }
+ bufused = 0;
+ memcpy(buffer, (int8_t*)outbuf, bytes_recv);
+ bufused += bytes_recv;
+ }
+ }
+ } while (PICO_STEP_BUSY == ret);
+
+ // synthesis is finished, notify caller and pass remaining samples
+ if (!picoSynthAbort)
+ {
+ picoSynthDoneCBPtr(userdata, 16000, AudioSystem::PCM_16_BIT, 1, buffer, bufused, TTS_SYNTH_DONE);
+ }
+ picoSynthAbort = 0;
+
+ if (ret != PICO_STEP_IDLE)
+ {
+ LOGE("Error occurred during synthesis [%d]", ret);
+ if (local_text) free(local_text);
+ return TTS_FAILURE;
+ }
+ }
+
+ if (local_text) free(local_text);
+ return TTS_SUCCESS;
+}
+
+/** synthesizeIpa
+ * synthesizes a phonetic string in IPA format
+ * @ipa - phonetic string to synthesize
+ * @buffer - buffer which will receive generated samples
+ * @bufferSize - size of buffer
+ * @userdata - pointer to user data which will be passed back to callback function
+ * return tts_result
+*/
+tts_result TtsEngine::synthesizeIpa(const char * /*ipa*/, int8_t * /*buffer*/, size_t /*bufferSize*/, void * /*userdata*/)
+{
+ LOGI("synthIPA not supported in this release");
+ return TTS_FEATURE_UNSUPPORTED;
+}
+
+/** stop
+ * aborts running synthesis
+ * return tts_result
+*/
+tts_result TtsEngine::stop()
+{
+ picoSynthAbort = 1;
+ return TTS_SUCCESS;
+}
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+TtsEngine* getTtsEngine()
+{
+ return new TtsEngine();
+}
+
+#ifdef __cplusplus
+}
+#endif
+