2019-09-13 09:33:49 +02:00
|
|
|
#include <assert.h>
|
|
|
|
#include <errno.h>
|
2020-01-13 12:14:33 +01:00
|
|
|
#include <inttypes.h>
|
2019-09-13 09:33:49 +02:00
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdlib.h>
|
2019-09-13 18:44:12 +02:00
|
|
|
#include <stdio.h>
|
2019-09-13 09:33:49 +02:00
|
|
|
#include <string.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include "libdrm_mock.h"
|
|
|
|
|
|
|
|
#define MAX_PLANES 64
|
2019-10-30 21:13:17 +01:00
|
|
|
#define MAX_LAYERS 512
|
|
|
|
#define MAX_PLANE_PROPS 64
|
|
|
|
#define MAX_REQ_PROPS 1024
|
2019-09-13 09:33:49 +02:00
|
|
|
|
|
|
|
uint32_t liftoff_mock_drm_crtc_id = 0xCC000000;
|
2019-10-14 17:34:32 +02:00
|
|
|
size_t liftoff_mock_commit_count = 0;
|
2019-09-13 09:33:49 +02:00
|
|
|
|
|
|
|
struct liftoff_mock_plane {
|
|
|
|
uint32_t id;
|
|
|
|
struct liftoff_layer *compatible_layers[MAX_LAYERS];
|
2020-05-22 17:36:27 +02:00
|
|
|
bool enabled_props[MAX_PLANE_PROPS];
|
2020-01-13 12:14:33 +01:00
|
|
|
uint64_t prop_values[MAX_PLANE_PROPS];
|
2019-09-13 09:33:49 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
struct liftoff_mock_prop {
|
|
|
|
uint32_t obj_id, prop_id;
|
|
|
|
uint64_t value;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct _drmModeAtomicReq {
|
2019-10-30 21:13:17 +01:00
|
|
|
struct liftoff_mock_prop props[MAX_REQ_PROPS];
|
2019-09-13 09:33:49 +02:00
|
|
|
int cursor;
|
|
|
|
};
|
|
|
|
|
|
|
|
static int mock_pipe[2] = {-1, -1};
|
|
|
|
static struct liftoff_mock_plane mock_planes[MAX_PLANES];
|
|
|
|
static struct liftoff_layer *mock_fbs[MAX_LAYERS];
|
|
|
|
|
|
|
|
enum plane_prop {
|
|
|
|
PLANE_TYPE,
|
|
|
|
PLANE_FB_ID,
|
|
|
|
PLANE_CRTC_ID,
|
|
|
|
};
|
|
|
|
|
2020-05-22 17:39:33 +02:00
|
|
|
static const char *basic_plane_props[] = {
|
2019-09-13 09:33:49 +02:00
|
|
|
[PLANE_TYPE] = "type",
|
|
|
|
[PLANE_FB_ID] = "FB_ID",
|
|
|
|
[PLANE_CRTC_ID] = "CRTC_ID",
|
|
|
|
"CRTC_X",
|
|
|
|
"CRTC_Y",
|
|
|
|
"CRTC_W",
|
|
|
|
"CRTC_H",
|
|
|
|
"SRC_X",
|
|
|
|
"SRC_Y",
|
|
|
|
"SRC_W",
|
|
|
|
"SRC_H",
|
|
|
|
};
|
|
|
|
|
2020-05-22 17:39:33 +02:00
|
|
|
static const size_t basic_plane_props_len = sizeof(basic_plane_props) /
|
|
|
|
sizeof(basic_plane_props[0]);
|
|
|
|
|
|
|
|
static drmModePropertyRes plane_props[MAX_PLANE_PROPS] = {0};
|
|
|
|
|
|
|
|
static size_t plane_props_len = 0;
|
|
|
|
|
2019-09-13 09:33:49 +02:00
|
|
|
static void assert_drm_fd(int fd)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
struct stat stat_got, stat_want;
|
|
|
|
|
|
|
|
ret = fstat(mock_pipe[0], &stat_want);
|
|
|
|
assert(ret == 0);
|
|
|
|
ret = fstat(fd, &stat_got);
|
|
|
|
assert(ret == 0);
|
|
|
|
|
|
|
|
assert(stat_got.st_dev == stat_want.st_dev &&
|
|
|
|
stat_got.st_ino == stat_want.st_ino);
|
|
|
|
}
|
|
|
|
|
2020-05-22 17:36:27 +02:00
|
|
|
static uint32_t register_prop(const drmModePropertyRes *prop)
|
|
|
|
{
|
|
|
|
drmModePropertyRes *dst;
|
|
|
|
|
|
|
|
assert(plane_props_len < MAX_PLANE_PROPS);
|
|
|
|
dst = &plane_props[plane_props_len];
|
|
|
|
memcpy(dst, prop, sizeof(*dst));
|
|
|
|
dst->prop_id = 0xB0000000 + plane_props_len;
|
|
|
|
plane_props_len++;
|
|
|
|
|
|
|
|
return dst->prop_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void init_basic_props(void)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (plane_props_len > 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (i = 0; i < basic_plane_props_len; i++) {
|
|
|
|
drmModePropertyRes prop = {0};
|
|
|
|
strncpy(prop.name, basic_plane_props[i], sizeof(prop.name) - 1);
|
|
|
|
/* TODO: fill flags */
|
|
|
|
register_prop(&prop);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-13 09:33:49 +02:00
|
|
|
int liftoff_mock_drm_open(void)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
assert(mock_pipe[0] < 0);
|
|
|
|
ret = pipe(mock_pipe);
|
|
|
|
assert(ret == 0);
|
|
|
|
|
2020-05-22 17:36:27 +02:00
|
|
|
init_basic_props();
|
2020-05-22 17:39:33 +02:00
|
|
|
|
2019-09-13 09:33:49 +02:00
|
|
|
return mock_pipe[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
struct liftoff_mock_plane *liftoff_mock_drm_create_plane(int type)
|
|
|
|
{
|
|
|
|
struct liftoff_mock_plane *plane;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
assert(mock_pipe[0] < 0);
|
|
|
|
|
2020-05-22 17:36:27 +02:00
|
|
|
init_basic_props();
|
|
|
|
|
2019-09-13 09:33:49 +02:00
|
|
|
i = 0;
|
|
|
|
plane = &mock_planes[0];
|
|
|
|
while (plane->id != 0) {
|
|
|
|
plane++;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
plane->id = 0xEE000000 + i;
|
2020-01-13 12:14:33 +01:00
|
|
|
plane->prop_values[PLANE_TYPE] = type;
|
2020-05-22 17:36:27 +02:00
|
|
|
|
|
|
|
for (size_t i = 0; i < basic_plane_props_len; i++) {
|
|
|
|
plane->enabled_props[i] = true;
|
|
|
|
}
|
|
|
|
|
2019-09-13 09:33:49 +02:00
|
|
|
return plane;
|
|
|
|
}
|
|
|
|
|
2019-09-13 18:46:21 +02:00
|
|
|
struct liftoff_mock_plane *liftoff_mock_drm_get_plane(uint32_t id)
|
|
|
|
{
|
|
|
|
struct liftoff_mock_plane *plane;
|
|
|
|
|
|
|
|
plane = &mock_planes[0];
|
|
|
|
while (plane->id != 0) {
|
|
|
|
if (plane->id == id) {
|
|
|
|
return plane;
|
|
|
|
}
|
|
|
|
plane++;
|
|
|
|
}
|
|
|
|
|
2020-05-19 12:10:17 +02:00
|
|
|
abort(); // unreachable
|
2019-09-13 18:46:21 +02:00
|
|
|
}
|
|
|
|
|
2019-09-13 09:33:49 +02:00
|
|
|
void liftoff_mock_plane_add_compatible_layer(struct liftoff_mock_plane *plane,
|
|
|
|
struct liftoff_layer *layer)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < MAX_LAYERS; i++) {
|
|
|
|
if (plane->compatible_layers[i] == NULL) {
|
|
|
|
plane->compatible_layers[i] = layer;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-19 12:10:17 +02:00
|
|
|
abort(); // unreachable
|
2019-09-13 09:33:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t liftoff_mock_drm_create_fb(struct liftoff_layer *layer)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
i = 0;
|
|
|
|
while (mock_fbs[i] != 0) {
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
mock_fbs[i] = layer;
|
|
|
|
|
|
|
|
return 0xFB000000 + i;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool mock_atomic_req_get_property(drmModeAtomicReq *req, uint32_t obj_id,
|
|
|
|
enum plane_prop prop, uint64_t *value)
|
|
|
|
{
|
|
|
|
ssize_t i;
|
|
|
|
uint32_t prop_id;
|
|
|
|
|
|
|
|
prop_id = 0xB0000000 + prop;
|
|
|
|
for (i = req->cursor - 1; i >= 0; i--) {
|
|
|
|
if (req->props[i].obj_id == obj_id &&
|
|
|
|
req->props[i].prop_id == prop_id) {
|
|
|
|
*value = req->props[i].value;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct liftoff_layer *mock_fb_get_layer(uint32_t fb_id)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (fb_id == 0) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert((fb_id & 0xFF000000) == 0xFB000000);
|
|
|
|
|
|
|
|
i = fb_id & 0x00FFFFFF;
|
|
|
|
assert(i < MAX_LAYERS);
|
|
|
|
|
|
|
|
return mock_fbs[i];
|
|
|
|
}
|
|
|
|
|
2020-01-13 12:14:33 +01:00
|
|
|
struct liftoff_layer *liftoff_mock_plane_get_layer(struct liftoff_mock_plane *plane)
|
2019-09-13 09:33:49 +02:00
|
|
|
{
|
2020-01-13 12:14:33 +01:00
|
|
|
return mock_fb_get_layer(plane->prop_values[PLANE_FB_ID]);
|
|
|
|
}
|
2019-09-13 09:33:49 +02:00
|
|
|
|
2020-01-13 12:14:33 +01:00
|
|
|
static size_t get_prop_index(uint32_t id)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
assert((id & 0xFF000000) == 0xB0000000);
|
|
|
|
|
|
|
|
i = id & 0x00FFFFFF;
|
2020-05-22 17:36:27 +02:00
|
|
|
assert(i < plane_props_len);
|
2019-09-13 09:33:49 +02:00
|
|
|
|
2020-01-13 12:14:33 +01:00
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
2020-05-22 17:36:27 +02:00
|
|
|
uint32_t liftoff_mock_plane_add_property(struct liftoff_mock_plane *plane,
|
|
|
|
const drmModePropertyRes *prop)
|
|
|
|
{
|
|
|
|
uint32_t prop_id;
|
|
|
|
|
|
|
|
prop_id = register_prop(prop);
|
|
|
|
plane->enabled_props[get_prop_index(prop_id)] = true;
|
|
|
|
return prop_id;
|
|
|
|
}
|
|
|
|
|
2020-01-13 12:14:33 +01:00
|
|
|
static void apply_atomic_req(drmModeAtomicReq *req)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
size_t prop_index;
|
|
|
|
struct liftoff_mock_prop *prop;
|
|
|
|
struct liftoff_mock_plane *plane;
|
|
|
|
|
|
|
|
for (i = 0; i < req->cursor; i++) {
|
|
|
|
prop = &req->props[i];
|
|
|
|
plane = liftoff_mock_drm_get_plane(prop->obj_id);
|
|
|
|
prop_index = get_prop_index(prop->prop_id);
|
|
|
|
plane->prop_values[prop_index] = prop->value;
|
|
|
|
fprintf(stderr, "plane %"PRIu32": setting %s = %"PRIu64"\n",
|
2020-05-22 17:39:33 +02:00
|
|
|
plane->id, plane_props[prop_index].name, prop->value);
|
2020-01-13 12:14:33 +01:00
|
|
|
}
|
2019-09-13 09:33:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int drmModeAtomicCommit(int fd, drmModeAtomicReq *req, uint32_t flags,
|
|
|
|
void *user_data)
|
|
|
|
{
|
|
|
|
size_t i, j;
|
|
|
|
struct liftoff_mock_plane *plane;
|
|
|
|
uint64_t fb_id, crtc_id;
|
|
|
|
bool has_fb, has_crtc, found;
|
|
|
|
struct liftoff_layer *layer;
|
|
|
|
|
|
|
|
assert_drm_fd(fd);
|
2020-05-26 18:47:52 +02:00
|
|
|
assert(flags == DRM_MODE_ATOMIC_TEST_ONLY || flags == 0);
|
2019-09-13 09:33:49 +02:00
|
|
|
|
2019-10-14 17:34:32 +02:00
|
|
|
liftoff_mock_commit_count++;
|
|
|
|
|
2019-09-13 09:33:49 +02:00
|
|
|
for (i = 0; i < MAX_PLANES; i++) {
|
|
|
|
plane = &mock_planes[i];
|
|
|
|
if (plane->id == 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-01-13 12:14:33 +01:00
|
|
|
fb_id = plane->prop_values[PLANE_FB_ID];
|
|
|
|
crtc_id = plane->prop_values[PLANE_CRTC_ID];
|
|
|
|
mock_atomic_req_get_property(req, plane->id, PLANE_FB_ID,
|
|
|
|
&fb_id);
|
|
|
|
mock_atomic_req_get_property(req, plane->id, PLANE_CRTC_ID,
|
|
|
|
&crtc_id);
|
2019-09-13 09:33:49 +02:00
|
|
|
|
2020-01-13 12:14:33 +01:00
|
|
|
has_fb = fb_id != 0;
|
|
|
|
has_crtc = crtc_id != 0;
|
2019-09-13 18:44:12 +02:00
|
|
|
|
2019-09-13 09:33:49 +02:00
|
|
|
if (has_fb != has_crtc) {
|
2019-09-13 18:44:12 +02:00
|
|
|
fprintf(stderr, "plane %u: both FB_ID and CRTC_ID must "
|
2020-01-13 12:14:33 +01:00
|
|
|
"be set or unset together (FB_ID = %"PRIu64", "
|
|
|
|
"CRTC_ID = %"PRIu64")\n", plane->id, fb_id,
|
|
|
|
crtc_id);
|
2019-09-13 09:33:49 +02:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (has_fb) {
|
|
|
|
if (crtc_id != liftoff_mock_drm_crtc_id) {
|
2019-09-13 18:44:12 +02:00
|
|
|
fprintf(stderr, "plane %u: invalid CRTC_ID\n",
|
|
|
|
plane->id);
|
2019-09-13 09:33:49 +02:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
layer = mock_fb_get_layer(fb_id);
|
|
|
|
if (layer == NULL) {
|
2019-09-13 18:44:12 +02:00
|
|
|
fprintf(stderr, "plane %u: invalid FB_ID\n",
|
|
|
|
plane->id);
|
2019-09-13 09:33:49 +02:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
found = false;
|
|
|
|
for (j = 0; j < MAX_LAYERS; j++) {
|
2019-09-13 21:34:34 +02:00
|
|
|
if (plane->compatible_layers[j] == layer) {
|
2019-09-13 09:33:49 +02:00
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!found) {
|
2019-09-13 18:44:12 +02:00
|
|
|
fprintf(stderr, "plane %u: layer %p is not "
|
|
|
|
"compatible\n", plane->id,
|
|
|
|
(void *)layer);
|
2019-09-13 09:33:49 +02:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-26 18:47:52 +02:00
|
|
|
if (!(flags & DRM_MODE_ATOMIC_TEST_ONLY)) {
|
|
|
|
apply_atomic_req(req);
|
|
|
|
}
|
2020-01-13 12:14:33 +01:00
|
|
|
|
2019-09-13 09:33:49 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
drmModeRes *drmModeGetResources(int fd)
|
|
|
|
{
|
|
|
|
drmModeRes *res;
|
|
|
|
|
|
|
|
assert_drm_fd(fd);
|
|
|
|
|
|
|
|
res = calloc(1, sizeof(*res));
|
|
|
|
res->count_crtcs = 1;
|
|
|
|
res->crtcs = &liftoff_mock_drm_crtc_id;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
void drmModeFreeResources(drmModeRes *res)
|
|
|
|
{
|
|
|
|
free(res);
|
|
|
|
}
|
|
|
|
|
|
|
|
drmModePlaneRes *drmModeGetPlaneResources(int fd)
|
|
|
|
{
|
|
|
|
static uint32_t plane_ids[MAX_PLANES];
|
|
|
|
drmModePlaneRes *res;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
assert_drm_fd(fd);
|
|
|
|
|
|
|
|
for (i = 0; i < MAX_PLANES; i++) {
|
|
|
|
if (mock_planes[i].id == 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
plane_ids[i] = mock_planes[i].id;
|
|
|
|
}
|
|
|
|
|
|
|
|
res = calloc(1, sizeof(*res));
|
|
|
|
res->count_planes = i;
|
|
|
|
res->planes = plane_ids;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
void drmModeFreePlaneResources(drmModePlaneRes *res)
|
|
|
|
{
|
|
|
|
free(res);
|
|
|
|
}
|
|
|
|
|
|
|
|
drmModePlane *drmModeGetPlane(int fd, uint32_t id)
|
|
|
|
{
|
|
|
|
drmModePlane *plane;
|
|
|
|
|
|
|
|
assert_drm_fd(fd);
|
|
|
|
|
|
|
|
plane = calloc(1, sizeof(*plane));
|
|
|
|
plane->plane_id = id;
|
|
|
|
plane->possible_crtcs = 1 << 0;
|
|
|
|
return plane;
|
|
|
|
}
|
|
|
|
|
|
|
|
void drmModeFreePlane(drmModePlane *plane) {
|
|
|
|
free(plane);
|
|
|
|
}
|
|
|
|
|
|
|
|
drmModeObjectProperties *drmModeObjectGetProperties(int fd, uint32_t obj_id,
|
|
|
|
uint32_t obj_type)
|
|
|
|
{
|
|
|
|
struct liftoff_mock_plane *plane;
|
|
|
|
drmModeObjectProperties *props;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
assert_drm_fd(fd);
|
|
|
|
assert(obj_type == DRM_MODE_OBJECT_PLANE);
|
|
|
|
|
|
|
|
plane = NULL;
|
|
|
|
for (i = 0; i < MAX_PLANES; i++) {
|
|
|
|
if (mock_planes[i].id == obj_id) {
|
|
|
|
plane = &mock_planes[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert(plane != NULL);
|
|
|
|
|
|
|
|
props = calloc(1, sizeof(*props));
|
2020-05-22 17:36:27 +02:00
|
|
|
props->props = calloc(plane_props_len, sizeof(uint32_t));
|
|
|
|
props->prop_values = calloc(plane_props_len, sizeof(uint64_t));
|
|
|
|
for (i = 0; i < plane_props_len; i++) {
|
|
|
|
if (!plane->enabled_props[i])
|
|
|
|
continue;
|
|
|
|
props->props[props->count_props] = plane_props[i].prop_id;
|
|
|
|
props->prop_values[props->count_props] = plane->prop_values[i];
|
|
|
|
props->count_props++;
|
2019-09-13 09:33:49 +02:00
|
|
|
}
|
|
|
|
return props;
|
|
|
|
}
|
|
|
|
|
|
|
|
void drmModeFreeObjectProperties(drmModeObjectProperties *props) {
|
2020-05-22 17:36:27 +02:00
|
|
|
free(props->props);
|
|
|
|
free(props->prop_values);
|
2019-09-13 09:33:49 +02:00
|
|
|
free(props);
|
|
|
|
}
|
|
|
|
|
|
|
|
drmModePropertyRes *drmModeGetProperty(int fd, uint32_t id)
|
|
|
|
{
|
|
|
|
assert_drm_fd(fd);
|
|
|
|
|
2020-05-22 17:39:33 +02:00
|
|
|
return &plane_props[get_prop_index(id)];
|
2019-09-13 09:33:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void drmModeFreeProperty(drmModePropertyRes *prop) {
|
2020-05-22 17:39:33 +02:00
|
|
|
/* Owned by plane_props */
|
2019-09-13 09:33:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
drmModeAtomicReq *drmModeAtomicAlloc(void)
|
|
|
|
{
|
|
|
|
return calloc(1, sizeof(drmModeAtomicReq));
|
|
|
|
}
|
|
|
|
|
|
|
|
void drmModeAtomicFree(drmModeAtomicReq *req)
|
|
|
|
{
|
|
|
|
free(req);
|
|
|
|
}
|
|
|
|
|
|
|
|
int drmModeAtomicAddProperty(drmModeAtomicReq *req, uint32_t obj_id,
|
|
|
|
uint32_t prop_id, uint64_t value)
|
|
|
|
{
|
|
|
|
assert((size_t)req->cursor < sizeof(req->props) / sizeof(req->props[0]));
|
|
|
|
req->props[req->cursor].obj_id = obj_id;
|
|
|
|
req->props[req->cursor].prop_id = prop_id;
|
|
|
|
req->props[req->cursor].value = value;
|
|
|
|
req->cursor++;
|
2019-11-29 11:06:57 +01:00
|
|
|
return req->cursor;
|
2019-09-13 09:33:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int drmModeAtomicGetCursor(drmModeAtomicReq *req)
|
|
|
|
{
|
|
|
|
return req->cursor;
|
|
|
|
}
|
|
|
|
|
|
|
|
void drmModeAtomicSetCursor(drmModeAtomicReq *req, int cursor)
|
|
|
|
{
|
|
|
|
req->cursor = cursor;
|
|
|
|
}
|