Resource Interchange File Format (RIFF): C++98/STL parsing library

The Resource Interchange File Format (RIFF) is a generic file container format for storing data in tagged chunks. It is primarily used to store multimedia such as sound and video, though it may also be used to store any arbitrary data.

The Microsoft implementation is mostly known through container formats like AVI, ANI and WAV, which use RIFF as their basis.

This page hosts a formal specification of Resource Interchange File Format (RIFF) using Kaitai Struct. This specification can be automatically translated into a variety of programming languages to get a parsing library.

Usage

Runtime library

All parsing code for C++98/STL generated by Kaitai Struct depends on the C++/STL runtime library. You have to install it before you can parse data.

For C++, the easiest way is to clone the runtime library sources and build them along with your project.

Code

Using Kaitai Struct in C++/STL usually consists of 3 steps.

  1. We need to create an STL input stream (std::istream). One can open local file for that, or use existing std::string or char* buffer.
    #include <fstream>
    
    std::ifstream is("path/to/local/file.bin", std::ifstream::binary);
    
    #include <sstream>
    
    std::istringstream is(str);
    
    #include <sstream>
    
    const char buf[] = { ... };
    std::string str(buf, sizeof buf);
    std::istringstream is(str);
    
  2. We need to wrap our input stream into Kaitai stream:
    #include "kaitai/kaitaistream.h"
    
    kaitai::kstream ks(&is);
    
  3. And finally, we can invoke the parsing:
    riff_t data(&ks);
    

After that, one can get various attributes from the structure by invoking getter methods like:

data.chunk() // => get chunk

C++98/STL source code to parse Resource Interchange File Format (RIFF)

riff.h

#ifndef RIFF_H_
#define RIFF_H_

// This is a generated file! Please edit source .ksy file and use kaitai-struct-compiler to rebuild

#include "kaitai/kaitaistruct.h"
#include <stdint.h>
#include <vector>

#if KAITAI_STRUCT_VERSION < 9000L
#error "Incompatible Kaitai Struct C++/STL API: version 0.9 or later is required"
#endif

/**
 * The Resource Interchange File Format (RIFF) is a generic file container format
 * for storing data in tagged chunks. It is primarily used to store multimedia
 * such as sound and video, though it may also be used to store any arbitrary data.
 * 
 * The Microsoft implementation is mostly known through container formats
 * like AVI, ANI and WAV, which use RIFF as their basis.
 * \sa https://www.johnloomis.org/cpe102/asgn/asgn1/riff.html Source
 */

class riff_t : public kaitai::kstruct {

public:
    class list_chunk_data_t;
    class chunk_t;
    class parent_chunk_data_t;
    class info_subchunk_t;
    class chunk_type_t;

    enum fourcc_t {
        FOURCC_RIFF = 1179011410,
        FOURCC_INFO = 1330007625,
        FOURCC_LIST = 1414744396
    };

    riff_t(kaitai::kstream* p__io, kaitai::kstruct* p__parent = 0, riff_t* p__root = 0);

private:
    void _read();
    void _clean_up();

public:
    ~riff_t();

    class list_chunk_data_t : public kaitai::kstruct {

    public:

        list_chunk_data_t(kaitai::kstream* p__io, riff_t::chunk_type_t* p__parent = 0, riff_t* p__root = 0);

    private:
        void _read();
        void _clean_up();

    public:
        ~list_chunk_data_t();

    private:
        bool f_parent_chunk_data_ofs;
        int32_t m_parent_chunk_data_ofs;

    public:
        int32_t parent_chunk_data_ofs();

    private:
        bool f_form_type;
        fourcc_t m_form_type;

    public:
        fourcc_t form_type();

    private:
        bool f_form_type_readable;
        std::string m_form_type_readable;

    public:
        std::string form_type_readable();

    private:
        bool f_subchunks;
        std::vector<kaitai::kstruct*>* m_subchunks;

    public:
        std::vector<kaitai::kstruct*>* subchunks();

    private:
        std::string m_save_parent_chunk_data_ofs;
        bool n_save_parent_chunk_data_ofs;

    public:
        bool _is_null_save_parent_chunk_data_ofs() { save_parent_chunk_data_ofs(); return n_save_parent_chunk_data_ofs; };

