С++ Bass и RapidXml на Raspberry Pi 1 B

У меня странное поведение на проекте, который я делаю для своих детей. Я хочу проанализировать xml с помощью rapidXml и воспроизвести mp3 с басом на моем RasPi1B. Все работает нормально, если я просто нажму ее в одной основной функции. Но как только я помещаю код в другой класс, мои xml-данные повреждаются. Я занимаюсь С++ только в свободное время и работаю с управляемыми языками, поэтому я не очень разбираюсь в управлении памятью в С++. Поэтому я надеюсь, что вы, люди, можете помочь мне (и моим детям) решить эту проблему. Конечно, я мог бы сделать все в одном классе, но мозг попытался бы задушить меня до смерти ;-)

Прежде всего в одной основной функции:

// test.cpp

#include <iostream>
#include <rapidxml.hpp>
#include <rapidxml_utils.hpp>
#include <rapidxml_print.hpp>
#include <bass.h>

using namespace std;
using namespace rapidxml;

int main(int argc, char **argv)
{
    file<> xmlFile("bin/test.xml");

    xml_document<> doc;
    doc.parse<0>(xmlFile.data());

    print(std::cout, doc, 0); // everything is fine

    bool init = BASS_Init(-1, 44100, 0, 0, NULL);
    cout << "BASS_Init " << (init ? "true" : "false") << " [error: " << BASS_ErrorGetCode() << "]" << endl;

    print(std::cout, doc, 0); // everything is fine

    return 0;
}

Затем (почти) то же самое с основной функцией и классом/заголовком:

//test2.cpp

#include <iostream>

#include "player.hpp"

using namespace std;
using namespace player;

int main(int argc, char **argv)
{
    Player* _pPlayer = new Player();
    _pPlayer->loadXml();

    _pPlayer->printXml();       
    _pPlayer->initPlayer();
    _pPlayer->printXml();

    return 0;
}



// player.cpp

#include "player.hpp"

using namespace std;
using namespace rapidxml;

namespace player {
    void Player::loadXml(void)
    {
        file<> xmlFile("bin/test.xml");
        doc.parse<0>(xmlFile.data());
    }

    void Player::printXml(void)
    {
        print(std::cout, doc, 0);
    }

    void Player::initPlayer(void)
    {
        bool init = BASS_Init(-1, 44100, 0, 0, NULL);
        cout << "BASS_Init " << (init ? "true" : "false") << " [error: " << BASS_ErrorGetCode() << "]" << endl;
    }
}



// player.hpp

#include <iostream>

#include <rapidxml.hpp>
#include <rapidxml_utils.hpp>
#include <rapidxml_print.hpp>
#include <bass.h>

using namespace std;
using namespace rapidxml;

namespace player {
    class Player
    {
        public:
            void loadXml(void);
            void printXml(void);
            void initPlayer(void);
        private:
            xml_document<> doc;
    };
}

Компиляция с:

BASS_PATH = ../libraries/bass_arm
BASS_INCLUDE_PATH = -I$(BASS_PATH)
BASS_LIBRARY_PATH = -L$(BASS_PATH) -L$(BASS_PATH)/hardfp
BASS_LIBS = -lbass -lm
BASS_LINKER_FLAGS = -Wl,-rpath,$(BASS_PATH):$(BASS_PATH)/hardfp

RAPIDXML_PATH = ../libraries/rapidxml-1.13
RAPIDXML_INCLUDE_PATH = -I$(RAPIDXML_PATH)
RAPIDXML_LIBRARY_PATH = -L$(RAPIDXML_PATH)

test:
    g++ -std=c++11 src_test/test.cpp -o bin/test $(BASS_INCLUDE_PATH) $(RAPIDXML_INCLUDE_PATH) $(BASS_LIBRARY_PATH) $(RAPIDXML_LIBRARY_PATH) $(BASS_LIBS) $(BASS_LINKER_FLAGS)

