videogen.c 7.76 KB
Newer Older
1
/*
2 3
 * Generate a synthetic YUV video sequence suitable for codec testing.
 * NOTE: No floats are used to guarantee bitexact output.
4 5 6
 *
 * Copyright (c) 2002 Fabrice Bellard
 *
7
 * This file is part of Libav.
8
 *
9
 * Libav is free software; you can redistribute it and/or
10 11 12 13
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
14
 * Libav is distributed in the hope that it will be useful,
15 16 17 18 19
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
20
 * License along with Libav; if not, write to the Free Software
21
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
22
 */
23

24
#include <stdlib.h>
25
#include <stdint.h>
26 27 28 29
#include <stdio.h>

#define SCALEBITS 8
#define ONE_HALF  (1 << (SCALEBITS - 1))
30
#define FIX(x)    ((int) ((x) * (1L << SCALEBITS) + 0.5))
31

32
static void rgb24_to_yuv420p(uint8_t *lum, uint8_t *cb, uint8_t *cr,
33
                             uint8_t *src, int width, int height)
34 35 36
{
    int wrap, wrap3, x, y;
    int r, g, b, r1, g1, b1;
37
    uint8_t *p;
38

39
    wrap  = width;
40
    wrap3 = width * 3;
41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89
    p     = src;
    for (y = 0; y < height; y += 2) {
        for (x = 0; x < width; x += 2) {
            r       = p[0];
            g       = p[1];
            b       = p[2];
            r1      = r;
            g1      = g;
            b1      = b;
            lum[0]  = (FIX(0.29900) * r + FIX(0.58700) * g +
                       FIX(0.11400) * b + ONE_HALF) >> SCALEBITS;
            r       = p[3];
            g       = p[4];
            b       = p[5];
            r1     += r;
            g1     += g;
            b1     += b;
            lum[1]  = (FIX(0.29900) * r + FIX(0.58700) * g +
                       FIX(0.11400) * b + ONE_HALF) >> SCALEBITS;
            p      += wrap3;
            lum    += wrap;

            r       = p[0];
            g       = p[1];
            b       = p[2];
            r1     += r;
            g1     += g;
            b1     += b;
            lum[0]  = (FIX(0.29900) * r + FIX(0.58700) * g +
                       FIX(0.11400) * b + ONE_HALF) >> SCALEBITS;
            r       = p[3];
            g       = p[4];
            b       = p[5];
            r1     += r;
            g1     += g;
            b1     += b;
            lum[1]  = (FIX(0.29900) * r + FIX(0.58700) * g +
                       FIX(0.11400) * b + ONE_HALF) >> SCALEBITS;

            cb[0]   = 128 + ((- FIX(0.16874) * r1 -
                                FIX(0.33126) * g1 +
                                FIX(0.50000) * b1 +
                              4 * ONE_HALF - 1)
                             >> (SCALEBITS + 2));
            cr[0]   = 128 + ((FIX(0.50000) * r1 -
                              FIX(0.41869) * g1 -
                              FIX(0.08131) * b1 +
                              4 * ONE_HALF - 1)
                             >> (SCALEBITS + 2));
90 91 92

            cb++;
            cr++;
93
            p   += -wrap3 + 2 * 3;
94 95
            lum += -wrap + 2;
        }
96
        p   += wrap3;
97 98 99 100 101 102 103
        lum += wrap;
    }
}

/* cif format */
#define DEFAULT_WIDTH   352
#define DEFAULT_HEIGHT  288
104
#define DEFAULT_NB_PICT 50 /* 2 seconds */
105

106 107
static void pgmyuv_save(const char *filename, int w, int h,
                        unsigned char *rgb_tab)
108 109 110 111 112 113 114
{
    FILE *f;
    int i, h2, w2;
    unsigned char *cb, *cr;
    unsigned char *lum_tab, *cb_tab, *cr_tab;

    lum_tab = malloc(w * h);
115 116
    cb_tab  = malloc((w * h) / 4);
    cr_tab  = malloc((w * h) / 4);
117 118 119

    rgb24_to_yuv420p(lum_tab, cb_tab, cr_tab, rgb_tab, w, h);

120
    f = fopen(filename, "wb");
121 122 123 124 125 126
    fprintf(f, "P5\n%d %d\n%d\n", w, (h * 3) / 2, 255);
    fwrite(lum_tab, 1, w * h, f);
    h2 = h / 2;
    w2 = w / 2;
    cb = cb_tab;
    cr = cr_tab;
127
    for (i = 0; i < h2; i++) {
128 129 130 131 132 133 134 135 136 137 138 139 140 141 142
        fwrite(cb, 1, w2, f);
        fwrite(cr, 1, w2, f);
        cb += w2;
        cr += w2;
    }
    fclose(f);

    free(lum_tab);
    free(cb_tab);
    free(cr_tab);
}

unsigned char *rgb_tab;
int width, height, wrap;

