1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133
|
/*
* June 1, 1992
* Copyright 1992 Guido van Rossum And Sundry Contributors
* This source code is freely redistributable and may be used for
* any purpose. This copyright notice must be maintained.
* Guido van Rossum And Sundry Contributors are not responsible for
* the consequences of using this software.
*/
/*
* "reverse" effect, uses a temporary file created by tmpfile().
*/
#include <math.h>
#include <stdio.h>
#ifdef HAVE_UNISTD_H
#include <unistd.h> /* For SEEK_* defines if not found in stdio */
#endif
#include "st.h"
IMPORT FILE *tmpfile();
/* Private data */
typedef struct reversestuff {
FILE *fp;
LONG pos;
int phase;
} *reverse_t;
#define WRITING 0
#define READING 1
/*
* Process options: none in our case.
*/
void reverse_getopts(effp, n, argv)
eff_t effp;
int n;
char **argv;
{
if (n)
fail("Reverse effect takes no options.");
}
/*
* Prepare processing: open temporary file.
*/
void reverse_start(effp)
eff_t effp;
{
reverse_t reverse = (reverse_t) effp->priv;
reverse->fp = tmpfile();
if (reverse->fp == NULL)
fail("Reverse effect can't create temporary file\n");
reverse->phase = WRITING;
}
/*
* Effect flow: a degenerate case: write input samples on temporary file,
* don't generate any output samples.
*/
void reverse_flow(effp, ibuf, obuf, isamp, osamp)
eff_t effp;
LONG *ibuf, *obuf;
int *isamp, *osamp;
{
reverse_t reverse = (reverse_t) effp->priv;
if (reverse->phase != WRITING)
fail("Internal error: reverse_flow called in wrong phase");
if (fwrite((char *)ibuf, sizeof(LONG), *isamp, reverse->fp)
!= *isamp)
fail("Reverse effect write error on temporary file\n");
*osamp = 0;
}
/*
* Effect drain: generate the actual samples in reverse order.
*/
void reverse_drain(effp, obuf, osamp)
eff_t effp;
LONG *obuf;
int *osamp;
{
reverse_t reverse = (reverse_t) effp->priv;
int len, nbytes;
register int i, j;
LONG temp;
if (reverse->phase == WRITING) {
fflush(reverse->fp);
fseek(reverse->fp, 0L, SEEK_END);
reverse->pos = ftell(reverse->fp);
if (reverse->pos % sizeof(LONG) != 0)
fail("Reverse effect finds odd temporary file\n");
reverse->phase = READING;
}
len = *osamp;
nbytes = len * sizeof(LONG);
if (reverse->pos < nbytes) {
nbytes = reverse->pos;
len = nbytes / sizeof(LONG);
}
reverse->pos -= nbytes;
fseek(reverse->fp, reverse->pos, SEEK_SET);
if (fread((char *)obuf, sizeof(LONG), len, reverse->fp) != len)
fail("Reverse effect read error from temporary file\n");
for (i = 0, j = len-1; i < j; i++, j--) {
temp = obuf[i];
obuf[i] = obuf[j];
obuf[j] = temp;
}
*osamp = len;
}
/*
* Close and unlink the temporary file.
*/
void reverse_stop(effp)
eff_t effp;
{
reverse_t reverse = (reverse_t) effp->priv;
fclose(reverse->fp);
}
|