X-Git-Url: https://git.gag.com/?a=blobdiff_plain;f=xfer-src%2Fsource-pattern.c;h=5a4aad836625c593341876f951845439898b757f;hb=66086d646c4cc92cf582b6765b315060117c6ec4;hp=2e8a5eb92b2cbf9d2628437265edbad7e4011ba5;hpb=fd48f3e498442f0cbff5f3606c7c403d0566150e;p=debian%2Famanda diff --git a/xfer-src/source-pattern.c b/xfer-src/source-pattern.c index 2e8a5eb..5a4aad8 100644 --- a/xfer-src/source-pattern.c +++ b/xfer-src/source-pattern.c @@ -1,6 +1,6 @@ /* * Amanda, The Advanced Maryland Automatic Network Disk Archiver - * Copyright (c) 2008,2009 Zmanda, Inc. All Rights Reserved. + * Copyright (c) 2008, 2009, 2011 Zmanda, Inc. All Rights Reserved. * * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 as published @@ -19,8 +19,8 @@ * Sunnyvale, CA 94085, USA, or: http://www.zmanda.com */ -#include "amxfer.h" #include "amanda.h" +#include "amxfer.h" #include "simpleprng.h" /* @@ -66,6 +66,41 @@ typedef struct { * Implementation */ +/* + * Utility function to fill a buffer buf of size len with the pattern defined + * for this instance. We start each copying from the offset where we left the + * previous one. + * + * Note that performance is important: amtapetype uses this very source and + * source-random.c to determine whether a tape device uses compression, and + * expects the two to run about the same speed. This is why this is a byte + * loop and does not use mempcy() and friends: the random source is also byte + * per byte. + */ + +static void fill_buffer_with_pattern(XferSourcePattern *self, char *buf, + size_t len) +{ + char *src = self->pattern, *dst = buf; + size_t plen = self->pattern_buffer_length, offset = self->current_offset; + size_t pos = 0; + + src += offset; + + while (pos < len) { + *dst++ = *src++; + pos++, offset++; + + if (G_LIKELY(offset < plen)) + continue; + + offset = 0; + src = self->pattern; + } + + self->current_offset = offset; +} + static gpointer pull_buffer_impl( XferElement *elt, @@ -73,12 +108,9 @@ pull_buffer_impl( { XferSourcePattern *self = (XferSourcePattern *)elt; char *rval; - char *s, *d; - size_t l; - size_t offset; /* indicate EOF on an cancel */ - if (elt->cancelled || (self->limited_length && self->length == 0)) { + if (elt->cancelled) { *size = 0; return NULL; } @@ -97,17 +129,7 @@ pull_buffer_impl( rval = malloc(*size); - /* fill the buffer "manually", instead of using fancy memcpy techniques, so - * that this runs at about the same speed as the random source */ - l = *size; - s = self->pattern; - offset = self->current_offset; - d = rval; - while (l--) { - *(d++) = *(s + offset++); - if (offset >= self->pattern_buffer_length) offset = 0; - } - self->current_offset = offset; + fill_buffer_with_pattern(self, rval, *size); return rval; } @@ -125,8 +147,8 @@ class_init( { XferElementClass *klass = XFER_ELEMENT_CLASS(selfc); static xfer_element_mech_pair_t mech_pairs[] = { - { XFER_MECH_NONE, XFER_MECH_PULL_BUFFER, 1, 0}, - { XFER_MECH_NONE, XFER_MECH_NONE, 0, 0}, + { XFER_MECH_NONE, XFER_MECH_PULL_BUFFER, XFER_NROPS(1), XFER_NTHREADS(0) }, + { XFER_MECH_NONE, XFER_MECH_NONE, XFER_NROPS(0), XFER_NTHREADS(0) }, }; klass->pull_buffer = pull_buffer_impl;