py: Switch to non-blocking eventfd
Blocking wait can be easily implemented on top in Python, so rather than supporting only blocking reads, or supporting both non-blocking and blocking reads, let's support only non-blocking reads. Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ideasonboard.com> Reviewed-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com> Signed-off-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
This commit is contained in:
parent
de7f1aa591
commit
bf3dbaece9
6 changed files with 41 additions and 12 deletions
|
@ -19,8 +19,9 @@ TIMEOUT_SEC = 3
|
||||||
|
|
||||||
|
|
||||||
def handle_camera_event(cm):
|
def handle_camera_event(cm):
|
||||||
# cm.get_ready_requests() will not block here, as we know there is an event
|
# cm.get_ready_requests() returns the ready requests, which in our case
|
||||||
# to read.
|
# should almost always return a single Request, but in some cases there
|
||||||
|
# could be multiple or none.
|
||||||
|
|
||||||
reqs = cm.get_ready_requests()
|
reqs = cm.get_ready_requests()
|
||||||
|
|
||||||
|
|
|
@ -14,6 +14,7 @@
|
||||||
|
|
||||||
import argparse
|
import argparse
|
||||||
import libcamera as libcam
|
import libcamera as libcam
|
||||||
|
import selectors
|
||||||
import sys
|
import sys
|
||||||
|
|
||||||
# Number of frames to capture
|
# Number of frames to capture
|
||||||
|
@ -107,11 +108,18 @@ def main():
|
||||||
# The main loop. Wait for the queued Requests to complete, process them,
|
# The main loop. Wait for the queued Requests to complete, process them,
|
||||||
# and re-queue them again.
|
# and re-queue them again.
|
||||||
|
|
||||||
|
sel = selectors.DefaultSelector()
|
||||||
|
sel.register(cm.event_fd, selectors.EVENT_READ)
|
||||||
|
|
||||||
while frames_done < TOTAL_FRAMES:
|
while frames_done < TOTAL_FRAMES:
|
||||||
# cm.get_ready_requests() blocks until there is an event and returns
|
# cm.get_ready_requests() does not block, so we use a Selector to wait
|
||||||
# all the ready requests. Here we should almost always get a single
|
# for a camera event. Here we should almost always get a single
|
||||||
# Request, but in some cases there could be multiple or none.
|
# Request, but in some cases there could be multiple or none.
|
||||||
|
|
||||||
|
events = sel.select()
|
||||||
|
if not events:
|
||||||
|
continue
|
||||||
|
|
||||||
reqs = cm.get_ready_requests()
|
reqs = cm.get_ready_requests()
|
||||||
|
|
||||||
for req in reqs:
|
for req in reqs:
|
||||||
|
|
|
@ -88,8 +88,9 @@ class CaptureContext:
|
||||||
camera_contexts: list[CameraCaptureContext] = []
|
camera_contexts: list[CameraCaptureContext] = []
|
||||||
|
|
||||||
def handle_camera_event(self):
|
def handle_camera_event(self):
|
||||||
# cm.get_ready_requests() will not block here, as we know there is an event
|
# cm.get_ready_requests() returns the ready requests, which in our case
|
||||||
# to read.
|
# should almost always return a single Request, but in some cases there
|
||||||
|
# could be multiple or none.
|
||||||
|
|
||||||
reqs = self.cm.get_ready_requests()
|
reqs = self.cm.get_ready_requests()
|
||||||
|
|
||||||
|
|
|
@ -24,7 +24,7 @@ PyCameraManager::PyCameraManager()
|
||||||
|
|
||||||
cameraManager_ = std::make_unique<CameraManager>();
|
cameraManager_ = std::make_unique<CameraManager>();
|
||||||
|
|
||||||
int fd = eventfd(0, EFD_CLOEXEC);
|
int fd = eventfd(0, EFD_CLOEXEC | EFD_NONBLOCK);
|
||||||
if (fd == -1)
|
if (fd == -1)
|
||||||
throw std::system_error(errno, std::generic_category(),
|
throw std::system_error(errno, std::generic_category(),
|
||||||
"Failed to create eventfd");
|
"Failed to create eventfd");
|
||||||
|
@ -62,7 +62,13 @@ py::list PyCameraManager::cameras()
|
||||||
|
|
||||||
std::vector<py::object> PyCameraManager::getReadyRequests()
|
std::vector<py::object> PyCameraManager::getReadyRequests()
|
||||||
{
|
{
|
||||||
readFd();
|
int ret = readFd();
|
||||||
|
|
||||||
|
if (ret == -EAGAIN)
|
||||||
|
return std::vector<py::object>();
|
||||||
|
|
||||||
|
if (ret != 0)
|
||||||
|
throw std::system_error(-ret, std::generic_category());
|
||||||
|
|
||||||
std::vector<py::object> py_reqs;
|
std::vector<py::object> py_reqs;
|
||||||
|
|
||||||
|
@ -96,12 +102,18 @@ void PyCameraManager::writeFd()
|
||||||
LOG(Python, Fatal) << "Unable to write to eventfd";
|
LOG(Python, Fatal) << "Unable to write to eventfd";
|
||||||
}
|
}
|
||||||
|
|
||||||
void PyCameraManager::readFd()
|
int PyCameraManager::readFd()
|
||||||
{
|
{
|
||||||
uint8_t buf[8];
|
uint8_t buf[8];
|
||||||
|
|
||||||
if (read(eventFd_.get(), buf, 8) != 8)
|
ssize_t ret = read(eventFd_.get(), buf, 8);
|
||||||
throw std::system_error(errno, std::generic_category());
|
|
||||||
|
if (ret == 8)
|
||||||
|
return 0;
|
||||||
|
else if (ret < 0)
|
||||||
|
return -errno;
|
||||||
|
else
|
||||||
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
void PyCameraManager::pushRequest(Request *req)
|
void PyCameraManager::pushRequest(Request *req)
|
||||||
|
|
|
@ -39,7 +39,7 @@ private:
|
||||||
LIBCAMERA_TSA_GUARDED_BY(completedRequestsMutex_);
|
LIBCAMERA_TSA_GUARDED_BY(completedRequestsMutex_);
|
||||||
|
|
||||||
void writeFd();
|
void writeFd();
|
||||||
void readFd();
|
int readFd();
|
||||||
void pushRequest(Request *req);
|
void pushRequest(Request *req);
|
||||||
std::vector<Request *> getCompletedRequests();
|
std::vector<Request *> getCompletedRequests();
|
||||||
};
|
};
|
||||||
|
|
|
@ -207,9 +207,16 @@ class SimpleCaptureMethods(CameraTesterBase):
|
||||||
reqs = None
|
reqs = None
|
||||||
gc.collect()
|
gc.collect()
|
||||||
|
|
||||||
|
sel = selectors.DefaultSelector()
|
||||||
|
sel.register(cm.event_fd, selectors.EVENT_READ)
|
||||||
|
|
||||||
reqs = []
|
reqs = []
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
|
events = sel.select()
|
||||||
|
if not events:
|
||||||
|
continue
|
||||||
|
|
||||||
ready_reqs = cm.get_ready_requests()
|
ready_reqs = cm.get_ready_requests()
|
||||||
|
|
||||||
reqs += ready_reqs
|
reqs += ready_reqs
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue