/* This file is part of Clementine. Copyright 2010, David Sansome Clementine 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 3 of the License, or (at your option) any later version. Clementine 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 Clementine. If not, see . */ #include "fingerprinter.h" #include "core/logging.h" #include #include #include Fingerprinter::Fingerprinter(const QString& filename) : filename_(filename), event_loop_(new QEventLoop), convert_element_(NULL) { } Fingerprinter::~Fingerprinter() { delete event_loop_; } bool Fingerprinter::GstreamerHasOfa() { GstElementFactory* factory = gst_element_factory_find("ofa"); if (!factory) { return false; } gst_object_unref(factory); return true; } GstElement* Fingerprinter::CreateElement(const QString &factory_name, GstElement *bin) { GstElement* ret = gst_element_factory_make( factory_name.toAscii().constData(), factory_name.toAscii().constData()); if (ret && bin) gst_bin_add(GST_BIN(bin), ret); if (!ret) { qLog(Warning) << "Couldn't create the gstreamer element" << factory_name; } return ret; } QString Fingerprinter::CreateFingerprint() { GstElement* pipeline = gst_pipeline_new("pipeline"); GstElement* src = CreateElement("filesrc", pipeline); GstElement* decode = CreateElement("decodebin2", pipeline); GstElement* convert = CreateElement("audioconvert", pipeline); GstElement* ofa = CreateElement("ofa", pipeline); GstElement* sink = CreateElement("fakesink", pipeline); if (!src || !decode || !convert || !ofa || !sink) { return QString(); } convert_element_ = convert; // Connect the elements gst_element_link_many(src, decode, NULL); gst_element_link_many(convert, ofa, sink, NULL); // Set the filename g_object_set(src, "location", filename_.toLocal8Bit().constData(), NULL); // Connect signals g_signal_connect(decode, "new-decoded-pad", G_CALLBACK(NewPadCallback), this); gst_bus_set_sync_handler(gst_pipeline_get_bus(GST_PIPELINE(pipeline)), BusCallbackSync, this); guint bus_callback_id = gst_bus_add_watch(gst_pipeline_get_bus(GST_PIPELINE(pipeline)), BusCallback, this); // Start playing gst_element_set_state(pipeline, GST_STATE_PLAYING); event_loop_->exec(); // Cleanup gst_bus_set_sync_handler(gst_pipeline_get_bus(GST_PIPELINE(pipeline)), NULL, NULL); g_source_remove(bus_callback_id); gst_object_unref(pipeline); return fingerprint_; } void Fingerprinter::NewPadCallback(GstElement*, GstPad* pad, gboolean, gpointer data) { Fingerprinter* instance = reinterpret_cast(data); GstPad* const audiopad = gst_element_get_pad(instance->convert_element_, "sink"); if (GST_PAD_IS_LINKED(audiopad)) { qLog(Warning) << "audiopad is already linked, unlinking old pad"; gst_pad_unlink(audiopad, GST_PAD_PEER(audiopad)); } gst_pad_link(pad, audiopad); gst_object_unref(audiopad); } void Fingerprinter::ReportError(GstMessage* msg) { GError* error; gchar* debugs; gst_message_parse_error(msg, &error, &debugs); QString message = QString::fromLocal8Bit(error->message); g_error_free(error); free(debugs); qLog(Error) << "Error processing" << filename_ << ":" << message; } gboolean Fingerprinter::BusCallback(GstBus*, GstMessage* msg, gpointer data) { Fingerprinter* instance = reinterpret_cast(data); switch (GST_MESSAGE_TYPE(msg)) { case GST_MESSAGE_ERROR: instance->ReportError(msg); instance->event_loop_->exit(); break; case GST_MESSAGE_TAG: instance->TagMessageReceived(msg); break; default: break; } return GST_BUS_DROP; } GstBusSyncReply Fingerprinter::BusCallbackSync(GstBus*, GstMessage* msg, gpointer data) { Fingerprinter* instance = reinterpret_cast(data); switch (GST_MESSAGE_TYPE(msg)) { case GST_MESSAGE_EOS: instance->event_loop_->exit(); break; case GST_MESSAGE_ERROR: instance->ReportError(msg); instance->event_loop_->exit(); break; case GST_MESSAGE_TAG: instance->TagMessageReceived(msg); break; default: break; } return GST_BUS_PASS; } void Fingerprinter::TagMessageReceived(GstMessage* message) { GstTagList* taglist = NULL; gst_message_parse_tag(message, &taglist); gchar* data = NULL; bool success = gst_tag_list_get_string(taglist, "ofa-fingerprint", &data); if (success && data) { fingerprint_ = QString::fromUtf8(data); g_free(data); } gst_tag_list_free(taglist); }