aboutsummaryrefslogtreecommitdiff
path: root/examples/example2.cpp
blob: 13e822840504ddce4d68c803ac95d418e0bbabd3 (plain)
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
#include <cstdlib>
#include <fstream>
#include <iostream>
#include "guttersynth.hpp"

using namespace std;


class Wavefile {
private:
    std::ofstream output;
    size_t data_chunk_pos;
    
    template <typename Word>
    void write_word(Word value, unsigned size=sizeof(Word)) {
        for (; size; --size, value >>= 8)
        output.put( static_cast <char> (value & 0xFF) );
    }
    
public:
    Wavefile(string path, int samplerate) : output(path.c_str(), ios::binary) {
    
        int bitdepth = 16;
        // Write the file headers
        output << "RIFF----WAVEfmt ";       // (chunk size to be filled in later)
        write_word(16, 4);                  // no extension data
        write_word(1, 2);                   // PCM - integer samples
        write_word(1, 2);                   // one channel (mono file)
        write_word(samplerate, 4);          // samples per second (Hz)
        write_word(samplerate * (bitdepth / 8), 4); // byte rate
        write_word(2, 2);        // data block size ; was bitdepth / 8
        write_word(bitdepth, 2);            // number of bits per sample (use a multiple of 8)
        
        data_chunk_pos = output.tellp();
        output << "data----";               // (chunk size to be filled in later)
    }
    
    ~Wavefile() {
        std::cout << "closing\n";
        size_t file_length = output.tellp();
        
        // Fix the data chunk header to contain the data size
        output.seekp(data_chunk_pos + 4);
        write_word(file_length - (data_chunk_pos + 8), 4);

        // Fix the file header to contain the proper RIFF chunk size, which is (file size - 8) bytes
        output.seekp(0 + 4);
        write_word(file_length - 8, 4);
        output.close();
    }
    
    void write(double value) {
        write_word((short) (value * SHRT_MAX * 0.5), 2);
    }
    
    
};


class Example2 {
private:
    GutterSynth* gs;
    Wavefile* wav;
    int samplerate;
    
public:
    
    Example2(std::string path, int samplerate) {
        this->samplerate = samplerate;
        gs = new GutterSynth(2, 24, samplerate);
        wav = new Wavefile(path, samplerate);
    }
    
    double rangerandom(double min, double max) {
        double random = ((double) rand()) / (double) RAND_MAX;
        double diff = max - min;
        double r = random * diff;
        return min + r;
    }
    
    void run() {
        int total_samples = 30 * samplerate;
        int sample = 0;
        FLT data;
        gs->filtersOn(true);
        gs->smoothing(true);
        gs->distortionMethod(2);
        gs->randomiseFilters();

        for (int sample = 0; sample < total_samples; sample ++) {
            data = gs->process();
            wav->write(data);

            if (sample % samplerate == 0) { // randomise every second
                gs->randomiseFilters();
                gs->gamma(rangerandom(2, 10));
                gs->omega(rangerandom(0.3, 0.9));
                gs->c(rangerandom(0.5, 0.9));
                gs->dt(rangerandom(100, 500));
                gs->singleGain(rangerandom(0.2, 0.7));
            }
        }
        delete wav;
    }
};


/*
 * 
 */
int main(int argc, char** argv) {
    Example2* e = new Example2("guttersynth-example2.wav", 44100);
    e->run();
    return 0;
}