    private:
        parent_chunk_data_t* m_parent_chunk_data;
        riff_t* m__root;
        riff_t::chunk_type_t* m__parent;

    public:
        std::string save_parent_chunk_data_ofs() const { return m_save_parent_chunk_data_ofs; }
        parent_chunk_data_t* parent_chunk_data() const { return m_parent_chunk_data; }
        riff_t* _root() const { return m__root; }
        riff_t::chunk_type_t* _parent() const { return m__parent; }
    };

    class chunk_t : public kaitai::kstruct {

    public:
        class slot_t;

        chunk_t(kaitai::kstream* p__io, kaitai::kstruct* p__parent = 0, riff_t* p__root = 0);

    private:
        void _read();
        void _clean_up();

    public:
        ~chunk_t();

        class slot_t : public kaitai::kstruct {

        public:

            slot_t(kaitai::kstream* p__io, riff_t::chunk_t* p__parent = 0, riff_t* p__root = 0);

        private:
            void _read();
            void _clean_up();

        public:
            ~slot_t();

        private:
            riff_t* m__root;
            riff_t::chunk_t* m__parent;

        public:
            riff_t* _root() const { return m__root; }
            riff_t::chunk_t* _parent() const { return m__parent; }
        };

    private:
        uint32_t m_id;
        uint32_t m_len;
        slot_t* m_data_slot;
        std::string m_pad_byte;
        riff_t* m__root;
        kaitai::kstruct* m__parent;
        std::string m__raw_data_slot;
        kaitai::kstream* m__io__raw_data_slot;

    public:
        uint32_t id() const { return m_id; }
        uint32_t len() const { return m_len; }
        slot_t* data_slot() const { return m_data_slot; }
        std::string pad_byte() const { return m_pad_byte; }
        riff_t* _root() const { return m__root; }
        kaitai::kstruct* _parent() const { return m__parent; }
        std::string _raw_data_slot() const { return m__raw_data_slot; }
        kaitai::kstream* _io__raw_data_slot() const { return m__io__raw_data_slot; }
    };

    class parent_chunk_data_t : public kaitai::kstruct {

    public:
        class slot_t;

        parent_chunk_data_t(kaitai::kstream* p__io, kaitai::kstruct* p__parent = 0, riff_t* p__root = 0);

    private:
        void _read();
        void _clean_up();

    public:
        ~parent_chunk_data_t();

        class slot_t : public kaitai::kstruct {

        public:

            slot_t(kaitai::kstream* p__io, riff_t::parent_chunk_data_t* p__parent = 0, riff_t* p__root = 0);

        private:
            void _read();
            void _clean_up();

        public:
            ~slot_t();

        private:
            riff_t* m__root;
            riff_t::parent_chunk_data_t* m__parent;

        public:
            riff_t* _root() const { return m__root; }
            riff_t::parent_chunk_data_t* _parent() const { return m__parent; }
        };

    private:
        uint32_t m_form_type;
        slot_t* m_subchunks_slot;
        riff_t* m__root;
        kaitai::kstruct* m__parent;
        std::string m__raw_subchunks_slot;
        kaitai::kstream* m__io__raw_subchunks_slot;

    public:
        uint32_t form_type() const { return m_form_type; }
        slot_t* subchunks_slot() const { return m_subchunks_slot; }
        riff_t* _root() const { return m__root; }
        kaitai::kstruct* _parent() const { return m__parent; }
        std::string _raw_subchunks_slot() const { return m__raw_subchunks_slot; }
        kaitai::kstream* _io__raw_subchunks_slot() const { return m__io__raw_subchunks_slot; }
    };

    /**
     * All registered subchunks in the INFO chunk are NULL-terminated strings,
     * but the unregistered might not be. By convention, the registered
     * chunk IDs are in uppercase and the unregistered IDs are in lowercase.
     * 
     * If the chunk ID of an INFO subchunk contains a lowercase
     * letter, this chunk is considered as unregistered and thus we can make
     * no assumptions about the type of data.
     */

    class info_subchunk_t : public kaitai::kstruct {

    public:

        info_subchunk_t(kaitai::kstream* p__io, riff_t::list_chunk_data_t* p__parent = 0, riff_t* p__root = 0);

