nkeynes@283: /** nkeynes@288: * $Id: testyuv.c,v 1.2 2007-01-15 10:41:30 nkeynes Exp $ nkeynes@283: * nkeynes@283: * Renderer test cases nkeynes@283: * nkeynes@283: * Copyright (c) 2006 Nathan Keynes. nkeynes@283: * nkeynes@283: * This program is free software; you can redistribute it and/or modify nkeynes@283: * it under the terms of the GNU General Public License as published by nkeynes@283: * the Free Software Foundation; either version 2 of the License, or nkeynes@283: * (at your option) any later version. nkeynes@283: * nkeynes@283: * This program is distributed in the hope that it will be useful, nkeynes@283: * but WITHOUT ANY WARRANTY; without even the implied warranty of nkeynes@283: * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the nkeynes@283: * GNU General Public License for more details. nkeynes@283: */ nkeynes@283: #include nkeynes@283: #include "timer.h" nkeynes@283: #include "testdata.h" nkeynes@283: #include "pvr.h" nkeynes@283: #include "lib.h" nkeynes@283: #include "asic.h" nkeynes@283: nkeynes@283: #define PVR_BASE 0xA05F8000 nkeynes@283: #define PVR_RESET (PVR_BASE+0x008) nkeynes@283: #define YUV_ADDR (PVR_BASE+0x148) nkeynes@283: #define YUV_CONFIG (PVR_BASE+0x14C) nkeynes@283: #define YUV_STATUS (PVR_BASE+0x150) nkeynes@288: #define TEXSIZE (PVR_BASE+0x0E4) nkeynes@283: #define DISPLAY_MODE (PVR_BASE+0x044) nkeynes@283: #define DISPLAY_ADDR1 (PVR_BASE+0x050) nkeynes@283: #define DISPLAY_ADDR2 (PVR_BASE+0x054) nkeynes@283: #define DISPLAY_SIZE (PVR_BASE+0x05C) nkeynes@283: nkeynes@288: #define OBJ_START 0x00010000 nkeynes@288: #define OBJ_LENGTH 0x00010000 nkeynes@288: #define TILE_START 0x00060000 nkeynes@288: #define TILE_LENGTH 0x00010000 nkeynes@288: #define TILEMAP_ADDR 0x050B2C8 nkeynes@288: #define RENDER_ADDR 0x00600000 nkeynes@288: nkeynes@288: #define OBJCFG 0xA05F807C nkeynes@288: #define ISPCFG 0xA05F8098 nkeynes@288: nkeynes@288: struct ta_config default_ta_config = { 0x00100002, GRID_SIZE(640,480), OBJ_START, nkeynes@288: OBJ_START+OBJ_LENGTH, TILE_START+TILE_LENGTH, nkeynes@288: TILE_START, TILE_START+TILE_LENGTH }; nkeynes@288: nkeynes@288: nkeynes@288: nkeynes@288: struct render_config default_render_config = { OBJ_START, TILEMAP_ADDR, RENDER_ADDR, 640, 480, nkeynes@288: 0x00000009, 0.0001f, 1.0f }; nkeynes@288: nkeynes@288: nkeynes@288: struct backplane { nkeynes@288: uint32_t mode1, mode2, mode3; nkeynes@288: float x1, y1, z1; nkeynes@288: uint32_t col1; nkeynes@288: float x2, y2, z2; nkeynes@288: uint32_t col2; nkeynes@288: float x3, y3, z3; nkeynes@288: uint32_t col3; nkeynes@288: }; nkeynes@288: nkeynes@288: nkeynes@288: #define START_BLEND 0 nkeynes@288: #define BG_COLOUR 0xFF808080 nkeynes@288: nkeynes@288: struct backplane default_backplane = { 0x90800000, 0x20800440, 0, nkeynes@288: 0.0, 0.0, 0.2, BG_COLOUR, nkeynes@288: 640.0, 0.0, 0.2, BG_COLOUR, nkeynes@288: 0.0, 480.0, 0.2, BG_COLOUR }; nkeynes@288: nkeynes@283: struct yuv_poly { nkeynes@283: uint32_t poly[8]; nkeynes@283: struct { nkeynes@283: uint32_t mode; nkeynes@283: float x,y,z,u,v; nkeynes@283: uint32_t colour; nkeynes@283: uint32_t pad; nkeynes@283: } vertex[4]; nkeynes@283: uint32_t end[8]; nkeynes@288: } test_yuv_poly = { { 0x808C010A, 0xE0000000, 0x2091A4F6, 0x1E000000, 0, 0, 0, 0 }, nkeynes@288: { { 0xE0000000, 0.0, 0.0, 2.0, 0.0, 0.0, 0xFFFFFFFF, 0 }, nkeynes@288: { 0xE0000000, 640.0, 0.0, 2.0, 0.625, 0.0, 0xFFFFFFFF, 0 }, nkeynes@288: { 0xE0000000, 0.0, 480.0, 2.0, 0.0, 0.9375, 0xFFFFFFFF, 0 }, nkeynes@288: { 0xF0000000, 640.0, 480.0, 2.0, 0.625, 0.9375, 0xFFFFFFFF, 0 } }, nkeynes@283: { 0,0,0,0,0,0,0,0 } }; nkeynes@283: nkeynes@288: nkeynes@288: nkeynes@288: nkeynes@283: int test_yuv( test_data_t test_case ) nkeynes@283: { nkeynes@283: int i; nkeynes@283: char tabuf[512]; nkeynes@283: char *p = DMA_ALIGN(&tabuf); nkeynes@283: nkeynes@283: /* Check input */ nkeynes@283: test_data_block_t input = get_test_data(test_case, "input"); nkeynes@283: if( input == NULL ) { nkeynes@283: fprintf( stderr, "Skipping test '%s' - no input\n", test_case->test_name ); nkeynes@283: return -1; nkeynes@283: } nkeynes@283: nkeynes@283: memset( (void *)(0xA5000000), 0xFE, 512000 ); nkeynes@283: nkeynes@283: pvr_init(); nkeynes@283: asic_clear(); nkeynes@283: nkeynes@283: fprintf( stdout, "Writing %d bytes\n", input->length ); nkeynes@283: long_write( YUV_CONFIG, 0x00001D14 ); nkeynes@283: long_write( YUV_ADDR, 0x00000000 ); nkeynes@283: timer_start(); nkeynes@283: uint32_t status1 = long_read( YUV_STATUS ); nkeynes@283: if( pvr_dma_write( 0x10800000, input->data, input->length, 0 ) != 0 ) { nkeynes@283: return -1; nkeynes@283: } nkeynes@283: uint32_t timeus = timer_gettime_us(); nkeynes@283: uint32_t status2 = long_read( YUV_STATUS ); nkeynes@283: nkeynes@288: struct ta_config *config = &default_ta_config; nkeynes@288: nkeynes@288: /* Send TA data */ nkeynes@288: asic_clear(); nkeynes@288: pvr_init(); nkeynes@288: ta_init(config); nkeynes@283: memcpy( p, &test_yuv_poly, sizeof(test_yuv_poly) ); nkeynes@288: if( pvr_dma_write( 0x10000000, p, sizeof(test_yuv_poly), 0 ) != 0 ) { nkeynes@288: return -1; nkeynes@288: } nkeynes@288: asic_wait( EVENT_PVR_OPAQUE_DONE ); nkeynes@288: if( asic_check( EVENT_PVR_PRIM_ALLOC_FAIL ) || nkeynes@288: asic_check( EVENT_PVR_MATRIX_ALLOC_FAIL ) || nkeynes@288: asic_check( EVENT_PVR_BAD_INPUT ) ) { nkeynes@288: asic_dump(stderr); nkeynes@283: return -1; nkeynes@283: } nkeynes@283: nkeynes@288: /* Write backplane (if any) */ nkeynes@288: uint32_t bgplane = pvr_get_objbuf_posn(); nkeynes@288: memcpy( (char *)(PVR_VRAM_BASE + bgplane), &default_backplane, sizeof(default_backplane) ); nkeynes@288: bgplane -= default_render_config.polybuf; nkeynes@288: render_set_backplane( (bgplane << 1) | 0x01000000 ); nkeynes@288: nkeynes@288: /* Render the damn thing */ nkeynes@288: long_write( OBJCFG, 0x0027DF77 ); nkeynes@288: long_write( ISPCFG, 0x00800409 ); nkeynes@288: long_write( TEXSIZE, 0x0000000A ); nkeynes@288: pvr_build_tilemap2( TILEMAP_ADDR, config, 0x10000000 ); nkeynes@288: render_start( &default_render_config ); nkeynes@288: if( asic_wait( EVENT_PVR_RENDER_DONE ) == -1 ) { nkeynes@288: fprintf( stderr, "Test render failed (timeout waiting for render)\n" ); nkeynes@288: asic_dump( stderr ); nkeynes@288: return -1; nkeynes@288: } nkeynes@288: nkeynes@288: asic_clear(); nkeynes@288: asic_wait( EVENT_RETRACE ); nkeynes@288: display_render( &default_render_config ); nkeynes@288: asic_clear(); nkeynes@288: asic_wait(EVENT_RETRACE); nkeynes@288: asic_clear(); nkeynes@288: asic_wait(EVENT_RETRACE); nkeynes@288: asic_clear(); nkeynes@288: asic_wait(EVENT_RETRACE); nkeynes@288: asic_clear(); nkeynes@288: asic_wait(EVENT_RETRACE); nkeynes@283: return 0; nkeynes@283: } nkeynes@283: nkeynes@283: int main( int argc, char *argv[] ) nkeynes@283: { nkeynes@283: int test_cases = 0; nkeynes@283: int test_failures = 0; nkeynes@283: test_data_t test_data = load_test_dataset(stdin); nkeynes@283: test_data_t test_case = test_data; nkeynes@283: nkeynes@283: asic_mask_all(); nkeynes@283: pvr_init(); nkeynes@283: nkeynes@283: while( test_case != NULL ) { nkeynes@283: test_cases++; nkeynes@283: int result = test_yuv(test_case); nkeynes@283: if( result != 0 ) { nkeynes@283: test_failures++; nkeynes@283: } nkeynes@283: test_case = test_case->next; nkeynes@283: } nkeynes@283: free_test_dataset(test_data); nkeynes@283: if( test_failures != 0 ) { nkeynes@283: fprintf( stderr, "%d/%d test failures!\n", test_failures, test_cases ); nkeynes@283: return 1; nkeynes@283: } else { nkeynes@283: fprintf( stderr, "%d tests OK\n", test_cases ); nkeynes@283: return 0; nkeynes@283: } nkeynes@283: }