AdPlugDecoderPlugin.cxx 3.2 KB
Newer Older
1
/*
2
 * Copyright (C) 2003-2013 The Music Player Daemon Project
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
 * http://www.musicpd.org
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
 */

#include "config.h"
#include "AdPlugDecoderPlugin.h"
22
#include "tag/TagHandler.hxx"
23
#include "DecoderAPI.hxx"
24
#include "CheckAudioFormat.hxx"
25
#include "util/Error.hxx"
26
#include "util/Macros.hxx"
27
#include "Log.hxx"
28 29 30 31 32 33 34 35 36

#include <adplug/adplug.h>
#include <adplug/emuopl.h>

#include <assert.h>

static unsigned sample_rate;

static bool
37
adplug_init(const config_param &param)
38
{
39
	Error error;
40

41
	sample_rate = param.GetBlockValue("sample_rate", 48000u);
42
	if (!audio_check_sample_rate(sample_rate, error)) {
43
		LogError(error);
44 45 46 47 48 49 50
		return false;
	}

	return true;
}

static void
51
adplug_file_decode(Decoder &decoder, const char *path_fs)
52 53 54 55 56 57 58 59
{
	CEmuopl opl(sample_rate, true, true);
	opl.init();

	CPlayer *player = CAdPlug::factory(path_fs, &opl);
	if (player == nullptr)
		return;

60 61
	const AudioFormat audio_format(sample_rate, SampleFormat::S16, 2);
	assert(audio_format.IsValid());
62

63
	decoder_initialized(decoder, audio_format, false,
64 65 66
		player->songlength() / 1000.);

	int16_t buffer[2048];
67
	const unsigned frames_per_buffer = ARRAY_SIZE(buffer) / 2;
68
	DecoderCommand cmd;
69 70 71 72 73 74

	do {
		if (!player->update())
			break;

		opl.update(buffer, frames_per_buffer);
75
		cmd = decoder_data(decoder, nullptr,
76 77
				   buffer, sizeof(buffer),
				   0);
78
	} while (cmd == DecoderCommand::NONE);
79 80 81 82 83

	delete player;
}

static void
84
adplug_scan_tag(TagType type, const std::string &value,
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
		const struct tag_handler *handler, void *handler_ctx)
{
	if (!value.empty())
		tag_handler_invoke_tag(handler, handler_ctx,
				       type, value.c_str());
}

static bool
adplug_scan_file(const char *path_fs,
		 const struct tag_handler *handler, void *handler_ctx)
{
	CEmuopl opl(sample_rate, true, true);
	opl.init();

	CPlayer *player = CAdPlug::factory(path_fs, &opl);
	if (player == nullptr)
		return false;

	tag_handler_invoke_duration(handler, handler_ctx,
				    player->songlength() / 1000);

	if (handler->tag != nullptr) {
		adplug_scan_tag(TAG_TITLE, player->gettitle(),
				handler, handler_ctx);
		adplug_scan_tag(TAG_ARTIST, player->getauthor(),
				handler, handler_ctx);
		adplug_scan_tag(TAG_COMMENT, player->getdesc(),
				handler, handler_ctx);
	}

	delete player;
	return true;
}

static const char *const adplug_suffixes[] = {
	"amd",
	"d00",
	"hsc",
	"laa",
	"rad",
	"raw",
	"sa2",
	nullptr
};

130
const struct DecoderPlugin adplug_decoder_plugin = {
131 132 133 134 135 136 137 138 139 140 141
	"adplug",
	adplug_init,
	nullptr,
	nullptr,
	adplug_file_decode,
	adplug_scan_file,
	nullptr,
	nullptr,
	adplug_suffixes,
	nullptr,
};