test: message: Test recursive Thread::dispatchMessages() calls

The Thread::dispatchMessages() function needs to support recursive
calls, for instance to allow flushing delivery of invoked methods. Add a
corresponding test, which currently fails with a double free.

Signed-off-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
Reviewed-by: Jacopo Mondi <jacopo@jmondi.org>
Reviewed-by: Kieran Bingham <kieran.bingham@ideasonboard.com>
This commit is contained in:
Laurent Pinchart 2021-07-01 22:15:54 +03:00
parent 4c63bb4938
commit 1f7f7a72ed

View file

@ -7,6 +7,7 @@
#include <chrono> #include <chrono>
#include <iostream> #include <iostream>
#include <memory>
#include <thread> #include <thread>
#include <libcamera/base/message.h> #include <libcamera/base/message.h>
@ -26,8 +27,8 @@ public:
MessageReceived, MessageReceived,
}; };
MessageReceiver() MessageReceiver(Object *parent = nullptr)
: status_(NoMessage) : Object(parent), status_(NoMessage)
{ {
} }
@ -52,6 +53,45 @@ private:
Status status_; Status status_;
}; };
class RecursiveMessageReceiver : public Object
{
public:
RecursiveMessageReceiver()
: child_(this), success_(false)
{
}
bool success() const { return success_; }
protected:
void message([[maybe_unused]] Message *msg)
{
if (msg->type() != Message::None) {
Object::message(msg);
return;
}
child_.postMessage(std::make_unique<Message>(Message::None));
/*
* If the child has already received the message, something is
* wrong.
*/
if (child_.status() != MessageReceiver::NoMessage)
return;
Thread::current()->dispatchMessages(Message::None);
/* The child should now have received the message. */
if (child_.status() == MessageReceiver::MessageReceived)
success_ = true;
}
private:
MessageReceiver child_;
bool success_;
};
class SlowMessageReceiver : public Object class SlowMessageReceiver : public Object
{ {
protected: protected:
@ -120,6 +160,28 @@ protected:
delete slowReceiver; delete slowReceiver;
this_thread::sleep_for(chrono::milliseconds(100));
/*
* Test recursive calls to Thread::dispatchMessages(). Messages
* should be delivered correctly, without crashes or memory
* leaks. Two messages need to be posted to ensure we don't only
* test the simple case of a queue containing a single message.
*/
std::unique_ptr<RecursiveMessageReceiver> recursiveReceiver =
std::make_unique<RecursiveMessageReceiver>();
recursiveReceiver->moveToThread(&thread_);
recursiveReceiver->postMessage(std::make_unique<Message>(Message::None));
recursiveReceiver->postMessage(std::make_unique<Message>(Message::UserMessage));
this_thread::sleep_for(chrono::milliseconds(10));
if (!recursiveReceiver->success()) {
cout << "Recursive message delivery failed" << endl;
return TestFail;
}
return TestPass; return TestPass;
} }