mirror of
				https://github.com/cookiengineer/audacity
				synced 2025-11-04 16:14:00 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			775 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			775 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
#include "assert.h"
 | 
						|
#include "stdlib.h"
 | 
						|
#include "string.h"
 | 
						|
#include "ctype.h"
 | 
						|
#include "trace.h"
 | 
						|
#include <string>
 | 
						|
#include <fstream>
 | 
						|
#include <algorithm>
 | 
						|
#include "strparse.h"
 | 
						|
#include "allegro.h"
 | 
						|
#include "algrd_internal.h"
 | 
						|
 | 
						|
using namespace std;
 | 
						|
 | 
						|
#define streql(s1, s2) (strcmp(s1, s2) == 0)
 | 
						|
#define field_max 80
 | 
						|
 | 
						|
class Alg_reader {
 | 
						|
public:
 | 
						|
    istream *file;
 | 
						|
    string input_line;
 | 
						|
    int line_no;
 | 
						|
    String_parse line_parser;
 | 
						|
    bool line_parser_flag;
 | 
						|
    string field;
 | 
						|
    bool error_flag;
 | 
						|
    Alg_seq_ptr seq;
 | 
						|
    double tsnum;
 | 
						|
    double tsden;
 | 
						|
    double offset;
 | 
						|
    bool offset_found;
 | 
						|
 | 
						|
    Alg_reader(istream *a_file, Alg_seq_ptr new_seq);
 | 
						|
    void readline();
 | 
						|
    Alg_parameters_ptr process_attributes(Alg_parameters_ptr attributes,
 | 
						|
                                          double time);
 | 
						|
    bool parse();
 | 
						|
    long parse_chan(string &field);
 | 
						|
    long parse_int(string &field);
 | 
						|
    int find_real_in(string &field, int n);
 | 
						|
    double parse_real(string &field);
 | 
						|
    void parse_error(string &field, long offset, char *message);
 | 
						|
    double parse_dur(string &field, double base);
 | 
						|
    double parse_after_dur(double dur, string &field, int n, double base);
 | 
						|
    double parse_loud(string &field);
 | 
						|
    long parse_key(string &field);
 | 
						|
    double parse_pitch(string &field);
 | 
						|
    long parse_after_key(int key, string &field, int n);
 | 
						|
    long find_int_in(string &field, int n);
 | 
						|
    bool parse_attribute(string &field, Alg_parameter_ptr parm);
 | 
						|
    bool parse_val(Alg_parameter_ptr param, string &s, int i);
 | 
						|
    bool check_type(char type_char, Alg_parameter_ptr param);
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
double Alg_reader::parse_pitch(string &field)
 | 
						|
{
 | 
						|
    if (isdigit(field[1])) {
 | 
						|
        int last = find_real_in(field, 1);
 | 
						|
        string real_string = field.substr(1, last - 1);
 | 
						|
        return atof(real_string.c_str());
 | 
						|
    } else {
 | 
						|
        return (double) parse_key(field);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
// it is the responsibility of the caller to delete
 | 
						|
// the seq
 | 
						|
Alg_reader::Alg_reader(istream *a_file, Alg_seq_ptr new_seq)
 | 
						|
{
 | 
						|
    file = a_file; // save the file
 | 
						|
    line_parser_flag = false;
 | 
						|
    line_no = 0;
 | 
						|
    tsnum = 4; // default time signature
 | 
						|
    tsden = 4;
 | 
						|
    seq = new_seq;
 | 
						|
    offset = 0.0;
 | 
						|
    offset_found = false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
Alg_error alg_read(istream &file, Alg_seq_ptr new_seq, double *offset_ptr)
 | 
						|
    // read a sequence from allegro file
 | 
						|
{
 | 
						|
    assert(new_seq);
 | 
						|
    Alg_reader alg_reader(&file, new_seq);
 | 
						|
    bool err = alg_reader.parse();
 | 
						|
    if (!err && offset_ptr) {
 | 
						|
        *offset_ptr = alg_reader.offset;
 | 
						|
    }
 | 
						|
    return (err ? alg_error_syntax : alg_no_error);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
void Alg_reader::readline()
 | 
						|
{
 | 
						|
    // a word about memory management: this Alg_reader has a
 | 
						|
    // member variable input_line that holds a line of input
 | 
						|
    // it is reused for each line. input_line is parsed by
 | 
						|
    // line_parser, which holds a reference to input_line
 | 
						|
    line_parser_flag = false;
 | 
						|
    if (getline(*file, input_line)) {
 | 
						|
        line_parser.init(&input_line);
 | 
						|
        line_parser_flag = true;
 | 
						|
        error_flag = false;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
Alg_parameters_ptr Alg_reader::process_attributes(
 | 
						|
        Alg_parameters_ptr attributes, double time)
 | 
						|
{
 | 
						|
    // print "process_attributes:", attributes
 | 
						|
    bool ts_flag = false;
 | 
						|
    if (attributes) {
 | 
						|
        Alg_parameters_ptr a;
 | 
						|
        bool in_seconds = seq->get_units_are_seconds();
 | 
						|
        if (a = Alg_parameters::remove_key(&attributes, "tempor")) {
 | 
						|
            double tempo = a->parm.r;
 | 
						|
            seq->insert_tempo(tempo, seq->get_time_map()->time_to_beat(time));
 | 
						|
        }
 | 
						|
        if (a = Alg_parameters::remove_key(&attributes, "beatr")) {
 | 
						|
            double beat = a->parm.r;
 | 
						|
            seq->insert_beat(time, beat);
 | 
						|
        }
 | 
						|
        if (a = Alg_parameters::remove_key(&attributes, "timesig_numr")) {
 | 
						|
            tsnum = a->parm.r;
 | 
						|
            ts_flag = true;
 | 
						|
        }
 | 
						|
        if (a = Alg_parameters::remove_key(&attributes, "timesig_denr")) {
 | 
						|
            tsden = a->parm.r;
 | 
						|
            ts_flag = true;
 | 
						|
        }
 | 
						|
        if (ts_flag) {
 | 
						|
            seq->set_time_sig(seq->get_time_map()->time_to_beat(time),
 | 
						|
            tsnum, tsden);
 | 
						|
        }
 | 
						|
        if (in_seconds) seq->convert_to_seconds();
 | 
						|
    }
 | 
						|
    return attributes; // in case it was modified
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
bool Alg_reader::parse()
 | 
						|
{
 | 
						|
    int voice = 0;
 | 
						|
    int key = 60;
 | 
						|
    double loud = 100.0;
 | 
						|
    double pitch = 60.0;
 | 
						|
    double dur = 1.0;
 | 
						|
    double time = 0.0;
 | 
						|
    int track_num = 0;
 | 
						|
    seq->convert_to_seconds();
 | 
						|
    //seq->set_real_dur(0.0); // just in case it's not initialized already
 | 
						|
    readline();
 | 
						|
    bool valid = false; // ignore blank lines
 | 
						|
    while (line_parser_flag) {
 | 
						|
        bool time_flag = false;
 | 
						|
        bool next_flag = false;
 | 
						|
        double next;
 | 
						|
        bool voice_flag = false;
 | 
						|
        bool loud_flag = false;
 | 
						|
        bool dur_flag = false;
 | 
						|
        bool new_pitch_flag = false; // "P" syntax or "A"-"G" syntax
 | 
						|
        double new_pitch = 0.0;
 | 
						|
        bool new_key_flag = false;   // "K" syntax
 | 
						|
        int new_key = 0;
 | 
						|
        Alg_parameters_ptr attributes = NULL;
 | 
						|
        if (line_parser.peek() == '#') {
 | 
						|
            // look for #track
 | 
						|
            line_parser.get_nonspace_quoted(field);
 | 
						|
            if (streql(field.c_str(), "#track")) {
 | 
						|
                line_parser.get_nonspace_quoted(field); // number
 | 
						|
                field.insert(0, " "); // need char at beginning because
 | 
						|
                // parse_int ignores the first character of the argument
 | 
						|
                track_num = parse_int(field);
 | 
						|
                seq->add_track(track_num);
 | 
						|
 | 
						|
                // maybe we have a sequence or track name
 | 
						|
                line_parser.get_remainder(field);
 | 
						|
                // if there is a non-space character after #track n then
 | 
						|
                // use it as sequence or track name. Note that because we
 | 
						|
                // skip over spaces, a sequence or track name cannot begin
 | 
						|
                // with leading blanks. Another decision is that the name
 | 
						|
                // must be at time zero
 | 
						|
                if (field.length() > 0) {
 | 
						|
                    // insert the field as sequence name or track name
 | 
						|
                    Alg_update_ptr update = new Alg_update;
 | 
						|
                    update->chan = -1;
 | 
						|
                    update->time = 0;
 | 
						|
                    update->set_identifier(-1);
 | 
						|
                    // sequence name is whatever is on track 0
 | 
						|
                    // other tracks have track names
 | 
						|
                    const char *attr =
 | 
						|
                            (track_num == 0 ? "seqnames" : "tracknames");
 | 
						|
                    update->parameter.set_attr(
 | 
						|
                            symbol_table.insert_string(attr));
 | 
						|
                    update->parameter.s = heapify(field.c_str());
 | 
						|
                    seq->add_event(update, track_num);
 | 
						|
                }
 | 
						|
            } else if (streql(field.c_str(), "#offset")) {
 | 
						|
                if (offset_found) {
 | 
						|
                    parse_error(field, 0, "#offset specified twice");
 | 
						|
                }
 | 
						|
                offset_found = true;
 | 
						|
                line_parser.get_nonspace_quoted(field); // number
 | 
						|
                field.insert(0, " "); // need char at beginning because
 | 
						|
                // parse_real ignores first character in the argument
 | 
						|
                offset = parse_real(field);
 | 
						|
            }
 | 
						|
        } else {
 | 
						|
            // we must have a track to insert into
 | 
						|
            if (seq->tracks() == 0) seq->add_track(0);
 | 
						|
            line_parser.get_nonspace_quoted(field);
 | 
						|
            char pk = line_parser.peek();
 | 
						|
            // attributes are parsed as two adjacent nonspace_quoted tokens
 | 
						|
            // so we have to conditionally call get_nonspace_quoted() again
 | 
						|
            if (pk && !isspace(pk)) {
 | 
						|
                string field2;
 | 
						|
                line_parser.get_nonspace_quoted(field2);
 | 
						|
                field.append(field2);
 | 
						|
            }
 | 
						|
            while (field[0]) {
 | 
						|
                char first = toupper(field[0]);
 | 
						|
                if (strchr("ABCDEFGKLPUSIQHW-", first)) {
 | 
						|
                    valid = true; // it's a note or event
 | 
						|
                }
 | 
						|
                if (first == 'V') {
 | 
						|
                    if (voice_flag) {
 | 
						|
                        parse_error(field, 0, "Voice specified twice");
 | 
						|
                    } else {
 | 
						|
                        voice = parse_chan(field);
 | 
						|
                    }
 | 
						|
                    voice_flag = true;
 | 
						|
                } else if (first == 'T') {
 | 
						|
                    if (time_flag) {
 | 
						|
                        parse_error(field, 0, "Time specified twice");
 | 
						|
                    } else {
 | 
						|
                        time = parse_dur(field, 0.0);
 | 
						|
                    }
 | 
						|
                    time_flag = true;
 | 
						|
                } else if (first == 'N') {
 | 
						|
                    if (next_flag) {
 | 
						|
                        parse_error(field, 0, "Next specified twice");
 | 
						|
                    } else {
 | 
						|
                        next = parse_dur(field, time);
 | 
						|
                    }
 | 
						|
                    next_flag = true;
 | 
						|
                } else if (first == 'K') {
 | 
						|
                    if (new_key_flag) {
 | 
						|
                        parse_error(field, 0, "Key specified twice");
 | 
						|
                    } else {
 | 
						|
                        new_key = parse_key(field);
 | 
						|
                        new_key_flag = true;
 | 
						|
                    }
 | 
						|
                } else if (first == 'L') {
 | 
						|
                    if (loud_flag) {
 | 
						|
                        parse_error(field, 0, "Loudness specified twice");
 | 
						|
                    } else {
 | 
						|
                        loud = parse_loud(field);
 | 
						|
                    }
 | 
						|
                    loud_flag = true;
 | 
						|
                } else if (first == 'P') {
 | 
						|
                    if (new_pitch_flag) {
 | 
						|
                        parse_error(field, 0, "Pitch specified twice");
 | 
						|
                    } else {
 | 
						|
                        new_pitch = parse_pitch(field);
 | 
						|
                        new_pitch_flag = true;
 | 
						|
                    }
 | 
						|
                } else if (first == 'U') {
 | 
						|
                    if (dur_flag) {
 | 
						|
                        parse_error(field, 0, "Dur specified twice");
 | 
						|
                    } else {
 | 
						|
                        dur = parse_dur(field, time);
 | 
						|
                        dur_flag = true;
 | 
						|
                    }
 | 
						|
                } else if (strchr("SIQHW", first)) {
 | 
						|
                    if (dur_flag) {
 | 
						|
                        parse_error(field, 0, "Dur specified twice");
 | 
						|
                    } else {
 | 
						|
                        // prepend 'U' to field, copy EOS too
 | 
						|
                        field.insert((unsigned int) 0, 1, 'U');
 | 
						|
                        dur = parse_dur(field, time);
 | 
						|
                        dur_flag = true;
 | 
						|
                    }
 | 
						|
                } else if (strchr("ABCDEFG", first)) {
 | 
						|
                    if (new_pitch_flag) {
 | 
						|
                        parse_error(field, 0, "Pitch specified twice");
 | 
						|
                    } else {
 | 
						|
                        // prepend 'P' to field
 | 
						|
                        field.insert((unsigned int) 0, 1, 'P');
 | 
						|
                        new_pitch = parse_pitch(field);
 | 
						|
                        new_pitch_flag = true;
 | 
						|
                    }
 | 
						|
                } else if (first == '-') {
 | 
						|
                    Alg_parameter parm;
 | 
						|
                    if (parse_attribute(field, &parm)) { // enter attribute-value pair
 | 
						|
                        attributes = new Alg_parameters(attributes);
 | 
						|
                        attributes->parm = parm;
 | 
						|
                        parm.s = NULL; // protect string from deletion by destructor
 | 
						|
                    }
 | 
						|
                } else {
 | 
						|
                    parse_error(field, 0, "Unknown field");
 | 
						|
                }
 | 
						|
 | 
						|
                if (error_flag) {
 | 
						|
                    field[0] = 0; // exit the loop
 | 
						|
                } else {
 | 
						|
                    line_parser.get_nonspace_quoted(field);
 | 
						|
                    pk = line_parser.peek();
 | 
						|
                    // attributes are parsed as two adjacent nonspace_quoted 
 | 
						|
                    // tokens so we have to conditionally call 
 | 
						|
                    // get_nonspace_quoted() again
 | 
						|
                    if (pk && !isspace(pk)) {
 | 
						|
                        string field2;
 | 
						|
                        line_parser.get_nonspace_quoted(field2);
 | 
						|
                        field.append(field2);
 | 
						|
                    }
 | 
						|
                }
 | 
						|
            }
 | 
						|
            // a case analysis:
 | 
						|
            // Key < 128 implies pitch unless pitch is explicitly given
 | 
						|
            // Pitch implies Key unless key is explicitly given,
 | 
						|
            // Pitch is rounded to nearest integer to determine the Key
 | 
						|
            //    if necessary, so MIDI files will lose the pitch fraction
 | 
						|
            // A-G is a Pitch specification (therefore it implies Key)
 | 
						|
            //   K60 P60 -- both are specified, use 'em
 | 
						|
            //   K60 P60 C4 -- overconstrained, an error
 | 
						|
            //   K60 C4 -- OK, but K60 is already implied by C4
 | 
						|
            //   K60 -- OK, pitch is 60
 | 
						|
            //   C4 P60 -- over constrained
 | 
						|
            //   P60 -- OK, key is 60
 | 
						|
            //   P60.1 -- OK, key is 60
 | 
						|
            //   C4 -- OK, key is 60, pitch is 60
 | 
						|
            //   <nothing> -- OK, key and pitch from before
 | 
						|
            //   K200 P60 -- ok, pitch is 60
 | 
						|
            //   K200 with neither P60 nor C4 uses 
 | 
						|
            //       pitch from before
 | 
						|
 | 
						|
            // figure out what the key/instance is:
 | 
						|
            if (new_key_flag) { // it was directly specified
 | 
						|
                key = new_key;
 | 
						|
            } else if (new_pitch_flag) {
 | 
						|
                // pitch was specified, but key was not; get key from pitch
 | 
						|
                key = (int) (new_pitch + 0.5); // round to integer key number
 | 
						|
            }
 | 
						|
            if (new_pitch_flag) {
 | 
						|
                pitch = new_pitch;
 | 
						|
            } else if (key < 128 && new_key_flag) {
 | 
						|
                // no explicit pitch, but key < 128, so it implies pitch
 | 
						|
                pitch = key;
 | 
						|
                new_pitch_flag = true;
 | 
						|
            }
 | 
						|
            // now we've acquired new parameters
 | 
						|
            // if (it is a note, then enter the note
 | 
						|
            if (valid) {
 | 
						|
                // change tempo or beat
 | 
						|
                attributes = process_attributes(attributes, time);
 | 
						|
                // if there's a duration or pitch, make a note:
 | 
						|
                if (new_pitch_flag || dur_flag) {
 | 
						|
                    Alg_note_ptr note_ptr = new Alg_note;
 | 
						|
                    note_ptr->chan = voice;
 | 
						|
                    note_ptr->time = time;
 | 
						|
                    note_ptr->dur = dur;
 | 
						|
                    note_ptr->set_identifier(key);
 | 
						|
                    note_ptr->pitch = (float) pitch;
 | 
						|
                    note_ptr->loud = (float) loud;
 | 
						|
                    note_ptr->parameters = attributes;
 | 
						|
                    seq->add_event(note_ptr, track_num); // sort later
 | 
						|
                    if (seq->get_real_dur() < (time + dur)) seq->set_real_dur(time + dur);
 | 
						|
                } else {
 | 
						|
                    int update_key = -1;
 | 
						|
                    // key must appear explicitly; otherwise
 | 
						|
                    //    update applies to channel
 | 
						|
                    if (new_key_flag) {
 | 
						|
                        update_key = key;
 | 
						|
                    }
 | 
						|
                    if (loud_flag) {
 | 
						|
                        Alg_update_ptr new_upd = new Alg_update;
 | 
						|
                        new_upd->chan = voice;
 | 
						|
                        new_upd->time = time;
 | 
						|
                        new_upd->set_identifier(update_key);
 | 
						|
                        new_upd->parameter.set_attr(symbol_table.insert_string("loudr"));
 | 
						|
                        new_upd->parameter.r = pitch;
 | 
						|
                        seq->add_event(new_upd, track_num);
 | 
						|
                        if (seq->get_real_dur() < time) seq->set_real_dur(time);
 | 
						|
                    }
 | 
						|
                    if (attributes) {
 | 
						|
                        while (attributes) {
 | 
						|
                            Alg_update_ptr new_upd = new Alg_update;
 | 
						|
                            new_upd->chan = voice;
 | 
						|
                            new_upd->time = time;
 | 
						|
                            new_upd->set_identifier(update_key);
 | 
						|
                            new_upd->parameter = attributes->parm;
 | 
						|
                            seq->add_event(new_upd, track_num);
 | 
						|
                            Alg_parameters_ptr p = attributes;
 | 
						|
                            attributes = attributes->next;
 | 
						|
                            p->parm.s = NULL; // so we don't delete the string
 | 
						|
                            delete p;
 | 
						|
                        }
 | 
						|
                    }
 | 
						|
                }
 | 
						|
                if (next_flag) {
 | 
						|
                    time = time + next;
 | 
						|
                } else if (dur_flag || new_pitch_flag) { // a note: incr by dur
 | 
						|
                    time = time + dur;
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
        readline();
 | 
						|
    }
 | 
						|
    if (!error_flag) { // why not convert even if there was an error? -RBD
 | 
						|
        seq->convert_to_seconds(); // make sure format is correct
 | 
						|
    }
 | 
						|
    // real_dur is valid, translate to beat_dur
 | 
						|
    seq->set_beat_dur((seq->get_time_map())->time_to_beat(seq->get_real_dur()));
 | 
						|
    return error_flag;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
long Alg_reader::parse_chan(string &field)
 | 
						|
{
 | 
						|
    const char *int_string = field.c_str() + 1;
 | 
						|
    char *msg = "Integer or - expected";
 | 
						|
    const char *p = int_string;
 | 
						|
    char c;
 | 
						|
    // check that all chars in int_string are digits or '-':
 | 
						|
    while (c = *p++) {
 | 
						|
        if (!isdigit(c) && c != '-') {
 | 
						|
            parse_error(field, p - field.c_str() - 1, msg);
 | 
						|
            return 0;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    p--; // p now points to end-of-string character
 | 
						|
    if (p - int_string == 0) {
 | 
						|
        // bad: string length is zero
 | 
						|
        parse_error(field, 1, msg);
 | 
						|
        return 0;
 | 
						|
    }
 | 
						|
    if (p - int_string == 1 && int_string[0] == '-') {
 | 
						|
        // special case: entire string is "-", interpret as -1
 | 
						|
        return -1;
 | 
						|
    }
 | 
						|
    return atoi(int_string);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
long Alg_reader::parse_int(string &field)
 | 
						|
{
 | 
						|
    const char *int_string = field.c_str() + 1;
 | 
						|
    char *msg = "Integer expected";
 | 
						|
    const char *p = int_string;
 | 
						|
    char c;
 | 
						|
    // check that all chars in int_string are digits:
 | 
						|
    while (c = *p++) {
 | 
						|
        if (!isdigit(c)) {
 | 
						|
            parse_error(field, p - field.c_str() - 1, msg);
 | 
						|
            return 0;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    p--; // p now points to end-of-string character
 | 
						|
    if (p - int_string == 0) {
 | 
						|
        // bad: string length is zero
 | 
						|
        parse_error(field, 1, msg);
 | 
						|
        return 0;
 | 
						|
    }
 | 
						|
    return atoi(int_string);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
int Alg_reader::find_real_in(string &field, int n)
 | 
						|
{
 | 
						|
    // scans from offset n to the end of a real constant
 | 
						|
    bool decimal = false;
 | 
						|
    int len = field.length();
 | 
						|
    if (n < len && field[n] == '-') n += 1; // parse one minus sign
 | 
						|
    for (int i = n; i < len; i++) {
 | 
						|
        char c = field[i];
 | 
						|
        if (!isdigit(c)) {
 | 
						|
            if (c == '.' && !decimal) {
 | 
						|
                decimal = true;
 | 
						|
            } else {
 | 
						|
                return i;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
    return len;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
double Alg_reader::parse_real(string &field)
 | 
						|
{
 | 
						|
    char *msg = "Real expected";
 | 
						|
    int last = find_real_in(field, 1);
 | 
						|
    string real_string = field.substr(1, last - 1);
 | 
						|
    if (last <= 1 || last < (int) field.length()) {
 | 
						|
       parse_error(field, 1, msg);
 | 
						|
       return 0;
 | 
						|
    }
 | 
						|
    return atof(real_string.c_str());
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
void Alg_reader::parse_error(string &field, long offset, char *message)
 | 
						|
{
 | 
						|
    int position = line_parser.pos - field.length() + offset;
 | 
						|
    error_flag = true;
 | 
						|
    puts(line_parser.str->c_str());
 | 
						|
    for (int i = 0; i < position; i++) {
 | 
						|
        putc(' ', stdout);
 | 
						|
    }
 | 
						|
    putc('^', stdout);
 | 
						|
    printf("    %s\n", message);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
double duration_lookup[] = { 0.25, 0.5, 1.0, 2.0, 4.0 };
 | 
						|
 | 
						|
 | 
						|
double Alg_reader::parse_dur(string &field, double base)
 | 
						|
{
 | 
						|
    char *msg = "Duration expected";
 | 
						|
    char *durs = "SIQHW";
 | 
						|
    char *p;
 | 
						|
    int last;
 | 
						|
    double dur;
 | 
						|
    if (field.length() < 2) {
 | 
						|
        // fall through to error message
 | 
						|
        return -1;
 | 
						|
    } else if (isdigit(field[1])) {
 | 
						|
        last = find_real_in(field, 1);
 | 
						|
        string real_string = field.substr(1, last - 1);
 | 
						|
        dur = atof(real_string.c_str());
 | 
						|
        // convert dur from seconds to beats
 | 
						|
        dur = seq->get_time_map()->time_to_beat(base + dur) - 
 | 
						|
              seq->get_time_map()->time_to_beat(base);
 | 
						|
    } else if (p = strchr(durs, toupper(field[1]))) {
 | 
						|
        dur = duration_lookup[p - durs];
 | 
						|
        last = 2;
 | 
						|
    } else {
 | 
						|
        parse_error(field, 1, msg);
 | 
						|
        return 0;
 | 
						|
    }
 | 
						|
    dur = parse_after_dur(dur, field, last, base);
 | 
						|
    dur = seq->get_time_map()->beat_to_time(
 | 
						|
              seq->get_time_map()->time_to_beat(base) + dur) - base;
 | 
						|
    return dur;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
double Alg_reader::parse_after_dur(double dur, string &field, 
 | 
						|
                                   int n, double base)
 | 
						|
{
 | 
						|
    if ((int) field.length() == n) {
 | 
						|
        return dur;
 | 
						|
    }
 | 
						|
    if (toupper(field[n]) == 'T') {
 | 
						|
        return parse_after_dur(dur * 2/3, field, n + 1, base);
 | 
						|
    }
 | 
						|
    if (field[n] == '.') {
 | 
						|
        return parse_after_dur(dur * 1.5, field, n + 1, base);
 | 
						|
    }
 | 
						|
    if (isdigit(field[n])) {
 | 
						|
        int last = find_real_in(field, n);
 | 
						|
        string a_string = field.substr(n, last - n);
 | 
						|
        double f = atof(a_string.c_str());
 | 
						|
        return parse_after_dur(dur * f, field, last, base);
 | 
						|
    }
 | 
						|
    if (field[n] == '+') {
 | 
						|
        string a_string = field.substr(n + 1);
 | 
						|
        return dur + parse_dur(
 | 
						|
                a_string, seq->get_time_map()->beat_to_time(
 | 
						|
                        seq->get_time_map()->time_to_beat(base) + dur));
 | 
						|
    }
 | 
						|
    parse_error(field, n, "Unexpected character in duration");
 | 
						|
    return dur;
 | 
						|
}
 | 
						|
 | 
						|
struct loud_lookup_struct {
 | 
						|
    char *str;
 | 
						|
    int val;
 | 
						|
} loud_lookup[] = { {"FFF", 127}, {"FF", 120}, {"F", 110}, {"MF", 100}, 
 | 
						|
                    {"MP", 90}, {"P", 80}, {"PP", 70}, {"PPP", 60}, 
 | 
						|
                    {NULL, 0} };
 | 
						|
 | 
						|
 | 
						|
double Alg_reader::parse_loud(string &field)
 | 
						|
{
 | 
						|
    char *msg = "Loudness expected";
 | 
						|
    if (isdigit(field[1])) {
 | 
						|
        return parse_int(field);
 | 
						|
    } else {
 | 
						|
        string dyn = field.substr(1);
 | 
						|
        transform(dyn.begin(), dyn.end(), dyn.begin(), ::toupper);
 | 
						|
        for (int i = 0; loud_lookup[i].str; i++) {
 | 
						|
            if (streql(loud_lookup[i].str, dyn.c_str())) {
 | 
						|
                return (double) loud_lookup[i].val;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
    parse_error(field, 1, msg);
 | 
						|
    return 100.0;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
int key_lookup[] = {21, 23, 12, 14, 16, 17, 19};
 | 
						|
 | 
						|
 | 
						|
// the field can be K<number> or K[A-G]<number> or P[A-G]<number>
 | 
						|
// (this can be called from parse_pitch() to handle [A-G])
 | 
						|
// Notice that the routine ignores the first character: K or P
 | 
						|
//
 | 
						|
long Alg_reader::parse_key(string &field)
 | 
						|
{
 | 
						|
    char *msg = "Pitch expected";
 | 
						|
    char *pitches = "ABCDEFG";
 | 
						|
    char *p;
 | 
						|
    if (isdigit(field[1])) {
 | 
						|
        // This routine would not have been called if field = "P<number>"
 | 
						|
        // so it must be "K<number>" so <number> must be an integer.
 | 
						|
        return parse_int(field);
 | 
						|
    } else if (p = strchr(pitches, toupper(field[1]))) {
 | 
						|
        long key = key_lookup[p - pitches];
 | 
						|
        key = parse_after_key(key, field, 2);
 | 
						|
        return key;
 | 
						|
    }
 | 
						|
    parse_error(field, 1, msg);
 | 
						|
    return 0;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
long Alg_reader::parse_after_key(int key, string &field, int n)
 | 
						|
{
 | 
						|
    if ((int) field.length() == n) {
 | 
						|
        return key;
 | 
						|
    }
 | 
						|
    char c = toupper(field[n]);
 | 
						|
    if (c == 'S') {
 | 
						|
        return parse_after_key(key + 1, field, n + 1);
 | 
						|
    }
 | 
						|
    if (c == 'F') {
 | 
						|
        return parse_after_key(key - 1, field, n + 1);
 | 
						|
    }
 | 
						|
    if (isdigit(field[n])) {
 | 
						|
        int last = find_int_in(field, n);
 | 
						|
        string octave = field.substr(n, last - n);
 | 
						|
        int oct = atoi(octave.c_str());
 | 
						|
        return parse_after_key(key + oct * 12, field, last);
 | 
						|
    }
 | 
						|
    parse_error(field, n, "Unexpected character in pitch");
 | 
						|
    return key;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
long Alg_reader::find_int_in(string &field, int n)
 | 
						|
{
 | 
						|
    while ((int) field.length() > n && isdigit(field[n])) {
 | 
						|
        n = n + 1;
 | 
						|
    }
 | 
						|
    return n;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
bool Alg_reader::parse_attribute(string &field, Alg_parameter_ptr param)
 | 
						|
{
 | 
						|
    int i = 1;
 | 
						|
    while (i < (int) field.length()) {
 | 
						|
        if (field[i] == ':') {
 | 
						|
            string attr = field.substr(1, i - 1);
 | 
						|
            char type_char = field[i - 1];
 | 
						|
            if (strchr("iarsl", type_char)) {
 | 
						|
                param->set_attr(symbol_table.insert_string(attr.c_str()));
 | 
						|
                parse_val(param, field, i + 1);
 | 
						|
            } else {
 | 
						|
                parse_error(field, 0, "attribute needs to end with typecode: i,a,r,s, or l");
 | 
						|
            }
 | 
						|
            return !error_flag;
 | 
						|
        }
 | 
						|
        i = i + 1;
 | 
						|
    }
 | 
						|
    return false;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
bool Alg_reader::parse_val(Alg_parameter_ptr param, string &s, int i)
 | 
						|
{
 | 
						|
    int len = (int) s.length();
 | 
						|
    if (i >= len) {
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
    if (s[i] == '"') {
 | 
						|
        if (!check_type('s', param)) {
 | 
						|
            return false;
 | 
						|
        }
 | 
						|
        // note: (len - i) includes 2 quote characters but no EOS character
 | 
						|
        // so total memory to allocate is (len - i) - 1
 | 
						|
        char *r = new char[(len - i) - 1];
 | 
						|
        strncpy(r, s.c_str() + i + 1, (len - i) - 2);
 | 
						|
        r[(len - i) - 2] = 0; // terminate the string
 | 
						|
        param->s = r;
 | 
						|
    } else if (s[i] == '\'') {
 | 
						|
        if (!check_type('a', param)) {
 | 
						|
            return false;
 | 
						|
        }
 | 
						|
        string r = s.substr(i + 1, len - i - 2);
 | 
						|
        param->a = symbol_table.insert_string(r.c_str());
 | 
						|
    } else if (param->attr_type() == 'l') {
 | 
						|
        if (streql(s.c_str() + i, "true") || 
 | 
						|
            streql(s.c_str() + i, "t")) {
 | 
						|
            param->l = true;
 | 
						|
        } else if (streql(s.c_str() + i, "false") || 
 | 
						|
                   streql(s.c_str() + i, "nil")) {
 | 
						|
            param->l = false;
 | 
						|
        } else return false;
 | 
						|
    } else if (isdigit(s[i]) || s[i] == '-' || s[i] == '.') {
 | 
						|
        int pos = i;
 | 
						|
        bool period = false;
 | 
						|
        int sign = 1;
 | 
						|
        if (s[pos] == '-') {
 | 
						|
            sign = -1;
 | 
						|
            pos++;
 | 
						|
        }
 | 
						|
        while (pos < len) {
 | 
						|
            if (isdigit(s[pos])) {
 | 
						|
                ;
 | 
						|
            } else if (!period && s[pos] == '.') {
 | 
						|
                period = true;
 | 
						|
            } else {
 | 
						|
                parse_error(s, pos, "Unexpected char in number");
 | 
						|
                return false;
 | 
						|
            }
 | 
						|
            pos = pos + 1;
 | 
						|
        }
 | 
						|
        string r = s.substr(i, len - i);
 | 
						|
        if (period) {
 | 
						|
            if (!check_type('r', param)) {
 | 
						|
                return false;
 | 
						|
            }
 | 
						|
            param->r = atof(r.c_str());
 | 
						|
        } else {
 | 
						|
            if (param->attr_type() == 'r') {
 | 
						|
                param->r = atoi(r.c_str());
 | 
						|
            } else if (!check_type('i', param)) {
 | 
						|
                return false;
 | 
						|
            } else {
 | 
						|
                param->i = atoi(r.c_str());
 | 
						|
            }
 | 
						|
        }
 | 
						|
    } else {
 | 
						|
        parse_error(s, i, "invalid value");
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
    return true;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
bool Alg_reader::check_type(char type_char, Alg_parameter_ptr param)
 | 
						|
{
 | 
						|
    return param->attr_type() == type_char;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
//duration_lookup = {"S": 0.5, "I": 0.5, "Q": 1, "H": 2, "W": 4}
 | 
						|
//key_lookup = {"C": 12, "D": 14, "E": 16, "F": 17, "G": 19, "A": 21, "B": 23}
 | 
						|
 | 
						|
/*
 | 
						|
def test():
 | 
						|
    reader = Alg_reader(open("data\\test.gro", "r"))
 | 
						|
    reader.parse()
 | 
						|
    score = reader->seq.notes
 | 
						|
    print "score:", score
 | 
						|
    reader = nil
 | 
						|
*/
 |