Bug 134663

Summary: Extreme memory usage when connecting to MTP MP3 Player
Product: [Applications] amarok Reporter: Josh Yavor <josh>
Component: Collections/MTP playerAssignee: Amarok Developers <amarok-bugs-dist>
Status: RESOLVED WORKSFORME    
Severity: wishlist    
Priority: NOR    
Version: unspecified   
Target Milestone: ---   
Platform: Ubuntu   
OS: Linux   
Latest Commit: Version Fixed In:
Sentry Crash Report:

Description Josh Yavor 2006-09-26 05:10:29 UTC
Version:           1.4-SVN Build Date 9/25/06 (using KDE KDE 3.5.2)
Installed from:    Ubuntu Packages
Compiler:          GCC 4.0.3-1 Ubuntu Package
OS:                Linux

Amarok's memory usage skyrockets when a MTP MP3 player, in this case a Creative Zen Vision:M with a large number of files is loaded.  It is expected that memory usage will increase with the addition of such a device, however on my system the usage goes from 42.1 MB when idle with no connected MP3 player to 749.4 MB + a few hundred megs of swap when idle (not playing a song) with the MP3 player connected.  This locks up my system for several minutes.

Basic System information:

Ubuntu 6.06
Kernel 2.6.15-27-686 (Ubuntu Package)
2.4 GHz P4
1 Gb RAM
Comment 1 Andy Kelk 2006-10-13 10:00:38 UTC
SVN commit 595044 by kelk:

Reduce memory usage of mtp plugin. Also simplify code by removing the trackValueList class.
Connecting to the device now also processes events so the UI remains responsive. 
CCBUG: 134663



 M  +1 -0      ChangeLog  
 M  +48 -201   src/mediadevice/mtp/mtpmediadevice.cpp  
 M  +2 -17     src/mediadevice/mtp/mtpmediadevice.h  


--- trunk/extragear/multimedia/amarok/ChangeLog #595043:595044
@@ -29,6 +29,7 @@
 
 
   CHANGES:
+    * Reduced memory usage for MTP media devices. (BR 134663)
     * Faster searching on playlist and startup, due to some optimizing in
       string usage. Patch by Ovidiu Gheorghioiu <ovidiug@gmail.com>.
     * Correctly translate media:, home:, ... style urls on KDE 3.5 and newer.
--- trunk/extragear/multimedia/amarok/src/mediadevice/mtp/mtpmediadevice.cpp #595043:595044
@@ -161,8 +161,7 @@
 
     QString genericError = i18n( "Could not send track" );
 
-    trackValueList::const_iterator it_track = m_trackList.findTrackByName( bundle.filename() );
-    if( it_track != m_trackList.end() )
+    if( m_fileNameToItem[ bundle.filename() ] != 0 )
     {
         // track already exists. don't do anything (for now).
         debug() << "Track already exists on device." << endl;
@@ -355,7 +354,6 @@
     kapp->processEvents( 100 );
 
     // cache the track
-    m_trackList.append( taggedTrack );
     return addTrackToView( taggedTrack );
 }
 
@@ -484,7 +482,10 @@
     int total,progress;
     total = items.count();
     progress = 0;
-    
+
+    if( total == 0 )
+        return 0;
+
     setProgress( progress, total );
     for( MtpMediaItem *it = dynamic_cast<MtpMediaItem*>(items.first()); it && !(m_canceled); it = dynamic_cast<MtpMediaItem*>(items.next()) )
     {
@@ -515,7 +516,7 @@
                 setProgress( progress );
             }
         }
-        else 
+        else
         {
             total--;
             setProgress( progress, total );
@@ -718,7 +719,7 @@
 }
 
 /**
- * Recursively remove MediaItem from the tracklist and the device
+ * Recursively remove MediaItem from the device and media view
  */
 int
 MtpMediaDevice::deleteItemFromDevice(MediaItem* item, int flags )
@@ -803,8 +804,7 @@
     }
     debug() << "track deleted" << endl;
 
-    // remove from the listview/tracklist
-    m_trackList.remove( trackItem->track() );
+    // remove from the media view
     delete trackItem;
     kapp->processEvents( 100 );
 
