{
}
+HTTPD::~HTTPD()
+{
+ MHD_quiesce_daemon(mhd);
+ for (Stream *stream : streams) {
+ stream->stop();
+ }
+ MHD_stop_daemon(mhd);
+}
+
void HTTPD::start(int port)
{
- MHD_start_daemon(MHD_USE_THREAD_PER_CONNECTION | MHD_USE_POLL_INTERNALLY | MHD_USE_DUAL_STACK,
- port,
- nullptr, nullptr,
- &answer_to_connection_thunk, this,
- MHD_OPTION_NOTIFY_COMPLETED, &request_completed_thunk, this,
- MHD_OPTION_END);
+ mhd = MHD_start_daemon(MHD_USE_THREAD_PER_CONNECTION | MHD_USE_POLL_INTERNALLY | MHD_USE_DUAL_STACK,
+ port,
+ nullptr, nullptr,
+ &answer_to_connection_thunk, this,
+ MHD_OPTION_NOTIFY_COMPLETED, &request_completed_thunk, this,
+ MHD_OPTION_END);
}
void HTTPD::add_data(const char *buf, size_t size, bool keyframe)
ssize_t HTTPD::Stream::reader_callback(uint64_t pos, char *buf, size_t max)
{
unique_lock<mutex> lock(buffer_mutex);
- has_buffered_data.wait(lock, [this]{ return !buffered_data.empty(); });
+ has_buffered_data.wait(lock, [this]{ return should_quit || !buffered_data.empty(); });
+ if (should_quit) {
+ return 0;
+ }
ssize_t ret = 0;
while (max > 0 && !buffered_data.empty()) {
has_buffered_data.notify_all();
}
+void HTTPD::Stream::stop()
+{
+ unique_lock<mutex> lock(buffer_mutex);
+ should_quit = true;
+ has_buffered_data.notify_all();
+}
#include <string>
struct MHD_Connection;
+struct MHD_Daemon;
class HTTPD {
public:
HTTPD();
+ ~HTTPD();
// Should be called before start().
void set_header(const std::string &data) {
DATA_TYPE_OTHER
};
void add_data(const char *buf, size_t size, DataType data_type);
+ void stop();
private:
Framing framing;
std::mutex buffer_mutex;
+ bool should_quit = false; // Under <buffer_mutex>.
std::condition_variable has_buffered_data;
- std::deque<std::string> buffered_data; // Protected by <mutex>.
+ std::deque<std::string> buffered_data; // Protected by <buffer_mutex>.
size_t used_of_buffered_data = 0; // How many bytes of the first element of <buffered_data> that is already used. Protected by <mutex>.
size_t seen_keyframe = false;
};
+ MHD_Daemon *mhd = nullptr;
std::mutex streams_mutex;
std::set<Stream *> streams; // Not owned.
std::string header;