/* * Copyright (C) 2003-2009 The Music Player Daemon Project * 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. */ /* * Parser functions for audio related objects. * */ #include "audio_parser.h" #include "audio_format.h" #include /** * The GLib quark used for errors reported by this library. */ static inline GQuark audio_parser_quark(void) { return g_quark_from_static_string("audio_parser"); } static bool parse_sample_rate(const char *src, uint32_t *sample_rate_r, const char **endptr_r, GError **error_r) { unsigned long value; char *endptr; value = strtoul(src, &endptr, 10); if (endptr == src) { g_set_error(error_r, audio_parser_quark(), 0, "Failed to parse the sample rate"); return false; } else if (!audio_valid_sample_rate(value)) { g_set_error(error_r, audio_parser_quark(), 0, "Invalid sample rate: %lu", value); return false; } *sample_rate_r = value; *endptr_r = endptr; return true; } static bool parse_sample_format(const char *src, uint8_t *bits_r, const char **endptr_r, GError **error_r) { unsigned long value; char *endptr; value = strtoul(src, &endptr, 10); if (endptr == src) { g_set_error(error_r, audio_parser_quark(), 0, "Failed to parse the sample format"); return false; } else if (!audio_valid_sample_format(value)) { g_set_error(error_r, audio_parser_quark(), 0, "Invalid sample format: %lu", value); return false; } *bits_r = value; *endptr_r = endptr; return true; } static bool parse_channel_count(const char *src, uint8_t *channels_r, const char **endptr_r, GError **error_r) { unsigned long value; char *endptr; value = strtoul(src, &endptr, 10); if (endptr == src) { g_set_error(error_r, audio_parser_quark(), 0, "Failed to parse the channel count"); return false; } else if (!audio_valid_channel_count(value)) { g_set_error(error_r, audio_parser_quark(), 0, "Invalid channel count: %lu", value); return false; } *channels_r = value; *endptr_r = endptr; return true; } bool audio_format_parse(struct audio_format *dest, const char *src, GError **error_r) { uint32_t rate; uint8_t bits, channels; audio_format_clear(dest); /* parse sample rate */ if (!parse_sample_rate(src, &rate, &src, error_r)) return false; if (*src++ != ':') { g_set_error(error_r, audio_parser_quark(), 0, "Sample format missing"); return false; } /* parse sample format */ if (!parse_sample_format(src, &bits, &src, error_r)) return false; if (*src++ != ':') { g_set_error(error_r, audio_parser_quark(), 0, "Channel count missing"); return false; } /* parse channel count */ if (!parse_channel_count(src, &channels, &src, error_r)) return false; if (*src != 0) { g_set_error(error_r, audio_parser_quark(), 0, "Extra data after channel count: %s", src); return false; } audio_format_init(dest, rate, bits, channels); return true; }