    private:
        void _read();
        void _clean_up();

    public:
        ~info_subchunk_t();

    private:
        bool f_chunk_data;
        std::string m_chunk_data;
        bool n_chunk_data;

    public:
        bool _is_null_chunk_data() { chunk_data(); return n_chunk_data; };

    private:

    public:
        std::string chunk_data();

    private:
        bool f_is_unregistered_tag;
        bool m_is_unregistered_tag;

    public:

        /**
         * Check if chunk_id contains lowercase characters ([a-z], ASCII 97 = a, ASCII 122 = z).
         */
        bool is_unregistered_tag();

    private:
        bool f_id_chars;
        std::string m_id_chars;

    public:
        std::string id_chars();

    private:
        bool f_chunk_id_readable;
        std::string m_chunk_id_readable;

    public:
        std::string chunk_id_readable();

    private:
        bool f_chunk_ofs;
        int32_t m_chunk_ofs;

    public:
        int32_t chunk_ofs();

    private:
        std::string m_save_chunk_ofs;
        bool n_save_chunk_ofs;

    public:
        bool _is_null_save_chunk_ofs() { save_chunk_ofs(); return n_save_chunk_ofs; };

    private:
        chunk_t* m_chunk;
        riff_t* m__root;
        riff_t::list_chunk_data_t* m__parent;

    public:
        std::string save_chunk_ofs() const { return m_save_chunk_ofs; }
        chunk_t* chunk() const { return m_chunk; }
        riff_t* _root() const { return m__root; }
        riff_t::list_chunk_data_t* _parent() const { return m__parent; }
    };

    class chunk_type_t : public kaitai::kstruct {

    public:

        chunk_type_t(kaitai::kstream* p__io, kaitai::kstruct* p__parent = 0, riff_t* p__root = 0);

    private:
        void _read();
        void _clean_up();

    public:
        ~chunk_type_t();

    private:
        bool f_chunk_ofs;
        int32_t m_chunk_ofs;

    public:
        int32_t chunk_ofs();

    private:
        bool f_chunk_id;
        fourcc_t m_chunk_id;

    public:
        fourcc_t chunk_id();

    private:
        bool f_chunk_id_readable;
        std::string m_chunk_id_readable;

    public:
        std::string chunk_id_readable();

    private:
        bool f_chunk_data;
        list_chunk_data_t* m_chunk_data;
        bool n_chunk_data;

    public:
        bool _is_null_chunk_data() { chunk_data(); return n_chunk_data; };

    private:

    public:
        list_chunk_data_t* chunk_data();

    private:
        std::string m_save_chunk_ofs;
        bool n_save_chunk_ofs;

    public:
        bool _is_null_save_chunk_ofs() { save_chunk_ofs(); return n_save_chunk_ofs; };

    private:
        chunk_t* m_chunk;
        riff_t* m__root;
        kaitai::kstruct* m__parent;

    public:
        std::string save_chunk_ofs() const { return m_save_chunk_ofs; }
        chunk_t* chunk() const { return m_chunk; }
        riff_t* _root() const { return m__root; }
        kaitai::kstruct* _parent() const { return m__parent; }
    };

private:
    bool f_chunk_id;
    fourcc_t m_chunk_id;

public:
    fourcc_t chunk_id();

private:
    bool f_is_riff_chunk;
    bool m_is_riff_chunk;

public:
    bool is_riff_chunk();

private:
    bool f_parent_chunk_data;
    parent_chunk_data_t* m_parent_chunk_data;
    bool n_parent_chunk_data;

public:
    bool _is_null_parent_chunk_data() { parent_chunk_data(); return n_parent_chunk_data; };

private:

public:
    parent_chunk_data_t* parent_chunk_data();

private:
    bool f_subchunks;
    std::vector<chunk_type_t*>* m_subchunks;
    bool n_subchunks;

public:
    bool _is_null_subchunks() { subchunks(); return n_subchunks; };

private:

public:
    std::vector<chunk_type_t*>* subchunks();

private:
    chunk_t* m_chunk;
    riff_t* m__root;
    kaitai::kstruct* m__parent;

public:
    chunk_t* chunk() const { return m_chunk; }
    riff_t* _root() const { return m__root; }
    kaitai::kstruct* _parent() const { return m__parent; }
};