@@ -985,7 +985,6 @@
     QString Information;
     if( isConnected() )
     {
-        QString tracksFound;
         QString batteryLevel;
         QString secureTime;
         QString storageInformation;
@@ -1006,9 +1005,6 @@
         LIBMTP_Get_Secure_Time( m_device, &sectime );
         m_critical_mutex.unlock();
 
-        tracksFound = i18n( "1 track found on device",
-                            "%n tracks found on device ",
-                            m_trackList.size() );
         batteryLevel = i18n("Battery level: ")
             + QString::number( (int) ( (float) currbattlevel / (float) maxbattlevel * 100.0 ) )
             + '%';
@@ -1113,36 +1109,6 @@
 }
 
 /**
- * Handle expanding a media item (show sub-items)
- */
-void
-MtpMediaDevice::expandItem( QListViewItem *item )
-{
-    if( item == 0 || !item->isExpandable() || m_transferring )
-        return;
-
-    // First clear the item's children to repopulate.
-    while( item->firstChild() )
-        delete item->firstChild();
-
-    MtpMediaItem *it = dynamic_cast<MtpMediaItem *>( item );
-
-    switch( it->type() )
-    {
-        case MediaItem::ARTIST:
-            if( it->childCount() == 0 ) // Just to be sure
-                addAlbums( item->text( 0 ), it );
-            break;
-        case MediaItem::ALBUM:
-            if( it->childCount() == 0 )
-                addTracks( it->bundle()->artist(), item->text( 0 ), it );
-            break;
-        default:
-            break;
-    }
-}
-
-/**
  * Add gui elements to the device configuration
  */
 void
@@ -1232,60 +1198,18 @@
         item->m_device = this;
         QString titleName = track->bundle()->title();
         item->setTrack( track );
-
+        item->m_order = track->bundle()->track();
         item->setText( 0, titleName );
         item->setType( MediaItem::TRACK );
         item->setBundle( track->bundle() );
         item->track()->setId( track->id() );
+        m_fileNameToItem[ track->bundle()->filename() ] = item;
+        m_idToTrack[track->id()] = track;
     }
     return item;
 }
 
 /**
- * Add new albums to the tracklist
- */
-MtpMediaItem
-*MtpMediaDevice::addAlbums( const QString &artist, MtpMediaItem *item )
-{
-    for( trackValueList::iterator it = m_trackList.begin(); it != m_trackList.end(); it++ )
-    {
-        if( item->findItem( (*it)->bundle()->album() ) == 0 && ( (*it)->bundle()->artist().string() == artist ) )
-        {
-            MtpMediaItem *album = new MtpMediaItem( item );
-            album->setText( 0, (*it)->bundle()->album() );
-            album->setType( MediaItem::ALBUM );
-            album->setExpandable( true );
-            album->setBundle( (*it)->bundle() );
-            album->m_device = this;
-            expandItem( album );
-        }
-    }
-    return item;
-}
-
-/**
- * Add new tracks to the tracklist
- */
-MtpMediaItem
-*MtpMediaDevice::addTracks(const QString &artist, const QString &album, MtpMediaItem *item)
-{
-    for( trackValueList::iterator it = m_trackList.begin(); it != m_trackList.end(); it++ )
-    {
-        if( ( (*it)->bundle()->album().string() == album ) && ( (*it)->bundle()->artist().string() == artist ))
-        {
-            MtpMediaItem *track = new MtpMediaItem( item );
-            track->setText( 0, (*it)->bundle()->title() );
-            track->setType( MediaItem::TRACK );
-            track->setBundle( (*it)->bundle() );
-            track->setTrack( (*it) );
-            track->m_device = this;
-            track->m_order = (*it)->bundle()->track();
-        }
-    }
-    return item;
-}
-
-/**
  * Get tracks and add them to the listview
  */
 int
@@ -1294,42 +1218,53 @@
 
     DEBUG_BLOCK
 
-    int result = 0;
+    clearItems();
 
-    if( m_trackList.isEmpty() )
-    {
-        m_critical_mutex.lock();
-        result = m_trackList.readFromDevice( this );
-        m_critical_mutex.unlock();
-    }
+    m_critical_mutex.lock();
 
