/************************************************************************** * * Copyright 2007 Tungsten Graphics, Inc., Cedar Park, Texas. * All Rights Reserved. * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the * "Software"), to deal in the Software without restriction, including * without limitation the rights to use, copy, modify, merge, publish, * distribute, sub license, and/or sell copies of the Software, and to * permit persons to whom the Software is furnished to do so, subject to * the following conditions: * * The above copyright notice and this permission notice (including the * next paragraph) shall be included in all copies or substantial portions * of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. * IN NO EVENT SHALL TUNGSTEN GRAPHICS AND/OR ITS SUPPLIERS BE LIABLE FOR * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. * **************************************************************************/ #ifndef LP_TILE_CACHE_H #define LP_TILE_CACHE_H #define TILE_CLEAR_OPTIMIZATION 1 #include "pipe/p_compiler.h" #include "lp_tile_soa.h" struct llvmpipe_context; struct llvmpipe_tile_cache; /* If we need to support > 4096, just expand this to be a 64 bit * union, or consider tiling in Z as well. */ union tile_address { struct { unsigned x:6; /* 4096 / TILE_SIZE */ unsigned y:6; /* 4096 / TILE_SIZE */ unsigned z:12; /* 4096 -- z not tiled */ unsigned face:3; unsigned level:4; unsigned invalid:1; } bits; unsigned value; }; struct llvmpipe_cached_tile { union tile_address addr; union { /** color in SOA format */ uint8_t ALIGN16_ATTRIB color[TILE_SIZE*TILE_SIZE*NUM_CHANNELS]; uint color32[TILE_SIZE][TILE_SIZE]; ubyte any[1]; } data; }; #define NUM_ENTRIES 50 /** XXX move these */ #define MAX_WIDTH 2048 #define MAX_HEIGHT 2048 struct llvmpipe_tile_cache { struct pipe_screen *screen; struct pipe_surface *surface; /**< the surface we're caching */ struct pipe_transfer *transfer; void *transfer_map; struct pipe_texture *texture; /**< if caching a texture */ unsigned timestamp; struct llvmpipe_cached_tile entries[NUM_ENTRIES]; uint clear_flags[(MAX_WIDTH / TILE_SIZE) * (MAX_HEIGHT / TILE_SIZE) / 32]; float clear_color[4]; /**< for color bufs */ uint clear_val; /**< for z+stencil, or packed color clear value */ struct pipe_transfer *tex_trans; void *tex_trans_map; int tex_face, tex_level, tex_z; struct llvmpipe_cached_tile tile; /**< scratch tile for clears */ struct llvmpipe_cached_tile *last_tile; /**< most recently retrieved tile */ }; extern struct llvmpipe_tile_cache * lp_create_tile_cache( struct pipe_screen *screen ); extern void lp_destroy_tile_cache(struct llvmpipe_tile_cache *tc); extern void lp_tile_cache_set_surface(struct llvmpipe_tile_cache *tc, struct pipe_surface *lps); extern struct pipe_surface * lp_tile_cache_get_surface(struct llvmpipe_tile_cache *tc); extern void lp_tile_cache_map_transfers(struct llvmpipe_tile_cache *tc); extern void lp_tile_cache_unmap_transfers(struct llvmpipe_tile_cache *tc); extern void lp_flush_tile_cache(struct llvmpipe_tile_cache *tc); extern void lp_tile_cache_clear(struct llvmpipe_tile_cache *tc, const float *rgba, uint clearValue); extern void * lp_find_cached_tile(struct llvmpipe_tile_cache *tc, union tile_address addr ); static INLINE const union tile_address tile_address( unsigned x, unsigned y, unsigned z, unsigned face, unsigned level ) { union tile_address addr; addr.value = 0; addr.bits.x = x / TILE_SIZE; addr.bits.y = y / TILE_SIZE; addr.bits.z = z; addr.bits.face = face; addr.bits.level = level; return addr; } /* Quickly retrieve tile if it matches last lookup. */ static INLINE void * lp_get_cached_tile(struct llvmpipe_tile_cache *tc, int x, int y ) { union tile_address addr = tile_address( x, y, 0, 0, 0 ); if (tc->last_tile->addr.value == addr.value) return &tc->last_tile->data.color; return lp_find_cached_tile( tc, addr ); } #endif /* LP_TILE_CACHE_H */