Line data Source code
1 : /*------------------------------------------------------------------------- 2 : * 3 : * execAsync.c 4 : * Support routines for asynchronous execution 5 : * 6 : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group 7 : * Portions Copyright (c) 1994, Regents of the University of California 8 : * 9 : * IDENTIFICATION 10 : * src/backend/executor/execAsync.c 11 : * 12 : *------------------------------------------------------------------------- 13 : */ 14 : 15 : #include "postgres.h" 16 : 17 : #include "executor/execAsync.h" 18 : #include "executor/executor.h" 19 : #include "executor/nodeAppend.h" 20 : #include "executor/nodeForeignscan.h" 21 : 22 : /* 23 : * Asynchronously request a tuple from a designed async-capable node. 24 : */ 25 : void 26 12150 : ExecAsyncRequest(AsyncRequest *areq) 27 : { 28 12150 : if (areq->requestee->chgParam != NULL) /* something changed? */ 29 4 : ExecReScan(areq->requestee); /* let ReScan handle this */ 30 : 31 : /* must provide our own instrumentation support */ 32 12150 : if (areq->requestee->instrument) 33 1620 : InstrStartNode(areq->requestee->instrument); 34 : 35 12150 : switch (nodeTag(areq->requestee)) 36 : { 37 12150 : case T_ForeignScanState: 38 12150 : ExecAsyncForeignScanRequest(areq); 39 12150 : break; 40 0 : default: 41 : /* If the node doesn't support async, caller messed up. */ 42 0 : elog(ERROR, "unrecognized node type: %d", 43 : (int) nodeTag(areq->requestee)); 44 : } 45 : 46 12150 : ExecAsyncResponse(areq); 47 : 48 : /* must provide our own instrumentation support */ 49 12150 : if (areq->requestee->instrument) 50 1620 : InstrStopNode(areq->requestee->instrument, 51 1620 : TupIsNull(areq->result) ? 0.0 : 1.0); 52 12150 : } 53 : 54 : /* 55 : * Give the asynchronous node a chance to configure the file descriptor event 56 : * for which it wishes to wait. We expect the node-type specific callback to 57 : * make a single call of the following form: 58 : * 59 : * AddWaitEventToSet(set, WL_SOCKET_READABLE, fd, NULL, areq); 60 : */ 61 : void 62 374 : ExecAsyncConfigureWait(AsyncRequest *areq) 63 : { 64 : /* must provide our own instrumentation support */ 65 374 : if (areq->requestee->instrument) 66 42 : InstrStartNode(areq->requestee->instrument); 67 : 68 374 : switch (nodeTag(areq->requestee)) 69 : { 70 374 : case T_ForeignScanState: 71 374 : ExecAsyncForeignScanConfigureWait(areq); 72 372 : break; 73 0 : default: 74 : /* If the node doesn't support async, caller messed up. */ 75 0 : elog(ERROR, "unrecognized node type: %d", 76 : (int) nodeTag(areq->requestee)); 77 : } 78 : 79 : /* must provide our own instrumentation support */ 80 372 : if (areq->requestee->instrument) 81 42 : InstrStopNode(areq->requestee->instrument, 0.0); 82 372 : } 83 : 84 : /* 85 : * Call the asynchronous node back when a relevant event has occurred. 86 : */ 87 : void 88 294 : ExecAsyncNotify(AsyncRequest *areq) 89 : { 90 : /* must provide our own instrumentation support */ 91 294 : if (areq->requestee->instrument) 92 28 : InstrStartNode(areq->requestee->instrument); 93 : 94 294 : switch (nodeTag(areq->requestee)) 95 : { 96 294 : case T_ForeignScanState: 97 294 : ExecAsyncForeignScanNotify(areq); 98 294 : break; 99 0 : default: 100 : /* If the node doesn't support async, caller messed up. */ 101 0 : elog(ERROR, "unrecognized node type: %d", 102 : (int) nodeTag(areq->requestee)); 103 : } 104 : 105 294 : ExecAsyncResponse(areq); 106 : 107 : /* must provide our own instrumentation support */ 108 294 : if (areq->requestee->instrument) 109 28 : InstrStopNode(areq->requestee->instrument, 110 28 : TupIsNull(areq->result) ? 0.0 : 1.0); 111 294 : } 112 : 113 : /* 114 : * Call the requestor back when an asynchronous node has produced a result. 115 : */ 116 : void 117 12454 : ExecAsyncResponse(AsyncRequest *areq) 118 : { 119 12454 : switch (nodeTag(areq->requestor)) 120 : { 121 12454 : case T_AppendState: 122 12454 : ExecAsyncAppendResponse(areq); 123 12454 : break; 124 0 : default: 125 : /* If the node doesn't support async, caller messed up. */ 126 0 : elog(ERROR, "unrecognized node type: %d", 127 : (int) nodeTag(areq->requestor)); 128 : } 129 12454 : } 130 : 131 : /* 132 : * A requestee node should call this function to deliver the tuple to its 133 : * requestor node. The requestee node can call this from its ExecAsyncRequest 134 : * or ExecAsyncNotify callback. 135 : */ 136 : void 137 12134 : ExecAsyncRequestDone(AsyncRequest *areq, TupleTableSlot *result) 138 : { 139 12134 : areq->request_complete = true; 140 12134 : areq->result = result; 141 12134 : } 142 : 143 : /* 144 : * A requestee node should call this function to indicate that it is pending 145 : * for a callback. The requestee node can call this from its ExecAsyncRequest 146 : * or ExecAsyncNotify callback. 147 : */ 148 : void 149 320 : ExecAsyncRequestPending(AsyncRequest *areq) 150 : { 151 320 : areq->callback_pending = true; 152 320 : areq->request_complete = false; 153 320 : areq->result = NULL; 154 320 : }