/* * 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, picoos_int32 r_mode) { 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 */