#endif  // RIFF_H_

riff.cpp

// This is a generated file! Please edit source .ksy file and use kaitai-struct-compiler to rebuild

#include "riff.h"

riff_t::riff_t(kaitai::kstream* p__io, kaitai::kstruct* p__parent, riff_t* p__root) : kaitai::kstruct(p__io) {
    m__parent = p__parent;
    m__root = this;
    m_chunk = 0;
    m_parent_chunk_data = 0;
    m_subchunks = 0;
    f_chunk_id = false;
    f_is_riff_chunk = false;
    f_parent_chunk_data = false;
    f_subchunks = false;

    try {
        _read();
    } catch(...) {
        _clean_up();
        throw;
    }
}

void riff_t::_read() {
    m_chunk = new chunk_t(m__io, this, m__root);
}

riff_t::~riff_t() {
    _clean_up();
}

void riff_t::_clean_up() {
    if (m_chunk) {
        delete m_chunk; m_chunk = 0;
    }
    if (f_parent_chunk_data && !n_parent_chunk_data) {
        if (m_parent_chunk_data) {
            delete m_parent_chunk_data; m_parent_chunk_data = 0;
        }
    }
    if (f_subchunks && !n_subchunks) {
        if (m_subchunks) {
            for (std::vector<chunk_type_t*>::iterator it = m_subchunks->begin(); it != m_subchunks->end(); ++it) {
                delete *it;
            }
            delete m_subchunks; m_subchunks = 0;
        }
    }
}

riff_t::list_chunk_data_t::list_chunk_data_t(kaitai::kstream* p__io, riff_t::chunk_type_t* p__parent, riff_t* p__root) : kaitai::kstruct(p__io) {
    m__parent = p__parent;
    m__root = p__root;
    m_parent_chunk_data = 0;
    m_subchunks = 0;
    f_parent_chunk_data_ofs = false;
    f_form_type = false;
    f_form_type_readable = false;
    f_subchunks = false;

    try {
        _read();
    } catch(...) {
        _clean_up();
        throw;
    }
}

void riff_t::list_chunk_data_t::_read() {
    n_save_parent_chunk_data_ofs = true;
    if (parent_chunk_data_ofs() < 0) {
        n_save_parent_chunk_data_ofs = false;
        m_save_parent_chunk_data_ofs = m__io->read_bytes(0);
    }
    m_parent_chunk_data = new parent_chunk_data_t(m__io, this, m__root);
}

riff_t::list_chunk_data_t::~list_chunk_data_t() {
    _clean_up();
}

void riff_t::list_chunk_data_t::_clean_up() {
    if (!n_save_parent_chunk_data_ofs) {
    }
    if (m_parent_chunk_data) {
        delete m_parent_chunk_data; m_parent_chunk_data = 0;
    }
    if (f_form_type_readable) {
    }
    if (f_subchunks) {
        if (m_subchunks) {
            for (std::vector<kaitai::kstruct*>::iterator it = m_subchunks->begin(); it != m_subchunks->end(); ++it) {
                delete *it;
            }
            delete m_subchunks; m_subchunks = 0;
        }
    }
}

int32_t riff_t::list_chunk_data_t::parent_chunk_data_ofs() {
    if (f_parent_chunk_data_ofs)
        return m_parent_chunk_data_ofs;
    m_parent_chunk_data_ofs = _io()->pos();
    f_parent_chunk_data_ofs = true;
    return m_parent_chunk_data_ofs;
}

riff_t::fourcc_t riff_t::list_chunk_data_t::form_type() {
    if (f_form_type)
        return m_form_type;
    m_form_type = static_cast<riff_t::fourcc_t>(parent_chunk_data()->form_type());
    f_form_type = true;
    return m_form_type;
}

std::string riff_t::list_chunk_data_t::form_type_readable() {
    if (f_form_type_readable)
        return m_form_type_readable;
    std::streampos _pos = m__io->pos();
    m__io->seek(parent_chunk_data_ofs());
    m_form_type_readable = kaitai::kstream::bytes_to_str(m__io->read_bytes(4), std::string("ASCII"));
    m__io->seek(_pos);
    f_form_type_readable = true;
    return m_form_type_readable;
}