test2:
    g++ -std=c++11 -g -c src_test/test2.cpp -o src_test/test2.o $(BASS_INCLUDE_PATH) $(RAPIDXML_INCLUDE_PATH)
    g++ -std=c++11 -g -c src_test/player.cpp -o src_test/player.o $(BASS_INCLUDE_PATH) $(RAPIDXML_INCLUDE_PATH)
    g++ -std=c++11 -g src_test/test2.o src_test/player.o -o bin/test2 $(BASS_LIBRARY_PATH) $(RAPIDXML_LIBRARY_PATH) $(BASS_LIBS) $(BASS_LINKER_FLAGS)

Мой test.xml выглядит так:

<?xml version="1.0" encoding="utf-8"?>
<library>
    <series>
        <title>Lorem ipsum A</title>
        <episode>
            <title>dolor sit amet 1</title>
            <cover></cover>
            <files rfid="x1" current_file="1" timestamp="0">
                <file>consetetur/sadipscing_elitr_11.mp3</file>
                <file>consetetur/sadipscing_elitr_12.mp3</file>
                <file>consetetur/sadipscing_elitr_13.mp3</file>
                <file>consetetur/sadipscing_elitr_14.mp3</file>
            </files>
        </episode>
        <episode>
            <title>dolor sit amet 2</title>
            <cover></cover>
            <files rfid="x2" current_file="1" timestamp="0">
                <file>consetetur/sadipscing_elitr_21.mp3</file>
                <file>consetetur/sadipscing_elitr_22.mp3</file>
                <file>consetetur/sadipscing_elitr_23.mp3</file>
                <file>consetetur/sadipscing_elitr_24.mp3</file>
            </files>
        </episode>
    </series>
</library>

А вот вывод sudo bin/test

<library>
        <series>
                <title>Lorem ipsum A</title>
                <episode>
                        <title>dolor sit amet 1</title>
                        <cover/>
                        <files rfid="x1" current_file="1" timestamp="0">
                                <file>consetetur/sadipscing_elitr_11.mp3</file>
                                <file>consetetur/sadipscing_elitr_12.mp3</file>
                                <file>consetetur/sadipscing_elitr_13.mp3</file>
                                <file>consetetur/sadipscing_elitr_14.mp3</file>
                        </files>
                </episode>
                <episode>
                        <title>dolor sit amet 2</title>
                        <cover/>
                        <files rfid="x2" current_file="1" timestamp="0">
                                <file>consetetur/sadipscing_elitr_21.mp3</file>
                                <file>consetetur/sadipscing_elitr_22.mp3</file>
                                <file>consetetur/sadipscing_elitr_23.mp3</file>
                                <file>consetetur/sadipscing_elitr_24.mp3</file>
                        </files>
                </episode>
        </series>
</library>
BASS_Init true [error: 0]
<library>
        <series>
                <title>Lorem ipsum A</title>
                <episode>
                        <title>dolor sit amet 1</title>
                        <cover/>
                        <files rfid="x1" current_file="1" timestamp="0">
                                <file>consetetur/sadipscing_elitr_11.mp3</file>
                                <file>consetetur/sadipscing_elitr_12.mp3</file>
                                <file>consetetur/sadipscing_elitr_13.mp3</file>
                                <file>consetetur/sadipscing_elitr_14.mp3</file>
                        </files>
                </episode>
                <episode>
                        <title>dolor sit amet 2</title>
                        <cover/>
                        <files rfid="x2" current_file="1" timestamp="0">
                                <file>consetetur/sadipscing_elitr_21.mp3</file>
                                <file>consetetur/sadipscing_elitr_22.mp3</file>
                                <file>consetetur/sadipscing_elitr_23.mp3</file>
                                <file>consetetur/sadipscing_elitr_24.mp3</file>
                        </files>
                </episode>
        </series>
</library>

И вывод sudo bin/test2:

