summaryrefslogtreecommitdiff
path: root/proto_proxy.c
diff options
context:
space:
mode:
authordormando <dormando@rydia.net>2023-01-12 12:15:23 -0800
committerdormando <dormando@rydia.net>2023-01-12 13:50:55 -0800
commit6537cfe41ead9c81dd3e9c9f5d3929a050692cbd (patch)
tree021699f628b831dee45800ba913ec53ce88a8cf1 /proto_proxy.c
parent0e87c6dd86d8a50ddd56fbce445fb18e498e7f5f (diff)
downloadmemcached-6537cfe41ead9c81dd3e9c9f5d3929a050692cbd.tar.gz
proxy: clean logic around lua yielding
We were duck typing the response code for a coroutine yield before. It would also pile random logic for overriding IO's in certain cases. This now makes everything explicit and more clear.
Diffstat (limited to 'proto_proxy.c')
-rw-r--r--proto_proxy.c82
1 files changed, 37 insertions, 45 deletions
diff --git a/proto_proxy.c b/proto_proxy.c
index 01b368d..7cea462 100644
--- a/proto_proxy.c
+++ b/proto_proxy.c
@@ -14,7 +14,6 @@
#define PROCESS_NORMAL false
static void proxy_process_command(conn *c, char *command, size_t cmdlen, bool multiget);
static void mcp_queue_io(conn *c, mc_resp *resp, int coro_ref, lua_State *Lc);
-static void proxy_out_errstring(mc_resp *resp, const char *str);
/******** EXTERNAL FUNCTIONS ******/
// functions starting with _ are breakouts for the public functions.
@@ -469,7 +468,7 @@ void proxy_lua_ferror(lua_State *L, const char *fmt, ...) {
}
// Need a custom function so we can prefix lua strings easily.
-static void proxy_out_errstring(mc_resp *resp, const char *str) {
+void proxy_out_errstring(mc_resp *resp, const char *str) {
size_t len;
const static char error_prefix[] = "SERVER_ERROR ";
const static int error_prefix_len = sizeof(error_prefix) - 1;
@@ -580,52 +579,45 @@ int proxy_run_coroutine(lua_State *Lc, mc_resp *resp, io_pending_proxy_t *p, con
} else {
proxy_out_errstring(resp, "bad response");
}
+
} else if (cores == LUA_YIELD) {
- if (nresults == 1) {
- // TODO (v2): try harder to validate; but we have so few yield cases
- // that I'm going to shortcut this here. A single yielded result
- // means it's probably an await(), so attempt to process this.
- if (p != NULL) {
- int coro_ref = p->coro_ref;
- mc_resp *resp = p->resp;
- assert((void *)p == (void *)resp->io_pending);
- resp->io_pending = NULL;
- c = p->c;
- do_cache_free(c->thread->io_cache, p);
- mcplib_await_run(c, resp, Lc, coro_ref);
- } else {
- // coroutine object sitting on the _main_ VM right now, so we grab
- // the reference from there, which also pops it.
- int coro_ref = luaL_ref(c->thread->L, LUA_REGISTRYINDEX);
- mcplib_await_run(c, c->resp, Lc, coro_ref);
- }
+ int coro_ref = 0;
+ int yield_type = lua_tointeger(Lc, -1);
+ assert(yield_type != 0);
+ lua_pop(Lc, 1);
+
+ // need to remove and free the io_pending, since c->resp owns it.
+ // so we call mcp_queue_io() again and let it override the
+ // mc_resp's io_pending object.
+ //
+ // p is not null only when being called from proxy_return_cb(),
+ // a pending IO is returning to resume.
+ if (p != NULL) {
+ coro_ref = p->coro_ref;
+ assert((void *)p == (void *)resp->io_pending);
+ resp->io_pending = NULL;
+ c = p->c;
+ // *p is now dead.
+ do_cache_free(c->thread->io_cache, p);
} else {
- // need to remove and free the io_pending, since c->resp owns it.
- // so we call mcp_queue_io() again and let it override the
- // mc_resp's io_pending object.
-
- int coro_ref = 0;
- mc_resp *resp;
- if (p != NULL) {
- coro_ref = p->coro_ref;
- resp = p->resp;
- c = p->c;
- do_cache_free(p->c->thread->io_cache, p);
- // *p is now dead.
- } else {
- // yielding from a top level call to the coroutine,
- // so we need to grab a reference to the coroutine thread.
- // TODO (v2): make this more explicit?
- // we only need to get the reference here, and error conditions
- // should instead drop it, but now it's not obvious to users that
- // we're reaching back into the main thread's stack.
- assert(c != NULL);
- coro_ref = luaL_ref(c->thread->L, LUA_REGISTRYINDEX);
- resp = c->resp;
- }
- // TODO (v2): c only used for cache alloc? push the above into the func?
- mcp_queue_io(c, resp, coro_ref, Lc);
+ // coroutine object sitting on the _main_ VM right now, so we grab
+ // the reference from there, which also pops it.
+ assert(c != NULL);
+ coro_ref = luaL_ref(c->thread->L, LUA_REGISTRYINDEX);
}
+
+ switch (yield_type) {
+ case MCP_YIELD_AWAIT:
+ mcplib_await_run(c, resp, Lc, coro_ref);
+ break;
+ case MCP_YIELD_POOL:
+ // TODO (v2): c only used for cache alloc?
+ mcp_queue_io(c, resp, coro_ref, Lc);
+ break;
+ default:
+ abort();
+ }
+
} else {
WSTAT_DECR(c, proxy_req_active, 1);
P_DEBUG("%s: Failed to run coroutine: %s\n", __func__, lua_tostring(Lc, -1));