EmbeddedCuePlaylistPlugin.cxx 4.45 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 22 23 24 25 26
 * 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.
 */

/** \file
 *
 * Playlist plugin that reads embedded cue sheets from the "CUESHEET"
 * tag of a music file.
 */

#include "config.h"
27
#include "EmbeddedCuePlaylistPlugin.hxx"
28
#include "PlaylistPlugin.hxx"
29 30
#include "tag.h"
#include "tag_handler.h"
31
#include "song.h"
32
#include "TagFile.hxx"
33 34

extern "C" {
35 36 37
#include "tag_ape.h"
#include "tag_id3.h"
#include "cue/cue_parser.h"
38
}
39 40 41 42 43 44 45 46 47 48 49

#include <glib.h>
#include <assert.h>
#include <string.h>

#undef G_LOG_DOMAIN
#define G_LOG_DOMAIN "cue"

struct embcue_playlist {
	struct playlist_provider base;

50 51 52 53 54 55 56
	/**
	 * This is an override for the CUE's "FILE".  An embedded CUE
	 * sheet must always point to the song file it is contained
	 * in.
	 */
	char *filename;

57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72
	/**
	 * The value of the file's "CUESHEET" tag.
	 */
	char *cuesheet;

	/**
	 * The offset of the next line within "cuesheet".
	 */
	char *next;

	struct cue_parser *parser;
};

static void
embcue_tag_pair(const char *name, const char *value, void *ctx)
{
73
	struct embcue_playlist *playlist = (struct embcue_playlist *)ctx;
74 75 76 77 78 79 80

	if (playlist->cuesheet == NULL &&
	    g_ascii_strcasecmp(name, "cuesheet") == 0)
		playlist->cuesheet = g_strdup(value);
}

static const struct tag_handler embcue_tag_handler = {
81 82 83
	nullptr,
	nullptr,
	embcue_tag_pair,
84 85 86 87
};

static struct playlist_provider *
embcue_playlist_open_uri(const char *uri,
88 89
			 gcc_unused Mutex &mutex,
			 gcc_unused Cond &cond)
90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111
{
	if (!g_path_is_absolute(uri))
		/* only local files supported */
		return NULL;

	struct embcue_playlist *playlist = g_new(struct embcue_playlist, 1);
	playlist_provider_init(&playlist->base, &embcue_playlist_plugin);
	playlist->cuesheet = NULL;

	tag_file_scan(uri, &embcue_tag_handler, playlist);
	if (playlist->cuesheet == NULL) {
		tag_ape_scan2(uri, &embcue_tag_handler, playlist);
		if (playlist->cuesheet == NULL)
			tag_id3_scan(uri, &embcue_tag_handler, playlist);
	}

	if (playlist->cuesheet == NULL) {
		/* no "CUESHEET" tag found */
		g_free(playlist);
		return NULL;
	}

112 113
	playlist->filename = g_path_get_basename(uri);

114 115 116 117 118 119 120 121 122 123 124 125 126
	playlist->next = playlist->cuesheet;
	playlist->parser = cue_parser_new();

	return &playlist->base;
}

static void
embcue_playlist_close(struct playlist_provider *_playlist)
{
	struct embcue_playlist *playlist = (struct embcue_playlist *)_playlist;

	cue_parser_free(playlist->parser);
	g_free(playlist->cuesheet);
127
	g_free(playlist->filename);
128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154
	g_free(playlist);
}

static struct song *
embcue_playlist_read(struct playlist_provider *_playlist)
{
	struct embcue_playlist *playlist = (struct embcue_playlist *)_playlist;

	struct song *song = cue_parser_get(playlist->parser);
	if (song != NULL)
		return song;

	while (*playlist->next != 0) {
		const char *line = playlist->next;
		char *eol = strpbrk(playlist->next, "\r\n");
		if (eol != NULL) {
			/* null-terminate the line */
			*eol = 0;
			playlist->next = eol + 1;
		} else
			/* last line; put the "next" pointer to the
			   end of the buffer */
			playlist->next += strlen(line);

		cue_parser_feed(playlist->parser, line);
		song = cue_parser_get(playlist->parser);
		if (song != NULL)
155
			return song_replace_uri(song, playlist->filename);
156 157 158
	}

	cue_parser_finish(playlist->parser);
159 160 161 162
	song = cue_parser_get(playlist->parser);
	if (song != NULL)
		song = song_replace_uri(song, playlist->filename);
	return song;
163 164 165 166 167 168 169 170 171 172 173 174 175 176 177
}

static const char *const embcue_playlist_suffixes[] = {
	/* a few codecs that are known to be supported; there are
	   probably many more */
	"flac",
	"mp3", "mp2",
	"mp4", "mp4a", "m4b",
	"ape",
	"wv",
	"ogg", "oga",
	NULL
};

const struct playlist_plugin embcue_playlist_plugin = {
178 179 180 181 182 183 184 185 186 187 188 189
	"cue",

	nullptr,
	nullptr,
	embcue_playlist_open_uri,
	nullptr,
	embcue_playlist_close,
	embcue_playlist_read,

	embcue_playlist_suffixes,
	nullptr,
	nullptr,
190
};