~ubuntu-branches/ubuntu/utopic/clementine/utopic

« back to all changes in this revision

Viewing changes to src/devices/cddadevice.cpp

  • Committer: Package Import Robot
  • Author(s): Thomas PIERSON
  • Date: 2012-01-01 20:43:39 UTC
  • mfrom: (1.1.1)
  • Revision ID: package-import@ubuntu.com-20120101204339-lsb6nndwhfy05sde
Tags: 1.0.1+dfsg-1
New upstream release. (Closes: #653926, #651611, #657391)

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
/* This file is part of Clementine.
 
2
   Copyright 2010, David Sansome <me@davidsansome.com>
 
3
 
 
4
   Clementine is free software: you can redistribute it and/or modify
 
5
   it under the terms of the GNU General Public License as published by
 
6
   the Free Software Foundation, either version 3 of the License, or
 
7
   (at your option) any later version.
 
8
 
 
9
   Clementine is distributed in the hope that it will be useful,
 
10
   but WITHOUT ANY WARRANTY; without even the implied warranty of
 
11
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
12
   GNU General Public License for more details.
 
13
 
 
14
   You should have received a copy of the GNU General Public License
 
15
   along with Clementine.  If not, see <http://www.gnu.org/licenses/>.
 
16
*/
 
17
 
 
18
#include <QMutexLocker>
 
19
 
 
20
#include "core/logging.h"
 
21
#include "core/timeconstants.h"
 
22
#include "library/librarybackend.h"
 
23
#include "library/librarymodel.h"
 
24
 
 
25
#include "cddadevice.h"
 
26
 
 
27
CddaDevice::CddaDevice(const QUrl& url, DeviceLister* lister,
 
28
    const QString& unique_id, DeviceManager* manager,
 
29
    int database_id, bool first_time)
 
30
      : ConnectedDevice(url, lister, unique_id, manager, database_id, first_time),
 
31
        cdda_(NULL),
 
32
        cdio_(NULL)
 
33
{
 
34
}
 
35
 
 
36
CddaDevice::~CddaDevice(){
 
37
  if (cdio_)
 
38
    cdio_destroy(cdio_);
 
39
}
 
40
 
 
41
void CddaDevice::Init() {
 
42
  QMutexLocker locker(&mutex_init_);
 
43
  song_count_ = 0; // Reset song count, in case it was already set
 
44
  cdio_ = cdio_open (url_.path().toLocal8Bit().constData(), DRIVER_DEVICE);
 
45
  if (cdio_ == NULL) {
 
46
    return;
 
47
  }
 
48
  // Create gstreamer cdda element
 
49
  cdda_ = gst_element_make_from_uri (GST_URI_SRC, "cdda://", NULL);
 
50
  if (cdda_ == NULL) {
 
51
    model_->Reset();
 
52
    return;
 
53
  }
 
54
 
 
55
  GST_CDDA_BASE_SRC(cdda_)->device = g_strdup (url_.path().toLocal8Bit().constData());
 
56
 
 
57
  // Change the element's state to ready and paused, to be able to query it
 
58
  if (gst_element_set_state(cdda_, GST_STATE_READY) == GST_STATE_CHANGE_FAILURE ||
 
59
      gst_element_set_state(cdda_, GST_STATE_PAUSED) == GST_STATE_CHANGE_FAILURE) {
 
60
    model_->Reset();
 
61
    gst_element_set_state(cdda_, GST_STATE_NULL);
 
62
    gst_object_unref(GST_OBJECT(cdda_));
 
63
    return;
 
64
  }
 
65
 
 
66
  // Get number of tracks
 
67
  GstFormat fmt = gst_format_get_by_nick ("track");
 
68
  GstFormat out_fmt = fmt;
 
69
  gint64 num_tracks = 0;
 
70
  if (!gst_element_query_duration (cdda_, &out_fmt, &num_tracks) || out_fmt != fmt) {
 
71
    qLog(Error) << "Error while querying cdda GstElement";
 
72
    model_->Reset();
 
73
    gst_object_unref(GST_OBJECT(cdda_));
 
74
    return;
 
75
  }
 
76
 
 
77
  SongList songs;
 
78
  for (int track_number = 1; track_number <= num_tracks; track_number++) {
 
79
    // Init song
 
80
    Song song;
 
81
    guint64 duration = 0;
 
82
    // quint64 == ulonglong and guint64 == ulong, therefore we must cast
 
83
    if (gst_tag_list_get_uint64 (GST_CDDA_BASE_SRC(cdda_)->tracks[track_number-1].tags,
 
84
                                 GST_TAG_DURATION, &duration)) {
 
85
      song.set_length_nanosec((quint64)duration);
 
86
    }
 
87
    song.set_id(track_number);
 
88
    song.set_valid(true);
 
89
    song.set_filetype(Song::Type_Cdda);
 
90
    song.set_url(QUrl(QString("cdda://%1/%2").arg(url_.path()).arg(track_number)));
 
91
    song.set_title(QString("Track %1").arg(track_number));
 
92
    song.set_track(track_number);
 
93
    songs << song;
 
94
  }
 
95
  song_count_ = num_tracks;
 
96
  connect(this, SIGNAL(SongsDiscovered(const SongList&)), model_, SLOT(SongsDiscovered(const SongList&)));
 
97
  emit SongsDiscovered(songs);
 
98
 
 
99
  // Generate MusicBrainz DiscId
 
100
  gst_tag_register_musicbrainz_tags();
 
101
  GstElement *pipe = gst_pipeline_new ("pipeline");
 
102
  gst_bin_add (GST_BIN (pipe), cdda_);
 
103
  gst_element_set_state (pipe, GST_STATE_READY);
 
104
  gst_element_set_state (pipe, GST_STATE_PAUSED);
 
105
  GstMessage *msg = gst_bus_timed_pop_filtered (GST_ELEMENT_BUS (pipe),
 
106
                    GST_CLOCK_TIME_NONE,
 
107
                    GST_MESSAGE_TAG);
 
108
  GstTagList *tags = NULL;
 
109
  gst_message_parse_tag (msg, &tags);
 
110
  char *string_mb = NULL;
 
111
  if (gst_tag_list_get_string (tags, GST_TAG_CDDA_MUSICBRAINZ_DISCID, &string_mb)) {
 
112
    QString musicbrainz_discid(string_mb);
 
113
    qLog(Info) << "MusicBrainz discid: " << musicbrainz_discid;
 
114
 
 
115
    MusicBrainzClient *musicbrainz_client = new MusicBrainzClient(this);
 
116
    connect(musicbrainz_client,
 
117
            SIGNAL(Finished(const QString&, const QString&, MusicBrainzClient::ResultList)),
 
118
            SLOT(AudioCDTagsLoaded(const QString&, const QString&, MusicBrainzClient::ResultList)));
 
119
    musicbrainz_client->StartDiscIdRequest(musicbrainz_discid);
 
120
    g_free(string_mb);
 
121
  }
 
122
 
 
123
  // Clean all the Gstreamer objects we have used: we don't need them anymore
 
124
  gst_element_set_state (pipe, GST_STATE_NULL);
 
125
  // This will also cause cdda_ to be unref'd.
 
126
  gst_object_unref(GST_OBJECT(pipe));
 
127
  gst_object_unref(GST_OBJECT(msg));
 
128
  gst_tag_list_free(tags);
 
129
 
 
130
}
 
131
 
 
132
void CddaDevice::AudioCDTagsLoaded(const QString& artist, const QString& album,
 
133
                                   const MusicBrainzClient::ResultList& results) {
 
134
  MusicBrainzClient *musicbrainz_client = qobject_cast<MusicBrainzClient*>(sender());
 
135
  musicbrainz_client->deleteLater();
 
136
  SongList songs;
 
137
  int track_number = 1;
 
138
  if (results.size() == 0)
 
139
    return;
 
140
  model_->Reset();
 
141
  foreach (const MusicBrainzClient::Result& ret, results) {
 
142
    Song song;
 
143
    song.set_artist(artist);
 
144
    song.set_album(album);
 
145
    song.set_title(ret.title_);
 
146
    song.set_length_nanosec(ret.duration_msec_ * kNsecPerMsec);
 
147
    song.set_track(track_number);
 
148
    song.set_year(ret.year_);
 
149
    song.set_id(track_number);
 
150
    // We need to set url: that's how playlist will find the correct item to update
 
151
    song.set_url(QUrl(QString("cdda://%1/%2").arg(unique_id()).arg(track_number++)));
 
152
    songs << song;
 
153
  }
 
154
  connect(this, SIGNAL(SongsDiscovered(const SongList&)), model_, SLOT(SongsDiscovered(const SongList&)));
 
155
  emit SongsDiscovered(songs);
 
156
}
 
157
 
 
158
void CddaDevice::Refresh() {
 
159
  if ((cdio_ && cdda_) &&                /* already init... */
 
160
      cdio_get_media_changed(cdio_) != 1 /* ...and hasn't change since last time */) {
 
161
    return;
 
162
  }
 
163
  // Check if mutex is already token (i.e. init is already taking place)
 
164
  if (!mutex_init_.tryLock()) {
 
165
    return;
 
166
  }
 
167
  mutex_init_.unlock();
 
168
  Init();
 
169
}