std::vector<kaitai::kstruct*>* riff_t::list_chunk_data_t::subchunks() {
    if (f_subchunks)
        return m_subchunks;
    kaitai::kstream *io = parent_chunk_data()->subchunks_slot()->_io();
    std::streampos _pos = io->pos();
    io->seek(0);
    m_subchunks = new std::vector<kaitai::kstruct*>();
    {
        int i = 0;
        while (!io->is_eof()) {
            switch (form_type()) {
            case riff_t::FOURCC_INFO: {
                m_subchunks->push_back(new info_subchunk_t(io, this, m__root));
                break;
            }
            default: {
                m_subchunks->push_back(new chunk_type_t(io, this, m__root));
                break;
            }
            }
            i++;
        }
    }
    io->seek(_pos);
    f_subchunks = true;
    return m_subchunks;
}

riff_t::chunk_t::chunk_t(kaitai::kstream* p__io, kaitai::kstruct* p__parent, riff_t* p__root) : kaitai::kstruct(p__io) {
    m__parent = p__parent;
    m__root = p__root;
    m_data_slot = 0;
    m__io__raw_data_slot = 0;

    try {
        _read();
    } catch(...) {
        _clean_up();
        throw;
    }
}

void riff_t::chunk_t::_read() {
    m_id = m__io->read_u4le();
    m_len = m__io->read_u4le();
    m__raw_data_slot = m__io->read_bytes(len());
    m__io__raw_data_slot = new kaitai::kstream(m__raw_data_slot);
    m_data_slot = new slot_t(m__io__raw_data_slot, this, m__root);
    m_pad_byte = m__io->read_bytes(kaitai::kstream::mod(len(), 2));
}

riff_t::chunk_t::~chunk_t() {
    _clean_up();
}

void riff_t::chunk_t::_clean_up() {
    if (m__io__raw_data_slot) {
        delete m__io__raw_data_slot; m__io__raw_data_slot = 0;
    }
    if (m_data_slot) {
        delete m_data_slot; m_data_slot = 0;
    }
}

riff_t::chunk_t::slot_t::slot_t(kaitai::kstream* p__io, riff_t::chunk_t* p__parent, riff_t* p__root) : kaitai::kstruct(p__io) {
    m__parent = p__parent;
    m__root = p__root;

    try {
        _read();
    } catch(...) {
        _clean_up();
        throw;
    }
}

void riff_t::chunk_t::slot_t::_read() {
}

riff_t::chunk_t::slot_t::~slot_t() {
    _clean_up();
}

void riff_t::chunk_t::slot_t::_clean_up() {
}

riff_t::parent_chunk_data_t::parent_chunk_data_t(kaitai::kstream* p__io, kaitai::kstruct* p__parent, riff_t* p__root) : kaitai::kstruct(p__io) {
    m__parent = p__parent;
    m__root = p__root;
    m_subchunks_slot = 0;
    m__io__raw_subchunks_slot = 0;

    try {
        _read();
    } catch(...) {
        _clean_up();
        throw;
    }
}

void riff_t::parent_chunk_data_t::_read() {
    m_form_type = m__io->read_u4le();
    m__raw_subchunks_slot = m__io->read_bytes_full();
    m__io__raw_subchunks_slot = new kaitai::kstream(m__raw_subchunks_slot);
    m_subchunks_slot = new slot_t(m__io__raw_subchunks_slot, this, m__root);
}

riff_t::parent_chunk_data_t::~parent_chunk_data_t() {
    _clean_up();
}

void riff_t::parent_chunk_data_t::_clean_up() {
    if (m__io__raw_subchunks_slot) {
        delete m__io__raw_subchunks_slot; m__io__raw_subchunks_slot = 0;
    }
    if (m_subchunks_slot) {
        delete m_subchunks_slot; m_subchunks_slot = 0;
    }
}

riff_t::parent_chunk_data_t::slot_t::slot_t(kaitai::kstream* p__io, riff_t::parent_chunk_data_t* p__parent, riff_t* p__root) : kaitai::kstruct(p__io) {
    m__parent = p__parent;
    m__root = p__root;

    try {
        _read();
    } catch(...) {
        _clean_up();
        throw;
    }
}