<library>
        <series>
                <title>Lorem ipsum A</title>
                <episode>
                        <title>dolor sit amet 1</title>
                        <cover/>
                        <files rfid="x1" current_file="1" timestamp="0">
                                <file>consetetur/sadipscing_elitr_11.mp3</file>
                                <file>consetetur/sadipscing_elitr_12.mp3</file>
                                <file>consetetur/sadipscing_elitr_13.mp3</file>
                                <file>consetetur/sadipscing_elitr_14.mp3</file>
                        </files>
                </episode>
                <episode>
                        <title>dolor sit amet 2</title>
                        <cover/>
                        <files rfid="x2" current_file="1" timestamp="0">
                                <file>consetetur/sadipscing_elitr_21.mp3</file>
                                <file>consetetur/sadipscing_elitr_22.mp3</file>
                                <file>consetetur/sadipscing_elitr_23.mp3</file>
                                <file>consetetur/sadipscing_elitr_24.mp3</file>
                        </files>
                </episode>
        </series>
</library>

BASS_Init true [error: 0]
<l1▒B>
        CJ>
                <>▒CJ▒BJ▒BJ</>
                <device>
                        <de>aults.pcm.device</de>
                        <`CJ/>
                        <8CJ BJ="" iec958="1" 1▒CJ="">
                                <>▒DJ▒BJ6Jcardp3</>
                                <ults>pcm.cardr/1▒CJ▒CJ</ults>
                                <▒CJ>devicesg_edefault</▒CJ>
                                <1>DJPDJ▒CJ(DJ▒CJ</1>
                        </8CJ>
                </device>
                <EJpE>
                        <▒>J▒CJ6J</▒>
                        <cover/>
                        <cm.ca did="" J▒DJ="" pEJ="J">
                                <>devicesdipdefaults.pcm.d</>
                                <>@EJ▒DJEJ▒DJ</>
                                <rmat>rawur/sadip1▒EJ▒EJ</rmat>
                                <▒DJ>6Jtrueltstr_fi</▒DJ>
                        </cm.ca>
                </EJpE>
        <CJ>
</l1▒B>

Простите за длинный пост! Я надеюсь, что кто-то может мне помочь. Я чувствую себя действительно потерянным... Заранее спасибо!


person FatalDiskError    schedule 21.12.2015    source источник


Ответы (1)


xmlFile уничтожается, когда Player::loadXml() выходит за рамки. Это позволяет назначить память, занятую данными xmlFile, чему-то другому (в этом случае BASE_Init, похоже, записывает в нее данные).

doc по-прежнему использует те же данные, что и xmlFile. Это приводит к неопределенному поведению.

Пожалуйста, сохраните xmlFile в классе Player, как вы сделали с doc.

Также не забудьте delete все, что вы new, например _pPlayer ;-)

person KompjoeFriek    schedule 21.12.2015
comment
о да, ты сделал мой день! Я все время думал, что xmlFile не используется после doc.parse()... Хотя этот rapidxml_utils (где определен файл‹›) тоже доставил мне несколько трудных минут, потому что я не знал, как обращаться с весь шаблонный материал (пока не в шаблонах, но немного похож на дженерики в java). Я решил эту проблему, добавив file<>* pXmlFile; в .hpp и pXmlFile = new file<>("bin/test.xml"); в .cpp. Пока работает нормально, но правильно ли это? (изменить: плохие параметры форматирования в комментариях ^^) - person FatalDiskError; 22.12.2015
comment
Также в качестве дополнения: да, в моем производственном коде я использую delete. Я просто хотел, чтобы пример был как можно короче. - person FatalDiskError; 22.12.2015
comment
Использование указателя на file<>* а правильно. Другой способ может быть без указателей, используя конструктор Player для инициализации вашего file<> xmlFile. После public: в классе Player можно добавить: Player() : xmlFile("bin/test.xml") {} - person KompjoeFriek; 22.12.2015
comment
Спасибо еще раз! Я могу жить с a правильно ;-) Я знаю, что в большинстве случаев есть более чем один хороший способ. Я намеренно не использовал конструктор-подход, потому что в реальной программе загруженный xml зависит от того, какой пользователь использует приложение. Но да, я тоже думал об этом решении. В любом случае, указатель работает как часы, и я очень рад продолжать эффективно работать над моим маленьким проектом. Я пытался исправить эту ошибку почти 3 недели (не каждый день и только вечером... но это стало надоедать). Так что в последний раз я должен сказать: большое спасибо! - person FatalDiskError; 22.12.2015