-    debug()<< "Result : " << result << endl;
+    QString genericError = i18n( "Could not get music from MTP Device" );
 
-    clearItems();
+    setProgress( 0, 100 ); // we don't know how many tracks. fake progress bar.
 
-    if( result == 0 )
-    {
-        kapp->processEvents( 100 );
+    kapp->processEvents( 100 );
 
-        for( trackValueList::iterator it = m_trackList.begin(); it != m_trackList.end(); it++ )
-        {
-            if( m_view->findItem( ((*it)->bundle()->artist().string()), 0 ) == 0 )
-            {
-                MtpMediaItem *artist = new MtpMediaItem( m_view );
-                artist->setText( 0, (*it)->bundle()->artist() );
-                artist->setType( MediaItem::ARTIST );
-                artist->setExpandable( true );
-                artist->setBundle( (*it)->bundle() );
-                artist->m_device = this;
+    LIBMTP_track_t *tracks = LIBMTP_Get_Tracklisting( m_device );
 
-                expandItem( artist );
-            }
+    debug() << "Got tracks from device" << endl;
+
+    if( tracks == 0 )
+    {
+        debug()<< "Error reading tracks. 0 returned." << endl;
+        Amarok::StatusBar::instance()->shortLongMessage(
+            genericError,
+            i18n( "Could not read MTP Device tracks" ),
+            KDE::StatusBar::Error
+        );
+        hideProgress();
+        return -1;
+    }
+    else
+    {
+        LIBMTP_track_t *tmp;
+        while( tracks != 0 )
+        {
+            MtpTrack *mtp_track = new MtpTrack( tracks );
+            mtp_track->readMetaData( tracks );
+            addTrackToView( mtp_track );
+            tmp = tracks;
+            tracks = tracks->next;
+            LIBMTP_destroy_track_t( tmp );
+            kapp->processEvents( 100 );
         }
     }
+    setProgress( 100 );
+    hideProgress();
 
+    m_critical_mutex.unlock();
+
     readPlaylists();
 
-    return result;
+    return 0;
 }
 
 /**
@@ -1354,7 +1289,7 @@
             uint32_t i;
             for( i = 0; i < playlists->no_tracks; i++ )
             {
-                MtpTrack *track = *(m_trackList.findTrackById( playlists->tracks[i] ));
+                MtpTrack *track = m_idToTrack[ playlists->tracks[i] ];
                 MtpMediaItem *item = new MtpMediaItem( playlist );
                 item->setText( 0, track->bundle()->artist() + " - " + track->bundle()->title() );
                 item->setType( MediaItem::PLAYLISTITEM );
@@ -1383,94 +1318,6 @@
 }
 
 /**
- * trackValueList Class
- */
-
-/**
- * Find a track by its id
- */
-trackValueList::iterator
-trackValueList::findTrackById( unsigned _id )
-{
-    trackValueList::iterator it;
-    for( it = begin(); it != end(); it++ )
-        if( (*it)->id() == _id )
-            break;
-    return it;
-}
-
-/**
- * Find a track by its id
- */
-trackValueList::const_iterator
-trackValueList::findTrackById( unsigned _id ) const
-{
-    trackValueList::const_iterator it;
-    for( it = begin(); it != end(); it++ )
-        if( (*it)->id() == _id )
-            break;
-    return it;
-}
-
-/**
- * Find a track by its name
- */
-trackValueList::iterator
-trackValueList::findTrackByName( const QString &_filename )
-{
-    trackValueList::iterator it;
-    for( it = begin(); it != end(); it++ )
-        if( (*it)->bundle()->url().path() == _filename )
-            break;
-    return it;
-}
-
-/**
- * Transfer info from the device to local structures
- */
-int
-trackValueList::readFromDevice( MtpMediaDevice *mtp )
-{
-
-    DEBUG_BLOCK
-
-    LIBMTP_mtpdevice_t *device = mtp->current_device();
-
-    QString genericError = i18n( "Could not get music from MTP Device" );
-
-    LIBMTP_track_t *tracks = LIBMTP_Get_Tracklisting( device );
-
-    debug() << "Got tracks from device" << endl;
-
-    if( tracks == 0 )
-    {
-        debug()<< "Error reading tracks. 0 returned." << endl;
-        Amarok::StatusBar::instance()->shortLongMessage(
-            genericError,
-            i18n( "Could not read MTP Device tracks" ),
-            KDE::StatusBar::Error
-        );
-        return -1;
-    }
-    else
-    {
-        LIBMTP_track_t *tmp;
-        while( tracks != 0 )
-        {
-            MtpTrack *mtp_track = new MtpTrack( tracks );
-            mtp_track->readMetaData( tracks );
-            append( mtp_track );
-            tmp = tracks;
-            tracks = tracks->next;
-            LIBMTP_destroy_track_t( tmp );
-        }
-        kapp->processEvents( 100 );
-    }
-
-    return 0;
-}
-
-/**
  * MtpTrack Class
  */
 MtpTrack::MtpTrack( LIBMTP_track_t *track )
--- trunk/extragear/multimedia/amarok/src/mediadevice/mtp/mtpmediadevice.h #595043:595044
@@ -102,17 +102,6 @@
         MtpPlaylist         *m_playlist;
 };
 
