LCOV - differential code coverage report
Current view: top level - src/test/modules/test_shm_mq - worker.c (source / functions) Coverage Total Hit UBC CBC
Current: Differential Code Coverage 16@8cea358b128 vs 17@8cea358b128 Lines: 86.4 % 44 38 6 38
Current Date: 2024-04-14 14:21:10 Functions: 100.0 % 3 3 3
Baseline: 16@8cea358b128 Branches: 38.5 % 26 10 16 10
Baseline Date: 2024-04-14 14:21:09 Line coverage date bins:
Legend: Lines: hit not hit | Branches: + taken - not taken # not executed (240..) days: 86.4 % 44 38 6 38
Function coverage date bins:
(240..) days: 100.0 % 3 3 3
Branch coverage date bins:
(240..) days: 38.5 % 26 10 16 10

 Age         Owner                    Branch data    TLA  Line data    Source code
                                  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-2024, 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/procarray.h"
                                 25                 :                : #include "storage/shm_mq.h"
                                 26                 :                : #include "storage/shm_toc.h"
                                 27                 :                : #include "tcop/tcopprot.h"
                                 28                 :                : 
                                 29                 :                : #include "test_shm_mq.h"
                                 30                 :                : 
                                 31                 :                : static void attach_to_queues(dsm_segment *seg, shm_toc *toc,
                                 32                 :                :                              int myworkernumber, shm_mq_handle **inqhp,
                                 33                 :                :                              shm_mq_handle **outqhp);
                                 34                 :                : static void copy_messages(shm_mq_handle *inqh, shm_mq_handle *outqh);
                                 35                 :                : 
                                 36                 :                : /*
                                 37                 :                :  * Background worker entrypoint.
                                 38                 :                :  *
                                 39                 :                :  * This is intended to demonstrate how a background worker can be used to
                                 40                 :                :  * facilitate a parallel computation.  Most of the logic here is fairly
                                 41                 :                :  * boilerplate stuff, designed to attach to the shared memory segment,
                                 42                 :                :  * notify the user backend that we're alive, and so on.  The
                                 43                 :                :  * application-specific bits of logic that you'd replace for your own worker
                                 44                 :                :  * are attach_to_queues() and copy_messages().
                                 45                 :                :  */
                                 46                 :                : void
 3743 rhaas@postgresql.org       47                 :CBC           7 : test_shm_mq_main(Datum main_arg)
                                 48                 :                : {
                                 49                 :                :     dsm_segment *seg;
                                 50                 :                :     shm_toc    *toc;
                                 51                 :                :     shm_mq_handle *inqh;
                                 52                 :                :     shm_mq_handle *outqh;
                                 53                 :                :     volatile test_shm_mq_header *hdr;
                                 54                 :                :     int         myworkernumber;
                                 55                 :                :     PGPROC     *registrant;
                                 56                 :                : 
                                 57                 :                :     /*
                                 58                 :                :      * Establish signal handlers.
                                 59                 :                :      *
                                 60                 :                :      * We want CHECK_FOR_INTERRUPTS() to kill off this worker process just as
                                 61                 :                :      * it would a normal user backend.  To make that happen, we use die().
                                 62                 :                :      */
 1234 fujii@postgresql.org       63                 :              7 :     pqsignal(SIGTERM, die);
 3743 rhaas@postgresql.org       64                 :              7 :     BackgroundWorkerUnblockSignals();
                                 65                 :                : 
                                 66                 :                :     /*
                                 67                 :                :      * Connect to the dynamic shared memory segment.
                                 68                 :                :      *
                                 69                 :                :      * The backend that registered this worker passed us the ID of a shared
                                 70                 :                :      * memory segment to which we must attach for further instructions.  Once
                                 71                 :                :      * we've mapped the segment in our address space, attach to the table of
                                 72                 :                :      * contents so we can locate the various data structures we'll need to
                                 73                 :                :      * find within the segment.
                                 74                 :                :      *
                                 75                 :                :      * Note: at this point, we have not created any ResourceOwner in this
                                 76                 :                :      * process.  This will result in our DSM mapping surviving until process
                                 77                 :                :      * exit, which is fine.  If there were a ResourceOwner, it would acquire
                                 78                 :                :      * ownership of the mapping, but we have no need for that.
                                 79                 :                :      */
                                 80                 :              7 :     seg = dsm_attach(DatumGetInt32(main_arg));
                                 81         [ -  + ]:              7 :     if (seg == NULL)
 3743 rhaas@postgresql.org       82         [ #  # ]:UBC           0 :         ereport(ERROR,
                                 83                 :                :                 (errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
                                 84                 :                :                  errmsg("unable to map dynamic shared memory segment")));
 3743 rhaas@postgresql.org       85                 :CBC           7 :     toc = shm_toc_attach(PG_TEST_SHM_MQ_MAGIC, dsm_segment_address(seg));
                                 86         [ -  + ]:              7 :     if (toc == NULL)
 3743 rhaas@postgresql.org       87         [ #  # ]:UBC           0 :         ereport(ERROR,
                                 88                 :                :                 (errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
                                 89                 :                :                  errmsg("bad magic number in dynamic shared memory segment")));
                                 90                 :                : 
                                 91                 :                :     /*
                                 92                 :                :      * Acquire a worker number.
                                 93                 :                :      *
                                 94                 :                :      * By convention, the process registering this background worker should
                                 95                 :                :      * have stored the control structure at key 0.  We look up that key to
                                 96                 :                :      * find it.  Our worker number gives our identity: there may be just one
                                 97                 :                :      * worker involved in this parallel operation, or there may be many.
                                 98                 :                :      */
 2505 tgl@sss.pgh.pa.us          99                 :CBC           7 :     hdr = shm_toc_lookup(toc, 0, false);
 3743 rhaas@postgresql.org      100         [ -  + ]:              7 :     SpinLockAcquire(&hdr->mutex);
                                101                 :              7 :     myworkernumber = ++hdr->workers_attached;
                                102                 :              7 :     SpinLockRelease(&hdr->mutex);
                                103         [ -  + ]:              7 :     if (myworkernumber > hdr->workers_total)
 3743 rhaas@postgresql.org      104         [ #  # ]:UBC           0 :         ereport(ERROR,
                                105                 :                :                 (errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
                                106                 :                :                  errmsg("too many message queue testing workers already")));
                                107                 :                : 
                                108                 :                :     /*
                                109                 :                :      * Attach to the appropriate message queues.
                                110                 :                :      */
 3743 rhaas@postgresql.org      111                 :CBC           7 :     attach_to_queues(seg, toc, myworkernumber, &inqh, &outqh);
                                112                 :                : 
                                113                 :                :     /*
                                114                 :                :      * Indicate that we're fully initialized and ready to begin the main part
                                115                 :                :      * of the parallel operation.
                                116                 :                :      *
                                117                 :                :      * Once we signal that we're ready, the user backend is entitled to assume
                                118                 :                :      * that our on_dsm_detach callbacks will fire before we disconnect from
                                119                 :                :      * the shared memory segment and exit.  Generally, that means we must have
                                120                 :                :      * attached to all relevant dynamic shared memory data structures by now.
                                121                 :                :      */
                                122         [ -  + ]:              7 :     SpinLockAcquire(&hdr->mutex);
                                123                 :              7 :     ++hdr->workers_ready;
                                124                 :              7 :     SpinLockRelease(&hdr->mutex);
                                125                 :              7 :     registrant = BackendPidGetProc(MyBgworkerEntry->bgw_notify_pid);
 3742 peter_e@gmx.net           126         [ -  + ]:              7 :     if (registrant == NULL)
                                127                 :                :     {
 3743 rhaas@postgresql.org      128         [ #  # ]:UBC           0 :         elog(DEBUG1, "registrant backend has exited prematurely");
                                129                 :              0 :         proc_exit(1);
                                130                 :                :     }
 3743 rhaas@postgresql.org      131                 :CBC           7 :     SetLatch(&registrant->procLatch);
                                132                 :                : 
                                133                 :                :     /* Do the work. */
                                134                 :              7 :     copy_messages(inqh, outqh);
                                135                 :                : 
                                136                 :                :     /*
                                137                 :                :      * We're done.  For cleanliness, explicitly detach from the shared memory
                                138                 :                :      * segment (that would happen anyway during process exit, though).
                                139                 :                :      */
                                140                 :              7 :     dsm_detach(seg);
                                141                 :              7 :     proc_exit(1);
                                142                 :                : }
                                143                 :                : 
                                144                 :                : /*
                                145                 :                :  * Attach to shared memory message queues.
                                146                 :                :  *
                                147                 :                :  * We use our worker number to determine to which queue we should attach.
                                148                 :                :  * The queues are registered at keys 1..<number-of-workers>.  The user backend
                                149                 :                :  * writes to queue #1 and reads from queue #<number-of-workers>; each worker
                                150                 :                :  * reads from the queue whose number is equal to its worker number and writes
                                151                 :                :  * to the next higher-numbered queue.
                                152                 :                :  */
                                153                 :                : static void
                                154                 :              7 : attach_to_queues(dsm_segment *seg, shm_toc *toc, int myworkernumber,
                                155                 :                :                  shm_mq_handle **inqhp, shm_mq_handle **outqhp)
                                156                 :                : {
                                157                 :                :     shm_mq     *inq;
                                158                 :                :     shm_mq     *outq;
                                159                 :                : 
 2505 tgl@sss.pgh.pa.us         160                 :              7 :     inq = shm_toc_lookup(toc, myworkernumber, false);
 3743 rhaas@postgresql.org      161                 :              7 :     shm_mq_set_receiver(inq, MyProc);
                                162                 :              7 :     *inqhp = shm_mq_attach(inq, seg, NULL);
 2505 tgl@sss.pgh.pa.us         163                 :              7 :     outq = shm_toc_lookup(toc, myworkernumber + 1, false);
 3743 rhaas@postgresql.org      164                 :              7 :     shm_mq_set_sender(outq, MyProc);
                                165                 :              7 :     *outqhp = shm_mq_attach(outq, seg, NULL);
                                166                 :              7 : }
                                167                 :                : 
                                168                 :                : /*
                                169                 :                :  * Loop, receiving and sending messages, until the connection is broken.
                                170                 :                :  *
                                171                 :                :  * This is the "real work" performed by this worker process.  Everything that
                                172                 :                :  * happens before this is initialization of one form or another, and everything
                                173                 :                :  * after this point is cleanup.
                                174                 :                :  */
                                175                 :                : static void
                                176                 :              7 : copy_messages(shm_mq_handle *inqh, shm_mq_handle *outqh)
                                177                 :                : {
                                178                 :                :     Size        len;
                                179                 :                :     void       *data;
                                180                 :                :     shm_mq_result res;
                                181                 :                : 
                                182                 :                :     for (;;)
                                183                 :                :     {
                                184                 :                :         /* Notice any interrupts that have occurred. */
                                185         [ -  + ]:          24608 :         CHECK_FOR_INTERRUPTS();
                                186                 :                : 
                                187                 :                :         /* Receive a message. */
                                188                 :          24608 :         res = shm_mq_receive(inqh, &len, &data, false);
                                189         [ +  + ]:          24608 :         if (res != SHM_MQ_SUCCESS)
                                190                 :              7 :             break;
                                191                 :                : 
                                192                 :                :         /* Send it back out. */
  913                           193                 :          24601 :         res = shm_mq_send(outqh, len, data, false, true);
 3743                           194         [ -  + ]:          24601 :         if (res != SHM_MQ_SUCCESS)
 3743 rhaas@postgresql.org      195                 :UBC           0 :             break;
                                196                 :                :     }
 3743 rhaas@postgresql.org      197                 :CBC           7 : }
        

Generated by: LCOV version 2.1-beta2-3-g6141622