]> git.sesse.net Git - cubemap/blobdiff - server.cpp
Implement sleeping/waking clients.
[cubemap] / server.cpp
index 9b91e0371254e67044a8138424334a21aea4b88c..b914b2ec4fbaa50175021038e42ed9161cd21bad 100644 (file)
@@ -107,6 +107,10 @@ void Server::set_header(const string &stream_id, const string &header)
        
 void Server::add_data(const string &stream_id, const char *data, size_t bytes)
 {
+       if (bytes == 0) {
+               return;
+       }
+
        MutexLock lock(&mutex);
        assert(streams.count(stream_id) != 0);
        Stream *stream = &streams[stream_id];
@@ -122,8 +126,7 @@ void Server::add_data(const string &stream_id, const char *data, size_t bytes)
        }
 
        memcpy(stream->data + pos, data, bytes);
-
-       // TODO: wake up clients
+       wake_up_all_clients();
 }
        
 void Server::process_client(Client *client)
@@ -231,10 +234,12 @@ void Server::process_client(Client *client)
                        close_client(client);
                        return;
                }
-               client->bytes_sent += ret;      
-
-               // TODO: put clients to sleep
+               client->bytes_sent += ret;
 
+               if (client->bytes_sent == stream.data_size) {
+                       // We don't have any more data for this client, so put it to sleep.
+                       put_client_to_sleep(client);
+               }
                break;
        }
        default:
@@ -276,3 +281,31 @@ void Server::close_client(Client *client)
        close(client->sock);
        clients.erase(client->sock);
 }
+       
+void Server::put_client_to_sleep(Client *client)
+{
+       epoll_event ev;
+       ev.events = EPOLLRDHUP;
+       ev.data.fd = client->sock;
+
+       if (epoll_ctl(epoll_fd, EPOLL_CTL_MOD, client->sock, &ev) == -1) {
+               perror("epoll_ctl(EPOLL_CTL_MOD)");
+               exit(1);
+       }
+
+       sleeping_clients.push_back(client->sock);
+}
+
+void Server::wake_up_all_clients()
+{
+       for (unsigned i = 0; i < sleeping_clients.size(); ++i) {
+               epoll_event ev;
+               ev.events = EPOLLOUT | EPOLLRDHUP;
+               ev.data.fd = sleeping_clients[i];
+               if (epoll_ctl(epoll_fd, EPOLL_CTL_MOD, sleeping_clients[i], &ev) == -1) {
+                       perror("epoll_ctl(EPOLL_CTL_MOD)");
+                       exit(1);
+               }
+       }
+       sleeping_clients.clear();
+}