summaryrefslogtreecommitdiffstats
path: root/gcbv/mirror/gcbv.c
diff options
context:
space:
mode:
Diffstat (limited to 'gcbv/mirror/gcbv.c')
-rw-r--r--gcbv/mirror/gcbv.c2401
1 files changed, 1422 insertions, 979 deletions
diff --git a/gcbv/mirror/gcbv.c b/gcbv/mirror/gcbv.c
index 0957014..61d1ca0 100644
--- a/gcbv/mirror/gcbv.c
+++ b/gcbv/mirror/gcbv.c
@@ -35,17 +35,17 @@
#define GCZONE_NONE 0
#define GCZONE_ALL (~0U)
#define GCZONE_MAPPING (1 << 0)
-#define GCZONE_BUFFER (1 << 1)
-#define GCZONE_FIXUP (1 << 2)
-#define GCZONE_FORMAT (1 << 3)
-#define GCZONE_COLOR (1 << 4)
-#define GCZONE_BLEND (1 << 5)
-#define GCZONE_DEST (1 << 6)
-#define GCZONE_SURF (1 << 7)
-#define GCZONE_DO_FILL (1 << 8)
-#define GCZONE_DO_BLIT (1 << 9)
-#define GCZONE_DO_FILTER (1 << 10)
-#define GCZONE_CLIP (1 << 11)
+#define GCZONE_BUFFER_ALLOC (1 << 1)
+#define GCZONE_BUFFER (1 << 2)
+#define GCZONE_FIXUP (1 << 3)
+#define GCZONE_FORMAT (1 << 4)
+#define GCZONE_COLOR (1 << 5)
+#define GCZONE_BLEND (1 << 6)
+#define GCZONE_DEST (1 << 7)
+#define GCZONE_SURF (1 << 8)
+#define GCZONE_DO_FILL (1 << 9)
+#define GCZONE_DO_BLIT (1 << 10)
+#define GCZONE_DO_FILTER (1 << 11)
#define GCZONE_BATCH (1 << 12)
#define GCZONE_BLIT (1 << 13)
#define GCZONE_CACHE (1 << 14)
@@ -62,7 +62,6 @@ GCDBG_FILTERDEF(gcbv, GCZONE_NONE,
"do_fill",
"do_blit",
"do_filter",
- "clip",
"batch",
"blit",
"cache")
@@ -72,9 +71,21 @@ GCDBG_FILTERDEF(gcbv, GCZONE_NONE,
** Miscellaneous defines and macros.
*/
+#if !defined(BVBATCH_DESTRECT)
+#define BVBATCH_DESTRECT (BVBATCH_DSTRECT_ORIGIN | BVBATCH_DSTRECT_SIZE)
+#endif
+
+#if !defined(BVBATCH_SRC1RECT)
+#define BVBATCH_SRC1RECT (BVBATCH_SRC1RECT_ORIGIN | BVBATCH_SRC1RECT_SIZE)
+#endif
+
+#if !defined(BVBATCH_SRC2RECT)
+#define BVBATCH_SRC2RECT (BVBATCH_SRC2RECT_ORIGIN | BVBATCH_SRC2RECT_SIZE)
+#endif
+
#define EQ_SIZE(rect1, rect2) \
( \
- (rect1.width == rect2.width) && (rect1.height == rect2.height) \
+ (rect1->width == rect2->width) && (rect1->height == rect2->height) \
)
#define STRUCTSIZE(structptr, lastmember) \
@@ -94,8 +105,6 @@ GCDBG_FILTERDEF(gcbv, GCZONE_NONE,
#define GC_CLIP_RESET_RIGHT ((unsigned short) ((1 << 15) - 1))
#define GC_CLIP_RESET_BOTTOM ((unsigned short) ((1 << 15) - 1))
-#define GC_BASE_ALIGN 64
-
#define GPU_CMD_SIZE (sizeof(unsigned int) * 2)
#define GC_BUFFER_RESERVE \
@@ -130,12 +139,69 @@ GCDBG_FILTERDEF(gcbv, GCZONE_NONE,
** Internal structures.
*/
+struct gcblendconfig {
+ unsigned char factor_mode;
+ unsigned char color_reverse;
+
+ unsigned char src1used;
+ unsigned char src2used;
+};
+
+struct bvblendxlate {
+ unsigned char match1;
+ unsigned char match2;
+
+ struct gcblendconfig k1;
+ struct gcblendconfig k2;
+};
+
+/* Alpha blending descriptor. */
+struct gcalpha {
+ unsigned int src_global_color;
+ unsigned int dst_global_color;
+
+ unsigned char src_global_alpha_mode;
+ unsigned char dst_global_alpha_mode;
+
+ struct gcblendconfig *k1;
+ struct gcblendconfig *k2;
+
+ struct gcblendconfig *srcconfig;
+ struct gcblendconfig *dstconfig;
+
+ unsigned int src1used;
+ unsigned int src2used;
+};
+
/* Used by blitters to define an array of valid sources. */
struct srcinfo {
+ /* BLTsville source index (0 for src1 and 1 for src2). */
int index;
+
+ /* Source surface buffer descriptor. */
union bvinbuff buf;
+
+ /* Source surface geometry. */
struct bvsurfgeom *geom;
+
+ /* Source rectangle. */
struct bvrect *rect;
+
+ /* Source surface format. */
+ struct bvformatxlate *format;
+
+ /* Source rotation angle. */
+ int angle;
+ unsigned int rot;
+
+ /* Mirror setting. */
+ unsigned int mirror;
+
+ /* ROP. */
+ unsigned short rop;
+
+ /* Blending info. */
+ struct gcalpha *gca;
};
/* bvbuffmap struct attachment. */
@@ -176,6 +242,9 @@ struct gcbatch {
/* State of the current operation. */
struct gcblit gcblit;
+ /* Destination format. */
+ struct bvformatxlate *dstformat;
+
/* Clipping deltas; used to correct the source coordinates for
* single source blits. */
int deltaleft;
@@ -183,17 +252,50 @@ struct gcbatch {
int deltaright;
int deltabottom;
- /* Destination format. */
- struct bvformatxlate *dstformat;
+ /* Clipped destination rectangle coordinates. */
+ unsigned short clippedleft;
+ unsigned short clippedtop;
+ unsigned short clippedright;
+ unsigned short clippedbottom;
+
+ /* Destination base address alignment in pixels. */
+ int dstalign;
+
+ /* Destination origin offset. */
+ unsigned int dstoffsetX;
+ unsigned int dstoffsetY;
+
+ /* Rotation angle. */
+ int dstangle;
+
+ /* Geometry size of the destination surface. */
+ unsigned int dstwidth;
+ unsigned int dstheight;
+
+ /* Physical size of the destination surface. */
+ unsigned int dstphyswidth;
+ unsigned int dstphysheight;
+
+ /* Computed destination rectangle coordinates; in multi-source
+ * setup can be modified to match new destination and source
+ * geometry. */
+ unsigned short left;
+ unsigned short top;
+ unsigned short right;
+ unsigned short bottom;
- /* Destination rectangle coordinates. */
- unsigned short dstleft;
- unsigned short dsttop;
- unsigned short dstright;
- unsigned short dstbottom;
+ /* Physical size of the matched destination and source surfaces
+ * for multi-source setup. */
+ unsigned int physwidth;
+ unsigned int physheight;
- /* Destination base address alignment offset in pixels. */
- unsigned int dstoffset;
+ /* Alignment byte offset for the destination surface; in multi-
+ * source setup can be modified to match new destination and source
+ * geometry. */
+ int dstbyteshift;
+
+ /* Block walker enable. */
+ int blockenable;
#if GCDEBUG_ENABLE
/* Rectangle validation storage. */
@@ -234,6 +336,9 @@ struct gccallbackinfo {
/* Driver context. */
struct gccontext {
+ /* Pipe initialization. */
+ int initdone;
+
/* Last generated error message. */
char bverrorstr[128];
@@ -269,24 +374,24 @@ static struct gccontext gccontext;
static void dumpbatch(struct gcbatch *batch)
{
- if ((GCDBGFILTER.zone & GCZONE_BATCH) != 0) {
+ if ((GCDBGFILTER.zone & (GCZONE_BUFFER)) != 0) {
struct gcbuffer *buffer;
unsigned int i, size;
struct gcfixup *fixup;
- GCDBG(GCZONE_BATCH, "BATCH DUMP (0x%08X)\n",
+ GCDBG(GCZONE_BUFFER, "BATCH DUMP (0x%08X)\n",
(unsigned int) batch);
buffer = batch->bufhead;
while (buffer != NULL) {
fixup = buffer->fixuphead;
while (fixup != NULL) {
- GCDBG(GCZONE_BATCH,
+ GCDBG(GCZONE_BUFFER,
" Fixup table @ 0x%08X, count = %d:\n",
(unsigned int) fixup, fixup->count);
for (i = 0; i < fixup->count; i += 1) {
- GCDBG(GCZONE_BATCH, " [%02d]"
+ GCDBG(GCZONE_BUFFER, " [%02d]"
" buffer offset = 0x%08X,"
" surface offset = 0x%08X\n",
i,
@@ -314,55 +419,28 @@ static void dumpbatch(struct gcbatch *batch)
* Error handling.
*/
-#define SETERROR(message) \
-do { \
- snprintf(gccontext.bverrorstr, sizeof(gccontext.bverrorstr), \
- "%s(%d): " message, __func__, __LINE__); \
- GCDUMPSTRING("%s\n", gccontext.bverrorstr); \
-} while (0)
-
-#define SETERRORARG(message, arg) \
+#define BVSETERROR(error, message, ...) \
do { \
snprintf(gccontext.bverrorstr, sizeof(gccontext.bverrorstr), \
- "%s(%d): " message, __func__, __LINE__, arg); \
- GCDUMPSTRING("%s\n", gccontext.bverrorstr); \
-} while (0)
-
-#define BVSETERROR(error, message) \
-do { \
- SETERROR(message); \
- bverror = error; \
-} while (0)
-
-#define BVSETERRORARG(error, message, arg) \
-do { \
- SETERRORARG(message, arg); \
+ message, ##__VA_ARGS__); \
+ GCDUMPSTRING("%s(%d): [ERROR] %s\n", __func__, __LINE__, \
+ gccontext.bverrorstr); \
bverror = error; \
} while (0)
-#define BVSETBLTERROR(error, message) \
-do { \
- BVSETERROR(error, message); \
- bltparams->errdesc = gccontext.bverrorstr; \
-} while (0)
-
-#define BVSETBLTERRORARG(error, message, arg) \
+#define BVSETBLTERROR(error, message, ...) \
do { \
- BVSETERRORARG(error, message, arg); \
+ BVSETERROR(error, message, ##__VA_ARGS__); \
bltparams->errdesc = gccontext.bverrorstr; \
} while (0)
-#define BVSETSURFERROR(errorid, errordesc) \
+#define BVSETBLTSURFERROR(errorid, errordesc) \
do { \
snprintf(gccontext.bverrorstr, sizeof(gccontext.bverrorstr), \
- g_surferr[errorid].message, __func__, __LINE__, errordesc.id); \
- GCDUMPSTRING("%s\n", gccontext.bverrorstr); \
+ g_surferr[errorid].message, errordesc.id); \
+ GCDUMPSTRING("%s(%d): [ERROR] %s\n", __func__, __LINE__, \
+ gccontext.bverrorstr); \
bverror = errordesc.base + g_surferr[errorid].offset; \
-} while (0)
-
-#define BVSETBLTSURFERROR(errorid, errordesc) \
-do { \
- BVSETSURFERROR(errorid, errordesc); \
bltparams->errdesc = gccontext.bverrorstr; \
} while (0)
@@ -388,31 +466,31 @@ struct bvsurferror {
static struct bvsurferror g_surferr[] = {
/* GCBVERR_DESC */
- { 0, "%s(%d): %s desc structure is not set" },
+ { 0, "%s desc structure is not set" },
/* GCBVERR_DESC_VERS */
- { 100, "%s(%d): %s desc structure has invalid size" },
+ { 100, "%s desc structure has invalid size" },
/* GCBVERR_DESC_VIRTADDR */
- { 200, "%s(%d): %s desc virtual pointer is not set" },
+ { 200, "%s desc virtual pointer is not set" },
/* GCBVERR_TILE: FIXME/TODO define error code */
- { 0, "%s(%d): %s tileparams structure is not set" },
+ { 0, "%s tileparams structure is not set" },
/* GCBVERR_TILE_VERS */
- { 3000, "%s(%d): %s tileparams structure has invalid size" },
+ { 3000, "%s tileparams structure has invalid size" },
/* GCBVERR_TILE_VIRTADDR: FIXME/TODO define error code */
- { 200, "%s(%d): %s tileparams virtual pointer is not set" },
+ { 200, "%s tileparams virtual pointer is not set" },
/* GCBVERR_GEOM */
- { 1000, "%s(%d): %s geom structure is not set" },
+ { 1000, "%s geom structure is not set" },
/* GCBVERR_GEOM_VERS */
- { 1100, "%s(%d): %s geom structure has invalid size" },
+ { 1100, "%s geom structure has invalid size" },
/* GCBVERR_GEOM_FORMAT */
- { 1200, "%s(%d): %s invalid format specified" },
+ { 1200, "%s invalid format specified" },
};
static struct bvsurferrorid g_destsurferr = { "dst", BVERR_DSTDESC };
@@ -506,7 +584,6 @@ static enum bverror do_map(struct bvbuffdesc *bvbuffdesc,
struct bvphysdesc *bvphysdesc;
bool mappedbyothers;
struct gcmap gcmap;
- unsigned int offset;
struct gcschedunmap *gcschedunmap;
GCENTERARG(GCZONE_MAPPING, "bvbuffdesc = 0x%08X\n",
@@ -559,49 +636,41 @@ static enum bverror do_map(struct bvbuffdesc *bvbuffdesc,
goto fail;
}
- gcmap.buf.offset = bvphysdesc->pageoffset
- & ~(GC_BASE_ALIGN - 1);
- offset = bvphysdesc->pageoffset
- & (GC_BASE_ALIGN - 1);
+ gcmap.buf.offset = bvphysdesc->pageoffset;
gcmap.pagesize = bvphysdesc->pagesize;
gcmap.pagearray = bvphysdesc->pagearray;
+ gcmap.size = bvbuffdesc->length;
GCDBG(GCZONE_MAPPING, "new mapping (%s):\n",
- (batch == NULL) ? "explicit" : "implicit");
+ (batch == NULL) ? "explicit" : "implicit");
GCDBG(GCZONE_MAPPING, "pagesize = %lu\n",
- bvphysdesc->pagesize);
+ bvphysdesc->pagesize);
GCDBG(GCZONE_MAPPING, "pagearray = 0x%08X\n",
- (unsigned int) bvphysdesc->pagearray);
- GCDBG(GCZONE_MAPPING, "specified pageoffset = %lu\n",
- bvphysdesc->pageoffset);
- GCDBG(GCZONE_MAPPING, "aligned pageoffset = %d\n",
- gcmap.buf.offset);
+ (unsigned int) bvphysdesc->pagearray);
+ GCDBG(GCZONE_MAPPING, "pageoffset = %lu\n",
+ bvphysdesc->pageoffset);
+ GCDBG(GCZONE_MAPPING, "mapping size = %d\n",
+ gcmap.size);
} else {
- gcmap.buf.logical
- = (void *) ((unsigned int) bvbuffdesc->virtaddr
- & ~(GC_BASE_ALIGN - 1));
- offset = (unsigned int) bvbuffdesc->virtaddr
- & (GC_BASE_ALIGN - 1);
+ gcmap.buf.logical = bvbuffdesc->virtaddr;
gcmap.pagesize = 0;
gcmap.pagearray = NULL;
+ gcmap.size = bvbuffdesc->length;
GCDBG(GCZONE_MAPPING, "new mapping (%s):\n",
- (batch == NULL) ? "explicit" : "implicit");
+ (batch == NULL) ? "explicit" : "implicit");
GCDBG(GCZONE_MAPPING, "specified virtaddr = 0x%08X\n",
- (unsigned int) bvbuffdesc->virtaddr);
+ (unsigned int) bvbuffdesc->virtaddr);
GCDBG(GCZONE_MAPPING, "aligned virtaddr = 0x%08X\n",
- (unsigned int) gcmap.buf.logical);
+ (unsigned int) gcmap.buf.logical);
+ GCDBG(GCZONE_MAPPING, "mapping size = %d\n",
+ gcmap.size);
}
- gcmap.size = offset + bvbuffdesc->length;
-
- GCDBG(GCZONE_MAPPING, "surface offset = %d\n", offset);
- GCDBG(GCZONE_MAPPING, "mapping size = %d\n", gcmap.size);
-
gc_map_wrapper(&gcmap);
if (gcmap.gcerror != GCERR_NONE) {
BVSETERROR(BVERR_OOM,
- "unable to allocate gccore memory");
+ "unable to allocate gccore memory");
goto fail;
}
@@ -808,9 +877,11 @@ static void unmap_implicit(struct gcbatch *batch)
* Batch memory manager.
*/
-static enum bverror allocate_batch(struct gcbatch **batch);
+static enum bverror allocate_batch(struct bvbltparams *bltparams,
+ struct gcbatch **batch);
static void free_batch(struct gcbatch *batch);
-static enum bverror append_buffer(struct gcbatch *batch);
+static enum bverror append_buffer(struct bvbltparams *bltparams,
+ struct gcbatch *batch);
static enum bverror do_end(struct bvbltparams *bltparams,
struct gcbatch *batch)
@@ -818,7 +889,8 @@ static enum bverror do_end(struct bvbltparams *bltparams,
return BVERR_NONE;
}
-static enum bverror allocate_batch(struct gcbatch **batch)
+static enum bverror allocate_batch(struct bvbltparams *bltparams,
+ struct gcbatch **batch)
{
enum bverror bverror;
struct gcbatch *temp;
@@ -831,19 +903,19 @@ static enum bverror allocate_batch(struct gcbatch **batch)
if (gccontext.vac_batches == NULL) {
temp = gcalloc(struct gcbatch, sizeof(struct gcbatch));
if (temp == NULL) {
- BVSETERROR(BVERR_OOM,
- "batch header allocation failed");
+ BVSETBLTERROR(BVERR_OOM,
+ "batch header allocation failed");
goto exit;
}
GCDBG(GCZONE_BATCH, "allocated new batch = 0x%08X\n",
- (unsigned int) temp);
+ (unsigned int) temp);
} else {
temp = (struct gcbatch *) gccontext.vac_batches;
gccontext.vac_batches = gccontext.vac_batches->next;
GCDBG(GCZONE_BATCH, "reusing batch = 0x%08X\n",
- (unsigned int) temp);
+ (unsigned int) temp);
}
memset(temp, 0, sizeof(struct gcbatch));
@@ -851,7 +923,7 @@ static enum bverror allocate_batch(struct gcbatch **batch)
temp->batchend = do_end;
INIT_LIST_HEAD(&temp->unmap);
- bverror = append_buffer(temp);
+ bverror = append_buffer(bltparams, temp);
if (bverror != BVERR_NONE) {
free_batch(temp);
goto exit;
@@ -866,7 +938,7 @@ exit:
GCUNLOCK(&gccontext.batchlock);
GCEXITARG(GCZONE_BATCH, "bv%s = %d\n",
- (bverror == BVERR_NONE) ? "result" : "error", bverror);
+ (bverror == BVERR_NONE) ? "result" : "error", bverror);
return bverror;
}
@@ -910,12 +982,14 @@ static void free_batch(struct gcbatch *batch)
GCEXIT(GCZONE_BATCH);
}
-static enum bverror append_buffer(struct gcbatch *batch)
+static enum bverror append_buffer(struct bvbltparams *bltparams,
+ struct gcbatch *batch)
{
enum bverror bverror;
struct gcbuffer *temp;
- GCENTERARG(GCZONE_BUFFER, "batch = 0x%08X\n", (unsigned int) batch);
+ GCENTERARG(GCZONE_BUFFER_ALLOC, "batch = 0x%08X\n",
+ (unsigned int) batch);
/* Lock access to buffer management. */
GCLOCK(&gccontext.bufferlock);
@@ -923,26 +997,26 @@ static enum bverror append_buffer(struct gcbatch *batch)
if (gccontext.vac_buffers == NULL) {
temp = gcalloc(struct gcbuffer, GC_BUFFER_SIZE);
if (temp == NULL) {
- BVSETERROR(BVERR_OOM,
- "command buffer allocation failed");
+ BVSETBLTERROR(BVERR_OOM,
+ "command buffer allocation failed");
goto exit;
}
- GCDBG(GCZONE_BUFFER, "allocated new buffer = 0x%08X\n",
- (unsigned int) temp);
+ GCDBG(GCZONE_BUFFER_ALLOC, "allocated new buffer = 0x%08X\n",
+ (unsigned int) temp);
} else {
temp = gccontext.vac_buffers;
gccontext.vac_buffers = temp->next;
- GCDBG(GCZONE_BUFFER, "reusing buffer = 0x%08X\n",
- (unsigned int) temp);
+ GCDBG(GCZONE_BUFFER_ALLOC, "reusing buffer = 0x%08X\n",
+ (unsigned int) temp);
}
memset(temp, 0, sizeof(struct gcbuffer));
temp->head =
temp->tail = (unsigned int *) (temp + 1);
temp->available = GC_BUFFER_SIZE - max(sizeof(struct gcbuffer),
- GC_BUFFER_RESERVE);
+ GC_BUFFER_RESERVE);
if (batch->bufhead == NULL)
batch->bufhead = temp;
@@ -950,8 +1024,8 @@ static enum bverror append_buffer(struct gcbatch *batch)
batch->buftail->next = temp;
batch->buftail = temp;
- GCDBG(GCZONE_BUFFER, "new buffer appended = 0x%08X\n",
- (unsigned int) temp);
+ GCDBG(GCZONE_BUFFER_ALLOC, "new buffer appended = 0x%08X\n",
+ (unsigned int) temp);
bverror = BVERR_NONE;
@@ -959,20 +1033,22 @@ exit:
/* Unlock access to buffer management. */
GCUNLOCK(&gccontext.bufferlock);
- GCEXITARG(GCZONE_BUFFER, "bv%s = %d\n",
- (bverror == BVERR_NONE) ? "result" : "error", bverror);
+ GCEXITARG(GCZONE_BUFFER_ALLOC, "bv%s = %d\n",
+ (bverror == BVERR_NONE) ? "result" : "error", bverror);
return bverror;
}
-static enum bverror add_fixup(struct gcbatch *batch, unsigned int *fixup,
+static enum bverror add_fixup(struct bvbltparams *bltparams,
+ struct gcbatch *batch,
+ unsigned int *fixup,
unsigned int surfoffset)
{
- enum bverror bverror;
+ enum bverror bverror = BVERR_NONE;
struct gcbuffer *buffer;
struct gcfixup *temp;
GCENTERARG(GCZONE_FIXUP, "batch = 0x%08X, fixup ptr = 0x%08X\n",
- (unsigned int) batch, (unsigned int) fixup);
+ (unsigned int) batch, (unsigned int) fixup);
/* Lock access to fixup management. */
GCLOCK(&gccontext.fixuplock);
@@ -981,26 +1057,26 @@ static enum bverror add_fixup(struct gcbatch *batch, unsigned int *fixup,
temp = buffer->fixuptail;
GCDBG(GCZONE_FIXUP, "buffer = 0x%08X, fixup struct = 0x%08X\n",
- (unsigned int) buffer, (unsigned int) temp);
+ (unsigned int) buffer, (unsigned int) temp);
if ((temp == NULL) || (temp->count == GC_FIXUP_MAX)) {
if (gccontext.vac_fixups == NULL) {
temp = gcalloc(struct gcfixup, sizeof(struct gcfixup));
if (temp == NULL) {
- BVSETERROR(BVERR_OOM,
- "fixup allocation failed");
+ BVSETBLTERROR(BVERR_OOM,
+ "fixup allocation failed");
goto exit;
}
GCDBG(GCZONE_FIXUP,
- "new fixup struct allocated = 0x%08X\n",
- (unsigned int) temp);
+ "new fixup struct allocated = 0x%08X\n",
+ (unsigned int) temp);
} else {
temp = gccontext.vac_fixups;
gccontext.vac_fixups = temp->next;
GCDBG(GCZONE_FIXUP, "fixup struct reused = 0x%08X\n",
- (unsigned int) temp);
+ (unsigned int) temp);
}
temp->next = NULL;
@@ -1013,7 +1089,7 @@ static enum bverror add_fixup(struct gcbatch *batch, unsigned int *fixup,
buffer->fixuptail = temp;
GCDBG(GCZONE_FIXUP, "new fixup struct allocated = 0x%08X\n",
- (unsigned int) temp);
+ (unsigned int) temp);
} else {
GCDBG(GCZONE_FIXUP, "fixups accumulated = %d\n", temp->count);
@@ -1026,35 +1102,34 @@ static enum bverror add_fixup(struct gcbatch *batch, unsigned int *fixup,
GCDBG(GCZONE_FIXUP, "fixup offset = 0x%08X\n", fixup - buffer->head);
GCDBG(GCZONE_FIXUP, "surface offset = 0x%08X\n", surfoffset);
- bverror = BVERR_NONE;
-
exit:
/* Unlock access to fixup management. */
GCUNLOCK(&gccontext.fixuplock);
GCEXITARG(GCZONE_FIXUP, "bv%s = %d\n",
- (bverror == BVERR_NONE) ? "result" : "error", bverror);
+ (bverror == BVERR_NONE) ? "result" : "error", bverror);
return bverror;
}
-static enum bverror claim_buffer(struct gcbatch *batch,
+static enum bverror claim_buffer(struct bvbltparams *bltparams,
+ struct gcbatch *batch,
unsigned int size,
void **buffer)
{
enum bverror bverror;
struct gcbuffer *curbuf;
- GCENTERARG(GCZONE_BUFFER, "batch = 0x%08X, size = %d\n",
- (unsigned int) batch, size);
+ GCENTERARG(GCZONE_BUFFER_ALLOC, "batch = 0x%08X, size = %d\n",
+ (unsigned int) batch, size);
/* Get the current command buffer. */
curbuf = batch->buftail;
- GCDBG(GCZONE_BUFFER, "buffer = 0x%08X, available = %d\n",
- (unsigned int) curbuf, curbuf->available);
+ GCDBG(GCZONE_BUFFER_ALLOC, "buffer = 0x%08X, available = %d\n",
+ (unsigned int) curbuf, curbuf->available);
if (curbuf->available < size) {
- bverror = append_buffer(batch);
+ bverror = append_buffer(bltparams, batch);
if (bverror != BVERR_NONE)
goto exit;
@@ -1062,9 +1137,9 @@ static enum bverror claim_buffer(struct gcbatch *batch,
}
if (curbuf->available < size) {
- GCDBG(GCZONE_BUFFER, "requested size is too large.\n");
- BVSETERROR(BVERR_OOM,
- "command buffer allocation failed");
+ GCDBG(GCZONE_BUFFER_ALLOC, "requested size is too large.\n");
+ BVSETBLTERROR(BVERR_OOM,
+ "command buffer allocation failed");
goto exit;
}
@@ -1075,8 +1150,8 @@ static enum bverror claim_buffer(struct gcbatch *batch,
bverror = BVERR_NONE;
exit:
- GCEXITARG(GCZONE_BUFFER, "bv%s = %d\n",
- (bverror == BVERR_NONE) ? "result" : "error", bverror);
+ GCEXITARG(GCZONE_BUFFER_ALLOC, "bv%s = %d\n",
+ (bverror == BVERR_NONE) ? "result" : "error", bverror);
return bverror;
}
@@ -1109,6 +1184,11 @@ exit:
#define OCDFMTDEF_BITS16 (2 << OCDFMTDEF_BITS_SHIFT)
#define OCDFMTDEF_BITS24 (3 << OCDFMTDEF_BITS_SHIFT)
+enum bvformattype {
+ BVFMT_RGB,
+ BVFMT_YUV
+};
+
struct bvcomponent {
unsigned int shift;
unsigned int size;
@@ -1123,6 +1203,7 @@ struct bvcsrgb {
};
struct bvformatxlate {
+ enum bvformattype type;
unsigned bitspp;
unsigned format;
unsigned swizzle;
@@ -1131,6 +1212,7 @@ struct bvformatxlate {
#define BVFORMATRGBA(BPP, Format, Swizzle, R, G, B, A) \
{ \
+ BVFMT_RGB, \
BPP, \
GCREG_DE_FORMAT_ ## Format, \
GCREG_DE_SWIZZLE_ ## Swizzle, \
@@ -1153,9 +1235,10 @@ struct bvformatxlate {
BVCOMP(Shift, Size)
#define BVFORMATINVALID \
- { 0, 0, 0, { { 0, 0, 0 }, { 0, 0, 0 }, { 0, 0, 0 }, { 0, 0, 0 } } }
+ { 0, 0, 0, 0, { { 0, 0, 0 }, { 0, 0, 0 }, { 0, 0, 0 }, { 0, 0, 0 } } }
static struct bvformatxlate g_format_nv12 = {
+ .type = BVFMT_YUV,
.bitspp = 8,
.format = GCREG_DE_FORMAT_NV12,
};
@@ -1504,36 +1587,6 @@ static unsigned int getinternalcolor(void *ptr, struct bvformatxlate *format)
* Alpha blending parser.
*/
-struct gcblendconfig {
- unsigned char factor_mode;
- unsigned char color_reverse;
-
- unsigned char src1used;
- unsigned char src2used;
-};
-
-struct bvblendxlate {
- unsigned char match1;
- unsigned char match2;
-
- struct gcblendconfig k1;
- struct gcblendconfig k2;
-};
-
-struct gcalpha {
- unsigned int src_global_color;
- unsigned int dst_global_color;
-
- unsigned char src_global_alpha_mode;
- unsigned char dst_global_alpha_mode;
-
- struct gcblendconfig *k1;
- struct gcblendconfig *k2;
-
- unsigned int src1used;
- unsigned int src2used;
-};
-
#define BVBLENDMATCH(Mode, Inverse, Normal) \
( \
BVBLENDDEF_ ## Mode | \
@@ -2192,7 +2245,7 @@ static enum bverror parse_blend(struct bvbltparams *bltparams,
if (((k3 != k1_xlate->match1) && (k3 != k1_xlate->match2)) ||
((k4 != k2_xlate->match1) && (k4 != k2_xlate->match2))) {
BVSETBLTERROR(BVERR_BLEND,
- "not supported coefficient combination");
+ "not supported coefficient combination");
goto exit;
}
@@ -2210,6 +2263,64 @@ exit:
/*******************************************************************************
+ * Rotation and mirror.
+ */
+
+#define BVFLAG_FLIP_MASK 0x00000003
+
+#define BVFLAG_FLIP_SRC1_SHIFT 14
+#define BVFLAG_FLIP_SRC2_SHIFT 16
+#define BVFLAG_FLIP_MASK_SHIFT 18
+
+#define GCREG_MIRROR_NONE 0x0
+#define GCREG_MIRROR_X 0x1
+#define GCREG_MIRROR_Y 0x2
+#define GCREG_MIRROR_XY 0x3
+
+#define GCREG_ROT_ANGLE_ROT0 0x0
+#define GCREG_ROT_ANGLE_ROT90 0x4
+#define GCREG_ROT_ANGLE_ROT180 0x5
+#define GCREG_ROT_ANGLE_ROT270 0x6
+
+#define ROT_ANGLE_INVALID -1
+#define ROT_ANGLE_0 0
+#define ROT_ANGLE_90 1
+#define ROT_ANGLE_180 2
+#define ROT_ANGLE_270 3
+
+/* NOTE: BLTsville rotation is defined conunter clock wise. */
+static const unsigned int rotencoding[] = {
+ GCREG_ROT_ANGLE_ROT0, /* ROT_ANGLE_0 */
+ GCREG_ROT_ANGLE_ROT270, /* ROT_ANGLE_90 */
+ GCREG_ROT_ANGLE_ROT180, /* ROT_ANGLE_180 */
+ GCREG_ROT_ANGLE_ROT90 /* ROT_ANGLE_270 */
+};
+
+static inline int get_angle(int orientation)
+{
+ int angle;
+
+ /* Normalize the angle. */
+ angle = orientation % 360;
+
+ /* Flip to positive. */
+ if (angle < 0)
+ angle = 360 + angle;
+
+ /* Translate the angle. */
+ switch (angle) {
+ case 0: return ROT_ANGLE_0;
+ case 90: return ROT_ANGLE_90;
+ case 180: return ROT_ANGLE_180;
+ case 270: return ROT_ANGLE_270;
+ }
+
+ /* Not supported angle. */
+ return ROT_ANGLE_INVALID;
+}
+
+
+/*******************************************************************************
* Surface compare and validation.
*/
@@ -2233,7 +2344,7 @@ static inline bool equal_rects(struct bvrect *rect1, struct bvrect *rect2)
/* The function verifies whether the two buffer descriptors and rectangles
define the same physical area. */
static bool same_phys_area(struct bvbuffdesc *surf1, struct bvrect *rect1,
- struct bvbuffdesc *surf2, struct bvrect *rect2)
+ struct bvbuffdesc *surf2, struct bvrect *rect2)
{
struct bvphysdesc *physdesc1;
struct bvphysdesc *physdesc2;
@@ -2305,53 +2416,32 @@ static int verify_surface(unsigned int tile,
union bvinbuff *surf,
struct bvsurfgeom *geom)
{
- int ret;
-
- GCENTER(GCZONE_SURF);
-
if (tile) {
- if (surf->tileparams == NULL) {
- ret = GCBVERR_TILE;
- goto exit;
- }
+ if (surf->tileparams == NULL)
+ return GCBVERR_TILE;
if (surf->tileparams->structsize <
- STRUCTSIZE(surf->tileparams, srcheight)) {
- ret = GCBVERR_TILE_VERS;
- goto exit;
- }
+ STRUCTSIZE(surf->tileparams, srcheight))
+ return GCBVERR_TILE_VERS;
/* FIXME/TODO */
- ret = GCBVERR_TILE;
- goto exit;
+ return GCBVERR_TILE;
} else {
- if (surf->desc == NULL) {
- ret = GCBVERR_DESC;
- goto exit;
- }
+ if (surf->desc == NULL)
+ return GCBVERR_DESC;
- if (surf->desc->structsize < STRUCTSIZE(surf->desc, map)) {
- ret = GCBVERR_DESC_VERS;
- goto exit;
- }
+ if (surf->desc->structsize < STRUCTSIZE(surf->desc, map))
+ return GCBVERR_DESC_VERS;
}
- if (geom == NULL) {
- ret = GCBVERR_GEOM;
- goto exit;
- }
+ if (geom == NULL)
+ return GCBVERR_GEOM;
- if (geom->structsize < STRUCTSIZE(geom, palette)) {
- ret = GCBVERR_GEOM_VERS;
- goto exit;
- }
+ if (geom->structsize < STRUCTSIZE(geom, palette))
+ return GCBVERR_GEOM_VERS;
/* Validation successful. */
- ret = -1;
-
-exit:
- GCEXITARG(GCZONE_SURF, "ret = %d\n", ret);
- return ret;
+ return -1;
}
#if GCDEBUG_ENABLE
@@ -2395,244 +2485,425 @@ static void verify_batch(unsigned int changeflags,
# define VERIFYBATCH(...)
#endif
+static inline int get_pixeloffset(struct bvbuffdesc *bvbuffdesc,
+ struct bvformatxlate *format,
+ int offset)
+{
+ unsigned int alignment;
+ int byteoffset;
+ unsigned int alignedoffset;
+ int pixeloffset;
+
+ alignment = (format->type == BVFMT_YUV)
+ ? (64 - 1)
+ : (16 - 1);
+
+ /* Determine offset in bytes from the base modified by the
+ * given offset. */
+ if (bvbuffdesc->auxtype == BVAT_PHYSDESC) {
+ struct bvphysdesc *bvphysdesc;
+ bvphysdesc = (struct bvphysdesc *) bvbuffdesc->auxptr;
+ byteoffset = bvphysdesc->pageoffset + offset;
+ } else {
+ byteoffset = (unsigned int) bvbuffdesc->virtaddr + offset;
+ }
-/*******************************************************************************
- * Primitive renderers.
- */
+ /* Compute the aligned offset. */
+ alignedoffset = byteoffset & alignment;
-static enum bverror set_dst(struct bvbltparams *bltparams,
- struct gcbatch *batch,
- struct bvbuffmap *dstmap)
+ /* Convert to pixels. */
+ pixeloffset = alignedoffset * 8 / format->bitspp;
+ return -pixeloffset;
+}
+
+static enum bverror parse_destination(struct bvbltparams *bltparams,
+ struct gcbatch *batch)
{
enum bverror bverror = BVERR_NONE;
- struct bvsurfgeom *dstgeom = bltparams->dstgeom;
- struct gcmodst *gcmodst;
- unsigned int dstoffset;
GCENTER(GCZONE_DEST);
- /* Did destination surface change? */
- if ((batch->batchflags & BVBATCH_DST) == 0)
- goto exit;
+ /* Did clipping/destination rects change? */
+ if ((batch->batchflags & (BVBATCH_CLIPRECT |
+ BVBATCH_DESTRECT)) != 0) {
+ struct bvrect *dstrect;
+ struct bvrect *cliprect;
+ int destleft, desttop, destright, destbottom;
+ int clipleft, cliptop, clipright, clipbottom;
+
+ /* Make shortcuts to the destination objects. */
+ dstrect = &bltparams->dstrect;
+ cliprect = &bltparams->cliprect;
+
+ /* Determine destination rectangle. */
+ destleft = dstrect->left;
+ desttop = dstrect->top;
+ destright = destleft + dstrect->width;
+ destbottom = desttop + dstrect->height;
+
+ GCDBG(GCZONE_DEST, "destination rectangle:\n");
+ GCDBG(GCZONE_DEST, " dstrect = (%d,%d)-(%d,%d), %dx%d\n",
+ destleft, desttop, destright, destbottom,
+ dstrect->width, dstrect->height);
+
+ /* Determine clipping. */
+ if ((bltparams->flags & BVFLAG_CLIP) == BVFLAG_CLIP) {
+ clipleft = cliprect->left;
+ cliptop = cliprect->top;
+ clipright = clipleft + cliprect->width;
+ clipbottom = cliptop + cliprect->height;
+
+ if ((clipleft < GC_CLIP_RESET_LEFT) ||
+ (cliptop < GC_CLIP_RESET_TOP) ||
+ (clipright > GC_CLIP_RESET_RIGHT) ||
+ (clipbottom > GC_CLIP_RESET_BOTTOM) ||
+ (clipright < clipleft) ||
+ (clipbottom < cliptop)) {
+ BVSETBLTERROR(BVERR_CLIP_RECT,
+ "invalid clipping rectangle");
+ goto exit;
+ }
- GCDBG(GCZONE_DEST, "destination surface changed.\n");
+ GCDBG(GCZONE_DEST,
+ " cliprect = (%d,%d)-(%d,%d), %dx%d\n",
+ clipleft, cliptop, clipright, clipbottom,
+ cliprect->width, cliprect->height);
+ } else {
+ clipleft = GC_CLIP_RESET_LEFT;
+ cliptop = GC_CLIP_RESET_TOP;
+ clipright = GC_CLIP_RESET_RIGHT;
+ clipbottom = GC_CLIP_RESET_BOTTOM;
+ }
- /* Parse the destination format. */
- GCDBG(GCZONE_FORMAT, "parsing the destination format.\n");
- if (!parse_format(dstgeom->format, &batch->dstformat)) {
- BVSETBLTERRORARG(BVERR_DSTGEOM_FORMAT,
- "invalid destination format (%d)",
- dstgeom->format);
- goto exit;
- }
+ /* Compute clipping deltas and the adjusted destination rect. */
+ if (clipleft <= destleft) {
+ batch->deltaleft = 0;
+ batch->clippedleft = destleft;
+ } else {
+ batch->deltaleft = clipleft - destleft;
+ batch->clippedleft = clipleft;
+ }
- /* Validate geometry. */
- if (!valid_geom(bltparams->dstdesc, bltparams->dstgeom,
- batch->dstformat)) {
- BVSETBLTERROR(BVERR_DSTGEOM,
- "destination geom exceeds surface size");
- goto exit;
- }
+ if (cliptop <= desttop) {
+ batch->deltatop = 0;
+ batch->clippedtop = desttop;
+ } else {
+ batch->deltatop = cliptop - desttop;
+ batch->clippedtop = cliptop;
+ }
- /* Compute the destination offset in pixels needed to compensate
- for the surface base address misalignment if any. */
- dstoffset = (((unsigned int) bltparams->dstdesc->virtaddr
- & (GC_BASE_ALIGN - 1)) * 8)
- / batch->dstformat->bitspp;
-
- /* Have to reset clipping with the new offset. */
- if (dstoffset != batch->dstoffset) {
- batch->dstoffset = dstoffset;
- batch->batchflags |= BVBATCH_CLIPRECT_ORIGIN;
- }
+ if (clipright >= destright) {
+ batch->deltaright = 0;
+ batch->clippedright = destright;
+ } else {
+ batch->deltaright = clipright - destright;
+ batch->clippedright = clipright;
+ }
- /* Allocate command buffer. */
- bverror = claim_buffer(batch, sizeof(struct gcmodst),
- (void **) &gcmodst);
- if (bverror != BVERR_NONE) {
- BVSETBLTERROR(BVERR_OOM,
- "failed to allocate command buffer");
- goto exit;
+ if (clipbottom >= destbottom) {
+ batch->deltabottom = 0;
+ batch->clippedbottom = destbottom;
+ } else {
+ batch->deltabottom = clipbottom - destbottom;
+ batch->clippedbottom = clipbottom;
+ }
+
+ /* Validate the rectangle. */
+ if ((batch->clippedright > (int) bltparams->dstgeom->width) ||
+ (batch->clippedbottom > (int) bltparams->dstgeom->height)) {
+ BVSETBLTERROR(BVERR_DSTRECT,
+ "destination rect exceeds surface size");
+ goto exit;
+ }
+
+ GCDBG(GCZONE_DEST,
+ " clipped dstrect = (%d,%d)-(%d,%d), %dx%d\n",
+ batch->clippedleft, batch->clippedtop,
+ batch->clippedright, batch->clippedbottom,
+ batch->clippedright - batch->clippedleft,
+ batch->clippedbottom - batch->clippedtop);
+ GCDBG(GCZONE_DEST,
+ " clipping delta = (%d,%d)-(%d,%d)\n",
+ batch->deltaleft, batch->deltatop,
+ batch->deltaright, batch->deltabottom);
}
- /* Add the address fixup. */
- add_fixup(batch, &gcmodst->address, 0);
+ /* Did the destination surface change? */
+ if ((batch->batchflags & BVBATCH_DST) != 0) {
+ struct bvbuffdesc *dstdesc;
+ struct bvsurfgeom *dstgeom;
+
+ /* Make shortcuts to the destination objects. */
+ dstdesc = bltparams->dstdesc;
+ dstgeom = bltparams->dstgeom;
+
+ /* Parse the destination format. */
+ GCDBG(GCZONE_FORMAT, "parsing destination format.\n");
+ if (!parse_format(dstgeom->format, &batch->dstformat)) {
+ BVSETBLTERROR(BVERR_DSTGEOM_FORMAT,
+ "invalid destination format (%d)",
+ dstgeom->format);
+ goto exit;
+ }
- /* Set surface parameters. */
- gcmodst->address_ldst = gcmodst_address_ldst;
- gcmodst->address = GET_MAP_HANDLE(dstmap);
- gcmodst->stride = dstgeom->virtstride;
-
- /* Set surface width and height. */
- gcmodst->rotation.raw = 0;
- gcmodst->rotation.reg.surf_width = dstgeom->width + dstoffset;
- gcmodst->rotationheight_ldst = gcmodst_rotationheight_ldst;
- gcmodst->rotationheight.raw = 0;
- gcmodst->rotationheight.reg.height = dstgeom->height;
-
- GCDBG(GCZONE_DEST, "destination surface:\n");
- GCDBG(GCZONE_DEST, " dstvirtaddr = 0x%08X\n",
- (unsigned int) bltparams->dstdesc->virtaddr);
- GCDBG(GCZONE_DEST, " dstaddr = 0x%08X\n",
- (unsigned int) GET_MAP_HANDLE(dstmap));
- GCDBG(GCZONE_DEST, " dstoffset = %d\n", dstoffset);
- GCDBG(GCZONE_DEST, " dstsurf = %dx%d, stride = %ld\n",
- dstgeom->width, dstgeom->height, dstgeom->virtstride);
+ /* Validate geometry. */
+ if (!valid_geom(dstdesc, dstgeom, batch->dstformat)) {
+ BVSETBLTERROR(BVERR_DSTGEOM,
+ "destination geom exceeds surface size");
+ goto exit;
+ }
+
+ /* Parse orientation. */
+ batch->dstangle = get_angle(dstgeom->orientation);
+ if (batch->dstangle == ROT_ANGLE_INVALID) {
+ BVSETBLTERROR(BVERR_DSTGEOM,
+ "unsupported destination orientation %d.",
+ dstgeom->orientation);
+ }
+
+ /* Compute the destination offset in pixels needed to compensate
+ * for the surface base address misalignment if any. */
+ batch->dstalign = get_pixeloffset(dstdesc, batch->dstformat, 0);
+
+ switch (batch->dstangle) {
+ case ROT_ANGLE_0:
+ /* Determine the physical size. */
+ batch->dstphyswidth = dstgeom->width - batch->dstalign;
+ batch->dstphysheight = dstgeom->height;
+
+ /* Determine geometry size. */
+ batch->dstwidth = dstgeom->width - batch->dstalign;
+ batch->dstheight = dstgeom->height;
+
+ /* Determine the origin offset. */
+ batch->dstoffsetX = -batch->dstalign;
+ batch->dstoffsetY = 0;
+ break;
+
+ case ROT_ANGLE_90:
+ /* Determine the physical size. */
+ batch->dstphyswidth = dstgeom->height - batch->dstalign;
+ batch->dstphysheight = dstgeom->width;
+
+ /* Determine geometry size. */
+ batch->dstwidth = dstgeom->width;
+ batch->dstheight = dstgeom->height - batch->dstalign;
+
+ /* Determine the origin offset. */
+ batch->dstoffsetX = 0;
+ batch->dstoffsetY = -batch->dstalign;
+ break;
+
+ case ROT_ANGLE_180:
+ /* Determine the physical size. */
+ batch->dstphyswidth = dstgeom->width - batch->dstalign;
+ batch->dstphysheight = dstgeom->height;
+
+ /* Determine geometry size. */
+ batch->dstwidth = dstgeom->width - batch->dstalign;
+ batch->dstheight = dstgeom->height;
+
+ /* Determine the origin offset. */
+ batch->dstoffsetX = 0;
+ batch->dstoffsetY = 0;
+ break;
+
+ case ROT_ANGLE_270:
+ /* Determine the physical size. */
+ batch->dstphyswidth = dstgeom->height - batch->dstalign;
+ batch->dstphysheight = dstgeom->width;
+
+ /* Determine geometry size. */
+ batch->dstwidth = dstgeom->width;
+ batch->dstheight = dstgeom->height - batch->dstalign;
+
+ /* Determine the origin offset. */
+ batch->dstoffsetX = 0;
+ batch->dstoffsetY = 0;
+ break;
+ }
+
+ GCDBG(GCZONE_DEST, "destination surface:\n");
+ GCDBG(GCZONE_SURF, " rotation %d degrees.\n",
+ batch->dstangle * 90);
+
+ if (dstdesc->auxtype == BVAT_PHYSDESC) {
+ struct bvphysdesc *bvphysdesc;
+ bvphysdesc = (struct bvphysdesc *) dstdesc->auxptr;
+ GCDBG(GCZONE_DEST, " page offset = 0x%08X\n",
+ bvphysdesc->pageoffset);
+ } else {
+ GCDBG(GCZONE_DEST, " virtual address = 0x%08X\n",
+ (unsigned int) dstdesc->virtaddr);
+ }
+
+ GCDBG(GCZONE_DEST, " stride = %ld\n",
+ dstgeom->virtstride);
+ GCDBG(GCZONE_DEST, " geometry size = %dx%d\n",
+ dstgeom->width, dstgeom->height);
+ GCDBG(GCZONE_DEST, " aligned geometry size = %dx%d\n",
+ batch->dstwidth, batch->dstheight);
+ GCDBG(GCZONE_DEST, " aligned physical size = %dx%d\n",
+ batch->dstphyswidth, batch->dstphysheight);
+ GCDBG(GCZONE_DEST, " origin offset (pixels) = %d,%d\n",
+ batch->dstoffsetX, batch->dstoffsetY);
+ GCDBG(GCZONE_DEST, " surface offset (pixels) = %d,0\n",
+ batch->dstalign);
+ }
exit:
GCEXITARG(GCZONE_DEST, "bv%s = %d\n",
- (bverror == BVERR_NONE) ? "result" : "error", bverror);
+ (bverror == BVERR_NONE) ? "result" : "error", bverror);
return bverror;
}
-static enum bverror set_clip(struct bvbltparams *bltparams,
- struct gcbatch *batch)
+static enum bverror parse_source(struct bvbltparams *bltparams,
+ struct gcbatch *batch,
+ struct srcinfo *srcinfo)
{
enum bverror bverror = BVERR_NONE;
- struct bvrect *dstrect = &bltparams->dstrect;
- struct bvrect *cliprect = &bltparams->cliprect;
- int destleft, desttop, destright, destbottom;
- int clipleft, cliptop, clipright, clipbottom;
- int clippedleft, clippedtop, clippedright, clippedbottom;
- int clipleftadj, cliprightadj;
- struct gcmoclip *gcmoclip;
+ struct bvbuffdesc *srcdesc;
+ struct bvsurfgeom *srcgeom;
+ struct bvrect *srcrect;
- GCENTER(GCZONE_CLIP);
+ /* Make shortcuts to the source objects. */
+ srcdesc = srcinfo->buf.desc;
+ srcgeom = srcinfo->geom;
+ srcrect = srcinfo->rect;
- /* Did clipping/destination rects change? */
- if ((batch->batchflags & (BVBATCH_CLIPRECT_ORIGIN |
- BVBATCH_CLIPRECT_SIZE |
- BVBATCH_DSTRECT_ORIGIN |
- BVBATCH_DSTRECT_SIZE)) == 0)
+ /* Parse the source format. */
+ GCDBG(GCZONE_FORMAT, "parsing source%d format.\n",
+ srcinfo->index + 1);
+ if (!parse_format(srcgeom->format, &srcinfo->format)) {
+ BVSETBLTERROR((srcinfo->index == 0)
+ ? BVERR_SRC1GEOM_FORMAT
+ : BVERR_SRC2GEOM_FORMAT,
+ "invalid source #%d format (%d)",
+ srcinfo->index + 1,
+ srcgeom->format);
goto exit;
-
- GCDBG(GCZONE_CLIP, "destination clipping changed.\n");
-
- /* Determine destination rectangle. */
- destleft = dstrect->left;
- desttop = dstrect->top;
- destright = dstrect->left + dstrect->width;
- destbottom = dstrect->top + dstrect->height;
-
- /* Determine clipping. */
- if ((bltparams->flags & BVFLAG_CLIP) == BVFLAG_CLIP) {
- clipleft = cliprect->left;
- cliptop = cliprect->top;
- clipright = cliprect->left + cliprect->width;
- clipbottom = cliprect->top + cliprect->height;
-
- if ((clipleft < GC_CLIP_RESET_LEFT) ||
- (cliptop < GC_CLIP_RESET_TOP) ||
- (clipright > GC_CLIP_RESET_RIGHT) ||
- (clipbottom > GC_CLIP_RESET_BOTTOM) ||
- (clipright < clipleft) || (clipbottom < cliptop)) {
- BVSETBLTERROR(BVERR_CLIP_RECT,
- "invalid clipping rectangle");
- goto exit;
- }
- } else {
- clipleft = GC_CLIP_RESET_LEFT;
- cliptop = GC_CLIP_RESET_TOP;
- clipright = GC_CLIP_RESET_RIGHT;
- clipbottom = GC_CLIP_RESET_BOTTOM;
}
- /* Compute clipping deltas and the adjusted destination rect. */
- if (clipleft <= destleft) {
- batch->deltaleft = 0;
- clippedleft = destleft;
- } else {
- batch->deltaleft = clipleft - destleft;
- clippedleft = destleft + batch->deltaleft;
+ /* Validate source geometry. */
+ if (!valid_geom(srcdesc, srcgeom, srcinfo->format)) {
+ BVSETBLTERROR((srcinfo->index == 0)
+ ? BVERR_SRC1GEOM
+ : BVERR_SRC2GEOM,
+ "source%d geom exceeds surface size",
+ srcinfo->index + 1);
+ goto exit;
}
- if (cliptop <= desttop) {
- batch->deltatop = 0;
- clippedtop = desttop;
- } else {
- batch->deltatop = cliptop - desttop;
- clippedtop = desttop + batch->deltatop;
+ /* Parse orientation. */
+ srcinfo->angle = get_angle(srcgeom->orientation);
+ if (srcinfo->angle == ROT_ANGLE_INVALID) {
+ BVSETBLTERROR((srcinfo->index == 0)
+ ? BVERR_SRC1GEOM
+ : BVERR_SRC2GEOM,
+ "unsupported source%d orientation %d.",
+ srcinfo->index + 1,
+ srcgeom->orientation);
}
- if (clipright >= destright) {
- batch->deltaright = 0;
- clippedright = destright;
+ /* Determine source mirror. */
+ srcinfo->mirror = (srcinfo->index == 0)
+ ? (bltparams->flags >> BVFLAG_FLIP_SRC1_SHIFT)
+ & BVFLAG_FLIP_MASK
+ : (bltparams->flags >> BVFLAG_FLIP_SRC2_SHIFT)
+ & BVFLAG_FLIP_MASK;
+
+ GCDBG(GCZONE_SURF, "source surface %d:\n", srcinfo->index + 1);
+ GCDBG(GCZONE_SURF, " rotation %d degrees.\n", srcinfo->angle * 90);
+
+ if (srcdesc->auxtype == BVAT_PHYSDESC) {
+ struct bvphysdesc *bvphysdesc;
+ bvphysdesc = (struct bvphysdesc *) srcdesc->auxptr;
+ GCDBG(GCZONE_DEST, " page offset = 0x%08X\n",
+ bvphysdesc->pageoffset);
} else {
- batch->deltaright = clipright - destright;
- clippedright = destright + batch->deltaright;
+ GCDBG(GCZONE_DEST, " virtual address = 0x%08X\n",
+ (unsigned int) srcdesc->virtaddr);
}
- if (clipbottom >= destbottom) {
- batch->deltabottom = 0;
- clippedbottom = destbottom;
- } else {
- batch->deltabottom = clipbottom - destbottom;
- clippedbottom = destbottom + batch->deltabottom;
- }
+ GCDBG(GCZONE_SURF, " stride = %ld\n",
+ srcgeom->virtstride);
+ GCDBG(GCZONE_SURF, " geometry size = %dx%d\n",
+ srcgeom->width, srcgeom->height);
+ GCDBG(GCZONE_SURF, " rect = (%d,%d)-(%d,%d), %dx%d\n",
+ srcrect->left, srcrect->top,
+ srcrect->left + srcrect->width,
+ srcrect->top + srcrect->height,
+ srcrect->width, srcrect->height);
+ GCDBG(GCZONE_SURF, " mirror = %d\n", srcinfo->mirror);
- /* Validate the rectangle. */
- if ((clippedright > (int) bltparams->dstgeom->width) ||
- (clippedbottom > (int) bltparams->dstgeom->height)) {
- BVSETBLTERROR(BVERR_DSTRECT,
- "destination rect exceeds surface size");
- goto exit;
- }
+exit:
+ return bverror;
+}
- /* Set the destination rectangle. */
- batch->dstleft = clippedleft;
- batch->dsttop = clippedtop;
- batch->dstright = clippedright;
- batch->dstbottom = clippedbottom;
- /* Allocate command buffer. */
- bverror = claim_buffer(batch, sizeof(struct gcmoclip),
- (void **) &gcmoclip);
- if (bverror != BVERR_NONE) {
- BVSETBLTERROR(BVERR_OOM,
- "failed to allocate command buffer");
- goto exit;
- }
+/*******************************************************************************
+ * Primitive renderers.
+ */
+
+static enum bverror set_dst(struct bvbltparams *bltparams,
+ struct gcbatch *batch,
+ struct bvbuffmap *dstmap)
+{
+ enum bverror bverror = BVERR_NONE;
+ struct gcmoclip *gcmoclip;
+ struct gcmodst *gcmodst;
+
+ GCENTER(GCZONE_DEST);
+
+ /* Did destination surface change? */
+ if ((batch->batchflags & BVBATCH_DST) != 0) {
+ /* Perform initialization. */
+ if (gccontext.initdone == 0) {
+ /* Allocate command buffer. */
+ bverror = claim_buffer(bltparams, batch,
+ sizeof(struct gcmoclip),
+ (void **) &gcmoclip);
+ if (bverror != BVERR_NONE)
+ goto exit;
- /* Compute adjusted clipping. */
- clipleftadj = min(clipleft + (int) batch->dstoffset,
- (int) GC_CLIP_RESET_RIGHT);
- cliprightadj = min(clipright + (int) batch->dstoffset,
- (int) GC_CLIP_RESET_RIGHT);
-
- /* Set clipping. */
- gcmoclip->lt_ldst = gcmoclip_lt_ldst;
- gcmoclip->lt.raw = 0;
- gcmoclip->lt.reg.left = clipleftadj;
- gcmoclip->lt.reg.top = cliptop;
- gcmoclip->rb.raw = 0;
- gcmoclip->rb.reg.right = cliprightadj;
- gcmoclip->rb.reg.bottom = clipbottom;
-
- GCDBG(GCZONE_CLIP, "destination rectangle:\n");
- GCDBG(GCZONE_CLIP, " dstrect = (%d,%d)-(%d,%d), %dx%d\n",
- destleft, desttop, destright, destbottom,
- destright - destleft, destbottom - desttop);
-
- GCDBG(GCZONE_CLIP, " clipping rect = (%d,%d)-(%d,%d), %dx%d\n",
- clipleft, cliptop, clipright, clipbottom,
- clipright - clipleft, clipbottom - cliptop);
-
- GCDBG(GCZONE_CLIP,
- " adjusted clipping rect = (%d,%d)-(%d,%d), %dx%d\n",
- clipleftadj, cliptop, cliprightadj, clipbottom,
- cliprightadj - clipleftadj, clipbottom - cliptop);
-
- GCDBG(GCZONE_CLIP, " clipping delta = (%d,%d)-(%d,%d)\n",
- batch->deltaleft, batch->deltatop,
- batch->deltaright, batch->deltabottom);
-
- GCDBG(GCZONE_CLIP, " clipped dstrect = (%d,%d)-(%d,%d), %dx%d\n",
- clippedleft, clippedtop, clippedright, clippedbottom,
- clippedright - clippedleft, clippedbottom - clippedtop);
+ /* Set clipping. */
+ gcmoclip->lt_ldst = gcmoclip_lt_ldst;
+ gcmoclip->lt.raw = 0;
+ gcmoclip->rb.raw = 0;
+ gcmoclip->rb.reg.right = GC_CLIP_RESET_RIGHT;
+ gcmoclip->rb.reg.bottom = GC_CLIP_RESET_BOTTOM;
+
+ /* Mark as initialized. */
+ gccontext.initdone = 1;
+ }
+
+ /* Allocate command buffer. */
+ bverror = claim_buffer(bltparams, batch,
+ sizeof(struct gcmodst),
+ (void **) &gcmodst);
+ if (bverror != BVERR_NONE)
+ goto exit;
+
+ /* Add the address fixup. */
+ add_fixup(bltparams, batch, &gcmodst->address,
+ batch->dstbyteshift);
+
+ /* Set surface parameters. */
+ gcmodst->address_ldst = gcmodst_address_ldst;
+ gcmodst->address = GET_MAP_HANDLE(dstmap);
+ gcmodst->stride = bltparams->dstgeom->virtstride;
+
+ /* Set surface width and height. */
+ gcmodst->rotation.raw = 0;
+ gcmodst->rotation.reg.surf_width = batch->physwidth;
+ gcmodst->rotationheight_ldst = gcmodst_rotationheight_ldst;
+ gcmodst->rotationheight.raw = 0;
+ gcmodst->rotationheight.reg.height = batch->physheight;
+ }
exit:
- GCEXITARG(GCZONE_CLIP, "bv%s = %d\n",
- (bverror == BVERR_NONE) ? "result" : "error", bverror);
+ GCEXITARG(GCZONE_DEST, "bv%s = %d\n",
+ (bverror == BVERR_NONE) ? "result" : "error", bverror);
return bverror;
}
@@ -2641,10 +2912,9 @@ static enum bverror do_fill(struct bvbltparams *bltparams,
struct srcinfo *srcinfo)
{
enum bverror bverror;
+ int dstbyteshift;
struct gcmofill *gcmofill;
unsigned char *fillcolorptr;
- struct bvformatxlate *srcformat;
- struct bvsurfgeom *dstgeom = bltparams->dstgeom;
struct bvbuffmap *dstmap = NULL;
GCENTER(GCZONE_DO_FILL);
@@ -2654,6 +2924,22 @@ static enum bverror do_fill(struct bvbltparams *bltparams,
if (bverror != BVERR_NONE)
goto exit;
+ /* Compute the surface offset in bytes. */
+ dstbyteshift = batch->dstalign * (int) batch->dstformat->bitspp / 8;
+
+ /* Verify if the destination parameter have been modified. */
+ if ((batch->dstbyteshift != dstbyteshift) ||
+ (batch->physwidth != batch->dstphyswidth) ||
+ (batch->physheight != batch->dstphysheight)) {
+ /* Set new values. */
+ batch->dstbyteshift = dstbyteshift;
+ batch->physwidth = batch->dstphyswidth;
+ batch->physheight = batch->dstphysheight;
+
+ /* Mark as modified. */
+ batch->batchflags |= BVBATCH_DST;
+ }
+
/* Map the destination. */
bverror = do_map(bltparams->dstdesc, batch, &dstmap);
if (bverror != BVERR_NONE) {
@@ -2666,51 +2952,36 @@ static enum bverror do_fill(struct bvbltparams *bltparams,
if (bverror != BVERR_NONE)
goto exit;
- /* Update clipping/destination rect if necessary. */
- bverror = set_clip(bltparams, batch);
- if (bverror != BVERR_NONE)
- goto exit;
+ /* Reset the modified flag. */
+ batch->batchflags &= ~(BVBATCH_DST |
+ BVBATCH_CLIPRECT |
+ BVBATCH_DESTRECT);
- /* Parse the source format. */
- GCDBG(GCZONE_FORMAT, "parsing the source format.\n");
-
- if (!parse_format(srcinfo->geom->format, &srcformat)) {
- BVSETBLTERRORARG((srcinfo->index == 0)
- ? BVERR_SRC1GEOM_FORMAT
- : BVERR_SRC2GEOM_FORMAT,
- "invalid source format (%d)",
- srcinfo->geom->format);
- goto exit;
- }
+ /***********************************************************************
+ ** Allocate command buffer.
+ */
- /* Validate soource geometry. */
- if (!valid_geom(srcinfo->buf.desc, srcinfo->geom, srcformat)) {
- BVSETBLTERRORARG((srcinfo->index == 0)
- ? BVERR_SRC1GEOM
- : BVERR_SRC2GEOM,
- "source%d geom exceeds surface size",
- srcinfo->index + 1);
+ bverror = claim_buffer(bltparams, batch,
+ sizeof(struct gcmofill),
+ (void **) &gcmofill);
+ if (bverror != BVERR_NONE)
goto exit;
- }
-
- bverror = claim_buffer(batch, sizeof(struct gcmofill),
- (void **) &gcmofill);
- if (bverror != BVERR_NONE) {
- BVSETBLTERROR(BVERR_OOM, "failed to allocate command buffer");
- goto exit;
- }
/***********************************************************************
- ** Set source.
+ ** Set dummy source.
*/
/* Set surface dummy width and height. */
gcmofill->src.rotation_ldst = gcmofillsrc_rotation_ldst;
gcmofill->src.rotation.raw = 0;
- gcmofill->src.rotation.reg.surf_width = dstgeom->width;
+ gcmofill->src.rotation.reg.surf_width = 1;
gcmofill->src.config.raw = 0;
+
gcmofill->src.rotationheight_ldst = gcmofillsrc_rotationheight_ldst;
- gcmofill->src.rotationheight.reg.height = dstgeom->height;
+ gcmofill->src.rotationheight.reg.height = 1;
+ gcmofill->src.rotationangle.raw = 0;
+ gcmofill->src.rotationangle.reg.dst = GCREG_ROT_ANGLE_ROT0;
+ gcmofill->src.rotationangle.reg.dst_mirror = GCREG_MIRROR_NONE;
/* Disable alpha blending. */
gcmofill->src.alphacontrol_ldst = gcmofillsrc_alphacontrol_ldst;
@@ -2724,607 +2995,765 @@ static enum bverror do_fill(struct bvbltparams *bltparams,
fillcolorptr
= (unsigned char *) srcinfo->buf.desc->virtaddr
+ srcinfo->rect->top * srcinfo->geom->virtstride
- + srcinfo->rect->left * srcformat->bitspp / 8;
+ + srcinfo->rect->left * srcinfo->format->bitspp / 8;
gcmofill->clearcolor_ldst = gcmofill_clearcolor_ldst;
- gcmofill->clearcolor.raw = getinternalcolor(fillcolorptr, srcformat);
+ gcmofill->clearcolor.raw = getinternalcolor(fillcolorptr,
+ srcinfo->format);
/***********************************************************************
** Configure and start fill.
*/
/* Set destination configuration. */
- gcmofill->start.config_ldst = gcmostart_config_ldst;
- gcmofill->start.config.raw = 0;
- gcmofill->start.config.reg.swizzle = batch->dstformat->swizzle;
- gcmofill->start.config.reg.format = batch->dstformat->format;
- gcmofill->start.config.reg.command = GCREG_DEST_CONFIG_COMMAND_CLEAR;
+ gcmofill->dstconfig_ldst = gcmofill_dstconfig_ldst;
+ gcmofill->dstconfig.raw = 0;
+ gcmofill->dstconfig.reg.swizzle = batch->dstformat->swizzle;
+ gcmofill->dstconfig.reg.format = batch->dstformat->format;
+ gcmofill->dstconfig.reg.command = GCREG_DEST_CONFIG_COMMAND_CLEAR;
/* Set ROP3. */
- gcmofill->start.rop_ldst = gcmostart_rop_ldst;
- gcmofill->start.rop.raw = 0;
- gcmofill->start.rop.reg.type = GCREG_ROP_TYPE_ROP3;
- gcmofill->start.rop.reg.fg = (unsigned char) bltparams->op.rop;
+ gcmofill->rop_ldst = gcmofill_rop_ldst;
+ gcmofill->rop.raw = 0;
+ gcmofill->rop.reg.type = GCREG_ROP_TYPE_ROP3;
+ gcmofill->rop.reg.fg = (unsigned char) bltparams->op.rop;
/* Set START_DE command. */
- gcmofill->start.startde.cmd.fld = gcfldstartde;
+ gcmofill->startde.cmd.fld = gcfldstartde;
/* Set destination rectangle. */
- gcmofill->start.rect.left = batch->dstleft + batch->dstoffset;
- gcmofill->start.rect.top = batch->dsttop;
- gcmofill->start.rect.right = batch->dstright + batch->dstoffset;
- gcmofill->start.rect.bottom = batch->dstbottom;
+ gcmofill->rect.left = batch->clippedleft + batch->dstoffsetX;
+ gcmofill->rect.top = batch->clippedtop + batch->dstoffsetY;
+ gcmofill->rect.right = batch->clippedright + batch->dstoffsetX;
+ gcmofill->rect.bottom = batch->clippedbottom + batch->dstoffsetY;
exit:
GCEXITARG(GCZONE_DO_FILL, "bv%s = %d\n",
- (bverror == BVERR_NONE) ? "result" : "error", bverror);
+ (bverror == BVERR_NONE) ? "result" : "error", bverror);
return bverror;
}
+#define ENABLE_HW_BLOCK 1
static enum bverror do_blit_end(struct bvbltparams *bltparams,
struct gcbatch *batch)
{
enum bverror bverror;
- struct gcmomultisrc *gcmomultisrc;
+ struct gcmobltconfig *gcmobltconfig;
struct gcmostart *gcmostart;
- unsigned int buffersize;
GCENTER(GCZONE_DO_BLIT);
- GCDBG(GCZONE_BATCH, "finalizing the blit, scrcount = %d\n",
- batch->gcblit.srccount);
+ GCDBG(GCZONE_DO_BLIT, "finalizing the blit, scrcount = %d\n",
+ batch->gcblit.srccount);
- /* Allocate command buffer. */
- buffersize
- = sizeof(struct gcmomultisrc)
- + sizeof(struct gcmostart);
+ /***********************************************************************
+ * Configure the operation.
+ */
- bverror = claim_buffer(batch, buffersize,
- (void **) &gcmomultisrc);
- if (bverror != BVERR_NONE) {
- BVSETBLTERROR(BVERR_OOM, "failed to allocate command buffer");
+ /* Allocate command buffer. */
+ bverror = claim_buffer(bltparams, batch,
+ sizeof(struct gcmobltconfig),
+ (void **) &gcmobltconfig);
+ if (bverror != BVERR_NONE)
goto exit;
- }
/* Configure multi-source control. */
- gcmomultisrc->control_ldst = gcmomultisrc_control_ldst;
- gcmomultisrc->control.raw = 0;
- gcmomultisrc->control.reg.srccount = batch->gcblit.srccount - 1;
- gcmomultisrc->control.reg.horblock
- = GCREG_DE_MULTI_SOURCE_HORIZONTAL_BLOCK_PIXEL128;
-
- /* Skip to the next structure. */
- gcmostart = (struct gcmostart *) (gcmomultisrc + 1);
+ gcmobltconfig->multisource_ldst = gcmobltconfig_multisource_ldst;
+ gcmobltconfig->multisource.raw = 0;
+ gcmobltconfig->multisource.reg.srccount = batch->gcblit.srccount - 1;
+
+ GCDBG(GCZONE_DO_BLIT, "blockenable = %d\n", batch->blockenable);
+ if (batch->blockenable && ENABLE_HW_BLOCK) {
+ gcmobltconfig->multisource.reg.horblock
+ = GCREG_DE_MULTI_SOURCE_HORIZONTAL_BLOCK_PIXEL16;
+ gcmobltconfig->multisource.reg.verblock
+ = GCREG_DE_MULTI_SOURCE_VERTICAL_BLOCK_LINE64;
+ } else {
+ gcmobltconfig->multisource.reg.horblock
+ = GCREG_DE_MULTI_SOURCE_HORIZONTAL_BLOCK_PIXEL128;
+ gcmobltconfig->multisource.reg.verblock
+ = GCREG_DE_MULTI_SOURCE_VERTICAL_BLOCK_LINE1;
+ }
/* Set destination configuration. */
GCDBG(GCZONE_DO_BLIT, "format entry = 0x%08X\n",
- (unsigned int) batch->dstformat);
+ (unsigned int) batch->dstformat);
GCDBG(GCZONE_DO_BLIT, " swizzle code = %d\n",
- batch->dstformat->swizzle);
+ batch->dstformat->swizzle);
GCDBG(GCZONE_DO_BLIT, " format code = %d\n",
- batch->dstformat->format);
+ batch->dstformat->format);
- gcmostart->config_ldst = gcmostart_config_ldst;
- gcmostart->config.raw = 0;
- gcmostart->config.reg.swizzle = batch->dstformat->swizzle;
- gcmostart->config.reg.format = batch->dstformat->format;
- gcmostart->config.reg.command = batch->gcblit.multisrc
+ gcmobltconfig->dstconfig_ldst = gcmobltconfig_dstconfig_ldst;
+ gcmobltconfig->dstconfig.raw = 0;
+ gcmobltconfig->dstconfig.reg.swizzle = batch->dstformat->swizzle;
+ gcmobltconfig->dstconfig.reg.format = batch->dstformat->format;
+ gcmobltconfig->dstconfig.reg.command = batch->gcblit.multisrc
? GCREG_DEST_CONFIG_COMMAND_MULTI_SOURCE_BLT
: GCREG_DEST_CONFIG_COMMAND_BIT_BLT;
/* Set ROP. */
- gcmostart->rop_ldst = gcmostart_rop_ldst;
- gcmostart->rop.raw = 0;
- gcmostart->rop.reg.type = GCREG_ROP_TYPE_ROP3;
- gcmostart->rop.reg.fg = (unsigned char) batch->gcblit.rop;
+ gcmobltconfig->rop_ldst = gcmobltconfig_rop_ldst;
+ gcmobltconfig->rop.raw = 0;
+ gcmobltconfig->rop.reg.type = GCREG_ROP_TYPE_ROP3;
+ gcmobltconfig->rop.reg.fg = (unsigned char) batch->gcblit.rop;
+
+ /***********************************************************************
+ * Start the operation.
+ */
+
+ /* Allocate command buffer. */
+ bverror = claim_buffer(bltparams, batch,
+ sizeof(struct gcmostart),
+ (void **) &gcmostart);
+ if (bverror != BVERR_NONE)
+ goto exit;
/* Set START_DE command. */
gcmostart->startde.cmd.fld = gcfldstartde;
/* Set destination rectangle. */
- gcmostart->rect.left = batch->dstleft + batch->dstoffset;
- gcmostart->rect.top = batch->dsttop;
- gcmostart->rect.right = batch->dstright + batch->dstoffset;
- gcmostart->rect.bottom = batch->dstbottom;
+ gcmostart->rect.left = batch->left;
+ gcmostart->rect.top = batch->top;
+ gcmostart->rect.right = batch->right;
+ gcmostart->rect.bottom = batch->bottom;
+
+ GCDBG(GCZONE_DO_BLIT, "dstrect = (%d,%d)-(%d,%d)\n",
+ gcmostart->rect.left, gcmostart->rect.top,
+ gcmostart->rect.right, gcmostart->rect.bottom);
/* Reset the finalizer. */
batch->batchend = do_end;
gc_debug_blt(batch->gcblit.srccount,
- abs(batch->dstright - batch->dstleft),
- abs(batch->dsttop - batch->dstbottom));
+ abs(batch->right - batch->left),
+ abs(batch->bottom - batch->top));
exit:
GCEXITARG(GCZONE_DO_BLIT, "bv%s = %d\n",
- (bverror == BVERR_NONE) ? "result" : "error", bverror);
+ (bverror == BVERR_NONE) ? "result" : "error", bverror);
return bverror;
}
-
-static inline int get_offset(struct bvbuffdesc *desc)
-{
- int offset;
-
- if (desc->auxtype == BVAT_PHYSDESC) {
- struct bvphysdesc *pdesc;
- pdesc = (struct bvphysdesc *) desc->auxptr;
-
- offset = pdesc->pageoffset &
- (GC_BASE_ALIGN - 1);
- } else {
- offset = (unsigned int) desc->virtaddr &
- (GC_BASE_ALIGN - 1);
- }
- return offset;
-}
-
static enum bverror do_blit(struct bvbltparams *bltparams,
struct gcbatch *batch,
- struct srcinfo *srcinfo,
- unsigned int srccount,
- struct gcalpha *gca)
+ struct srcinfo *srcinfo)
{
enum bverror bverror = BVERR_NONE;
struct gcmosrc *gcmosrc;
struct gcmosrcalpha *gcmosrcalpha;
- unsigned int i, index;
+ unsigned int index;
- struct bvbuffmap *dstmap = NULL;
- struct bvsurfgeom *dstgeom = bltparams->dstgeom;
- struct bvrect *dstrect = &bltparams->dstrect;
+ struct bvbuffmap *dstmap;
+ struct bvbuffdesc *dstdesc;
+ struct bvsurfgeom *dstgeom;
+ struct bvformatxlate *dstformat;
+ int dstshiftX, dstshiftY;
+ int dstalign, dstbyteshift;
- struct bvformatxlate *srcformat;
- struct bvbuffmap *srcmap[2] = { NULL, NULL };
+ struct bvbuffmap *srcmap;
+ struct gcalpha *srcgca;
+ struct bvrect *srcrect;
struct bvbuffdesc *srcdesc;
struct bvsurfgeom *srcgeom;
- struct bvrect *srcrect;
- int srcleft, srctop, srcright, srcbottom;
- int srcoffset, srcsurfleft, srcsurftop;
- int srcshift, srcadjust, srcalign;
+ struct bvformatxlate *srcformat;
+ int srcshiftX, srcshiftY;
+ int srcalign, srcbyteshift;
+
+ int srcleft, srctop;
+ int dstleft, dsttop, dstright, dstbottom;
+ int srcsurfwidth, srcsurfheight;
+ unsigned int physwidth, physheight;
+ int orthogonal;
int multisrc;
GCENTER(GCZONE_DO_BLIT);
- /* Did destination change? */
- if ((batch->batchflags & (BVBATCH_DST |
- BVBATCH_CLIPRECT_ORIGIN |
- BVBATCH_CLIPRECT_SIZE |
- BVBATCH_DSTRECT_ORIGIN |
- BVBATCH_DSTRECT_SIZE)) != 0) {
- /* Finalize the previous blit if any. */
- bverror = batch->batchend(bltparams, batch);
- if (bverror != BVERR_NONE)
- goto exit;
- }
-
- /* Map the destination. */
- bverror = do_map(bltparams->dstdesc, batch, &dstmap);
- if (bverror != BVERR_NONE) {
- bltparams->errdesc = gccontext.bverrorstr;
- goto exit;
- }
+ /* Create source object shortcuts. */
+ srcmap = NULL;
+ srcgca = srcinfo->gca;
+ srcrect = srcinfo->rect;
+ srcdesc = srcinfo->buf.desc;
+ srcgeom = srcinfo->geom;
+ srcformat = srcinfo->format;
- /* Set the new destination. */
- bverror = set_dst(bltparams, batch, dstmap);
- if (bverror != BVERR_NONE)
- goto exit;
+ /* Create destination object shortcuts. */
+ dstmap = NULL;
+ dstdesc = bltparams->dstdesc;
+ dstgeom = bltparams->dstgeom;
+ dstformat = batch->dstformat;
- /* Update clipping/destination rect if necessary. */
- bverror = set_clip(bltparams, batch);
- if (bverror != BVERR_NONE)
- goto exit;
+ /***********************************************************************
+ * Determine source surface alignment offset.
+ */
+
+ /* Determine whether the source and the destination are orthogonal
+ * to each other. */
+ orthogonal = (srcinfo->angle % 2) != (batch->dstangle % 2);
+
+ /* Compute adjusted destination rectangle. */
+ dstleft = batch->clippedleft + batch->dstoffsetX;
+ dsttop = batch->clippedtop + batch->dstoffsetY;
+ dstright = batch->clippedright + batch->dstoffsetX;
+ dstbottom = batch->clippedbottom + batch->dstoffsetY;
+
+ /* Compute clipped source origin. */
+ srcleft = srcrect->left + batch->deltaleft;
+ srctop = srcrect->top + batch->deltatop;
+
+ GCDBG(GCZONE_SURF, "adjusted dstrect = (%d,%d)-(%d,%d), %dx%d\n",
+ dstleft, dsttop, dstright, dstbottom,
+ dstright - dstleft, dstbottom - dsttop);
+
+ /* Compute the source surface shift. */
+ switch (srcinfo->angle) {
+ case ROT_ANGLE_0:
+ srcshiftX = srcleft - dstleft;
+ srcshiftY = srctop - dsttop;
+ break;
- for (i = 0; i < srccount; i += 1) {
- /***************************************************************
- ** Parse the source format.
- */
+ case ROT_ANGLE_90:
+ srcshiftX = srctop - dsttop;
+ srcshiftY = (srcgeom->width - srcleft)
+ - (batch->dstwidth - dstleft);
+ break;
- srcdesc = srcinfo[i].buf.desc;
- srcgeom = srcinfo[i].geom;
- srcrect = srcinfo[i].rect;
+ case ROT_ANGLE_180:
+ srcshiftX = (srcgeom->width - srcleft)
+ - (batch->dstwidth - dstleft);
+ srcshiftY = (srcgeom->height - srctop)
+ - (batch->dstheight - dsttop);
+ break;
- /* Parse the source format. */
- GCDBG(GCZONE_FORMAT, "parsing the source format.\n");
+ case ROT_ANGLE_270:
+ srcshiftX = (srcgeom->height - srctop)
+ - (batch->dstheight - dsttop);
+ srcshiftY = srcleft - dstleft;
+ break;
- if (!parse_format(srcgeom->format, &srcformat)) {
- BVSETBLTERRORARG((srcinfo[i].index == 0)
- ? BVERR_SRC1GEOM_FORMAT
- : BVERR_SRC2GEOM_FORMAT,
- "invalid source format (%d)",
- srcgeom->format);
- goto exit;
- }
+ default:
+ srcshiftX = 0;
+ srcshiftY = 0;
+ }
- /* Validate soource geometry. */
- if (!valid_geom(srcdesc, srcgeom, srcformat)) {
- BVSETBLTERRORARG((srcinfo->index == 0)
- ? BVERR_SRC1GEOM
- : BVERR_SRC2GEOM,
- "source%d geom exceeds surface size",
- srcinfo->index + 1);
- goto exit;
- }
+ /* Compute the source surface offset in bytes. */
+ srcbyteshift = srcshiftY * (int) srcgeom->virtstride
+ + srcshiftX * (int) srcformat->bitspp / 8;
- srcright = srcrect->left + srcrect->width + batch->deltaright;
- srcbottom = srcrect->top + srcrect->height + batch->deltabottom;
+ /* Compute the source offset in pixels needed to compensate
+ * for the surface base address misalignment if any. */
+ srcalign = get_pixeloffset(srcdesc, srcformat, srcbyteshift);
- /* Validate the rectangle. */
- if ((srcright > (int) srcgeom->width) ||
- (srcbottom > (int) srcgeom->height)) {
- BVSETBLTERRORARG((srcinfo->index == 0)
- ? BVERR_SRC1RECT
- : BVERR_SRC2RECT,
- "source%d rect exceeds surface size",
- srcinfo->index + 1);
- goto exit;
- }
+ GCDBG(GCZONE_SURF, "source surface %d:\n", srcinfo->index + 1);
+ GCDBG(GCZONE_SURF, " surface offset (pixels) = %d,%d\n",
+ srcshiftX, srcshiftY);
+ GCDBG(GCZONE_SURF, " surface offset (bytes) = 0x%08X\n",
+ srcbyteshift);
+ GCDBG(GCZONE_SURF, " srcalign = %d\n",
+ srcalign);
- GCDBG(GCZONE_SURF, "source surface %d:\n", i);
- GCDBG(GCZONE_SURF, " srcaddr[%d] = 0x%08X\n",
- i, (unsigned int) srcdesc->virtaddr);
+ /* Apply the source alignment. */
+ srcbyteshift += srcalign * (int) srcformat->bitspp / 8;
+ srcshiftX += srcalign;
- GCDBG(GCZONE_SURF, " srcsurf = %dx%d, stride = %ld\n",
- srcgeom->width, srcgeom->height,
- srcgeom->virtstride);
+ GCDBG(GCZONE_SURF, " adjusted surface offset (pixels) = %d,%d\n",
+ srcshiftX, srcshiftY);
+ GCDBG(GCZONE_SURF, " adjusted surface offset (bytes) = 0x%08X\n",
+ srcbyteshift);
- GCDBG(GCZONE_SURF, " srcrect = (%d,%d)-(%d,%d), %dx%d\n",
- srcrect->left, srcrect->top,
- srcrect->left + srcrect->width,
- srcrect->top + srcrect->height,
- srcrect->width, srcrect->height);
+ /* Determine the destination surface shift. */
+ dstshiftX = batch->dstalign;
+ dstshiftY = (((srcinfo->angle + 3) % 4) == batch->dstangle)
+ ? srcalign : 0;
- /***************************************************************
- ** Determine source placement.
- */
+ /* Compute the destination surface offset in bytes. */
+ dstbyteshift = dstshiftY * (int) dstgeom->virtstride
+ + dstshiftX * (int) dstformat->bitspp / 8;
+ /* Compute the destination offset in pixels needed to compensate
+ * for the surface base address misalignment if any. */
+ dstalign = get_pixeloffset(dstdesc, dstformat, dstbyteshift);
+
+ GCDBG(GCZONE_SURF, "destination surface:\n");
+ GCDBG(GCZONE_SURF, " surface offset (pixels) = %d,%d\n",
+ dstshiftX, dstshiftY);
+ GCDBG(GCZONE_SURF, " surface offset (bytes) = 0x%08X\n",
+ dstbyteshift);
+ GCDBG(GCZONE_SURF, " realignment = %d\n",
+ dstalign);
+
+ if ((srcformat->type == BVFMT_YUV) || (dstalign != 0) ||
+ ((srcalign != 0) && (srcinfo->angle == batch->dstangle))) {
/* Compute the source offset in pixels needed to compensate
- for the surface base address misalignment if any. */
- srcoffset = get_offset(srcdesc) * 8 / srcformat->bitspp;
-
- GCDBG(GCZONE_SURF, " source offset = %d\n", srcoffset);
+ * for the surface base address misalignment if any. */
+ srcalign = get_pixeloffset(srcdesc, srcformat, 0);
+
+ /* Compute the surface offsets in bytes. */
+ srcbyteshift = srcalign * (int) srcformat->bitspp / 8;
+ dstbyteshift = batch->dstalign * (int) dstformat->bitspp / 8;
+
+ GCDBG(GCZONE_SURF, "recomputed for single-source setup:\n");
+ GCDBG(GCZONE_SURF, " srcalign = %d\n",
+ srcalign);
+ GCDBG(GCZONE_SURF, " srcsurf offset (bytes) = 0x%08X\n",
+ srcbyteshift);
+ GCDBG(GCZONE_SURF, " dstsurf offset (bytes) = 0x%08X\n",
+ dstbyteshift);
+
+ switch (srcinfo->angle) {
+ case ROT_ANGLE_0:
+ /* Adjust left coordinate. */
+ srcleft -= srcalign;
+
+ /* Determine source size. */
+ srcsurfwidth = srcgeom->width - srcalign;
+ srcsurfheight = srcgeom->height;
+ break;
- srcsurfleft = srcrect->left - dstrect->left + srcoffset;
- srcsurftop = srcrect->top - dstrect->top;
+ case ROT_ANGLE_90:
+ /* Adjust top coordinate. */
+ srctop -= srcalign;
- GCDBG(GCZONE_SURF, " source surfaceorigin = %d,%d\n",
- srcsurfleft, srcsurftop);
+ /* Determine source size. */
+ srcsurfwidth = srcgeom->height - srcalign;
+ srcsurfheight = srcgeom->width;
+ break;
- /* (srcsurfleft,srcsurftop) are the coordinates of the source
- surface origin. PE requires alignment of the base
- address. Compute the alignment requirement in pixels. */
- srcalign = 16 * 8 / srcformat->bitspp;
+ case ROT_ANGLE_180:
+ /* Determine source size. */
+ srcsurfwidth = srcgeom->width - srcalign;
+ srcsurfheight = srcgeom->height;
+ break;
- GCDBG(GCZONE_SURF, " srcalign = %d\n", srcalign);
+ case ROT_ANGLE_270:
+ /* Determine source size. */
+ srcsurfwidth = srcgeom->height - srcalign;
+ srcsurfheight = srcgeom->width;
+ break;
- /* Compute the number of pixels the origin has to be
- aligned by. */
- srcadjust = srcsurfleft % srcalign;
+ default:
+ srcsurfwidth = 0;
+ srcsurfheight = 0;
+ }
- GCDBG(GCZONE_SURF, " srcadjust = %d\n", srcadjust);
+ GCDBG(GCZONE_SURF, "srcrect origin = %d,%d\n",
+ srcleft, srctop);
+ GCDBG(GCZONE_SURF, "source physical size = %dx%d\n",
+ srcsurfwidth, srcsurfheight);
- /* TODO: multisrc with nv12 format */
- multisrc = (srcformat->format == GCREG_DE_FORMAT_NV12) ? 0 :
- (srcadjust == 0);
+ /* Set the physical destination size. */
+ physwidth = batch->dstphyswidth;
+ physheight = batch->dstphysheight;
- GCDBG(GCZONE_DO_BLIT, " multisrc = %d\n", multisrc);
+ /* Disable multi source for YUV and for the cases where
+ * the destination and the base address alignment does
+ * not match. */
+ multisrc = 0;
+ GCDBG(GCZONE_SURF, "multi-source disabled.\n");
+ } else {
+ /* Source origin is not used in multi-source setup. */
+ srcleft = 0;
+ srctop = 0;
+
+ /* Adjust the destination to match the source geometry. */
+ switch (srcinfo->angle) {
+ case ROT_ANGLE_0:
+ dstleft -= srcalign;
+ dstright -= srcalign;
+
+ /* Apply the source alignment. */
+ if ((batch->dstangle == ROT_ANGLE_0) ||
+ (batch->dstangle == ROT_ANGLE_180)) {
+ physwidth = batch->dstphyswidth - srcalign;
+ physheight = batch->dstphysheight;
+ } else {
+ physwidth = batch->dstphyswidth;
+ physheight = batch->dstphysheight - srcalign;
+ }
+ break;
- if (multisrc) {
- /* Adjust the origin. */
- srcsurfleft -= srcadjust;
-
- GCDBG(GCZONE_SURF,
- " adjusted source surface origin = %d,%d\n",
- srcsurfleft, srcsurftop);
-
- /* Adjust the source rectangle for the
- * single source walker.
- */
- srcleft = dstrect->left + batch->deltaleft + srcadjust;
- srctop = dstrect->top + batch->deltatop;
-
- GCDBG(GCZONE_SURF,
- " source %d rectangle origin = %d,%d\n",
- i, srcleft, srctop);
-
- /* Compute the surface offset in bytes. */
- srcshift
- = srcsurftop * (int) srcgeom->virtstride
- + srcsurfleft * (int) srcformat->bitspp / 8;
-
- GCDBG(GCZONE_SURF, " srcshift = 0x%08X (%d)\n",
- srcshift, srcshift);
- } else {
- srcadjust = srcoffset % srcalign;
+ case ROT_ANGLE_90:
+ dsttop -= srcalign;
+ dstbottom -= srcalign;
- GCDBG(GCZONE_SURF, " single source srcadjust = %d\n",
- srcadjust);
+ /* Apply the source alignment. */
+ if ((batch->dstangle == ROT_ANGLE_0) ||
+ (batch->dstangle == ROT_ANGLE_180)) {
+ physwidth = batch->dstphyswidth;
+ physheight = batch->dstphysheight - srcalign;
+ } else {
+ physwidth = batch->dstphyswidth - srcalign;
+ physheight = batch->dstphysheight;
+ }
+ break;
- srcleft = srcrect->left + batch->deltaleft + srcadjust;
- srctop = srcrect->top + batch->deltatop;
+ case ROT_ANGLE_180:
+ /* Apply the source alignment. */
+ if ((batch->dstangle == ROT_ANGLE_0) ||
+ (batch->dstangle == ROT_ANGLE_180)) {
+ physwidth = batch->dstphyswidth - srcalign;
+ physheight = batch->dstphysheight;
+ } else {
+ physwidth = batch->dstphyswidth;
+ physheight = batch->dstphysheight - srcalign;
+ }
+ break;
- GCDBG(GCZONE_SURF,
- " source %d rectangle origin = %d,%d\n",
- i, srcleft, srctop);
+ case ROT_ANGLE_270:
+ /* Apply the source alignment. */
+ if ((batch->dstangle == ROT_ANGLE_0) ||
+ (batch->dstangle == ROT_ANGLE_180)) {
+ physwidth = batch->dstphyswidth;
+ physheight = batch->dstphysheight - srcalign;
+ } else {
+ physwidth = batch->dstphyswidth - srcalign;
+ physheight = batch->dstphysheight;
+ }
+ break;
- /* No shift needed for single source walker */
- srcshift = 0;
+ default:
+ physwidth = 0;
+ physheight = 0;
}
- /***************************************************************
- ** Finalize existing blit if necessary.
- */
-
- if ((batch->batchend != do_blit_end) ||
- (batch->gcblit.srccount == 4) ||
- !batch->gcblit.multisrc || !multisrc) {
- /* Finalize the previous blit if any. */
- bverror = batch->batchend(bltparams, batch);
- if (bverror != BVERR_NONE)
- goto exit;
-
- /* Initialize the new batch. */
- batch->batchend = do_blit_end;
- batch->gcblit.srccount = 0;
- batch->gcblit.multisrc = multisrc;
- batch->gcblit.rop = (gca != NULL)
- ? 0xCC : bltparams->op.rop;
+ /* Source geometry is now the same as the destination. */
+ if (orthogonal) {
+ srcsurfwidth = physheight;
+ srcsurfheight = physwidth;
+ } else {
+ srcsurfwidth = physwidth;
+ srcsurfheight = physheight;
}
- /***************************************************************
- ** Map the source.
- */
+ /* Enable multi-source. */
+ multisrc = 1;
+ GCDBG(GCZONE_SURF, "multi-source enabled.\n");
+ }
- bverror = do_map(srcdesc, batch, &srcmap[i]);
- if (bverror != BVERR_NONE) {
- bltparams->errdesc = gccontext.bverrorstr;
+ /* Verify if the destination has been modified. */
+ if ((batch->dstbyteshift != dstbyteshift) ||
+ (batch->physwidth != physwidth) ||
+ (batch->physheight != physheight)) {
+ /* Set new values. */
+ batch->dstbyteshift = dstbyteshift;
+ batch->physwidth = physwidth;
+ batch->physheight = physheight;
+
+ /* Mark as modified. */
+ batch->batchflags |= BVBATCH_DST;
+ }
+
+ /* Check if we need to finalize existing batch. */
+ if ((batch->batchend != do_blit_end) ||
+ (batch->gcblit.srccount == 4) ||
+ (batch->gcblit.multisrc == 0) ||
+ (multisrc == 0) ||
+ ((batch->batchflags & (BVBATCH_DST |
+ BVBATCH_CLIPRECT |
+ BVBATCH_DESTRECT)) != 0)) {
+ /* Finalize existing batch if any. */
+ bverror = batch->batchend(bltparams, batch);
+ if (bverror != BVERR_NONE)
goto exit;
- }
- /***************************************************************
- ** Configure source.
- */
+ /* Initialize the new batch. */
+ batch->batchend = do_blit_end;
+ batch->blockenable = 0;
+ batch->gcblit.srccount = 0;
+ batch->gcblit.multisrc = multisrc;
+ batch->gcblit.rop = srcinfo->rop;
+ }
- /* Allocate command buffer. */
- bverror = claim_buffer(batch, sizeof(struct gcmosrc),
- (void **) &gcmosrc);
- if (bverror != BVERR_NONE) {
- BVSETBLTERROR(BVERR_OOM,
- "failed to allocate command buffer");
- goto exit;
- }
+ /* Set destination coordinates. */
+ batch->left = dstleft;
+ batch->top = dsttop;
+ batch->right = dstright;
+ batch->bottom = dstbottom;
- /* Shortcut to the register index. */
- index = batch->gcblit.srccount;
+ /* Map the destination. */
+ bverror = do_map(dstdesc, batch, &dstmap);
+ if (bverror != BVERR_NONE) {
+ bltparams->errdesc = gccontext.bverrorstr;
+ goto exit;
+ }
- add_fixup(batch, &gcmosrc->address, srcshift);
+ /* Set the new destination. */
+ bverror = set_dst(bltparams, batch, dstmap);
+ if (bverror != BVERR_NONE)
+ goto exit;
- /* Set surface parameters. */
- gcmosrc->address_ldst = gcmosrc_address_ldst[index];
- gcmosrc->address = GET_MAP_HANDLE(srcmap[i]);
-
- gcmosrc->stride_ldst = gcmosrc_stride_ldst[index];
- gcmosrc->stride = srcgeom->virtstride;
-
- gcmosrc->rotation_ldst = gcmosrc_rotation_ldst[index];
- gcmosrc->rotation.raw = 0;
- gcmosrc->rotation.reg.surf_width
- = dstrect->left + srcgeom->width;
-
- gcmosrc->config_ldst = gcmosrc_config_ldst[index];
- gcmosrc->config.raw = 0;
- gcmosrc->config.reg.swizzle = srcformat->swizzle;
- gcmosrc->config.reg.format = srcformat->format;
-
- gcmosrc->origin_ldst = gcmosrc_origin_ldst[index];
- if (multisrc)
- gcmosrc->origin.reg = gcregsrcorigin_min;
- else {
- gcmosrc->origin.reg.x = srcleft;
- gcmosrc->origin.reg.y = srctop;
- }
+ /* Reset the modified flag. */
+ batch->batchflags &= ~(BVBATCH_DST |
+ BVBATCH_CLIPRECT |
+ BVBATCH_DESTRECT);
- gcmosrc->size_ldst = gcmosrc_size_ldst[index];
- gcmosrc->size.reg = gcregsrcsize_max;
+ /* Map the source. */
+ bverror = do_map(srcdesc, batch, &srcmap);
+ if (bverror != BVERR_NONE) {
+ bltparams->errdesc = gccontext.bverrorstr;
+ goto exit;
+ }
- gcmosrc->rotationheight_ldst
- = gcmosrc_rotationheight_ldst[index];
- gcmosrc->rotationheight.reg.height
- = dstrect->top + srcgeom->height;
+ /***************************************************************
+ ** Configure source.
+ */
- gcmosrc->rop_ldst = gcmosrc_rop_ldst[index];
- gcmosrc->rop.raw = 0;
- gcmosrc->rop.reg.type = GCREG_ROP_TYPE_ROP3;
- gcmosrc->rop.reg.fg = (unsigned char) batch->gcblit.rop;
+ /* We need to walk in blocks if the source and the destination
+ * surfaces are orthogonal to each other. */
+ batch->blockenable = orthogonal;
- gcmosrc->mult_ldst = gcmosrc_mult_ldst[index];
- gcmosrc->mult.raw = 0;
- gcmosrc->mult.reg.srcglobalpremul
- = GCREG_COLOR_MULTIPLY_MODES_SRC_GLOBAL_PREMULTIPLY_DISABLE;
+ /* Allocate command buffer. */
+ bverror = claim_buffer(bltparams, batch,
+ sizeof(struct gcmosrc),
+ (void **) &gcmosrc);
+ if (bverror != BVERR_NONE)
+ goto exit;
- if ((srcgeom->format & OCDFMTDEF_NON_PREMULT) != 0)
- gcmosrc->mult.reg.srcpremul
- = GCREG_COLOR_MULTIPLY_MODES_SRC_PREMULTIPLY_ENABLE;
- else
- gcmosrc->mult.reg.srcpremul
- = GCREG_COLOR_MULTIPLY_MODES_SRC_PREMULTIPLY_DISABLE;
+ /* Shortcut to the register index. */
+ index = batch->gcblit.srccount;
- if ((dstgeom->format & OCDFMTDEF_NON_PREMULT) != 0) {
- gcmosrc->mult.reg.dstpremul
- = GCREG_COLOR_MULTIPLY_MODES_SRC_PREMULTIPLY_ENABLE;
+ add_fixup(bltparams, batch, &gcmosrc->address, srcbyteshift);
- gcmosrc->mult.reg.dstdemul
- = GCREG_COLOR_MULTIPLY_MODES_DST_DEMULTIPLY_ENABLE;
- } else {
- gcmosrc->mult.reg.dstpremul
- = GCREG_COLOR_MULTIPLY_MODES_SRC_PREMULTIPLY_DISABLE;
+ /* Set surface parameters. */
+ gcmosrc->address_ldst = gcmosrc_address_ldst[index];
+ gcmosrc->address = GET_MAP_HANDLE(srcmap);
+
+ gcmosrc->stride_ldst = gcmosrc_stride_ldst[index];
+ gcmosrc->stride = srcgeom->virtstride;
+
+ gcmosrc->rotation_ldst = gcmosrc_rotation_ldst[index];
+ gcmosrc->rotation.raw = 0;
+ gcmosrc->rotation.reg.surf_width = srcsurfwidth;
+
+ gcmosrc->config_ldst = gcmosrc_config_ldst[index];
+ gcmosrc->config.raw = 0;
+ gcmosrc->config.reg.swizzle = srcformat->swizzle;
+ gcmosrc->config.reg.format = srcformat->format;
+
+ gcmosrc->origin_ldst = gcmosrc_origin_ldst[index];
+ gcmosrc->origin.reg.x = srcleft;
+ gcmosrc->origin.reg.y = srctop;
+
+ gcmosrc->size_ldst = gcmosrc_size_ldst[index];
+ gcmosrc->size.reg = gcregsrcsize_max;
+
+ gcmosrc->rotationheight_ldst
+ = gcmosrc_rotationheight_ldst[index];
+ gcmosrc->rotationheight.reg.height = srcsurfheight;
+
+ gcmosrc->rotationangle_ldst
+ = gcmosrc_rotationangle_ldst[index];
+ gcmosrc->rotationangle.raw = 0;
+ gcmosrc->rotationangle.reg.src = rotencoding[srcinfo->angle];
+ gcmosrc->rotationangle.reg.dst = rotencoding[batch->dstangle];
+ gcmosrc->rotationangle.reg.src_mirror = srcinfo->mirror;
+ gcmosrc->rotationangle.reg.dst_mirror = GCREG_MIRROR_NONE;
+
+ gcmosrc->rop_ldst = gcmosrc_rop_ldst[index];
+ gcmosrc->rop.raw = 0;
+ gcmosrc->rop.reg.type = GCREG_ROP_TYPE_ROP3;
+ gcmosrc->rop.reg.fg = (unsigned char) batch->gcblit.rop;
+
+ gcmosrc->mult_ldst = gcmosrc_mult_ldst[index];
+ gcmosrc->mult.raw = 0;
+ gcmosrc->mult.reg.srcglobalpremul
+ = GCREG_COLOR_MULTIPLY_MODES_SRC_GLOBAL_PREMULTIPLY_DISABLE;
+
+ if ((srcgeom->format & OCDFMTDEF_NON_PREMULT) != 0)
+ gcmosrc->mult.reg.srcpremul
+ = GCREG_COLOR_MULTIPLY_MODES_SRC_PREMULTIPLY_ENABLE;
+ else
+ gcmosrc->mult.reg.srcpremul
+ = GCREG_COLOR_MULTIPLY_MODES_SRC_PREMULTIPLY_DISABLE;
- gcmosrc->mult.reg.dstdemul
- = GCREG_COLOR_MULTIPLY_MODES_DST_DEMULTIPLY_DISABLE;
- }
+ if ((dstgeom->format & OCDFMTDEF_NON_PREMULT) != 0) {
+ gcmosrc->mult.reg.dstpremul
+ = GCREG_COLOR_MULTIPLY_MODES_SRC_PREMULTIPLY_ENABLE;
- if (srcformat->format == GCREG_DE_FORMAT_NV12) {
- struct gcmosrcplanaryuv *yuv;
- int uvshift;
+ gcmosrc->mult.reg.dstdemul
+ = GCREG_COLOR_MULTIPLY_MODES_DST_DEMULTIPLY_ENABLE;
+ } else {
+ gcmosrc->mult.reg.dstpremul
+ = GCREG_COLOR_MULTIPLY_MODES_SRC_PREMULTIPLY_DISABLE;
- if (multisrc && srcsurftop % 2) {
- /* We can't shift the uv plane by
- * an odd number of rows.
- */
- BVSETBLTERROR(BVERR_SRC1RECT,
- "src/dst y coordinate combination not supported");
- goto exit;
- }
+ gcmosrc->mult.reg.dstdemul
+ = GCREG_COLOR_MULTIPLY_MODES_DST_DEMULTIPLY_DISABLE;
+ }
- bverror = claim_buffer(batch,
- sizeof(struct gcmosrcplanaryuv),
- (void **) &yuv);
- if (bverror != BVERR_NONE) {
- BVSETBLTERROR(BVERR_OOM,
- "failed to allocate command buffer for planar yuv");
- goto exit;
- }
+ if (srcformat->type == BVFMT_YUV) {
+ struct gcmosrcplanaryuv *yuv;
+ int uvshift;
+
+#if 0
+ if (multisrc && (srcsurftop % 2)) {
+ /* We can't shift the uv plane by an odd number
+ * of rows. */
+ BVSETBLTERROR(BVERR_SRC1RECT,
+ "src/dst y coordinate combination"
+ " not supported");
+ goto exit;
+ }
+#endif
- yuv->uplaneaddress_ldst =
- gcmosrc_uplaneaddress_ldst[index];
- yuv->uplanestride_ldst =
- gcmosrc_uplanestride_ldst[index];
- yuv->vplaneaddress_ldst =
- gcmosrc_vplaneaddress_ldst[index];
- yuv->vplanestride_ldst =
- gcmosrc_vplanestride_ldst[index];
-
- if (multisrc) {
- /* uv plane is half height */
- uvshift = (srcsurftop / 2) *
- (int) srcgeom->virtstride +
- srcsurfleft *
- (int) srcformat->bitspp / 8;
- } else {
- /* No shift needed for single source walker */
- uvshift = 0;
- }
+ bverror = claim_buffer(bltparams, batch,
+ sizeof(struct gcmosrcplanaryuv),
+ (void **) &yuv);
+ if (bverror != BVERR_NONE)
+ goto exit;
- GCDBG(GCZONE_SURF, " uvshift = 0x%08X (%d)\n",
- uvshift, uvshift);
+ yuv->uplaneaddress_ldst =
+ gcmosrc_uplaneaddress_ldst[index];
+ yuv->uplanestride_ldst =
+ gcmosrc_uplanestride_ldst[index];
+ yuv->vplaneaddress_ldst =
+ gcmosrc_vplaneaddress_ldst[index];
+ yuv->vplanestride_ldst =
+ gcmosrc_vplanestride_ldst[index];
+#if 0
- /* add fixed offset from Y plane */
- uvshift += srcgeom->virtstride * srcgeom->height;
+ if (multisrc) {
+ /* UV plane is half height. */
+ uvshift = (srcsurftop / 2)
+ * (int) srcgeom->virtstride
+ + srcsurfleft
+ * (int) srcformat->bitspp / 8;
+ } else {
+ /* No shift needed for single source walker. */
+ uvshift = 0;
+ }
+#else
+ /* No shift needed for single source walker. */
+ uvshift = 0;
+#endif
- GCDBG(GCZONE_SURF, " final uvshift = 0x%08X (%d)\n",
- uvshift, uvshift);
+ GCDBG(GCZONE_SURF, " uvshift = 0x%08X (%d)\n",
+ uvshift, uvshift);
- yuv->uplaneaddress = GET_MAP_HANDLE(srcmap[i]);
- add_fixup(batch, &yuv->uplaneaddress, uvshift);
+ /* add fixed offset from Y plane */
+ uvshift += srcgeom->virtstride * srcgeom->height;
- yuv->uplanestride = srcgeom->virtstride;
+ GCDBG(GCZONE_SURF, " final uvshift = 0x%08X (%d)\n",
+ uvshift, uvshift);
- yuv->vplaneaddress = GET_MAP_HANDLE(srcmap[i]);
- add_fixup(batch, &yuv->vplaneaddress, uvshift);
+ yuv->uplaneaddress = GET_MAP_HANDLE(srcmap);
+ add_fixup(bltparams, batch, &yuv->uplaneaddress, uvshift);
- yuv->vplanestride = srcgeom->virtstride;
- }
+ yuv->uplanestride = srcgeom->virtstride;
- if ((gca != NULL) && ((srccount == 1) || (i > 0))) {
- gcmosrc->alphacontrol_ldst
- = gcmosrc_alphacontrol_ldst[index];
- gcmosrc->alphacontrol.raw = 0;
- gcmosrc->alphacontrol.reg.enable
- = GCREG_ALPHA_CONTROL_ENABLE_ON;
+ yuv->vplaneaddress = GET_MAP_HANDLE(srcmap);
+ add_fixup(bltparams, batch, &yuv->vplaneaddress, uvshift);
- /* Allocate command buffer. */
- bverror = claim_buffer(batch,
- sizeof(struct gcmosrcalpha),
- (void **) &gcmosrcalpha);
- if (bverror != BVERR_NONE) {
- BVSETBLTERROR(BVERR_OOM,
- "failed to allocate command buffer");
- goto exit;
- }
+ yuv->vplanestride = srcgeom->virtstride;
+ }
- gcmosrcalpha->alphamodes_ldst
- = gcmosrcalpha_alphamodes_ldst[index];
- gcmosrcalpha->alphamodes.raw = 0;
- gcmosrcalpha->alphamodes.reg.src_global_alpha
- = gca->src_global_alpha_mode;
- gcmosrcalpha->alphamodes.reg.dst_global_alpha
- = gca->dst_global_alpha_mode;
-
- if (srccount == 1) {
- GCDBG(GCZONE_BLEND, "k1 sets src blend.\n");
- GCDBG(GCZONE_BLEND, "k2 sets dst blend.\n");
-
- gcmosrcalpha->alphamodes.reg.src_blend
- = gca->k1->factor_mode;
- gcmosrcalpha->alphamodes.reg.src_color_reverse
- = gca->k1->color_reverse;
-
- gcmosrcalpha->alphamodes.reg.dst_blend
- = gca->k2->factor_mode;
- gcmosrcalpha->alphamodes.reg.dst_color_reverse
- = gca->k2->color_reverse;
- } else {
- GCDBG(GCZONE_BLEND, "k1 sets dst blend.\n");
- GCDBG(GCZONE_BLEND, "k2 sets src blend.\n");
-
- gcmosrcalpha->alphamodes.reg.src_blend
- = gca->k2->factor_mode;
- gcmosrcalpha->alphamodes.reg.src_color_reverse
- = gca->k2->color_reverse;
-
- gcmosrcalpha->alphamodes.reg.dst_blend
- = gca->k1->factor_mode;
- gcmosrcalpha->alphamodes.reg.dst_color_reverse
- = gca->k1->color_reverse;
- }
+ if (srcgca != NULL) {
+ gcmosrc->alphacontrol_ldst
+ = gcmosrc_alphacontrol_ldst[index];
+ gcmosrc->alphacontrol.raw = 0;
+ gcmosrc->alphacontrol.reg.enable
+ = GCREG_ALPHA_CONTROL_ENABLE_ON;
- GCDBG(GCZONE_BLEND, "dst blend:\n");
- GCDBG(GCZONE_BLEND, " factor = %d\n",
- gcmosrcalpha->alphamodes.reg.dst_blend);
- GCDBG(GCZONE_BLEND, " inverse = %d\n",
- gcmosrcalpha->alphamodes.reg.dst_color_reverse);
-
- GCDBG(GCZONE_BLEND, "src blend:\n");
- GCDBG(GCZONE_BLEND, " factor = %d\n",
- gcmosrcalpha->alphamodes.reg.src_blend);
- GCDBG(GCZONE_BLEND, " inverse = %d\n",
- gcmosrcalpha->alphamodes.reg.src_color_reverse);
-
- gcmosrcalpha->srcglobal_ldst
- = gcmosrcalpha_srcglobal_ldst[index];
- gcmosrcalpha->srcglobal.raw = gca->src_global_color;
-
- gcmosrcalpha->dstglobal_ldst
- = gcmosrcalpha_dstglobal_ldst[index];
- gcmosrcalpha->dstglobal.raw = gca->dst_global_color;
- } else {
- GCDBG(GCZONE_BLEND, "blending disabled.\n");
+ /* Allocate command buffer. */
+ bverror = claim_buffer(bltparams, batch,
+ sizeof(struct gcmosrcalpha),
+ (void **) &gcmosrcalpha);
+ if (bverror != BVERR_NONE)
+ goto exit;
- gcmosrc->alphacontrol_ldst
- = gcmosrc_alphacontrol_ldst[index];
- gcmosrc->alphacontrol.raw = 0;
- gcmosrc->alphacontrol.reg.enable
- = GCREG_ALPHA_CONTROL_ENABLE_OFF;
- }
+ gcmosrcalpha->alphamodes_ldst
+ = gcmosrcalpha_alphamodes_ldst[index];
+ gcmosrcalpha->alphamodes.raw = 0;
+ gcmosrcalpha->alphamodes.reg.src_global_alpha
+ = srcgca->src_global_alpha_mode;
+ gcmosrcalpha->alphamodes.reg.dst_global_alpha
+ = srcgca->dst_global_alpha_mode;
+
+ gcmosrcalpha->alphamodes.reg.src_blend
+ = srcgca->srcconfig->factor_mode;
+ gcmosrcalpha->alphamodes.reg.src_color_reverse
+ = srcgca->srcconfig->color_reverse;
+
+ gcmosrcalpha->alphamodes.reg.dst_blend
+ = srcgca->dstconfig->factor_mode;
+ gcmosrcalpha->alphamodes.reg.dst_color_reverse
+ = srcgca->dstconfig->color_reverse;
+
+ GCDBG(GCZONE_BLEND, "dst blend:\n");
+ GCDBG(GCZONE_BLEND, " factor = %d\n",
+ gcmosrcalpha->alphamodes.reg.dst_blend);
+ GCDBG(GCZONE_BLEND, " inverse = %d\n",
+ gcmosrcalpha->alphamodes.reg.dst_color_reverse);
+
+ GCDBG(GCZONE_BLEND, "src blend:\n");
+ GCDBG(GCZONE_BLEND, " factor = %d\n",
+ gcmosrcalpha->alphamodes.reg.src_blend);
+ GCDBG(GCZONE_BLEND, " inverse = %d\n",
+ gcmosrcalpha->alphamodes.reg.src_color_reverse);
+
+ gcmosrcalpha->srcglobal_ldst
+ = gcmosrcalpha_srcglobal_ldst[index];
+ gcmosrcalpha->srcglobal.raw = srcgca->src_global_color;
+
+ gcmosrcalpha->dstglobal_ldst
+ = gcmosrcalpha_dstglobal_ldst[index];
+ gcmosrcalpha->dstglobal.raw = srcgca->dst_global_color;
+ } else {
+ GCDBG(GCZONE_BLEND, "blending disabled.\n");
- batch->gcblit.srccount += 1;
+ gcmosrc->alphacontrol_ldst
+ = gcmosrc_alphacontrol_ldst[index];
+ gcmosrc->alphacontrol.raw = 0;
+ gcmosrc->alphacontrol.reg.enable
+ = GCREG_ALPHA_CONTROL_ENABLE_OFF;
}
+ batch->gcblit.srccount += 1;
+
exit:
GCEXITARG(GCZONE_DO_BLIT, "bv%s = %d\n",
- (bverror == BVERR_NONE) ? "result" : "error", bverror);
+ (bverror == BVERR_NONE) ? "result" : "error", bverror);
return bverror;
}
static enum bverror do_filter(struct bvbltparams *bltparams,
- struct gcbatch *batch)
+ struct gcbatch *batch,
+ struct srcinfo *srcinfo)
{
enum bverror bverror;
GCENTER(GCZONE_DO_FILTER);
- BVSETBLTERROR(BVERR_SRC1_HORZSCALE, "scaling not supported");
+
+ GCDBG(GCZONE_DO_FILTER, "source rectangle:\n");
+ GCDBG(GCZONE_DO_FILTER, " stride = %d, geom = %dx%d\n",
+ srcinfo->geom->virtstride,
+ srcinfo->geom->width, srcinfo->geom->height);
+ GCDBG(GCZONE_DO_FILTER, " rotation = %d\n",
+ get_angle(srcinfo->geom->orientation));
+ GCDBG(GCZONE_DO_FILTER, " rect = (%d,%d)-(%d,%d), %dx%d\n",
+ srcinfo->rect->left, srcinfo->rect->top,
+ srcinfo->rect->left + srcinfo->rect->width,
+ srcinfo->rect->top + srcinfo->rect->height,
+ srcinfo->rect->width, srcinfo->rect->height);
+
+ GCDBG(GCZONE_DO_FILTER, "destination rectangle:\n");
+ GCDBG(GCZONE_DO_FILTER, " stride = %d, geom size = %dx%d\n",
+ bltparams->dstgeom->virtstride,
+ bltparams->dstgeom->width, bltparams->dstgeom->height);
+ GCDBG(GCZONE_DO_FILTER, " rotaton = %d\n",
+ get_angle(bltparams->dstgeom->orientation));
+ GCDBG(GCZONE_DO_FILTER, " rect = (%d,%d)-(%d,%d), %dx%d\n",
+ bltparams->dstrect.left, bltparams->dstrect.top,
+ bltparams->dstrect.left + bltparams->dstrect.width,
+ bltparams->dstrect.top + bltparams->dstrect.height,
+ bltparams->dstrect.width, bltparams->dstrect.height);
+
+ BVSETBLTERROR((srcinfo->index == 0)
+ ? BVERR_SRC1_HORZSCALE
+ : BVERR_SRC2_HORZSCALE,
+ "scaling not supported");
+
GCEXITARG(GCZONE_DO_FILTER, "bv%s = %d\n",
- (bverror == BVERR_NONE) ? "result" : "error", bverror);
+ (bverror == BVERR_NONE) ? "result" : "error", bverror);
return bverror;
}
@@ -3414,7 +3843,7 @@ enum bverror bv_map(struct bvbuffdesc *buffdesc)
exit:
GCEXITARG(GCZONE_MAPPING, "bv%s = %d\n",
- (bverror == BVERR_NONE) ? "result" : "error", bverror);
+ (bverror == BVERR_NONE) ? "result" : "error", bverror);
return bverror;
}
@@ -3547,7 +3976,7 @@ exit:
GCUNLOCK(&gccontext.maplock);
GCEXITARG(GCZONE_MAPPING, "bv%s = %d\n",
- (bverror == BVERR_NONE) ? "result" : "error", bverror);
+ (bverror == BVERR_NONE) ? "result" : "error", bverror);
return bverror;
}
@@ -3560,14 +3989,15 @@ enum bverror bv_blt(struct bvbltparams *bltparams)
unsigned int batchexec = 0;
bool nop = false;
struct gcbatch *batch;
+ struct bvrect *dstrect;
int src1used, src2used, maskused;
struct srcinfo srcinfo[2];
unsigned short rop;
struct gccommit gccommit;
- int srccount, res;
+ int i, srccount, res;
GCENTERARG(GCZONE_BLIT, "bltparams = 0x%08X\n",
- (unsigned int) bltparams);
+ (unsigned int) bltparams);
/* Verify blt parameters structure. */
if (bltparams == NULL) {
@@ -3599,7 +4029,7 @@ enum bverror bv_blt(struct bvbltparams *bltparams)
switch (type) {
case (BVFLAG_BATCH_NONE >> BVFLAG_BATCH_SHIFT):
- bverror = allocate_batch(&batch);
+ bverror = allocate_batch(bltparams, &batch);
if (bverror != BVERR_NONE) {
bltparams->errdesc = gccontext.bverrorstr;
goto exit;
@@ -3613,7 +4043,7 @@ enum bverror bv_blt(struct bvbltparams *bltparams)
break;
case (BVFLAG_BATCH_BEGIN >> BVFLAG_BATCH_SHIFT):
- bverror = allocate_batch(&batch);
+ bverror = allocate_batch(bltparams, &batch);
if (bverror != BVERR_NONE) {
bltparams->errdesc = gccontext.bverrorstr;
goto exit;
@@ -3679,9 +4109,12 @@ enum bverror bv_blt(struct bvbltparams *bltparams)
GCDUMPSCHEDULE();
if (!nop) {
+ /* Get a shortcut to the destination rectangle. */
+ dstrect = &bltparams->dstrect;
+
/* Verify the batch change flags. */
- VERIFYBATCH(batch->batchflags >> 12, &batch->prevdstrect,
- &bltparams->dstrect);
+ VERIFYBATCH(batch->batchflags >> 12,
+ &batch->prevdstrect, dstrect);
switch (op) {
case (BVFLAG_ROP >> BVFLAG_OP_SHIFT):
@@ -3689,11 +4122,11 @@ enum bverror bv_blt(struct bvbltparams *bltparams)
rop = bltparams->op.rop;
src1used = ((rop & 0xCCCC) >> 2)
- ^ (rop & 0x3333);
+ ^ (rop & 0x3333);
src2used = ((rop & 0xF0F0) >> 4)
- ^ (rop & 0x0F0F);
+ ^ (rop & 0x0F0F);
maskused = ((rop & 0xFF00) >> 8)
- ^ (rop & 0x00FF);
+ ^ (rop & 0x00FF);
break;
case (BVFLAG_BLEND >> BVFLAG_OP_SHIFT):
@@ -3701,7 +4134,7 @@ enum bverror bv_blt(struct bvbltparams *bltparams)
blend = bltparams->op.blend;
format = (blend & BVBLENDDEF_FORMAT_MASK)
- >> BVBLENDDEF_FORMAT_SHIFT;
+ >> BVBLENDDEF_FORMAT_SHIFT;
bverror = parse_blend(bltparams, blend, &_gca);
if (bverror != BVERR_NONE)
@@ -3719,7 +4152,7 @@ enum bverror bv_blt(struct bvbltparams *bltparams)
default:
BVSETBLTERROR(BVERR_BLEND,
- "unrecognized blend format");
+ "unrecognized blend format");
goto exit;
}
break;
@@ -3727,7 +4160,7 @@ enum bverror bv_blt(struct bvbltparams *bltparams)
case (BVFLAG_FILTER >> BVFLAG_OP_SHIFT):
GCDBG(GCZONE_BLIT, "BVFLAG_FILTER\n");
BVSETBLTERROR(BVERR_OP,
- "filter operation not supported");
+ "filter operation not supported");
goto exit;
default:
@@ -3735,6 +4168,11 @@ enum bverror bv_blt(struct bvbltparams *bltparams)
goto exit;
}
+ /* Parse destination parameters. */
+ bverror = parse_destination(bltparams, batch);
+ if (bverror != BVERR_NONE)
+ goto exit;
+
srccount = 0;
/* Verify the src1 parameters structure. */
@@ -3752,14 +4190,14 @@ enum bverror bv_blt(struct bvbltparams *bltparams)
/* Verify the batch change flags. */
VERIFYBATCH(batch->batchflags >> 14,
- &batch->prevsrc1rect,
- &bltparams->src1rect);
+ &batch->prevsrc1rect,
+ &bltparams->src1rect);
/* Same as the destination? */
if (same_phys_area(bltparams->src1.desc,
- &bltparams->src1rect,
- bltparams->dstdesc,
- &bltparams->dstrect)) {
+ &bltparams->src1rect,
+ bltparams->dstdesc,
+ dstrect)) {
GCDBG(GCZONE_BLIT, "src1 is the same as dst\n");
} else {
srcinfo[srccount].index = 0;
@@ -3767,6 +4205,11 @@ enum bverror bv_blt(struct bvbltparams *bltparams)
srcinfo[srccount].geom = bltparams->src1geom;
srcinfo[srccount].rect = &bltparams->src1rect;
+ bverror = parse_source(bltparams, batch,
+ &srcinfo[srccount]);
+ if (bverror != BVERR_NONE)
+ goto exit;
+
srccount += 1;
}
}
@@ -3786,14 +4229,14 @@ enum bverror bv_blt(struct bvbltparams *bltparams)
/* Verify the batch change flags. */
VERIFYBATCH(batch->batchflags >> 16,
- &batch->prevsrc2rect,
- &bltparams->src2rect);
+ &batch->prevsrc2rect,
+ &bltparams->src2rect);
/* Same as the destination? */
if (same_phys_area(bltparams->src2.desc,
- &bltparams->src2rect,
- bltparams->dstdesc,
- &bltparams->dstrect)) {
+ &bltparams->src2rect,
+ bltparams->dstdesc,
+ dstrect)) {
GCDBG(GCZONE_BLIT, "src2 is the same as dst\n");
} else {
srcinfo[srccount].index = 1;
@@ -3801,6 +4244,11 @@ enum bverror bv_blt(struct bvbltparams *bltparams)
srcinfo[srccount].geom = bltparams->src2geom;
srcinfo[srccount].rect = &bltparams->src2rect;
+ bverror = parse_source(bltparams, batch,
+ &srcinfo[srccount]);
+ if (bverror != BVERR_NONE)
+ goto exit;
+
srccount += 1;
}
}
@@ -3820,55 +4268,55 @@ enum bverror bv_blt(struct bvbltparams *bltparams)
/* Verify the batch change flags. */
VERIFYBATCH(batch->batchflags >> 18,
- &batch->prevmaskrect,
- &bltparams->maskrect);
+ &batch->prevmaskrect,
+ &bltparams->maskrect);
- BVSETERROR(BVERR_OP,
- "operation with mask not supported");
+ BVSETBLTERROR(BVERR_OP,
+ "operation with mask not supported");
goto exit;
}
GCDBG(GCZONE_BLIT, "srccount = %d\n", srccount);
- switch (srccount) {
- case 0:
- bverror = do_blit(bltparams, batch, NULL, 0, gca);
- if (bverror != BVERR_NONE)
- goto exit;
- break;
-
- case 1:
- if (EQ_SIZE((*srcinfo[0].rect), bltparams->dstrect))
- bverror = do_blit(bltparams, batch,
- srcinfo, 1, gca);
- else if ((srcinfo[0].rect->width == 1) &&
- (srcinfo[0].rect->height == 1))
- bverror = do_fill(bltparams, batch, srcinfo);
- else
- bverror = do_filter(bltparams, batch);
- if (bverror != BVERR_NONE)
- goto exit;
- break;
+ if (srccount == 0) {
+ BVSETBLTERROR(BVERR_OP,
+ "operation not supported");
+ goto exit;
+ } else {
+ for (i = 0; i < srccount; i += 1) {
+ if (gca == NULL) {
+ srcinfo[i].rop = bltparams->op.rop;
+ srcinfo[i].gca = NULL;
+ } else if ((i + 1) != srccount) {
+ srcinfo[i].rop = 0xCC;
+ srcinfo[i].gca = NULL;
+ } else {
+ srcinfo[i].rop = 0xCC;
+ srcinfo[i].gca = gca;
+
+ if (srccount == 1) {
+ gca->srcconfig = gca->k1;
+ gca->dstconfig = gca->k2;
+ } else {
+ gca->srcconfig = gca->k2;
+ gca->dstconfig = gca->k1;
+ }
+ }
- case 2:
- if (EQ_SIZE((*srcinfo[0].rect), bltparams->dstrect))
- if (EQ_SIZE((*srcinfo[1].rect),
- bltparams->dstrect))
+ if (EQ_SIZE(srcinfo[i].rect, dstrect))
bverror = do_blit(bltparams, batch,
- srcinfo, 2, gca);
- else
- BVSETBLTERROR(BVERR_SRC2_HORZSCALE,
- "scaling not supported");
- else
- if (EQ_SIZE((*srcinfo[1].rect),
- bltparams->dstrect))
- BVSETBLTERROR(BVERR_SRC1_HORZSCALE,
- "scaling not supported");
+ &srcinfo[i]);
+ else if ((srcinfo[i].rect->width == 1) &&
+ (srcinfo[i].rect->height == 1))
+ bverror = do_fill(bltparams, batch,
+ &srcinfo[i]);
else
- BVSETBLTERROR(BVERR_SRC1_HORZSCALE,
- "scaling not supported");
- if (bverror != BVERR_NONE)
- goto exit;
+ bverror = do_filter(bltparams, batch,
+ &srcinfo[i]);
+
+ if (bverror != BVERR_NONE)
+ goto exit;
+ }
}
}
@@ -3884,13 +4332,11 @@ enum bverror bv_blt(struct bvbltparams *bltparams)
/* Add PE flush. */
GCDBG(GCZONE_BLIT, "appending the flush.\n");
- bverror = claim_buffer(batch, sizeof(struct gcmoflush),
+ bverror = claim_buffer(bltparams, batch,
+ sizeof(struct gcmoflush),
(void **) &flush);
- if (bverror != BVERR_NONE) {
- BVSETBLTERROR(BVERR_OOM,
- "failed to allocate command buffer");
+ if (bverror != BVERR_NONE)
goto exit;
- }
flush->flush_ldst = gcmoflush_flush_ldst;
flush->flush.reg = gcregflush_pe2D;
@@ -3915,7 +4361,8 @@ enum bverror bv_blt(struct bvbltparams *bltparams)
bverror = get_callbackinfo(&gccallbackinfo);
if (bverror != BVERR_NONE) {
BVSETBLTERROR(BVERR_OOM,
- "callback allocation failed");
+ "callback allocation "
+ "failed");
goto exit;
}
@@ -3972,11 +4419,12 @@ enum bverror bv_blt(struct bvbltparams *bltparams)
case GCERR_OODM:
case GCERR_CTX_ALLOC:
BVSETBLTERROR(BVERR_OOM,
- "unable to allocate gccore memory");
+ "unable to allocate gccore "
+ "memory");
goto exit;
default:
BVSETBLTERROR(BVERR_RSRC,
- "gccore error");
+ "gccore error");
goto exit;
}
@@ -3993,7 +4441,7 @@ exit:
}
GCEXITARG(GCZONE_BLIT, "bv%s = %d\n",
- (bverror == BVERR_NONE) ? "result" : "error", bverror);
+ (bverror == BVERR_NONE) ? "result" : "error", bverror);
return bverror;
}
@@ -4027,18 +4475,23 @@ enum bverror bv_cache(struct bvcopparams *copparams)
case OCDFMTDEF_CONTAINER_8BIT:
container_size = 8;
break;
+
case OCDFMTDEF_CONTAINER_16BIT:
container_size = 16;
break;
+
case OCDFMTDEF_CONTAINER_24BIT:
container_size = 24;
break;
+
case OCDFMTDEF_CONTAINER_32BIT:
container_size = 32;
break;
+
case OCDFMTDEF_CONTAINER_48BIT:
container_size = 48;
break;
+
case OCDFMTDEF_CONTAINER_64BIT:
container_size = 64;
break;
@@ -4061,16 +4514,6 @@ enum bverror bv_cache(struct bvcopparams *copparams)
bytespp = (container_size / 2) / 8;
break;
- case OCDFMTDEF_SUBSAMPLE_420_YCbCr:
- bverror = BVERR_FORMAT;
- goto exit;
- break;
-
- case OCDFMTDEF_SUBSAMPLE_411_YCbCr:
- bverror = BVERR_FORMAT;
- goto exit;
- break;
-
default:
bverror = BVERR_FORMAT;
goto exit;