void riff_t::parent_chunk_data_t::slot_t::_read() {
}

riff_t::parent_chunk_data_t::slot_t::~slot_t() {
    _clean_up();
}

void riff_t::parent_chunk_data_t::slot_t::_clean_up() {
}

riff_t::info_subchunk_t::info_subchunk_t(kaitai::kstream* p__io, riff_t::list_chunk_data_t* p__parent, riff_t* p__root) : kaitai::kstruct(p__io) {
    m__parent = p__parent;
    m__root = p__root;
    m_chunk = 0;
    f_chunk_data = false;
    f_is_unregistered_tag = false;
    f_id_chars = false;
    f_chunk_id_readable = false;
    f_chunk_ofs = false;

    try {
        _read();
    } catch(...) {
        _clean_up();
        throw;
    }
}

void riff_t::info_subchunk_t::_read() {
    n_save_chunk_ofs = true;
    if (chunk_ofs() < 0) {
        n_save_chunk_ofs = false;
        m_save_chunk_ofs = m__io->read_bytes(0);
    }
    m_chunk = new chunk_t(m__io, this, m__root);
}

riff_t::info_subchunk_t::~info_subchunk_t() {
    _clean_up();
}

void riff_t::info_subchunk_t::_clean_up() {
    if (!n_save_chunk_ofs) {
    }
    if (m_chunk) {
        delete m_chunk; m_chunk = 0;
    }
    if (f_chunk_data && !n_chunk_data) {
    }
    if (f_id_chars) {
    }
}

std::string riff_t::info_subchunk_t::chunk_data() {
    if (f_chunk_data)
        return m_chunk_data;
    kaitai::kstream *io = chunk()->data_slot()->_io();
    std::streampos _pos = io->pos();
    io->seek(0);
    n_chunk_data = true;
    {
        bool on = is_unregistered_tag();
        if (on == false) {
            n_chunk_data = false;
            m_chunk_data = kaitai::kstream::bytes_to_str(io->read_bytes_term(0, false, true, true), std::string("UTF-8"));
        }
    }
    io->seek(_pos);
    f_chunk_data = true;
    return m_chunk_data;
}

bool riff_t::info_subchunk_t::is_unregistered_tag() {
    if (f_is_unregistered_tag)
        return m_is_unregistered_tag;
    m_is_unregistered_tag =  (( ((id_chars()[0] >= 97) && (id_chars()[0] <= 122)) ) || ( ((id_chars()[1] >= 97) && (id_chars()[1] <= 122)) ) || ( ((id_chars()[2] >= 97) && (id_chars()[2] <= 122)) ) || ( ((id_chars()[3] >= 97) && (id_chars()[3] <= 122)) )) ;
    f_is_unregistered_tag = true;
    return m_is_unregistered_tag;
}

std::string riff_t::info_subchunk_t::id_chars() {
    if (f_id_chars)
        return m_id_chars;
    std::streampos _pos = m__io->pos();
    m__io->seek(chunk_ofs());
    m_id_chars = m__io->read_bytes(4);
    m__io->seek(_pos);
    f_id_chars = true;
    return m_id_chars;
}

std::string riff_t::info_subchunk_t::chunk_id_readable() {
    if (f_chunk_id_readable)
        return m_chunk_id_readable;
    m_chunk_id_readable = kaitai::kstream::bytes_to_str(id_chars(), std::string("ASCII"));
    f_chunk_id_readable = true;
    return m_chunk_id_readable;
}

int32_t riff_t::info_subchunk_t::chunk_ofs() {
    if (f_chunk_ofs)
        return m_chunk_ofs;
    m_chunk_ofs = _io()->pos();
    f_chunk_ofs = true;
    return m_chunk_ofs;
}

riff_t::chunk_type_t::chunk_type_t(kaitai::kstream* p__io, kaitai::kstruct* p__parent, riff_t* p__root) : kaitai::kstruct(p__io) {
    m__parent = p__parent;
    m__root = p__root;
    m_chunk = 0;
    f_chunk_ofs = false;
    f_chunk_id = false;
    f_chunk_id_readable = false;
    f_chunk_data = false;

    try {
        _read();
    } catch(...) {
        _clean_up();
        throw;
    }
}

