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:
parent
4c63bb4938
commit
1f7f7a72ed
1 changed files with 64 additions and 2 deletions
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue