PostgreSQL Source Code git master
Loading...
Searching...
No Matches
worker.c
Go to the documentation of this file.
1/*--------------------------------------------------------------------------
2 *
3 * worker.c
4 * Code for sample worker making use of shared memory message queues.
5 * Our test worker simply reads messages from one message queue and
6 * writes them back out to another message queue. In a real
7 * application, you'd presumably want the worker to do some more
8 * complex calculation rather than simply returning the input,
9 * but it should be possible to use much of the control logic just
10 * as presented here.
11 *
12 * Copyright (c) 2013-2026, PostgreSQL Global Development Group
13 *
14 * IDENTIFICATION
15 * src/test/modules/test_shm_mq/worker.c
16 *
17 * -------------------------------------------------------------------------
18 */
19
20#include "postgres.h"
21
22#include "miscadmin.h"
23#include "storage/ipc.h"
24#include "storage/latch.h"
25#include "storage/proc.h"
26#include "storage/procarray.h"
27#include "storage/shm_mq.h"
28#include "storage/shm_toc.h"
29#include "tcop/tcopprot.h"
30
31#include "test_shm_mq.h"
32
33static void attach_to_queues(dsm_segment *seg, shm_toc *toc,
37
38/*
39 * Background worker entrypoint.
40 *
41 * This is intended to demonstrate how a background worker can be used to
42 * facilitate a parallel computation. Most of the logic here is fairly
43 * boilerplate stuff, designed to attach to the shared memory segment,
44 * notify the user backend that we're alive, and so on. The
45 * application-specific bits of logic that you'd replace for your own worker
46 * are attach_to_queues() and copy_messages().
47 */
48void
50{
51 dsm_segment *seg;
52 shm_toc *toc;
55 volatile test_shm_mq_header *hdr;
58
59 /* Unblock signals. The standard signal handlers are OK for us. */
61
62 /*
63 * Connect to the dynamic shared memory segment.
64 *
65 * The backend that registered this worker passed us the ID of a shared
66 * memory segment to which we must attach for further instructions. Once
67 * we've mapped the segment in our address space, attach to the table of
68 * contents so we can locate the various data structures we'll need to
69 * find within the segment.
70 *
71 * Note: at this point, we have not created any ResourceOwner in this
72 * process. This will result in our DSM mapping surviving until process
73 * exit, which is fine. If there were a ResourceOwner, it would acquire
74 * ownership of the mapping, but we have no need for that.
75 */
77 if (seg == NULL)
80 errmsg("unable to map dynamic shared memory segment")));
82 if (toc == NULL)
85 errmsg("bad magic number in dynamic shared memory segment")));
86
87 /*
88 * Acquire a worker number.
89 *
90 * By convention, the process registering this background worker should
91 * have stored the control structure at key 0. We look up that key to
92 * find it. Our worker number gives our identity: there may be just one
93 * worker involved in this parallel operation, or there may be many.
94 */
95 hdr = shm_toc_lookup(toc, 0, false);
102 errmsg("too many message queue testing workers already")));
103
104 /*
105 * Attach to the appropriate message queues.
106 */
108
109 /*
110 * Indicate that we're fully initialized and ready to begin the main part
111 * of the parallel operation.
112 *
113 * Once we signal that we're ready, the user backend is entitled to assume
114 * that our on_dsm_detach callbacks will fire before we disconnect from
115 * the shared memory segment and exit. Generally, that means we must have
116 * attached to all relevant dynamic shared memory data structures by now.
117 */
118 SpinLockAcquire(&hdr->mutex);
119 ++hdr->workers_ready;
120 SpinLockRelease(&hdr->mutex);
122 if (registrant == NULL)
123 {
124 elog(DEBUG1, "registrant backend has exited prematurely");
125 proc_exit(1);
126 }
127 SetLatch(&registrant->procLatch);
128
129 /* Do the work. */
131
132 /*
133 * We're done. For cleanliness, explicitly detach from the shared memory
134 * segment (that would happen anyway during process exit, though).
135 */
136 dsm_detach(seg);
137 proc_exit(1);
138}
139
140/*
141 * Attach to shared memory message queues.
142 *
143 * We use our worker number to determine to which queue we should attach.
144 * The queues are registered at keys 1..<number-of-workers>. The user backend
145 * writes to queue #1 and reads from queue #<number-of-workers>; each worker
146 * reads from the queue whose number is equal to its worker number and writes
147 * to the next higher-numbered queue.
148 */
149static void
163
164/*
165 * Loop, receiving and sending messages, until the connection is broken.
166 *
167 * This is the "real work" performed by this worker process. Everything that
168 * happens before this is initialization of one form or another, and everything
169 * after this point is cleanup.
170 */
171static void
173{
174 Size len;
175 void *data;
176 shm_mq_result res;
177
178 for (;;)
179 {
180 /* Notice any interrupts that have occurred. */
182
183 /* Receive a message. */
184 res = shm_mq_receive(inqh, &len, &data, false);
185 if (res != SHM_MQ_SUCCESS)
186 break;
187
188 /* Send it back out. */
189 res = shm_mq_send(outqh, len, data, false, true);
190 if (res != SHM_MQ_SUCCESS)
191 break;
192 }
193}
void BackgroundWorkerUnblockSignals(void)
Definition bgworker.c:933
size_t Size
Definition c.h:652
void dsm_detach(dsm_segment *seg)
Definition dsm.c:803
void * dsm_segment_address(dsm_segment *seg)
Definition dsm.c:1095
dsm_segment * dsm_attach(dsm_handle h)
Definition dsm.c:665
int errcode(int sqlerrcode)
Definition elog.c:874
int errmsg(const char *fmt,...)
Definition elog.c:1093
#define DEBUG1
Definition elog.h:30
#define ERROR
Definition elog.h:39
#define elog(elevel,...)
Definition elog.h:226
#define ereport(elevel,...)
Definition elog.h:150
void proc_exit(int code)
Definition ipc.c:105
void SetLatch(Latch *latch)
Definition latch.c:290
#define CHECK_FOR_INTERRUPTS()
Definition miscadmin.h:123
const void size_t len
const void * data
static uint32 DatumGetUInt32(Datum X)
Definition postgres.h:232
uint64_t Datum
Definition postgres.h:70
BackgroundWorker * MyBgworkerEntry
Definition postmaster.c:200
static int fb(int x)
PGPROC * BackendPidGetProc(int pid)
Definition procarray.c:3160
void shm_mq_set_sender(shm_mq *mq, PGPROC *proc)
Definition shm_mq.c:225
void shm_mq_set_receiver(shm_mq *mq, PGPROC *proc)
Definition shm_mq.c:207
shm_mq_result shm_mq_receive(shm_mq_handle *mqh, Size *nbytesp, void **datap, bool nowait)
Definition shm_mq.c:573
shm_mq_result shm_mq_send(shm_mq_handle *mqh, Size nbytes, const void *data, bool nowait, bool force_flush)
Definition shm_mq.c:330
shm_mq_handle * shm_mq_attach(shm_mq *mq, dsm_segment *seg, BackgroundWorkerHandle *handle)
Definition shm_mq.c:291
shm_mq_result
Definition shm_mq.h:39
@ SHM_MQ_SUCCESS
Definition shm_mq.h:40
void * shm_toc_lookup(shm_toc *toc, uint64 key, bool noError)
Definition shm_toc.c:232
shm_toc * shm_toc_attach(uint64 magic, void *address)
Definition shm_toc.c:64
static void SpinLockRelease(volatile slock_t *lock)
Definition spin.h:62
static void SpinLockAcquire(volatile slock_t *lock)
Definition spin.h:56
PGPROC * MyProc
Definition proc.c:67
pid_t bgw_notify_pid
Definition bgworker.h:107
Definition proc.h:176
void test_shm_mq_main(Datum main_arg)
Definition worker.c:49
static void attach_to_queues(dsm_segment *seg, shm_toc *toc, int myworkernumber, shm_mq_handle **inqhp, shm_mq_handle **outqhp)
Definition worker.c:150
static void copy_messages(shm_mq_handle *inqh, shm_mq_handle *outqh)
Definition worker.c:172
#define PG_TEST_SHM_MQ_MAGIC
Definition test_shm_mq.h:22