-
-class trackValueList: public QValueList<MtpTrack *>
-{
-    public:
-        trackValueList::iterator        findTrackById( unsigned );
-        trackValueList::const_iterator  findTrackById( unsigned ) const;
-        trackValueList::iterator        findTrackByName( const QString& );
-        int                             readFromDevice( MtpMediaDevice *mtp );
-};
-
-
 class MtpMediaDevice : public MediaDevice
 {
     Q_OBJECT
@@ -135,8 +124,6 @@
         virtual void            applyConfig();
         virtual void            loadConfig();
         static int              progressCallback( uint64_t const sent, uint64_t const total, void const * const data );
-    public slots:
-        void                    expandItem( QListViewItem *item );
 
     protected:
         MediaItem*              trackExists( const MetaBundle &bundle );
@@ -157,9 +144,6 @@
         virtual void            updateRootItems() {};
 
     private:
-        MtpMediaItem            *addAlbums( const QString &artist, MtpMediaItem *item );
-        MtpMediaItem            *addTracks( const QString &artist, const QString &track, MtpMediaItem *item );
-        MtpMediaItem            *addArtist( MtpTrack *track );
         MtpMediaItem            *addTrackToView(MtpTrack *track, MtpMediaItem *item=0 );
         int                     readMtpMusic( void );
         void                    clearItems();
@@ -172,7 +156,6 @@
         void                    initView( void );
         void                    readPlaylists( void );
         void                    playlistFromItem( MtpMediaItem *item);
-        trackValueList          m_trackList;
         LIBMTP_mtpdevice_t      *m_device;
         QMutex                  m_mutex;
         QMutex                  m_critical_mutex;
@@ -183,6 +166,8 @@
         QLabel                  *m_folderLabel;
         QStringList             m_supportedFiles;
         QMap<int,QString>       mtpFileTypes;
+        QMap<uint32_t,MtpTrack*> m_idToTrack;
+        QMap<QString,MtpMediaItem*> m_fileNameToItem;
 };
 
 #endif
Comment 2 Andy Kelk 2006-10-13 10:57:39 UTC
On my lightly loaded mp3 player, I've managed to reduce the memory usage. There's 819 tracks on there. Previously, memory usage went from 33,656 Kb (unplugged) to 74,680 Kb (plugged).
With the changes applied to SVN (above), the memory usage goes from 34,660 Kb (unplugged) to 35,028 Kb (plugged).
By my calculations that's 39 Mb of memory begin saved (about 50 Kb per track).
I'd be interested to see if the effects are as dramatic on your system.
Comment 3 Josh Yavor 2006-10-13 13:12:43 UTC
Thanks for your effort.  My mp3 player is about 50% loaded, about 1700 tracks total.  I will try out the new SVN ASAP and post the results.
Comment 4 Josh Yavor 2006-10-14 02:39:24 UTC
Bug resolution confirmed.  Memory usage is back to normal.  Thanks Andy!