1. 07 Sep, 2008 6 commits
    • Max Kellermann's avatar
      playlist: added is_valid_playlist_name() · dc353eca
      Max Kellermann authored
      The function valid_playlist_name() checks the name, but it insists on
      reporting an eventual error to the client.  The new function
      is_valid_playlist_name() is more generic: it just returns a boolean,
      and does not care what the caller will use it for.  The old function
      valid_playlist_name() will be removed later.
      dc353eca
    • Max Kellermann's avatar
      tag: added buffered versions of the tag_print.c code · 59efed3e
      Max Kellermann authored
      Currently, when the tag cache is being serialized to hard disk, the
      stdio buffer is flushed before every song, because tag_print.c
      performs unbuffered writes on the raw file descriptor.  Unfortunately,
      the fdprintf() API allows buffered I/O only for a client connection by
      looking up the client pointer owning the file descriptor - for stdio,
      this is not possible.  To re-enable proper stdio buffering, we have to
      duplicate the tag_print.c code without fprintf() instead of our custom
      fdprintf() hack.  Add this duplicated code to tag_save.c.
      59efed3e
    • Max Kellermann's avatar
      song: moved code to song_print.c, song_save.c · 75aa8dad
      Max Kellermann authored
      Move everything which dumps song information (via tag_print.c) to a
      separate source file.  song_print.c gets code which writes song data
      to the client; song_save.c is responsible for serializing songs from
      the tag cache.
      75aa8dad
    • Max Kellermann's avatar
      tag: moved code to tag_print.c · 386c3031
      Max Kellermann authored
      Move everything which dumps a tag to a file descriptor to tag_print.c.
      This relaxes dependencies and splits the code into smaller parts.
      386c3031
    • Max Kellermann's avatar
      client: added client_printf() · f73319c0
      Max Kellermann authored
      Based on client_puts(), client_printf() is the successor of
      fdprintf().  As soon as all fdprintf() callers have been rewritten to
      use client_printf(), we can remove fdprintf().
      f73319c0
    • Max Kellermann's avatar
      client: added client_write() and client_puts() · 33aec0d6
      Max Kellermann authored
      client_write() writes a buffer to the client and buffers it if
      required.  client_puts() does the same for a C string.  The next patch
      will add more tools which will replace fdprintf() later.
      33aec0d6
  2. 06 Sep, 2008 5 commits
    • Max Kellermann's avatar
      include cleanup · a34e1d2b
      Max Kellermann authored
      As usual, include only headers which are really needed.
      a34e1d2b
    • Eric Wong's avatar
      tag: fix segfault on update · 092bdf3d
      Eric Wong authored
      clearMpdTag could be called on a tag that was still in a
      tag_begin_add transaction before tag_end_add is called.  This
      was causing free() to attempt to operate on bulk.items; which is
      un-free()-able.  Now instead we unmark the bulk.busy to avoid
      committing the tags to the heap only to be immediately freed.
      
      Additionally, we need to remember to call tag_end_add() when
      a song is updated before we NULL song->tag to avoid tripping
      an assertion the next time tag_begin_add() is called.
      092bdf3d
    • Max Kellermann's avatar
      client: don't check FD_ISSET(client->fd) on expired client · 6146d4f5
      Max Kellermann authored
      client->fd becomes -1 when the client expires.  Don't use FD_ISSET()
      with this expired client; doing so would cause a crash due to SIGBUS.
      6146d4f5
    • Max Kellermann's avatar
      client: removed assert(client->fd)>=0 · 77b16716
      Max Kellermann authored
      Since client->fd==-1 has become our "expired" flag, it may already be
      -1 when client_close() is called.  Don't assert that it is still
      non-negative, and call client_set_expired() instead.
      77b16716
    • Max Kellermann's avatar
      dbUtils, playlist, directory: pass constant pointers · 35c0b84f
      Max Kellermann authored
      The usual bunch of const pointer conversions.
      35c0b84f
  3. 29 Aug, 2008 29 commits
    • Max Kellermann's avatar
      tag: optimize tag_dup(), copy item references · d38d2bc3
      Max Kellermann authored
      Don't call tag_pool_get_item() for duplicating tags, just increase the
      item's reference counter instead.
      d38d2bc3
    • Max Kellermann's avatar
      oggflac: fix GCC warnings · d8ad109e
      Max Kellermann authored
      Fix lots of "unused parameter" warnings in the OggFLAC decoder
      plugin.  Not sure if anybody uses it anymore, since newer libflac
      obsoletes it.
      d8ad109e
    • Max Kellermann's avatar
      tag: fix the shout and oggflac plugins · 01f9684f
      Max Kellermann authored
      During the tag library refactoring, the shout plugin was disabled, and
      I forgot about adapting it to the new API.  Apply the same fixes to
      the oggflac decoder plugin.
      01f9684f
    • Max Kellermann's avatar
      const pointers · 37d77caa
      Max Kellermann authored
      Yet another patch which converts pointer arguments to "const".
      37d77caa
    • Max Kellermann's avatar
      tag: static directory name · 5bd55516
      Max Kellermann authored
      While parsing the tag cache, don't allocate the directory name from
      the heap, but copy it into a buffer on the stack.  This reduces heap
      fragmentation by 1%.
      5bd55516
    • Max Kellermann's avatar
      tag: try not to reallocate tag.items in every add() call · 1aa34573
      Max Kellermann authored
      If many tag_items are added at once while the tag cache is being
      loaded, manage these items in a static fixed list, instead of
      reallocating the list with every newly created item.  This reduces
      heap fragmentation.
      
      Massif results again:
      
       mk before:  total 12,837,632; useful 10,626,383; extra 2,211,249
       mk now:     total 12,736,720; useful 10,626,383; extra 2,110,337
      
      The "useful" value is the same since this patch only changes the way
      we allocate the same amount of memory, but heap fragmentation was
      reduced by 5%.
      1aa34573
    • Max Kellermann's avatar
      song: don't export newNullSong() · 03152206
      Max Kellermann authored
      The function newNullSong() is only used internally in song.c.
      03152206
    • Max Kellermann's avatar
      tag: try not to duplicate the input string · a208d654
      Max Kellermann authored
      Try to detect if the string needs Latin1-UTF8 conversion, or
      whitespace cleanup.  If not, we don't need to allocate temporary
      memory, leading to decreased heap fragmentation.
      a208d654
    • Max Kellermann's avatar
      tag: pass length to fix_utf8() · 9352fc9e
      Max Kellermann authored
      Same as the previous patch, prepare the function fix_utf8() this time.
      9352fc9e
    • Max Kellermann's avatar
      added "length" parameter to validUtf8String() · 43c389b9
      Max Kellermann authored
      At several places, we create temporary copies of non-null-terminated
      strings, just to use them in functions like validUtf8String().  We can
      save this temporary allocation and avoid heap fragmentation if we
      add a length parameter instead of expecting a null-terminated string.
      43c389b9
    • Max Kellermann's avatar
      assert value!=NULL in fix_utf8() · 92b75767
      Max Kellermann authored
      We must never pass value==NULL to fix_utf().  Replace the run-time
      check with an assertion.
      92b75767
    • Max Kellermann's avatar
      tag: converted macro fixUtf8() to an inline function · f99fe808
      Max Kellermann authored
      Since the inline function cannot modify its caller's variables (which
      is a good thing for code readability), the new string pointer is the
      return value.  The resulting binary should be the same as with the
      macro.
      f99fe808
    • Max Kellermann's avatar
      tag: added a pool for tag items · c855415c
      Max Kellermann authored
      The new source tag_pool.c manages a pool of reference counted tag_item
      objects.  This is used to merge tag items of the same type and value,
      saving lots of memory.  Formerly, only the value itself was pooled,
      wasting memory for all the pointers and tag_item structs.
      
      The following results were measured with massif.  Started MPD on
      amd64, typed "mpc", no song being played.  My music database contains
      35k tagged songs.  The results are what massif reports as "peak".
      
       0.13.2:     total 14,131,392; useful 11,408,972; extra 2,722,420
       eric:       total 18,370,696; useful 15,648,182; extra 2,722,514
       mk f34f694e: total 15,833,952; useful 13,111,470; extra 2,722,482
       mk now:     total 12,837,632; useful 10,626,383; extra 2,211,249
      
      This patch set saves 20% memory, and does a good job in reducing heap
      fragmentation.
      c855415c
    • Max Kellermann's avatar
      tag: converted tag_item.value to a char array · e5a78798
      Max Kellermann authored
      The value is stored in the same memory allocation as the tag_item
      struct; this saves memory because we do not store the value pointer
      anymore.  Also remove the getTagItemString()/removeTagItemString()
      dummies.
      e5a78798
    • Max Kellermann's avatar
      removed tree.c · b731bbe9
      Max Kellermann authored
      This patch makes MPD consume much more memory because string pooling
      is disabled, but it prepares the next bunch of patches.  Replace the
      code in tagTracker.c with naive algorithms without the tree code.  For
      now, this should do; later we should find better algorithms,
      especially for getNumberOfTagItems(), which has become wasteful with
      temporary memory.
      b731bbe9
    • Max Kellermann's avatar
      tag: converted MpdTag.items to a pointer list · ad0e09b2
      Max Kellermann authored
      This prepares the following patches, which aim to reduce MPD's memory
      usage: we plan to share tag_item instances, instead of just their
      values.
      ad0e09b2
    • Max Kellermann's avatar
      tag: moved code to tag_id3.c · 6f72fe3e
      Max Kellermann authored
      The ID3 code uses only the public tag API, but is otherwise
      unrelated.  Move it to a separate source file.
      6f72fe3e
    • Max Kellermann's avatar
      wavpack: tag_new() cannot fail · 055f4a41
      Max Kellermann authored
      Since tag_new() uses xmalloc(), it cannot fail - if we're really out
      of memory, the process will abort.
      055f4a41
    • Max Kellermann's avatar
      5e1feb8f
    • Max Kellermann's avatar
      tag: renamed functions, no CamelCase · 91502cd7
      Max Kellermann authored
      91502cd7
    • Max Kellermann's avatar
      tag: renamed MpdTag and MpdTagItem to struct tag, struct mpd_tag_item · d0556dc9
      Max Kellermann authored
      Getting rid of CamelCase; not having typedefs also allows us to
      forward-declare the structures.
      d0556dc9
    • Max Kellermann's avatar
      added xfree() which takes a const pointer · f42de62a
      Max Kellermann authored
      Unfortunately, the C standard postulates that the argument to free()
      must be non-const.  This does not makes sense, and virtually prevents
      every pointer which must be freed at some time to be non-const.  Use
      the deconst hack (sorry for that) to allow us to free constant
      pointers.
      f42de62a
    • Max Kellermann's avatar
      export the function client_is_expired() · 8811c0e0
      Max Kellermann authored
      Instead of passing the pointer to the "expired" flag to
      processListOfCommands(), this function should use the client API to
      check this flag.  We can now remove the "global_expired" hack
      introduced recently.
      8811c0e0
    • Max Kellermann's avatar
      client: pass the client struct to processCommand() · 8b1b82b3
      Max Kellermann authored
      Start exporting the client struct as an opaque struct.  For now, pass
      it only to processCommand() and processListOfCommands(), and provide a
      function to extract the socket handle.  Later, we will propagate the
      pointer to all command implementations, and of course to
      client_print() etc.
      8b1b82b3
    • Max Kellermann's avatar
      client: reorder function declarations · 2c8aa8ef
      Max Kellermann authored
      Change the order of function declarations in client.h, to make it well
      arranged and readable.
      2c8aa8ef
    • Max Kellermann's avatar
      client: check "expired" after command execution · e743d71b
      Max Kellermann authored
      The old code tried to write a response to the client, without even
      checking if it was already closed.  Now that we have added more
      assertions, these may fail...  perform the "expired" check earlier.
      e743d71b
    • Max Kellermann's avatar
      client: added global "expired" flag · 76ecc302
      Max Kellermann authored
      Patch bdeb8e14 ("client: moved "expired" accesses into inline
      function") was created under the wrong assumption that
      processListOfCommands() could modify the expired flag, which is not
      the case.  Although "expired" is a non-const pointer,
      processListOfCommands() just reads it, using it as the break condition
      in a "while" loop.  I will address this issue with a better overall
      solution, but for now provide a pointer to a global "expired" flag.
      76ecc302
    • Max Kellermann's avatar
      pass constant pointers · 12bcba8b
      Max Kellermann authored
      And again, convert arguments to const.
      12bcba8b
    • Max Kellermann's avatar
      client: removed superfluous assertion · d8a8fa63
      Max Kellermann authored
      client_defer_output() was modified so that it can create the
      deferred_send list.  With this patch, the assertion on
      "deferred_send!=NULL" has become invalid.  Remove it.
      d8a8fa63