From 6054e1011983d34202128619544492242e9e99bd Mon Sep 17 00:00:00 2001 From: Norihiro Kamae Date: Mon, 16 Sep 2024 18:49:33 +0900 Subject: [PATCH] ptz-http-backend: Implement seek method When communicating with a PTZ camera, libcurl left a error below. > necessary data rewind wasn't possible --- src/ptz-http-backend.cpp | 35 +++++++++++++++++++++++++++++++---- 1 file changed, 31 insertions(+), 4 deletions(-) diff --git a/src/ptz-http-backend.cpp b/src/ptz-http-backend.cpp index 8aaf660..b15644c 100644 --- a/src/ptz-http-backend.cpp +++ b/src/ptz-http-backend.cpp @@ -183,27 +183,52 @@ void *ptz_http_backend::thread_main(void *data) struct read_cb_data { const char *data; + size_t offset; size_t size; }; static size_t read_cb(char *ptr, size_t size, size_t nmemb, void *userdata) { auto *ctx = static_cast(userdata); - size_t ret = std::min(nmemb, ctx->size / size); - memcpy(ptr, ctx->data, ret * size); - ctx->data += ret * size; - ctx->size -= ret * size; + if (ctx->offset > ctx->size) + return 0; + size_t ret = std::min(nmemb, (ctx->size - ctx->offset) / size); + + memcpy(ptr, ctx->data + ctx->offset, ret * size); + ctx->offset += ret * size; return ret; } +static int seek_cb(void *userdata, curl_off_t offset, int origin) +{ + auto *ctx = static_cast(userdata); + + switch (origin) { + case SEEK_SET: + ctx->offset = offset; + break; + case SEEK_CUR: + ctx->offset += offset; + break; + case SEEK_END: + ctx->offset = ctx->size + offset; + break; + default: + return CURL_SEEKFUNC_FAIL; + } + + return CURL_SEEKFUNC_OK; +} + static void call_url(obs_data_t *data, const char *method, const char *url, const char *payload) { blog(LOG_DEBUG, "call_url(method='%s', url='%s', payload='%s')", method, url, payload); struct read_cb_data read_cb_data = { .data = payload, + .offset = 0, .size = strlen(payload), }; @@ -229,6 +254,8 @@ static void call_url(obs_data_t *data, const char *method, const char *url, cons curl_easy_setopt(c, CURLOPT_READFUNCTION, read_cb); curl_easy_setopt(c, CURLOPT_READDATA, &read_cb_data); + curl_easy_setopt(c, CURLOPT_SEEKFUNCTION, seek_cb); + curl_easy_setopt(c, CURLOPT_SEEKDATA, &read_cb_data); } char error[CURL_ERROR_SIZE];