void riff_t::chunk_type_t::_read() {
    n_save_chunk_ofs = true;
    if (chunk_ofs() < 0) {
        n_save_chunk_ofs = false;
        m_save_chunk_ofs = m__io->read_bytes(0);
    }
    m_chunk = new chunk_t(m__io, this, m__root);
}

riff_t::chunk_type_t::~chunk_type_t() {
    _clean_up();
}

void riff_t::chunk_type_t::_clean_up() {
    if (!n_save_chunk_ofs) {
    }
    if (m_chunk) {
        delete m_chunk; m_chunk = 0;
    }
    if (f_chunk_id_readable) {
    }
    if (f_chunk_data && !n_chunk_data) {
        if (m_chunk_data) {
            delete m_chunk_data; m_chunk_data = 0;
        }
    }
}

int32_t riff_t::chunk_type_t::chunk_ofs() {
    if (f_chunk_ofs)
        return m_chunk_ofs;
    m_chunk_ofs = _io()->pos();
    f_chunk_ofs = true;
    return m_chunk_ofs;
}

riff_t::fourcc_t riff_t::chunk_type_t::chunk_id() {
    if (f_chunk_id)
        return m_chunk_id;
    m_chunk_id = static_cast<riff_t::fourcc_t>(chunk()->id());
    f_chunk_id = true;
    return m_chunk_id;
}

std::string riff_t::chunk_type_t::chunk_id_readable() {
    if (f_chunk_id_readable)
        return m_chunk_id_readable;
    std::streampos _pos = m__io->pos();
    m__io->seek(chunk_ofs());
    m_chunk_id_readable = kaitai::kstream::bytes_to_str(m__io->read_bytes(4), std::string("ASCII"));
    m__io->seek(_pos);
    f_chunk_id_readable = true;
    return m_chunk_id_readable;
}

riff_t::list_chunk_data_t* riff_t::chunk_type_t::chunk_data() {
    if (f_chunk_data)
        return m_chunk_data;
    kaitai::kstream *io = chunk()->data_slot()->_io();
    std::streampos _pos = io->pos();
    io->seek(0);
    n_chunk_data = true;
    switch (chunk_id()) {
    case riff_t::FOURCC_LIST: {
        n_chunk_data = false;
        m_chunk_data = new list_chunk_data_t(io, this, m__root);
        break;
    }
    }
    io->seek(_pos);
    f_chunk_data = true;
    return m_chunk_data;
}

riff_t::fourcc_t riff_t::chunk_id() {
    if (f_chunk_id)
        return m_chunk_id;
    m_chunk_id = static_cast<riff_t::fourcc_t>(chunk()->id());
    f_chunk_id = true;
    return m_chunk_id;
}

bool riff_t::is_riff_chunk() {
    if (f_is_riff_chunk)
        return m_is_riff_chunk;
    m_is_riff_chunk = chunk_id() == riff_t::FOURCC_RIFF;
    f_is_riff_chunk = true;
    return m_is_riff_chunk;
}

riff_t::parent_chunk_data_t* riff_t::parent_chunk_data() {
    if (f_parent_chunk_data)
        return m_parent_chunk_data;
    n_parent_chunk_data = true;
    if (is_riff_chunk()) {
        n_parent_chunk_data = false;
        kaitai::kstream *io = chunk()->data_slot()->_io();
        std::streampos _pos = io->pos();
        io->seek(0);
        m_parent_chunk_data = new parent_chunk_data_t(io, this, m__root);
        io->seek(_pos);
        f_parent_chunk_data = true;
    }
    return m_parent_chunk_data;
}

std::vector<riff_t::chunk_type_t*>* riff_t::subchunks() {
    if (f_subchunks)
        return m_subchunks;
    n_subchunks = true;
    if (is_riff_chunk()) {
        n_subchunks = false;
        kaitai::kstream *io = parent_chunk_data()->subchunks_slot()->_io();
        std::streampos _pos = io->pos();
        io->seek(0);
        m_subchunks = new std::vector<chunk_type_t*>();
        {
            int i = 0;
            while (!io->is_eof()) {
                m_subchunks->push_back(new chunk_type_t(io, this, m__root));
                i++;
            }
        }
        io->seek(_pos);
        f_subchunks = true;
    }
    return m_subchunks;
}