Commit e4e4576a authored by Max Kellermann's avatar Max Kellermann

Merge tag 'v0.22.9'

release v0.22.9
parents f7894510 18628bf8
...@@ -16,11 +16,16 @@ ver 0.23 (not yet released) ...@@ -16,11 +16,16 @@ ver 0.23 (not yet released)
- new tags "ComposerSort", "Ensemble", "Movement", "MovementNumber", and "Location" - new tags "ComposerSort", "Ensemble", "Movement", "MovementNumber", and "Location"
* new build-time dependency: libfmt * new build-time dependency: libfmt
ver 0.22.9 (not yet released) ver 0.22.9 (2021/06/23)
* database * database
- simple: load all .mpdignore files of all parent directories - simple: load all .mpdignore files of all parent directories
* tags
- fix "readcomments" and "readpicture" on remote files with ID3 tags
* decoder * decoder
- ffmpeg: support the tags "sort_album", "album-sort", "artist-sort" - ffmpeg: support the tags "sort_album", "album-sort", "artist-sort"
- ffmpeg: fix build failure with FFmpeg 3.4
* Android
- fix auto-start on boot in Android 8 or later
* Windows * Windows
- fix build failure with SQLite - fix build failure with SQLite
......
...@@ -14,7 +14,7 @@ For basic installation instructions ...@@ -14,7 +14,7 @@ For basic installation instructions
- [Manual](http://www.musicpd.org/doc/user/) - [Manual](http://www.musicpd.org/doc/user/)
- [Forum](http://forum.musicpd.org/) - [Forum](http://forum.musicpd.org/)
- [IRC](irc://chat.freenode.net/#mpd) - [IRC](ircs://irc.libera.chat:6697/#mpd)
- [Bug tracker](https://github.com/MusicPlayerDaemon/MPD/issues/) - [Bug tracker](https://github.com/MusicPlayerDaemon/MPD/issues/)
# Developers # Developers
......
...@@ -414,6 +414,15 @@ public class Main extends Service implements Runnable { ...@@ -414,6 +414,15 @@ public class Main extends Service implements Runnable {
* start Main service without any callback * start Main service without any callback
*/ */
public static void start(Context context, boolean wakelock) { public static void start(Context context, boolean wakelock) {
context.startService(new Intent(context, Main.class).putExtra("wakelock", wakelock)); Intent intent = new Intent(context, Main.class)
.putExtra("wakelock", wakelock);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O)
/* in Android 8+, we need to use this method
or else we'll get "IllegalStateException:
app is in background" */
context.startForegroundService(intent);
else
context.startService(intent);
} }
} }
...@@ -760,7 +760,7 @@ A resampler using `libsamplerate <http://www.mega-nerd.com/SRC/>`_ a.k.a. Secret ...@@ -760,7 +760,7 @@ A resampler using `libsamplerate <http://www.mega-nerd.com/SRC/>`_ a.k.a. Secret
* - Name * - Name
- Description - Description
* - **type** * - **type**
- The interpolator type. See below for a list of known types. - The interpolator type. Defaults to :samp:`2`. See below for a list of known types.
The following converter types are provided by libsamplerate: The following converter types are provided by libsamplerate:
......
...@@ -689,6 +689,8 @@ The State File ...@@ -689,6 +689,8 @@ The State File
- Specify the state file location. The parent directory must be writable by the :program:`MPD` user (+wx). - Specify the state file location. The parent directory must be writable by the :program:`MPD` user (+wx).
* - **state_file_interval SECONDS** * - **state_file_interval SECONDS**
- Auto-save the state file this number of seconds after each state change. Defaults to 120 (2 minutes). - Auto-save the state file this number of seconds after each state change. Defaults to 120 (2 minutes).
* - **restore_paused yes|no**
- If set to :samp:`yes`, then :program:`MPD` is put into pause mode instead of starting playback after startup. Default is :samp:`no`.
The Sticker Database The Sticker Database
^^^^^^^^^^^^^^^^^^^^ ^^^^^^^^^^^^^^^^^^^^
...@@ -1121,7 +1123,7 @@ Support ...@@ -1121,7 +1123,7 @@ Support
Getting Help Getting Help
^^^^^^^^^^^^ ^^^^^^^^^^^^
The :program:`MPD` project runs a `forum <https://forum.musicpd.org/>`_ and an IRC channel (#mpd on Freenode) for requesting help. Visit the MPD help page for details on how to get help. The :program:`MPD` project runs a `forum <https://forum.musicpd.org/>`_ and an IRC channel (#mpd on Libera.Chat) for requesting help. Visit the MPD help page for details on how to get help.
Common Problems Common Problems
^^^^^^^^^^^^^^^ ^^^^^^^^^^^^^^^
......
...@@ -20,7 +20,7 @@ class Project: ...@@ -20,7 +20,7 @@ class Project:
self.base = base self.base = base
if name is None or version is None: if name is None or version is None:
m = re.match(r'^([-\w]+)-(\d[\d.]*[a-z]?[\d.]*(?:-alpha\d+)?)$', self.base) m = re.match(r'^([-\w]+)-(\d[\d.]*[a-z]?[\d.]*(?:-alpha\d+)?)(\+.*)?$', self.base)
if name is None: name = m.group(1) if name is None: name = m.group(1)
if version is None: version = m.group(2) if version is None: version = m.group(2)
......
...@@ -489,6 +489,15 @@ MainConfigured(const struct options &options, const ConfigData &raw_config) ...@@ -489,6 +489,15 @@ MainConfigured(const struct options &options, const ConfigData &raw_config)
LogError(std::current_exception(), LogError(std::current_exception(),
"Zeroconf initialization failed"); "Zeroconf initialization failed");
} }
AtScopeExit(&zeroconf, &instance) {
if (zeroconf) {
auto &event_loop = instance.io_thread.GetEventLoop();
BlockingCall(event_loop, [&](){
zeroconf.reset();
});
}
};
#endif #endif
#ifdef ENABLE_DATABASE #ifdef ENABLE_DATABASE
...@@ -550,16 +559,6 @@ MainConfigured(const struct options &options, const ConfigData &raw_config) ...@@ -550,16 +559,6 @@ MainConfigured(const struct options &options, const ConfigData &raw_config)
instance.state_file->Write(); instance.state_file->Write();
instance.BeginShutdownUpdate(); instance.BeginShutdownUpdate();
#ifdef HAVE_ZEROCONF
if (zeroconf) {
auto &event_loop = instance.io_thread.GetEventLoop();
BlockingCall(event_loop, [&](){
zeroconf.reset();
});
}
#endif
instance.BeginShutdownPartitions(); instance.BeginShutdownPartitions();
} }
......
...@@ -25,6 +25,7 @@ ...@@ -25,6 +25,7 @@
#include "client/Client.hxx" #include "client/Client.hxx"
#include "protocol/Ack.hxx" #include "protocol/Ack.hxx"
#include "fs/AllocatedPath.hxx" #include "fs/AllocatedPath.hxx"
#include "input/InputStream.hxx"
#include "util/Compiler.h" #include "util/Compiler.h"
#include "util/UriExtract.hxx" #include "util/UriExtract.hxx"
#include "LocateUri.hxx" #include "LocateUri.hxx"
...@@ -32,8 +33,13 @@ ...@@ -32,8 +33,13 @@
static void static void
TagScanStream(const char *uri, TagHandler &handler) TagScanStream(const char *uri, TagHandler &handler)
{ {
if (!tag_stream_scan(uri, handler)) Mutex mutex;
auto is = InputStream::OpenReady(uri, mutex);
if (!tag_stream_scan(*is, handler))
throw ProtocolError(ACK_ERROR_NO_EXIST, "Failed to load file"); throw ProtocolError(ACK_ERROR_NO_EXIST, "Failed to load file");
ScanGenericTags(*is, handler);
} }
static void static void
......
...@@ -430,6 +430,7 @@ SendGroup(mpd_connection *connection, TagType group) ...@@ -430,6 +430,7 @@ SendGroup(mpd_connection *connection, TagType group)
return mpd_search_add_group_tag(connection, tag); return mpd_search_add_group_tag(connection, tag);
#else #else
(void)connection; (void)connection;
(void)group;
throw std::runtime_error("Grouping requires libmpdclient 2.12"); throw std::runtime_error("Grouping requires libmpdclient 2.12");
#endif #endif
......
...@@ -470,6 +470,7 @@ IsSeekable(const AVFormatContext &format_context) noexcept ...@@ -470,6 +470,7 @@ IsSeekable(const AVFormatContext &format_context) noexcept
#if LIBAVFORMAT_VERSION_INT >= AV_VERSION_INT(58, 6, 100) #if LIBAVFORMAT_VERSION_INT >= AV_VERSION_INT(58, 6, 100)
return (format_context.ctx_flags & AVFMTCTX_UNSEEKABLE) != 0; return (format_context.ctx_flags & AVFMTCTX_UNSEEKABLE) != 0;
#else #else
(void)format_context;
return false; return false;
#endif #endif
} }
...@@ -658,6 +659,8 @@ ffmpeg_scan_stream(InputStream &is, TagHandler &handler) ...@@ -658,6 +659,8 @@ ffmpeg_scan_stream(InputStream &is, TagHandler &handler)
return FfmpegScanStream(*f, handler); return FfmpegScanStream(*f, handler);
} }
#if LIBAVFORMAT_VERSION_INT >= AV_VERSION_INT(58, 9, 100)
static void static void
ffmpeg_uri_decode(DecoderClient &client, const char *uri) ffmpeg_uri_decode(DecoderClient &client, const char *uri)
{ {
...@@ -689,6 +692,8 @@ ffmpeg_protocols() noexcept ...@@ -689,6 +692,8 @@ ffmpeg_protocols() noexcept
return protocols; return protocols;
} }
#endif
/** /**
* A list of extensions found for the formats supported by ffmpeg. * A list of extensions found for the formats supported by ffmpeg.
* This list is current as of 02-23-09; To find out if there are more * This list is current as of 02-23-09; To find out if there are more
...@@ -812,6 +817,8 @@ static const char *const ffmpeg_mime_types[] = { ...@@ -812,6 +817,8 @@ static const char *const ffmpeg_mime_types[] = {
constexpr DecoderPlugin ffmpeg_decoder_plugin = constexpr DecoderPlugin ffmpeg_decoder_plugin =
DecoderPlugin("ffmpeg", ffmpeg_decode, ffmpeg_scan_stream) DecoderPlugin("ffmpeg", ffmpeg_decode, ffmpeg_scan_stream)
.WithInit(ffmpeg_init, ffmpeg_finish) .WithInit(ffmpeg_init, ffmpeg_finish)
#if LIBAVFORMAT_VERSION_INT >= AV_VERSION_INT(58, 9, 100)
.WithProtocols(ffmpeg_protocols, ffmpeg_uri_decode) .WithProtocols(ffmpeg_protocols, ffmpeg_uri_decode)
#endif
.WithSuffixes(ffmpeg_suffixes) .WithSuffixes(ffmpeg_suffixes)
.WithMimeTypes(ffmpeg_mime_types); .WithMimeTypes(ffmpeg_mime_types);
...@@ -32,6 +32,7 @@ LastInputStream::~LastInputStream() noexcept = default; ...@@ -32,6 +32,7 @@ LastInputStream::~LastInputStream() noexcept = default;
void void
LastInputStream::Close() noexcept LastInputStream::Close() noexcept
{ {
uri.clear();
is.reset(); is.reset();
close_timer.Cancel(); close_timer.Cancel();
} }
......
...@@ -64,8 +64,7 @@ public: ...@@ -64,8 +64,7 @@ public:
return is.get(); return is.get();
} }
is.reset(); Close();
close_timer.Cancel();
is = open(new_uri, mutex); is = open(new_uri, mutex);
uri = std::forward<U>(new_uri); uri = std::forward<U>(new_uri);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment