ap_strange.c 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829
  1. /*
  2. * Copyright (c) 1983-2013 Trevor Wishart and Composers Desktop Project Ltd
  3. * http://www.trevorwishart.co.uk
  4. * http://www.composersdesktop.com
  5. *
  6. This file is part of the CDP System.
  7. The CDP System is free software; you can redistribute it
  8. and/or modify it under the terms of the GNU Lesser General Public
  9. License as published by the Free Software Foundation; either
  10. version 2.1 of the License, or (at your option) any later version.
  11. The CDP System is distributed in the hope that it will be useful,
  12. but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. GNU Lesser General Public License for more details.
  15. You should have received a copy of the GNU Lesser General Public
  16. License along with the CDP System; if not, write to the Free Software
  17. Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
  18. 02111-1307 USA
  19. *
  20. */
  21. #include <stdio.h>
  22. #include <stdlib.h>
  23. #include <structures.h>
  24. #include <cdpmain.h>
  25. #include <tkglobals.h>
  26. #include <pnames.h>
  27. #include <strange.h>
  28. #include <processno.h>
  29. #include <modeno.h>
  30. #include <globcon.h>
  31. #include <logic.h>
  32. #include <filetype.h>
  33. #include <mixxcon.h>
  34. #include <speccon.h>
  35. #include <flags.h>
  36. #include <arrays.h>
  37. #include <formants.h>
  38. #include <sfsys.h>
  39. #include <osbind.h>
  40. #include <string.h>
  41. #include <math.h>
  42. #include <srates.h>
  43. /********************************************************************************************/
  44. /********************************** FORMERLY IN buffers.c ***********************************/
  45. /********************************************************************************************/
  46. static int allocate_warp_buffer(dataptr dz);
  47. /********************************************************************************************/
  48. /********************************** FORMERLY IN preprocess.c ********************************/
  49. /********************************************************************************************/
  50. #define ROLL_OFF (500.0)
  51. static int setup_internal_arrays_for_specshift(dataptr dz);
  52. static int establish_internal_params_for_glis(dataptr dz);
  53. static int setup_internal_arrays_for_waver(dataptr dz);
  54. static int setup_array_of_botfrq_of_each_channel(int *botchan,dataptr dz);
  55. static int warp_preprocess(dataptr dz);
  56. static int convert_params_for_warp(dataptr dz);
  57. static int setup_internal_arrays_and_params_for_specwarp(dataptr dz);
  58. static int convert_pitchdata_to_midi_for_warp(dataptr dz);
  59. static int invert_preprocess(dataptr dz);
  60. static int setup_internal_arrays_for_invert(dataptr dz);
  61. static int extract_normalising_criteria_by_reading_infile(dataptr dz);
  62. /***************************************************************************************/
  63. /****************************** FORMERLY IN aplinit.c **********************************/
  64. /***************************************************************************************/
  65. /***************************** ESTABLISH_BUFPTRS_AND_EXTRA_BUFFERS **************************/
  66. int establish_bufptrs_and_extra_buffers(dataptr dz)
  67. {
  68. int exit_status;
  69. int is_spec = FALSE;
  70. dz->extra_bufcnt = -1; /* ENSURE EVERY CASE HAS A PAIR OF ENTRIES !! */
  71. dz->bptrcnt = 0;
  72. dz->bufcnt = 0;
  73. switch(dz->process) {
  74. case(SHIFT): dz->extra_bufcnt = 0; dz->bptrcnt = 1; is_spec = TRUE; break;
  75. case(GLIS): dz->extra_bufcnt = 0; dz->bptrcnt = 1; is_spec = TRUE; break;
  76. case(WAVER): dz->extra_bufcnt = 0; dz->bptrcnt = 1; is_spec = TRUE; break;
  77. case(WARP): dz->extra_bufcnt = 1; dz->bptrcnt = 7; is_spec = TRUE; break;
  78. case(INVERT): dz->extra_bufcnt = 0; dz->bptrcnt = 1; is_spec = TRUE; break;
  79. default:
  80. sprintf(errstr,"Unknown program type [%d] in establish_bufptrs_and_extra_buffers()\n",dz->process);
  81. return(PROGRAM_ERROR);
  82. }
  83. if(dz->extra_bufcnt < 0) {
  84. sprintf(errstr,"bufcnts have not been set: establish_bufptrs_and_extra_buffers()\n");
  85. return(PROGRAM_ERROR);
  86. }
  87. if(is_spec)
  88. return establish_spec_bufptrs_and_extra_buffers(dz);
  89. else if((dz->process==HOUSE_SPEC && dz->mode==HOUSE_CONVERT) || dz->process==INFO_DIFF) {
  90. if((exit_status = establish_spec_bufptrs_and_extra_buffers(dz))<0)
  91. return(exit_status);
  92. }
  93. return establish_groucho_bufptrs_and_extra_buffers(dz);
  94. }
  95. /***************************** SETUP_INTERNAL_ARRAYS_AND_ARRAY_POINTERS **************************/
  96. int setup_internal_arrays_and_array_pointers(dataptr dz)
  97. {
  98. int n;
  99. dz->ptr_cnt = -1; /* base constructor...process */
  100. dz->array_cnt = -1;
  101. dz->iarray_cnt = -1;
  102. dz->larray_cnt = -1;
  103. switch(dz->process) {
  104. case(SHIFT): dz->array_cnt =2; dz->iarray_cnt =2; dz->larray_cnt = 0; dz->ptr_cnt = 0; dz->fptr_cnt = 0; break;
  105. case(GLIS): dz->array_cnt = 0; dz->iarray_cnt = 0; dz->larray_cnt = 0; dz->ptr_cnt = 0; dz->fptr_cnt = 0; break;
  106. case(WAVER): dz->array_cnt =1; dz->iarray_cnt = 0; dz->larray_cnt = 0; dz->ptr_cnt = 0; dz->fptr_cnt = 0; break;
  107. case(WARP): dz->array_cnt =2; dz->iarray_cnt =1; dz->larray_cnt = 0; dz->ptr_cnt = 0; dz->fptr_cnt = 0; break;
  108. case(INVERT): dz->array_cnt =1; dz->iarray_cnt = 0; dz->larray_cnt = 0; dz->ptr_cnt = 0; dz->fptr_cnt = 0; break;
  109. }
  110. /*** WARNING ***
  111. ANY APPLICATION DEALING WITH A NUMLIST INPUT: MUST establish AT LEAST 1 double array: i.e. dz->array_cnt = at least 1
  112. **** WARNING ***/
  113. if(dz->array_cnt < 0 || dz->iarray_cnt < 0 || dz->larray_cnt < 0 || dz->ptr_cnt < 0 || dz->fptr_cnt < 0) {
  114. sprintf(errstr,"array_cnt not set in setup_internal_arrays_and_array_pointers()\n");
  115. return(PROGRAM_ERROR);
  116. }
  117. if(dz->array_cnt > 0) {
  118. if((dz->parray = (double **)malloc(dz->array_cnt * sizeof(double *)))==NULL) {
  119. sprintf(errstr,"INSUFFICIENT MEMORY for internal double arrays.\n");
  120. return(MEMORY_ERROR);
  121. }
  122. for(n=0;n<dz->array_cnt;n++)
  123. dz->parray[n] = NULL;
  124. }
  125. if(dz->iarray_cnt > 0) {
  126. if((dz->iparray = (int **)malloc(dz->iarray_cnt * sizeof(int *)))==NULL) {
  127. sprintf(errstr,"INSUFFICIENT MEMORY for internal int arrays.\n");
  128. return(MEMORY_ERROR);
  129. }
  130. for(n=0;n<dz->iarray_cnt;n++)
  131. dz->iparray[n] = NULL;
  132. }
  133. if(dz->larray_cnt > 0) {
  134. if((dz->lparray = (int **)malloc(dz->larray_cnt * sizeof(int *)))==NULL) {
  135. sprintf(errstr,"INSUFFICIENT MEMORY for internal long arrays.\n");
  136. return(MEMORY_ERROR);
  137. }
  138. for(n=0;n<dz->larray_cnt;n++)
  139. dz->lparray[n] = NULL;
  140. }
  141. if(dz->ptr_cnt > 0) {
  142. if((dz->ptr = (double **)malloc(dz->ptr_cnt * sizeof(double *)))==NULL) {
  143. sprintf(errstr,"INSUFFICIENT MEMORY for internal pointer arrays.\n");
  144. return(MEMORY_ERROR);
  145. }
  146. for(n=0;n<dz->ptr_cnt;n++)
  147. dz->ptr[n] = NULL;
  148. }
  149. if(dz->fptr_cnt > 0) {
  150. if((dz->fptr = (float **)malloc(dz->fptr_cnt * sizeof(float *)))==NULL) {
  151. sprintf(errstr,"INSUFFICIENT MEMORY for internal float-pointer arrays.\n");
  152. return(MEMORY_ERROR);
  153. }
  154. for(n=0;n<dz->fptr_cnt;n++)
  155. dz->fptr[n] = NULL;
  156. }
  157. return(FINISHED);
  158. }
  159. /****************************** ASSIGN_PROCESS_LOGIC *********************************/
  160. int assign_process_logic(dataptr dz)
  161. {
  162. switch(dz->process) {
  163. case(SHIFT): setup_process_logic(ANALFILE_ONLY, EQUAL_ANALFILE, ANALFILE_OUT, dz); break;
  164. case(GLIS): setup_process_logic(ANALFILE_ONLY, EQUAL_ANALFILE, ANALFILE_OUT, dz); break;
  165. case(WAVER): setup_process_logic(ANALFILE_ONLY, EQUAL_ANALFILE, ANALFILE_OUT, dz); break;
  166. case(WARP): setup_process_logic(ANAL_WITH_PITCHDATA, BIG_ANALFILE, ANALFILE_OUT, dz); break;
  167. case(INVERT): setup_process_logic(ANALFILE_ONLY, EQUAL_ANALFILE, ANALFILE_OUT, dz); break;
  168. default:
  169. sprintf(errstr,"Unknown process: assign_process_logic()\n");
  170. return(PROGRAM_ERROR);
  171. break;
  172. }
  173. if(dz->has_otherfile) {
  174. switch(dz->input_data_type) {
  175. case(ALL_FILES):
  176. case(TWO_SNDFILES):
  177. case(SNDFILE_AND_ENVFILE):
  178. case(SNDFILE_AND_BRKFILE):
  179. case(SNDFILE_AND_UNRANGED_BRKFILE):
  180. case(SNDFILE_AND_DB_BRKFILE):
  181. break;
  182. case(MANY_SNDFILES):
  183. if(dz->process==INFO_TIMELIST)
  184. break;
  185. /* fall thro */
  186. default:
  187. sprintf(errstr,"Most processes accepting files with different properties\n"
  188. "can only take 2 sound infiles.\n");
  189. return(PROGRAM_ERROR);
  190. }
  191. }
  192. return(FINISHED);
  193. }
  194. /***************************** SET_LEGAL_INFILE_STRUCTURE **************************
  195. *
  196. * Allows 2nd infile to have different props to first infile.
  197. */
  198. void set_legal_infile_structure(dataptr dz)
  199. {
  200. switch(dz->process) {
  201. default:
  202. dz->has_otherfile = FALSE;
  203. break;
  204. }
  205. }
  206. /***************************************************************************************/
  207. /****************************** FORMERLY IN internal.c *********************************/
  208. /***************************************************************************************/
  209. /****************************** SET_LEGAL_INTERNALPARAM_STRUCTURE *********************************/
  210. int set_legal_internalparam_structure(int process,int mode,aplptr ap)
  211. {
  212. int exit_status = FINISHED;
  213. switch(process) {
  214. case(SHIFT): return(FINISHED);
  215. case(GLIS): exit_status = set_internalparam_data("ddddddd",ap); break;
  216. case(WAVER): exit_status = set_internalparam_data("ii",ap); break;
  217. case(WARP): exit_status = set_internalparam_data("iii",ap); break;
  218. case(INVERT): return(FINISHED);
  219. default:
  220. sprintf(errstr,"Unknown process in set_legal_internalparam_structure()\n");
  221. return(PROGRAM_ERROR);
  222. }
  223. return(exit_status);
  224. }
  225. /********************************************************************************************/
  226. /********************************** FORMERLY IN specialin.c *********************************/
  227. /********************************************************************************************/
  228. /********************** READ_SPECIAL_DATA ************************/
  229. int read_special_data(char *str,dataptr dz)
  230. {
  231. aplptr ap = dz->application;
  232. switch(ap->special_data) {
  233. default:
  234. sprintf(errstr,"Unknown special_data type: read_special_data()\n");
  235. return(PROGRAM_ERROR);
  236. }
  237. return(FINISHED); /* NOTREACHED */
  238. }
  239. /********************************************************************************************/
  240. /********************************** FORMERLY IN preprocess.c ********************************/
  241. /********************************************************************************************/
  242. /****************************** PARAM_PREPROCESS *********************************/
  243. int param_preprocess(dataptr dz)
  244. {
  245. switch(dz->process) {
  246. case(SHIFT): return setup_internal_arrays_for_specshift(dz);
  247. case(GLIS): return establish_internal_params_for_glis(dz);
  248. case(WAVER): return setup_internal_arrays_for_waver(dz);
  249. case(WARP): return warp_preprocess(dz);
  250. case(INVERT): return invert_preprocess(dz);
  251. default:
  252. sprintf(errstr,"PROGRAMMING PROBLEM: Unknown process in param_preprocess()\n");
  253. return(PROGRAM_ERROR);
  254. }
  255. return(FINISHED); /* NOTREACHED */
  256. }
  257. /************ SETUP_INTERNAL_ARRAYS_FOR_SPECSHIFT *************/
  258. int setup_internal_arrays_for_specshift(dataptr dz)
  259. {
  260. int n;
  261. if((dz->iparray[SHIFT_OVER] = (int *)malloc(dz->clength * sizeof(int)))==NULL) {
  262. sprintf(errstr,"INSUFFICIENT MEMORY for shift over array.\n");
  263. return(MEMORY_ERROR);
  264. }
  265. if((dz->iparray[SHIFT_DONE] = (int *)malloc(dz->clength * sizeof(int)))==NULL) {
  266. sprintf(errstr,"INSUFFICIENT MEMORY for shift done array.\n");
  267. return(MEMORY_ERROR);
  268. }
  269. if((dz->parray[SHIFT_CHTOP]
  270. = (double *)malloc(dz->clength * sizeof(double)))==NULL) {
  271. sprintf(errstr,"INSUFFICIENT MEMORY for shift channel top array.\n");
  272. return(MEMORY_ERROR);
  273. }
  274. if((dz->parray[SHIFT_CHMID]
  275. = (double *)malloc(dz->clength * sizeof(double)))==NULL) {
  276. sprintf(errstr,"INSUFFICIENT MEMORY for shift channel mid array.\n");
  277. return(MEMORY_ERROR);
  278. }
  279. dz->parray[SHIFT_CHTOP][0] = dz->halfchwidth;
  280. dz->parray[SHIFT_CHMID][0] = 0.0;
  281. for(n = 1;n < dz->clength; n++) {
  282. dz->parray[SHIFT_CHTOP][n] = dz->parray[SHIFT_CHTOP][n-1] + dz->chwidth;
  283. dz->parray[SHIFT_CHMID][n] = dz->parray[SHIFT_CHMID][n-1] + dz->chwidth;
  284. }
  285. dz->parray[SHIFT_CHMID][0] = 1.0; /* arbitrary, but NOT zero!! */
  286. return(FINISHED);
  287. }
  288. /********************** ESTABLISH_INTERNAL_PARAMS_FOR_GLIS **********************/
  289. int establish_internal_params_for_glis(dataptr dz)
  290. {
  291. dz->param[GLIS_CONVERTOR] = dz->frametime/SEMITONES_PER_OCTAVE;
  292. if(dz->brksize[GLIS_RATE]==0)
  293. dz->param[GLIS_RATE] *= dz->param[GLIS_CONVERTOR];
  294. dz->param[GLIS_HALF_SHIFT] = dz->param[GLIS_SHIFT]/2.0;
  295. dz->param[GLIS_REFPITCH] = log(GLIS_REFERENCE_FRQ);
  296. dz->param[GLIS_HALF_REFPITCH] = log(GLIS_REFERENCE_FRQ/2.0);
  297. if(dz->vflag[GLIS_FTOP]) {
  298. dz->param[GLIS_FRQTOP_TOP] = min(dz->nyquist,dz->param[GLIS_HIFRQ] + ROLL_OFF);
  299. dz->param[GLIS_ROLL_OFF] =
  300. min(dz->param[GLIS_FRQTOP_TOP] - dz->param[GLIS_HIFRQ], ROLL_OFF);
  301. }
  302. dz->param[GLIS_BASEFRQ] = GLIS_REFERENCE_FRQ;
  303. return(FINISHED);
  304. }
  305. /************ SETUP_INTERNAL_ARRAYS_FOR_WAVER *************/
  306. int setup_internal_arrays_for_waver(dataptr dz)
  307. {
  308. int exit_status;
  309. if((dz->parray[WAVER_CHFBOT] = (double *)malloc((dz->clength + 1) * sizeof(double)))==NULL) {
  310. sprintf(errstr,"INSUFFICIENT MEMORY for waver channel bottom array.\n");
  311. return(MEMORY_ERROR);
  312. }
  313. if((exit_status = setup_array_of_botfrq_of_each_channel(&(dz->iparam[WAVER_BOTCHAN]),dz))<0)
  314. return(exit_status);
  315. dz->iparam[WAVER_STRCHANS] = dz->clength - dz->iparam[WAVER_BOTCHAN];
  316. /* No of chans to be stretched */
  317. return(FINISHED);
  318. }
  319. /*********************** SETUP_ARRAY_OF_BOTFRQ_OF_EACH_CHANNEL *********************/
  320. int setup_array_of_botfrq_of_each_channel(int *botchan,dataptr dz)
  321. {
  322. int n = 1;
  323. double boundary;
  324. *botchan = 0;
  325. dz->parray[WAVER_CHFBOT][0] = 0.0;
  326. boundary = dz->chwidth/2.0;
  327. while(boundary < dz->nyquist && n < dz->clength) {
  328. dz->parray[WAVER_CHFBOT][n] = boundary;
  329. if(*botchan==0 && (dz->param[WAVER_LOFRQ] < boundary)) /* IF botchan not set */
  330. *botchan = n; /* AND if botfrq BELOW this chan */
  331. /* set botchan to THIS chan */
  332. boundary += dz->chwidth;
  333. n++;
  334. }
  335. (*botchan)--; /* RESET botchan to its true location */
  336. if(n!=dz->clength) {
  337. sprintf(errstr,"arithmetic error in setup_array_of_botfrq_of_each_channel()\n");
  338. return(PROGRAM_ERROR);
  339. }
  340. dz->parray[WAVER_CHFBOT][dz->clength] = dz->nyquist;
  341. /* upper boundary channel clength-1 */
  342. return(FINISHED);
  343. }
  344. /************************** WARP_PREPROCESS ******************************/
  345. int warp_preprocess(dataptr dz)
  346. {
  347. int exit_status;
  348. if((exit_status = convert_params_for_warp(dz))<0)
  349. return(exit_status);
  350. if((exit_status = setup_internal_arrays_and_params_for_specwarp(dz))<0)
  351. return(exit_status);
  352. return convert_pitchdata_to_midi_for_warp(dz);
  353. }
  354. /************ CONVERT_PARAMS_FOR_WARP *************/
  355. int convert_params_for_warp(dataptr dz)
  356. {
  357. int exit_status;
  358. if((exit_status = convert_msecs_to_secs(WARP_TRNG,dz))<0)
  359. return(exit_status);
  360. return convert_msecs_to_secs(WARP_SRNG,dz);
  361. }
  362. /************ SETUP_INTERNAL_ARRAYS_AND_PARAMS_FOR_SPECWARP *************/
  363. int setup_internal_arrays_and_params_for_specwarp(dataptr dz)
  364. {
  365. double d;
  366. int cc;
  367. if((dz->parray[WARP_P2]
  368. = (double *)malloc(dz->wlength * sizeof(double)))==NULL) {
  369. sprintf(errstr,"INSUFFICIENT MEMORY for warp array.\n");
  370. return(MEMORY_ERROR);
  371. }
  372. if((dz->parray[WARP_AVP] = (double *)malloc(((dz->wlength/BLOKCNT) + 2) * sizeof(double)))==NULL) {
  373. sprintf(errstr,"INSUFFICIENT MEMORY for warp average array.\n");
  374. return(MEMORY_ERROR);
  375. }
  376. if((dz->iparray[WARP_CHANGE] = (int *)malloc(((dz->wlength/BLOKCNT) + 2) * sizeof(int)))==NULL) {
  377. sprintf(errstr,"INSUFFICIENT MEMORY for warp change array.\n");
  378. return(MEMORY_ERROR);
  379. }
  380. dz->flbufptr[WCHTOP] = dz->windowbuf[0]; /* use existing windowbuf for chtop and chbot store */
  381. dz->flbufptr[WCHBOT] = dz->windowbuf[0] + dz->clength;
  382. d = 0.0;
  383. for(cc=0; cc < dz->clength;cc++) {
  384. dz->flbufptr[WCHTOP][cc] = (float)min(dz->nyquist,d + (dz->chwidth * (double)CHANSPAN));
  385. dz->flbufptr[WCHBOT][cc] = (float)max(0.0,d - (dz->chwidth * (double)CHANSPAN));
  386. d += dz->chwidth;
  387. }
  388. dz->iparam[WARP_PART_INBUF] = 0;
  389. dz->iparam[WARP_ATEND] = 0;
  390. return(FINISHED);
  391. }
  392. /************ CONVERT_PITCHDATA_TO_MIDI_FOR_WARP *************/
  393. int convert_pitchdata_to_midi_for_warp(dataptr dz)
  394. {
  395. int exit_status;
  396. int n;
  397. double val;
  398. for(n=0;n<dz->wlength;n++) {
  399. if(dz->pitches[n]<MINPITCH)
  400. dz->pitches[n] = -1.0f;
  401. else {
  402. if((exit_status = hztomidi(&val,dz->pitches[n]))<0)
  403. return(exit_status);
  404. dz->pitches[n] = (float)val;
  405. }
  406. }
  407. return(FINISHED);
  408. }
  409. /************************** INVERT_PREPROCESS ******************************/
  410. int invert_preprocess(dataptr dz)
  411. {
  412. int exit_status;
  413. if((exit_status = setup_internal_arrays_for_invert(dz))<0)
  414. return(exit_status);
  415. return extract_normalising_criteria_by_reading_infile(dz);
  416. }
  417. /************ SETUP_INTERNAL_ARRAYS_FOR_INVERT *************/
  418. int setup_internal_arrays_for_invert(dataptr dz)
  419. {
  420. int cc, vc;
  421. if((dz->parray[INV_AMPRATIO] = (double *)malloc(dz->clength * sizeof(double)))==NULL) {
  422. sprintf(errstr,"INSUFFICIENT MEMORY for invert amplitude ratio array.\n");
  423. return(MEMORY_ERROR);
  424. }
  425. for(cc = 0, vc = 0; cc < dz->clength; cc++, vc += 2) {
  426. dz->amp[cc] = (float)(-BIGAMP);
  427. dz->parray[INV_AMPRATIO][cc] = -BIGAMP;
  428. }
  429. return(FINISHED);
  430. }
  431. /*************** EXTRACT_NORMALISING_CRITERIA_BY_READING_INFILE ********************/
  432. int extract_normalising_criteria_by_reading_infile(dataptr dz)
  433. {
  434. int exit_status;
  435. int samps_read, w_to_buf, wc;
  436. double pre_totalamp;
  437. int cc, vc;
  438. fprintf(stdout,"INFO: Extracting normalisation criteria.\n");
  439. fflush(stdout);
  440. while((samps_read = fgetfbufEx(dz->bigfbuf,dz->big_fsize,dz->ifd[0],0)) > 0) {
  441. dz->flbufptr[0] = dz->bigfbuf;
  442. w_to_buf = samps_read/dz->wanted;
  443. for(wc=0; wc<w_to_buf; wc++) {
  444. switch(dz->mode) {
  445. case(INV_NORMAL):
  446. for(cc = 0, vc = 0; cc < dz->clength; cc++, vc += 2)
  447. dz->amp[cc] = max(dz->amp[cc],dz->flbufptr[0][vc]);
  448. break;
  449. case(INV_KEEPAMP):
  450. if((exit_status = get_totalamp(&pre_totalamp,dz->flbufptr[0],dz->wanted))<0)
  451. return(exit_status);
  452. if(pre_totalamp > MINAMP) {
  453. for(cc = 0, vc = 0; cc < dz->clength; cc++, vc += 2)
  454. dz->parray[INV_AMPRATIO][cc] =
  455. max(dz->parray[INV_AMPRATIO][cc],dz->flbufptr[0][vc]/pre_totalamp);
  456. }
  457. break;
  458. default:
  459. sprintf(errstr,"Unknown progmode in extract_normalising_criteria_by_reading_infile()\n");
  460. return(PROGRAM_ERROR);
  461. }
  462. dz->flbufptr[0] += dz->wanted;
  463. }
  464. }
  465. if(samps_read < 0) {
  466. sprintf(errstr,"Sound read error.\n");
  467. return(SYSTEM_ERROR);
  468. }
  469. if(sndseekEx(dz->ifd[0],0,0)<0) {
  470. sprintf(errstr,"seek failed in extract_normalising_criteria_by_reading_infile()\n");
  471. return(SYSTEM_ERROR);
  472. }
  473. fprintf(stdout,"INFO: Processing the file.\n");
  474. fflush(stdout);
  475. return(FINISHED);
  476. }
  477. /********************************************************************************************/
  478. /********************************** FORMERLY IN procspec.c **********************************/
  479. /********************************************************************************************/
  480. /**************************** SPEC_PROCESS_FILE ****************************/
  481. int spec_process_file(dataptr dz)
  482. {
  483. dz->total_windows = 0;
  484. display_virtual_time(0L,dz);
  485. switch(dz->process) {
  486. case(SHIFT): return specshift(dz);
  487. case(GLIS): return outer_loop(dz);
  488. case(INVERT): return outer_loop(dz);
  489. case(WAVER): return specwaver(dz);
  490. case(WARP): return specwarp(dz);
  491. default:
  492. sprintf(errstr,"Unknown process in procspec()\n");
  493. return(PROGRAM_ERROR);
  494. }
  495. return(FINISHED); /* NOTREACHED */
  496. }
  497. /**************************** INNER_LOOP ****************************/
  498. int inner_loop
  499. (int *peakscore,int *descnt,int *in_start_portion,int *least,int *pitchcnt,int windows_in_buf,dataptr dz)
  500. {
  501. int exit_status;
  502. int local_zero_set = FALSE;
  503. int wc;
  504. for(wc=0; wc<windows_in_buf; wc++) {
  505. if(dz->total_windows==0) {
  506. if((exit_status = skip_or_special_operation_on_window_zero(dz))<0)
  507. return(exit_status);
  508. if(exit_status==TRUE) {
  509. dz->flbufptr[0] += dz->wanted;
  510. dz->total_windows++;
  511. dz->time = (float)(dz->time + dz->frametime);
  512. continue;
  513. }
  514. }
  515. if((exit_status = read_values_from_all_existing_brktables((double)dz->time,dz))<0)
  516. return(exit_status);
  517. switch(dz->process) {
  518. case(GLIS): exit_status = specglis(dz); break;
  519. case(INVERT): exit_status = specinvert(dz); break;
  520. default:
  521. sprintf(errstr,"unknown process in inner_loop()\n");
  522. return(PROGRAM_ERROR);
  523. }
  524. if(exit_status<0)
  525. return(exit_status);
  526. dz->flbufptr[0] += dz->wanted;
  527. dz->total_windows++;
  528. dz->time = (float)(dz->time + dz->frametime);
  529. }
  530. if(!dz->zeroset && local_zero_set==TRUE) {
  531. fprintf(stdout,"WARNING: Zero-amp spectral window(s) encountered: orig window(s) substituted.\n");
  532. fflush(stdout);
  533. dz->zeroset = TRUE;
  534. }
  535. return(FINISHED);
  536. }
  537. /***************** SKIP_OR_SPECIAL_OPERATION_ON_WINDOW_ZERO ************/
  538. int skip_or_special_operation_on_window_zero(dataptr dz)
  539. {
  540. int exit_status = FINISHED;
  541. switch(dz->process) {
  542. case(INVERT): case(GLIS):
  543. switch(dz->process) {
  544. case(GLIS):
  545. if((exit_status = initialise_window_frqs(dz))<0)
  546. return(exit_status);
  547. break;
  548. }
  549. return(TRUE);
  550. }
  551. return(FALSE);
  552. }
  553. /********************************************************************************************/
  554. /********************************** FORMERLY IN pconsistency.c ******************************/
  555. /********************************************************************************************/
  556. /****************************** CHECK_PARAM_VALIDITY_AND_CONSISTENCY *********************************/
  557. int check_param_validity_and_consistency(dataptr dz)
  558. {
  559. handle_pitch_zeros(dz);
  560. return(FINISHED);
  561. }
  562. /********************************************************************************************/
  563. /********************************** FORMERLY IN buffers.c ***********************************/
  564. /********************************************************************************************/
  565. /**************************** ALLOCATE_LARGE_BUFFERS ******************************/
  566. int allocate_large_buffers(dataptr dz)
  567. {
  568. switch(dz->process) {
  569. case(SHIFT): case(GLIS): case(WAVER):
  570. case(INVERT):
  571. return allocate_single_buffer(dz);
  572. case(WARP):
  573. return allocate_warp_buffer(dz);
  574. default:
  575. sprintf(errstr,"Unknown program no. in allocate_large_buffers()\n");
  576. return(PROGRAM_ERROR);
  577. }
  578. return(FINISHED); /* NOTREACHED */
  579. }
  580. /*************************** ALLOCATE_WARP_BUFFER ****************************
  581. *
  582. * 2 equal large buffs and an overflow of dz->wanted floats.
  583. */
  584. int allocate_warp_buffer(dataptr dz)
  585. {
  586. unsigned int buffersize;
  587. if(dz->bptrcnt < 7) {
  588. sprintf(errstr,"Insufficient bufptrs established in allocate_warp_buffer()\n");
  589. return(PROGRAM_ERROR);
  590. }
  591. //TW REVISED, secsize mult of bufs not needed
  592. buffersize = dz->wanted * BUF_MULTIPLIER;
  593. dz->buflen = buffersize;
  594. buffersize *= 2;
  595. buffersize += dz->wanted;
  596. if((dz->bigfbuf = (float*)malloc((size_t)buffersize * sizeof(float)))==NULL) {
  597. sprintf(errstr,"INSUFFICIENT MEMORY for sound buffers.\n");
  598. return(MEMORY_ERROR);
  599. }
  600. dz->big_fsize = dz->buflen;
  601. dz->flbufptr[0] = dz->bigfbuf;
  602. dz->flbufptr[2] = dz->flbufptr[0] + dz->big_fsize;
  603. dz->flbufptr[1] = dz->flbufptr[2];
  604. dz->flbufptr[3] = dz->flbufptr[2] + dz->big_fsize;
  605. return(FINISHED);
  606. }
  607. /********************************************************************************************/
  608. /********************************** FORMERLY IN cmdline.c ***********************************/
  609. /********************************************************************************************/
  610. int get_process_no(char *prog_identifier_from_cmdline,dataptr dz)
  611. {
  612. if (!strcmp(prog_identifier_from_cmdline,"shift")) dz->process = SHIFT;
  613. else if(!strcmp(prog_identifier_from_cmdline,"glis")) dz->process = GLIS;
  614. else if(!strcmp(prog_identifier_from_cmdline,"waver")) dz->process = WAVER;
  615. else if(!strcmp(prog_identifier_from_cmdline,"warp")) dz->process = WARP;
  616. else if(!strcmp(prog_identifier_from_cmdline,"invert")) dz->process = INVERT;
  617. else {
  618. sprintf(errstr,"Unknown program identification string '%s'\n",prog_identifier_from_cmdline);
  619. return(USAGE_ONLY);
  620. }
  621. //TW UPDATE
  622. return(FINISHED);
  623. }
  624. /********************************************************************************************/
  625. /********************************** FORMERLY IN usage.c *************************************/
  626. /********************************************************************************************/
  627. /******************************** USAGE1 ********************************/
  628. int usage1(void)
  629. {
  630. sprintf(errstr,
  631. "\nSTRANGE OPERATIONS ON A SPECTRAL FILE\n\n"
  632. "USAGE: strange NAME (mode) infile outfile parameters: \n"
  633. "\n"
  634. "where NAME can be any one of\n"
  635. "\n"
  636. "shift glis waver invert\n\n"
  637. "Type 'strange shift' for more info on strange shift..ETC.\n");
  638. return(USAGE_ONLY);
  639. }
  640. /******************************** USAGE2 ********************************/
  641. int usage2(char *str)
  642. {
  643. if(!strcmp(str,"shift")) {
  644. fprintf(stdout,
  645. "strange shift 1 infile outfile frqshift [-l]\n"
  646. "strange shift 2-3 infile outfile frqshift frq_divide [-l]\n"
  647. "strange shift 4-5 infile outfile frqshift frqlo frqhi [-l]\n"
  648. "\n"
  649. "LINEAR FREQUENCY SHIFT OF (PART OF) THE SPECTRUM\n"
  650. "\n"
  651. "MODES :-\n"
  652. "1 Shift the whole spectrum.\n"
  653. "2 Shift the spectrum above frq_divide.\n"
  654. "3 Shift the spectrum below frq_divide.\n"
  655. "4 Shift the spectrum only in the range frqlo and frqhi.\n"
  656. "5 Shift the spectrum outside the range frqlo to frqhi.\n"
  657. "\n"
  658. "frqshift linear shift of spectral frequencies (same for all).\n"
  659. "frq_divide frq at which shifting starts or stops.\n"
  660. "frqlo & frqhi define a range inside or outside of which shifting takes place.\n"
  661. "-l log interpolation between varying frq vals (Default: linear).\n"
  662. " (Useful only if any of above input parameters are time-varying,\n"
  663. " AND frqshift vals must be all +ve, or all -ve for this to work).\n"
  664. "\n"
  665. "frqshift,frq_divide,frqlo & frqhi may vary over time.\n");
  666. } else if(!strcmp(str,"glis")) {
  667. fprintf(stdout,
  668. "strange glis 1 infile outfile -fN|-pN [-i] glisrate [-ttopfrq] \n"
  669. "strange glis 2 infile outfile -fN|-pN [-i] glisrate hzstep [-ttopfrq] \n"
  670. "strange glis 3 infile outfile -fN|-pN [-i] glisrate [-ttopfrq] \n"
  671. "\n"
  672. "CREATE GLISSANDI INSIDE THE (CHANGING) SPECTRAL ENVELOPE OF ORIGINAL SOUND\n"
  673. "\n"
  674. "MODES :-\n"
  675. "1 shepard tones.\n"
  676. "2 inharmonic glide.\n"
  677. "3 self-glissando.\n"
  678. "\n"
  679. "-f extract formant envelope linear frqwise,\n"
  680. " using 1 point for every N equally-spaced frequency-channels.\n"
  681. "-p extract formant envelope linear pitchwise,\n"
  682. " using N equally-spaced pitch-bands per octave.\n"
  683. "-i quicksearch for formants (less accurate).\n"
  684. "\n"
  685. "glisrate rate of glissing in semitones per second (-ve val for downward gliss).\n"
  686. "hzstep partials-spacing in inharmonic glide.\n"
  687. " Range: FROM channel-frq-width TO nyquist/2.\n"
  688. "topfrq top of spectrum: must be > 2*chanwidth of analysis (default:nyquist)\n"
  689. "\n"
  690. "glisrate may vary over time.\n");
  691. } else if(!strcmp(str,"waver")) {
  692. fprintf(stdout,
  693. "strange waver 1 infile outfile vibfrq stretch botfrq\n"
  694. "strange waver 2 infile outfile vibfrq stretch botfrq expon\n"
  695. "\n"
  696. "OSCILLATE BETWEEN HARMONIC AND INHARMONIC STATE\n"
  697. "\n"
  698. "MODES :-\n"
  699. "1 Standard spectral stretching for inharmonic state.\n"
  700. "2 Specify spectral stretching for inharmonic state.\n"
  701. "\n"
  702. "vibfrq is frq of oscillation. \n"
  703. "stretch is maximum spectral stretch in inharmonic state. \n"
  704. "botfrq is frq above which spectral stretching happens.\n"
  705. "expon defines type of stretch (must be > 0.0).\n"
  706. "\n"
  707. "vibfrq and stretch may vary over time.\n");
  708. /***
  709. } else if(!strcmp(str,"warp")) {
  710. fprintf(stdout,
  711. "strange warp infile pitchfile outfile [-pprange] [-ttrange] [-ssrange]\n\n"
  712. "PRODUCE AN APPROXIMATE COPY OF SPECTRUM.\n\n"
  713. "PITCHFILE must be derived from infile.\n"
  714. "PRANGE Interval (semitones) over which pitch varies\n"
  715. " +- randomly from orig. Range > 0.0.\n"
  716. "TRANGE Time-interval (msecs) by which pitch can stray from orig time.\n"
  717. " Range: duration of 1 analysis window to dur of entire file.\n"
  718. "SRANGE Time-interval (msecs) over which pitch contour scanned.\n"
  719. " Pitchshift by %.2lf semitones within scanrange, indicates\n"
  720. " pitch rise or fall.\n"
  721. " Range: from dur of %d anal windows to (approx) dur of entire file.\n\n"
  722. "prange, trange and srange may vary over time.\n",WARP_MININT,2*BLOKCNT);
  723. ***/
  724. } else if(!strcmp(str,"invert")) {
  725. fprintf(stdout,
  726. "strange invert mode infile outfile\n"
  727. "\n"
  728. "INVERT THE SPECTRUM\n"
  729. "\n"
  730. "MODES :-\n"
  731. "1 Normal inversion.\n"
  732. "2 Output sound retains amplitude envelope of source sound.\n");
  733. } else
  734. fprintf(stdout,"Unknown option '%s'\n",str);
  735. return(USAGE_ONLY);
  736. }
  737. /******************************** USAGE3 ********************************/
  738. int usage3(char *str1,char *str2)
  739. {
  740. sprintf(errstr,"Insufficient parameters on command line.\n");
  741. return(USAGE_ONLY);
  742. }