OS: Make ServerPort acquisition idempotent

This is required for the ssl and http modules and probably others.
This commit is contained in:
Tony Wasserka 2024-03-20 22:31:11 +01:00
parent 51df05c7d1
commit c2b79585ac
2 changed files with 8 additions and 13 deletions

View file

@ -1101,16 +1101,12 @@ void ObserverSubject::Unregister(std::shared_ptr<Thread> thread) {
// TODOTEST: What happens if the application were doing something stupid like // TODOTEST: What happens if the application were doing something stupid like
// SVCWaitSynchronizationN'ing on multiple server port handles with the // SVCWaitSynchronizationN'ing on multiple server port handles with the
// wait_for_all flag set to true? In our implementation, incoming_session // wait_for_all flag set to true?
// would be overwritten and hence the first server port acquisition being
// silently dropped!
bool ServerPort::TryAcquireImpl(std::shared_ptr<Thread> thread) { bool ServerPort::TryAcquireImpl(std::shared_ptr<Thread> thread) {
if (session_queue.empty()) // NOTE: Some modules call SVCWaitSynchronization again after waking up
return false; // on a ServerPort handle in SVCReplyAndReceive. Hence, acquisition
// must be idempotent.
thread->incoming_session = session_queue.front(); return !session_queue.empty();
session_queue.pop();
return true;
} }
void ServerPort::OfferSession(std::shared_ptr<Session> session) { void ServerPort::OfferSession(std::shared_ptr<Session> session) {
@ -3304,11 +3300,12 @@ SVCFuture<OS::Result,HandleTable::Entry<ServerSession>,HandleTable::Entry<Client
SVCFuture<OS::Result,HandleTable::Entry<ServerSession>> OS::SVCAcceptSession(Thread& source, ServerPort& server_port) { SVCFuture<OS::Result,HandleTable::Entry<ServerSession>> OS::SVCAcceptSession(Thread& source, ServerPort& server_port) {
source.GetLogger()->info("{}SVCAcceptSession: server_port={}", ThreadPrinter{source}, ObjectRefPrinter{server_port}); source.GetLogger()->info("{}SVCAcceptSession: server_port={}", ThreadPrinter{source}, ObjectRefPrinter{server_port});
if (!source.incoming_session) { if (server_port.session_queue.empty()) {
throw Mikage::Exceptions::Invalid("Attempted to accept nonexisting session"); throw Mikage::Exceptions::Invalid("Attempted to accept nonexisting session");
} }
auto session = server_port.session_queue.front();
server_port.session_queue.pop();
std::shared_ptr<Session> session = std::move(source.incoming_session);
source.GetLogger()->info("{}SVCAcceptSession: accepting session {} / {}", ThreadPrinter{source}, source.GetLogger()->info("{}SVCAcceptSession: accepting session {} / {}", ThreadPrinter{source},
ObjectPrinter{session}, ObjectPrinter{session->client.lock()}); ObjectPrinter{session}, ObjectPrinter{session->client.lock()});

View file

@ -363,8 +363,6 @@ public:
*/ */
std::weak_ptr<Object> woken_object; std::weak_ptr<Object> woken_object;
std::shared_ptr<Session> incoming_session;
// Notify thread of a resource being ready for being TryAcquire'ed // Notify thread of a resource being ready for being TryAcquire'ed
// TODO: This function is unused now! // TODO: This function is unused now!
void SignalResourceReady(); void SignalResourceReady();