Fix base backup rate limiting in presence of slow i/o
authorMagnus Hagander <magnus@hagander.net>
Mon, 19 Dec 2016 09:11:04 +0000 (10:11 +0100)
committerMagnus Hagander <magnus@hagander.net>
Mon, 19 Dec 2016 09:15:52 +0000 (10:15 +0100)
When source i/o on disk was too slow compared to the rate limiting
specified, the system could end up with a negative value for sleep that
it never got out of, which caused rate limiting to effectively be
turned off.

Discussion: https://postgr.es/m/CABUevEy_-e0YvL4ayoX8bH_Ja9w%2BBHoP6jUgdxZuG2nEj3uAfQ%40mail.gmail.com

Analysis by me, patch by Antonin Houska

src/backend/replication/basebackup.c

index 426f6c85e1b08289d941d28a8f100a82f5247ee8..08d6733259611c87082c8ee72ede54fdfc3e847f 100644 (file)
@@ -1303,26 +1303,16 @@ throttle(size_t increment)
                if (wait_result & WL_LATCH_SET)
                        CHECK_FOR_INTERRUPTS();
        }
-       else
-       {
-               /*
-                * The actual transfer rate is below the limit.  A negative value
-                * would distort the adjustment of throttled_last.
-                */
-               wait_result = 0;
-               sleep = 0;
-       }
 
        /*
-        * Only a whole multiple of throttling_sample was processed. The rest will
-        * be done during the next call of this function.
+        * As we work with integers, only whole multiple of throttling_sample was
+        * processed. The rest will be done during the next call of this function.
         */
        throttling_counter %= throttling_sample;
 
-       /* Once the (possible) sleep has ended, new period starts. */
-       if (wait_result & WL_TIMEOUT)
-               throttled_last += elapsed + sleep;
-       else if (sleep > 0)
-               /* Sleep was necessary but might have been interrupted. */
-               throttled_last = GetCurrentIntegerTimestamp();
+       /*
+        * Time interval for the remaining amount and possible next increments
+        * starts now.
+        */
+       throttled_last = GetCurrentIntegerTimestamp();
 }