143
static void put_pixel(int x, int y, int r, int g, int b)
144 145 146 147 148 149 150
{
    unsigned char *p;

    if (x < 0 || x >= width ||
        y < 0 || y >= height)
        return;

151
    p    = rgb_tab + y * wrap + x * 3;
152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179
    p[0] = r;
    p[1] = g;
    p[2] = b;
}

static unsigned int myrnd(unsigned int *seed_ptr, int n)
{
    unsigned int seed, val;

    seed = *seed_ptr;
    seed = (seed * 314159) + 1;
    if (n == 256) {
        val = seed >> 24;
    } else {
        val = seed % n;
    }
    *seed_ptr = seed;
    return val;
}

#define NOISE_X  10
#define NOISE_Y  30
#define NOISE_W  26

#define FRAC_BITS 8
#define FRAC_ONE (1 << FRAC_BITS)

/* cosine approximate with 1-x^2 */
180
static int int_cos(int a)
181 182 183 184 185 186 187 188
{
    int v, neg;
    a = a & (FRAC_ONE - 1);
    if (a >= (FRAC_ONE / 2))
        a = FRAC_ONE - a;
    neg = 0;
    if (a > (FRAC_ONE / 4)) {
        neg = -1;
189
        a   = (FRAC_ONE / 2) - a;
190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206
    }
    v = FRAC_ONE - ((a * a) >> 4);
    v = (v ^ neg) - neg;
    return v;
}

#define NB_OBJS  10

typedef struct VObj {
    int x, y, w, h;
    int r, g, b;
} VObj;

VObj objs[NB_OBJS];

unsigned int seed = 1;

207
static void gen_image(int num, int w, int h)
208 209 210 211 212
{
    int r, g, b, x, y, i, dx, dy, x1, y1;
    unsigned int seed1;

    if (num == 0) {
213
        for (i = 0; i < NB_OBJS; i++) {
214 215 216 217 218 219 220 221 222 223 224 225 226 227
            objs[i].x = myrnd(&seed, w);
            objs[i].y = myrnd(&seed, h);
            objs[i].w = myrnd(&seed, w / 4) + 10;
            objs[i].h = myrnd(&seed, h / 4) + 10;
            objs[i].r = myrnd(&seed, 256);
            objs[i].g = myrnd(&seed, 256);
            objs[i].b = myrnd(&seed, 256);
        }
    }

    /* first a moving background with gradients */
    /* test motion estimation */
    dx = int_cos(num * FRAC_ONE / 50) * 35;
    dy = int_cos(num * FRAC_ONE / 50 + FRAC_ONE / 10) * 30;
228 229
    for (y = 0; y < h; y++) {
        for (x = 0; x < w; x++) {
230
            x1 = (x << FRAC_BITS) + dx;
231
            y1 = (y << FRAC_BITS) + dy;
232 233 234
            r  =       ((y1  * 7) >> FRAC_BITS) & 0xff;
            g  = (((x1 + y1) * 9) >> FRAC_BITS) & 0xff;
            b  =  ((x1       * 5) >> FRAC_BITS) & 0xff;
235 236 237 238 239 240
            put_pixel(x, y, r, g, b);
        }
    }

    /* then some noise with very high intensity to test saturation */
    seed1 = num;
241 242
    for (y = 0; y < NOISE_W; y++) {
        for (x = 0; x < NOISE_W; x++) {
243 244 245 246 247 248
            r = myrnd(&seed1, 256);
            g = myrnd(&seed1, 256);
            b = myrnd(&seed1, 256);
            put_pixel(x + NOISE_X, y + NOISE_Y, r, g, b);
        }
    }
249

250
    /* then moving objects */
251
    for (i = 0; i < NB_OBJS; i++) {
252 253
        VObj *p = &objs[i];
        seed1 = i;
254 255
        for (y = 0; y < p->h; y++) {
            for (x = 0; x < p->w; x++) {
256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285
                r = p->r;
                g = p->g;
                b = p->b;
                /* add a per object noise */
                r += myrnd(&seed1, 50);
                g += myrnd(&seed1, 50);
                b += myrnd(&seed1, 50);
                put_pixel(x + p->x, y + p->y, r, g, b);
            }
        }
        p->x += myrnd(&seed, 21) - 10;
        p->y += myrnd(&seed, 21) - 10;
    }
}

int main(int argc, char **argv)
{
    int w, h, i;
    char buf[1024];

    if (argc != 2) {
        printf("usage: %s file\n"
               "generate a test video stream\n", argv[0]);
        exit(1);
    }

    w = DEFAULT_WIDTH;
    h = DEFAULT_HEIGHT;

    rgb_tab = malloc(w * h * 3);
286 287 288
    wrap    = w * 3;
    width   = w;
    height  = h;
289

290
    for (i = 0; i < DEFAULT_NB_PICT; i++) {
291
        snprintf(buf, sizeof(buf), "%s%02d.pgm", argv[1], i);
292 293 294
        gen_image(i, w, h);
        pgmyuv_save(buf, w, h, rgb_tab);
    }
295

296 297 298
    free(rgb_tab);
    return 0;
}