diff options
Diffstat (limited to 'drivers')
-rw-r--r-- | drivers/misc/gcx/gcbv/Makefile | 8 | ||||
-rw-r--r-- | drivers/misc/gcx/gcbv/gcblit.c | 750 | ||||
-rw-r--r-- | drivers/misc/gcx/gcbv/gcbuffer.c | 405 | ||||
-rw-r--r-- | drivers/misc/gcx/gcbv/gcbv.c | 3964 | ||||
-rw-r--r-- | drivers/misc/gcx/gcbv/gcbv.h | 459 | ||||
-rw-r--r-- | drivers/misc/gcx/gcbv/gcfill.c | 272 | ||||
-rw-r--r-- | drivers/misc/gcx/gcbv/gcfilter.c | 128 | ||||
-rw-r--r-- | drivers/misc/gcx/gcbv/gcmap.c | 364 | ||||
-rw-r--r-- | drivers/misc/gcx/gcbv/gcparser.c | 1602 | ||||
-rw-r--r-- | drivers/misc/gcx/gccore/gcdbglog.c | 38 | ||||
-rw-r--r-- | drivers/misc/gcx/gccore/gcmain.c | 9 | ||||
-rw-r--r-- | drivers/misc/gcx/gccore/gcmmu.c | 23 | ||||
-rw-r--r-- | drivers/misc/gcx/gccore/gcmmu.h | 3 | ||||
-rw-r--r-- | drivers/misc/gcx/gcioctl/gcif.c | 280 |
14 files changed, 4527 insertions, 3778 deletions
diff --git a/drivers/misc/gcx/gcbv/Makefile b/drivers/misc/gcx/gcbv/Makefile index 8f02893..9e7e6a5 100644 --- a/drivers/misc/gcx/gcbv/Makefile +++ b/drivers/misc/gcx/gcbv/Makefile @@ -2,4 +2,10 @@ obj-$(CONFIG_GCBV) += gcbv2d.o gcbv2d-y := \ gcmain.o \ - gcbv.o + gcbv.o \ + gcparser.o \ + gcmap.o \ + gcbuffer.o \ + gcfill.o \ + gcblit.o \ + gcfilter.o diff --git a/drivers/misc/gcx/gcbv/gcblit.c b/drivers/misc/gcx/gcbv/gcblit.c new file mode 100644 index 0000000..8fa2a00 --- /dev/null +++ b/drivers/misc/gcx/gcbv/gcblit.c @@ -0,0 +1,750 @@ +/* + * This file is provided under a dual BSD/GPLv2 license. When using or + * redistributing this file, you may do so under either license. + * + * GPL LICENSE SUMMARY + * + * Copyright(c) 2012 Vivante Corporation. All rights reserved. + * + * This program is free software; you can redistribute it and/or modify it + * under the terms and conditions of the GNU General Public License, + * version 2, as published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for + * more details. + * + * You should have received a copy of the GNU General Public License along with + * this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. + * + * The full GNU General Public License is included in this distribution in + * the file called "COPYING". + * + * BSD LICENSE + * + * Copyright(c) 2012 Vivante Corporation. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * * Neither the name of Vivante Corporation nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE + * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + */ + +#include "gcbv.h" + +#define GCZONE_NONE 0 +#define GCZONE_ALL (~0U) +#define GCZONE_BLEND (1 << 0) +#define GCZONE_SURF (1 << 1) +#define GCZONE_BLIT (1 << 3) + +GCDBG_FILTERDEF(gcblit, GCZONE_NONE, + "blend", + "surf", + "blit") + + +static enum bverror do_blit_end(struct bvbltparams *bltparams, + struct gcbatch *batch) +{ + enum bverror bverror; + struct gcmobltconfig *gcmobltconfig; + struct gcmostart *gcmostart; + + GCENTER(GCZONE_BLIT); + + GCDBG(GCZONE_BLIT, "finalizing the blit, scrcount = %d\n", + batch->gcblit.srccount); + + /*********************************************************************** + * Configure the operation. + */ + + /* Allocate command buffer. */ + bverror = claim_buffer(bltparams, batch, + sizeof(struct gcmobltconfig), + (void **) &gcmobltconfig); + if (bverror != BVERR_NONE) + goto exit; + + /* Configure multi-source control. */ + gcmobltconfig->multisource_ldst = gcmobltconfig_multisource_ldst; + gcmobltconfig->multisource.raw = 0; + gcmobltconfig->multisource.reg.srccount = batch->gcblit.srccount - 1; + + GCDBG(GCZONE_BLIT, "blockenable = %d\n", batch->blockenable); + if (batch->blockenable) { + 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_BLIT, "format entry = 0x%08X\n", + (unsigned int) batch->dstformat); + GCDBG(GCZONE_BLIT, " swizzle code = %d\n", + batch->dstformat->swizzle); + GCDBG(GCZONE_BLIT, " format code = %d\n", + batch->dstformat->format); + + 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. */ + 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->left; + gcmostart->rect.top = batch->top; + gcmostart->rect.right = batch->right; + gcmostart->rect.bottom = batch->bottom; + + GCDBG(GCZONE_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->right - batch->left), + abs(batch->bottom - batch->top)); + +exit: + GCEXITARG(GCZONE_BLIT, "bv%s = %d\n", + (bverror == BVERR_NONE) ? "result" : "error", bverror); + return bverror; +} + +enum bverror do_blit(struct bvbltparams *bltparams, + struct gcbatch *batch, + struct srcinfo *srcinfo) +{ + enum bverror bverror = BVERR_NONE; + struct gccontext *gccontext = get_context(); + + struct gcmosrc *gcmosrc; + struct gcmosrcalpha *gcmosrcalpha; + + unsigned int index; + + struct bvbuffmap *dstmap; + struct bvbuffdesc *dstdesc; + struct bvsurfgeom *dstgeom; + struct bvformatxlate *dstformat; + int dstshiftX, dstshiftY; + int dstalign, dstbyteshift; + + struct bvbuffmap *srcmap; + struct gcalpha *srcgca; + struct bvrect *srcrect; + struct bvbuffdesc *srcdesc; + struct bvsurfgeom *srcgeom; + 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_BLIT); + + /* Create source object shortcuts. */ + srcmap = NULL; + srcgca = srcinfo->gca; + srcrect = srcinfo->rect; + srcdesc = srcinfo->buf.desc; + srcgeom = srcinfo->geom; + srcformat = srcinfo->format; + + /* Create destination object shortcuts. */ + dstmap = NULL; + dstdesc = bltparams->dstdesc; + dstgeom = bltparams->dstgeom; + dstformat = batch->dstformat; + + /*********************************************************************** + * 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; + + case ROT_ANGLE_90: + srcshiftX = srctop - dsttop; + srcshiftY = (srcgeom->width - srcleft) + - (batch->dstwidth - dstleft); + break; + + case ROT_ANGLE_180: + srcshiftX = (srcgeom->width - srcleft) + - (batch->dstwidth - dstleft); + srcshiftY = (srcgeom->height - srctop) + - (batch->dstheight - dsttop); + break; + + case ROT_ANGLE_270: + srcshiftX = (srcgeom->height - srctop) + - (batch->dstheight - dsttop); + srcshiftY = srcleft - dstleft; + break; + + default: + srcshiftX = 0; + srcshiftY = 0; + } + + /* Compute the source surface offset in bytes. */ + srcbyteshift = srcshiftY * (int) srcgeom->virtstride + + srcshiftX * (int) srcformat->bitspp / 8; + + /* Compute the source offset in pixels needed to compensate + * for the surface base address misalignment if any. */ + srcalign = get_pixel_offset(srcdesc, srcformat, srcbyteshift); + + 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); + + /* Apply the source alignment. */ + srcbyteshift += srcalign * (int) srcformat->bitspp / 8; + srcshiftX += srcalign; + + GCDBG(GCZONE_SURF, " adjusted surface offset (pixels) = %d,%d\n", + srcshiftX, srcshiftY); + GCDBG(GCZONE_SURF, " adjusted surface offset (bytes) = 0x%08X\n", + srcbyteshift); + + /* Determine the destination surface shift. */ + dstshiftX = batch->dstalign; + dstshiftY = (((srcinfo->angle + 3) % 4) == batch->dstangle) + ? srcalign : 0; + + /* 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_pixel_offset(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->format == GCREG_DE_FORMAT_NV12) || + (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. */ + srcalign = get_pixel_offset(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; + + case ROT_ANGLE_90: + /* Adjust top coordinate. */ + srctop -= srcalign; + + /* Determine source size. */ + srcsurfwidth = srcgeom->height - srcalign; + srcsurfheight = srcgeom->width; + break; + + case ROT_ANGLE_180: + /* Determine source size. */ + srcsurfwidth = srcgeom->width - srcalign; + srcsurfheight = srcgeom->height; + break; + + case ROT_ANGLE_270: + /* Determine source size. */ + srcsurfwidth = srcgeom->height - srcalign; + srcsurfheight = srcgeom->width; + break; + + default: + srcsurfwidth = 0; + srcsurfheight = 0; + } + + GCDBG(GCZONE_SURF, "srcrect origin = %d,%d\n", + srcleft, srctop); + GCDBG(GCZONE_SURF, "source physical size = %dx%d\n", + srcsurfwidth, srcsurfheight); + + /* Set the physical destination size. */ + physwidth = batch->dstphyswidth; + physheight = batch->dstphysheight; + + /* 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; + + case ROT_ANGLE_90: + dsttop -= srcalign; + dstbottom -= srcalign; + + /* 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; + + 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; + + 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; + + default: + physwidth = 0; + physheight = 0; + } + + /* Source geometry is now the same as the destination. */ + if (orthogonal) { + srcsurfwidth = physheight; + srcsurfheight = physwidth; + } else { + srcsurfwidth = physwidth; + srcsurfheight = physheight; + } + + /* Enable multi-source. */ + multisrc = 1; + GCDBG(GCZONE_SURF, "multi-source enabled.\n"); + } + + /* 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; + + /* 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; + } + + /* Set destination coordinates. */ + batch->left = dstleft; + batch->top = dsttop; + batch->right = dstright; + batch->bottom = dstbottom; + + /* Map the destination. */ + bverror = do_map(dstdesc, batch, &dstmap); + if (bverror != BVERR_NONE) { + bltparams->errdesc = gccontext->bverrorstr; + goto exit; + } + + /* Set the new destination. */ + bverror = set_dst(bltparams, batch, dstmap); + if (bverror != BVERR_NONE) + goto exit; + + /* Reset the modified flag. */ + batch->batchflags &= ~(BVBATCH_DST | + BVBATCH_CLIPRECT | + BVBATCH_DESTRECT); + + /* Map the source. */ + bverror = do_map(srcdesc, batch, &srcmap); + if (bverror != BVERR_NONE) { + bltparams->errdesc = gccontext->bverrorstr; + goto exit; + } + + /*************************************************************** + ** Configure source. + */ + + /* We need to walk in blocks if the source and the destination + * surfaces are orthogonal to each other. */ + batch->blockenable = orthogonal; + + /* Allocate command buffer. */ + bverror = claim_buffer(bltparams, batch, + sizeof(struct gcmosrc), + (void **) &gcmosrc); + if (bverror != BVERR_NONE) + goto exit; + + /* Shortcut to the register index. */ + index = batch->gcblit.srccount; + + add_fixup(bltparams, batch, &gcmosrc->address, srcbyteshift); + + /* 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; + + if ((dstgeom->format & OCDFMTDEF_NON_PREMULT) != 0) { + gcmosrc->mult.reg.dstpremul + = GCREG_COLOR_MULTIPLY_MODES_SRC_PREMULTIPLY_ENABLE; + + gcmosrc->mult.reg.dstdemul + = GCREG_COLOR_MULTIPLY_MODES_DST_DEMULTIPLY_ENABLE; + } else { + gcmosrc->mult.reg.dstpremul + = GCREG_COLOR_MULTIPLY_MODES_SRC_PREMULTIPLY_DISABLE; + + gcmosrc->mult.reg.dstdemul + = GCREG_COLOR_MULTIPLY_MODES_DST_DEMULTIPLY_DISABLE; + } + + if (srcformat->format == GCREG_DE_FORMAT_NV12) { + struct gcmosrcplanaryuv *yuv; + int uvshift = srcbyteshift; + +#if 0 + /* TODO: needs rework */ + 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 + + bverror = claim_buffer(bltparams, batch, + sizeof(struct gcmosrcplanaryuv), + (void **) &yuv); + if (bverror != BVERR_NONE) + goto exit; + + 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 + /* TODO: needs rework */ + 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; + } +#endif + + GCDBG(GCZONE_SURF, " uvshift = 0x%08X (%d)\n", + uvshift, uvshift); + + /* add fixed offset from Y plane */ + uvshift += srcgeom->virtstride * srcgeom->height; + + GCDBG(GCZONE_SURF, " final uvshift = 0x%08X (%d)\n", + uvshift, uvshift); + + yuv->uplaneaddress = GET_MAP_HANDLE(srcmap); + add_fixup(bltparams, batch, &yuv->uplaneaddress, uvshift); + + yuv->uplanestride = srcgeom->virtstride; + + yuv->vplaneaddress = GET_MAP_HANDLE(srcmap); + add_fixup(bltparams, batch, &yuv->vplaneaddress, uvshift); + + yuv->vplanestride = srcgeom->virtstride; + } + + if (srcgca != NULL) { + gcmosrc->alphacontrol_ldst + = gcmosrc_alphacontrol_ldst[index]; + gcmosrc->alphacontrol.raw = 0; + gcmosrc->alphacontrol.reg.enable + = GCREG_ALPHA_CONTROL_ENABLE_ON; + + /* Allocate command buffer. */ + bverror = claim_buffer(bltparams, batch, + sizeof(struct gcmosrcalpha), + (void **) &gcmosrcalpha); + if (bverror != BVERR_NONE) + goto exit; + + 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"); + + 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_BLIT, "bv%s = %d\n", + (bverror == BVERR_NONE) ? "result" : "error", bverror); + return bverror; +} diff --git a/drivers/misc/gcx/gcbv/gcbuffer.c b/drivers/misc/gcx/gcbv/gcbuffer.c new file mode 100644 index 0000000..484eb08 --- /dev/null +++ b/drivers/misc/gcx/gcbv/gcbuffer.c @@ -0,0 +1,405 @@ +/* + * This file is provided under a dual BSD/GPLv2 license. When using or + * redistributing this file, you may do so under either license. + * + * GPL LICENSE SUMMARY + * + * Copyright(c) 2012 Vivante Corporation. All rights reserved. + * + * This program is free software; you can redistribute it and/or modify it + * under the terms and conditions of the GNU General Public License, + * version 2, as published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for + * more details. + * + * You should have received a copy of the GNU General Public License along with + * this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. + * + * The full GNU General Public License is included in this distribution in + * the file called "COPYING". + * + * BSD LICENSE + * + * Copyright(c) 2012 Vivante Corporation. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * * Neither the name of Vivante Corporation nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE + * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + */ + +#include "gcbv.h" + +#define GCZONE_NONE 0 +#define GCZONE_ALL (~0U) +#define GCZONE_BATCH_ALLOC (1 << 0) +#define GCZONE_BUFFER_ALLOC (1 << 1) +#define GCZONE_FIXUP_ALLOC (1 << 2) +#define GCZONE_FIXUP (1 << 3) + +GCDBG_FILTERDEF(gcbuffer, GCZONE_NONE, + "batchalloc", + "bufferalloc" + "fixupalloc", + "fixup") + + +/******************************************************************************* +** Miscellaneous defines and macros. +*/ + +#define GC_BUFFER_INIT_SIZE \ +( \ + GC_BUFFER_SIZE - max(sizeof(struct gcbuffer), GC_BUFFER_RESERVE) \ +) + +#define GC_BUFFER_RESERVE \ +( \ + sizeof(struct gcmopipesel) + \ + sizeof(struct gcmommumaster) + \ + sizeof(struct gcmommuflush) + \ + sizeof(struct gcmosignal) + \ + sizeof(struct gccmdend) \ +) + + +/******************************************************************************* + * Batch/command buffer management. + */ + +enum bverror do_end(struct bvbltparams *bvbltparams, + struct gcbatch *gcbatch) +{ + return BVERR_NONE; +} + +enum bverror allocate_batch(struct bvbltparams *bvbltparams, + struct gcbatch **gcbatch) +{ + enum bverror bverror; + struct gccontext *gccontext = get_context(); + struct gcbatch *temp; + struct gcbuffer *gcbuffer; + + GCENTER(GCZONE_BATCH_ALLOC); + + /* Lock access to batch management. */ + GCLOCK(&gccontext->batchlock); + + if (list_empty(&gccontext->batchvac)) { + temp = gcalloc(struct gcbatch, sizeof(struct gcbatch)); + if (temp == NULL) { + BVSETBLTERROR(BVERR_OOM, + "batch header allocation failed"); + goto exit; + } + + GCDBG(GCZONE_BATCH_ALLOC, "allocated new batch = 0x%08X\n", + (unsigned int) temp); + } else { + struct list_head *head; + head = gccontext->batchvac.next; + temp = list_entry(head, struct gcbatch, link); + list_del(head); + + GCDBG(GCZONE_BATCH_ALLOC, "reusing batch = 0x%08X\n", + (unsigned int) temp); + } + + memset(temp, 0, sizeof(struct gcbatch)); + temp->structsize = sizeof(struct gcbatch); + temp->batchend = do_end; + INIT_LIST_HEAD(&temp->buffer); + INIT_LIST_HEAD(&temp->unmap); + INIT_LIST_HEAD(&temp->link); + + bverror = append_buffer(bvbltparams, temp, &gcbuffer); + if (bverror != BVERR_NONE) { + free_batch(temp); + goto exit; + } + + *gcbatch = temp; + + GCDBG(GCZONE_BATCH_ALLOC, "batch allocated = 0x%08X\n", + (unsigned int) temp); + +exit: + /* Unlock access to batch management. */ + GCUNLOCK(&gccontext->batchlock); + + GCEXITARG(GCZONE_BATCH_ALLOC, "bv%s = %d\n", + (bverror == BVERR_NONE) ? "result" : "error", bverror); + return bverror; +} + +void free_batch(struct gcbatch *gcbatch) +{ + struct list_head *head; + struct gccontext *gccontext = get_context(); + struct gcbuffer *gcbuffer; + + GCENTERARG(GCZONE_BATCH_ALLOC, "batch = 0x%08X\n", + (unsigned int) gcbatch); + + /* Lock access. */ + GCLOCK(&gccontext->batchlock); + GCLOCK(&gccontext->bufferlock); + GCLOCK(&gccontext->fixuplock); + GCLOCK(&gccontext->maplock); + + /* Free implicit unmappings. */ + list_splice_init(&gcbatch->unmap, &gccontext->unmapvac); + + /* Free command buffers. */ + while (!list_empty(&gcbatch->buffer)) { + head = gcbatch->buffer.next; + gcbuffer = list_entry(head, struct gcbuffer, link); + + /* Free fixups. */ + list_splice_init(&gcbuffer->fixup, &gccontext->fixupvac); + + /* Free the command buffer. */ + list_move(&gcbuffer->link, &gccontext->buffervac); + } + + /* Free the batch. */ + list_add(&gcbatch->link, &gccontext->batchvac); + + /* Unlock access. */ + GCUNLOCK(&gccontext->maplock); + GCUNLOCK(&gccontext->fixuplock); + GCUNLOCK(&gccontext->bufferlock); + GCUNLOCK(&gccontext->batchlock); + + GCEXIT(GCZONE_BATCH_ALLOC); +} + +enum bverror append_buffer(struct bvbltparams *bvbltparams, + struct gcbatch *gcbatch, + struct gcbuffer **gcbuffer) +{ + enum bverror bverror; + struct gccontext *gccontext = get_context(); + struct gcbuffer *temp; + + GCENTERARG(GCZONE_BUFFER_ALLOC, "batch = 0x%08X\n", + (unsigned int) gcbatch); + + /* Lock access to buffer management. */ + GCLOCK(&gccontext->bufferlock); + + if (list_empty(&gccontext->buffervac)) { + temp = gcalloc(struct gcbuffer, GC_BUFFER_SIZE); + if (temp == NULL) { + BVSETBLTERROR(BVERR_OOM, + "command buffer allocation failed"); + goto exit; + } + + list_add_tail(&temp->link, &gcbatch->buffer); + + GCDBG(GCZONE_BUFFER_ALLOC, "allocated new buffer = 0x%08X\n", + (unsigned int) temp); + } else { + struct list_head *head; + head = gccontext->buffervac.next; + temp = list_entry(head, struct gcbuffer, link); + + list_move_tail(&temp->link, &gcbatch->buffer); + + GCDBG(GCZONE_BUFFER_ALLOC, "reusing buffer = 0x%08X\n", + (unsigned int) temp); + } + + INIT_LIST_HEAD(&temp->fixup); + temp->pixelcount = 0; + temp->head = temp->tail = (unsigned int *) (temp + 1); + temp->available = GC_BUFFER_INIT_SIZE; + + GCDBG(GCZONE_BUFFER_ALLOC, "new buffer appended = 0x%08X\n", + (unsigned int) temp); + + *gcbuffer = temp; + bverror = BVERR_NONE; + +exit: + /* Unlock access to buffer management. */ + GCUNLOCK(&gccontext->bufferlock); + + GCEXITARG(GCZONE_BUFFER_ALLOC, "bv%s = %d\n", + (bverror == BVERR_NONE) ? "result" : "error", bverror); + return bverror; +} + +static enum bverror allocate_fixup(struct bvbltparams *bvbltparams, + struct gcbuffer *gcbuffer, + struct gcfixup **gcfixup) +{ + enum bverror bverror = BVERR_NONE; + struct gccontext *gccontext = get_context(); + struct gcfixup *temp; + + if (list_empty(&gccontext->fixupvac)) { + temp = gcalloc(struct gcfixup, sizeof(struct gcfixup)); + if (temp == NULL) { + BVSETBLTERROR(BVERR_OOM, "fixup allocation failed"); + goto exit; + } + + list_add_tail(&temp->link, &gcbuffer->fixup); + + GCDBG(GCZONE_FIXUP_ALLOC, + "new fixup struct allocated = 0x%08X\n", + (unsigned int) temp); + } else { + struct list_head *head; + head = gccontext->fixupvac.next; + temp = list_entry(head, struct gcfixup, link); + + list_move_tail(&temp->link, &gcbuffer->fixup); + + GCDBG(GCZONE_FIXUP_ALLOC, "fixup struct reused = 0x%08X\n", + (unsigned int) temp); + } + + temp->count = 0; + *gcfixup = temp; + +exit: + return bverror; +} + +enum bverror add_fixup(struct bvbltparams *bvbltparams, + struct gcbatch *gcbatch, + unsigned int *ptr, + unsigned int surfoffset) +{ + enum bverror bverror = BVERR_NONE; + struct gccontext *gccontext = get_context(); + struct list_head *head; + struct gcbuffer *buffer; + struct gcfixup *gcfixup; + + GCENTERARG(GCZONE_FIXUP, "batch = 0x%08X, fixup ptr = 0x%08X\n", + (unsigned int) gcbatch, (unsigned int) ptr); + + /* Lock access to fixup management. */ + GCLOCK(&gccontext->fixuplock); + + /* Get the current command buffer. */ + if (list_empty(&gcbatch->buffer)) { + GCERR("no command buffers are allocated"); + goto exit; + } + head = gcbatch->buffer.prev; + buffer = list_entry(head, struct gcbuffer, link); + + /* No fixups? Allocate one. */ + if (list_empty(&buffer->fixup)) { + GCDBG(GCZONE_FIXUP_ALLOC, "no fixups allocated.\n"); + bverror = allocate_fixup(bvbltparams, buffer, &gcfixup); + if (bverror != BVERR_NONE) + goto exit; + } else { + /* Get the current fixup. */ + head = buffer->fixup.prev; + gcfixup = list_entry(head, struct gcfixup, link); + + /* No more room? */ + if (gcfixup->count == GC_FIXUP_MAX) { + GCDBG(GCZONE_FIXUP_ALLOC, + "out of room, allocating new.\n"); + bverror = allocate_fixup(bvbltparams, buffer, &gcfixup); + if (bverror != BVERR_NONE) + goto exit; + } + } + + GCDBG(GCZONE_FIXUP, "buffer = 0x%08X, fixup struct = 0x%08X\n", + (unsigned int) buffer, (unsigned int) gcfixup); + + gcfixup->fixup[gcfixup->count].dataoffset = ptr - buffer->head; + gcfixup->fixup[gcfixup->count].surfoffset = surfoffset; + gcfixup->count += 1; + + GCDBG(GCZONE_FIXUP, "fixup offset = 0x%08X\n", ptr - buffer->head); + GCDBG(GCZONE_FIXUP, "surface offset = 0x%08X\n", surfoffset); + +exit: + /* Unlock access to fixup management. */ + GCUNLOCK(&gccontext->fixuplock); + + GCEXITARG(GCZONE_FIXUP, "bv%s = %d\n", + (bverror == BVERR_NONE) ? "result" : "error", bverror); + return bverror; +} + +enum bverror claim_buffer(struct bvbltparams *bvbltparams, + struct gcbatch *gcbatch, + unsigned int size, + void **buffer) +{ + enum bverror bverror; + struct list_head *head; + struct gcbuffer *gcbuffer; + + GCENTERARG(GCZONE_BUFFER_ALLOC, "batch = 0x%08X, size = %d\n", + (unsigned int) gcbatch, size); + + if (size > GC_BUFFER_INIT_SIZE) { + GCERR("requested size is too big.\n"); + BVSETBLTERROR(BVERR_OOM, + "command buffer allocation failed"); + goto exit; + } + + /* Get the current command buffer. */ + head = gcbatch->buffer.prev; + gcbuffer = list_entry(head, struct gcbuffer, link); + + GCDBG(GCZONE_BUFFER_ALLOC, "buffer = 0x%08X, available = %d\n", + (unsigned int) gcbuffer, gcbuffer->available); + + if (gcbuffer->available < size) { + bverror = append_buffer(bvbltparams, gcbatch, &gcbuffer); + if (bverror != BVERR_NONE) + goto exit; + } + + *buffer = gcbuffer->tail; + gcbuffer->tail = (unsigned int *) + ((unsigned char *) gcbuffer->tail + size); + gcbuffer->available -= size; + gcbatch->size += size; + bverror = BVERR_NONE; + +exit: + GCEXITARG(GCZONE_BUFFER_ALLOC, "bv%s = %d\n", + (bverror == BVERR_NONE) ? "result" : "error", bverror); + return bverror; +} diff --git a/drivers/misc/gcx/gcbv/gcbv.c b/drivers/misc/gcx/gcbv/gcbv.c index cd666d5..170d4b4 100644 --- a/drivers/misc/gcx/gcbv/gcbv.c +++ b/drivers/misc/gcx/gcbv/gcbv.c @@ -52,335 +52,39 @@ * POSSIBILITY OF SUCH DAMAGE. */ -#include "gcmain.h" +#include "gcbv.h" #define GCZONE_NONE 0 #define GCZONE_ALL (~0U) #define GCZONE_MAPPING (1 << 0) -#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) +#define GCZONE_BUFFER (1 << 1) +#define GCZONE_DEST (1 << 2) +#define GCZONE_SRC (1 << 3) +#define GCZONE_MASK (1 << 4) +#define GCZONE_BATCH (1 << 5) +#define GCZONE_BLIT (1 << 6) +#define GCZONE_CACHE (1 << 7) GCDBG_FILTERDEF(gcbv, GCZONE_NONE, "mapping", "buffer", - "fixup", - "format", - "color", - "blend", "dest", - "surf", - "do_fill", - "do_blit", - "do_filter", + "source", + "mask", "batch", "blit", "cache") /******************************************************************************* -** Miscellaneous defines and macros. +** Global driver data access. */ -#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) \ -) - -#define STRUCTSIZE(structptr, lastmember) \ -( \ - (size_t) &structptr->lastmember + \ - sizeof(structptr->lastmember) - \ - (size_t) structptr \ -) - -#define GET_MAP_HANDLE(map) \ -( \ - ((struct bvbuffmapinfo *) map->handle)->handle \ -) - -#define GC_CLIP_RESET_LEFT ((unsigned short) 0) -#define GC_CLIP_RESET_TOP ((unsigned short) 0) -#define GC_CLIP_RESET_RIGHT ((unsigned short) ((1 << 15) - 1)) -#define GC_CLIP_RESET_BOTTOM ((unsigned short) ((1 << 15) - 1)) - -#define GPU_CMD_SIZE (sizeof(unsigned int) * 2) - -#define GC_BUFFER_RESERVE \ -( \ - sizeof(struct gcmopipesel) + \ - sizeof(struct gcmommumaster) + \ - sizeof(struct gcmommuflush) + \ - sizeof(struct gcmosignal) + \ - sizeof(struct gccmdend) \ -) - -#if GCDEBUG_ENABLE -#define GCDUMPSCHEDULE() { \ - struct list_head *__temphead__; \ - struct gcschedunmap *__tempunmap__; \ - GCDBG(GCZONE_MAPPING, "scheduled unmaps:\n"); \ - list_for_each(__temphead__, &batch->unmap) { \ - __tempunmap__ = list_entry(__temphead__, \ - struct gcschedunmap, \ - link); \ - GCDBG(GCZONE_MAPPING, \ - " handle = 0x%08X\n", \ - (unsigned int) __tempunmap__->handle); \ - } \ +struct gccontext *get_context(void) +{ + static struct gccontext gccontext; + return &gccontext; } -#else -#define GCDUMPSCHEDULE() -#endif - - -/******************************************************************************* -** 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. */ -struct bvbuffmapinfo { - /* Mapped handle for the buffer. */ - unsigned long handle; - - /* Number of times the client explicitly mapped this buffer. */ - int usermap; - - /* Number of times implicit mapping happened. */ - int automap; -}; - -/* Operation finalization call. */ -struct gcbatch; -typedef enum bverror (*gcbatchend) (struct bvbltparams *bltparams, - struct gcbatch *batch); - -/* Blit states. */ -struct gcblit { - unsigned int srccount; - unsigned int multisrc; - unsigned short rop; -}; - -/* Batch header. */ -struct gcbatch { - /* Used to ID structure version. */ - unsigned int structsize; - - /* Batch change flags. */ - unsigned long batchflags; - - /* Pointer to the function to finalize the current operation. */ - gcbatchend batchend; - - /* 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; - int deltatop; - int deltaright; - int deltabottom; - - /* 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; - - /* Physical size of the matched destination and source surfaces - * for multi-source setup. */ - unsigned int physwidth; - unsigned int physheight; - - /* 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. */ - struct bvrect prevdstrect; - struct bvrect prevsrc1rect; - struct bvrect prevsrc2rect; - struct bvrect prevmaskrect; -#endif - - /* Total size of the command buffer. */ - unsigned int size; - - /* Command buffer list. */ - struct gcbuffer *bufhead; - struct gcbuffer *buftail; - - /* Scheduled implicit unmappings (gcschedunmap). */ - struct list_head unmap; -}; - -/* Vacant batch header. */ -struct gcvacbatch { - struct gcvacbatch *next; -}; - -/* Callback information. */ -struct gccallbackinfo { - /* BLTsville callback function. */ - void (*callbackfn) (struct bvcallbackerror *err, - unsigned long callbackdata); - - /* Callback data. */ - unsigned long callbackdata; - - /* Previous/next callback information. */ - struct list_head link; -}; - -/* Driver context. */ -struct gccontext { - /* Last generated error message. */ - char bverrorstr[128]; - - /* Dynamically allocated structure cache. */ - struct list_head vac_unmap; /* gcschedunmap */ - struct bvbuffmap *vac_buffmap; - struct gcbuffer *vac_buffers; - struct gcfixup *vac_fixups; - struct gcvacbatch *vac_batches; - - /* Callback lists. */ - struct list_head callbacklist; - struct list_head callbackvac; - - /* Access locks. */ - GCLOCK_TYPE batchlock; - GCLOCK_TYPE bufferlock; - GCLOCK_TYPE fixuplock; - GCLOCK_TYPE maplock; - GCLOCK_TYPE callbacklock; -}; - -static struct gccontext gccontext; /******************************************************************************* @@ -391,46 +95,85 @@ static struct gccontext gccontext; #define GCDUMPBATCH(batch) \ dumpbatch(batch) -static void dumpbatch(struct gcbatch *batch) -{ - if ((GCDBGFILTER.zone & (GCZONE_BUFFER)) != 0) { - struct gcbuffer *buffer; - unsigned int i, size; - struct gcfixup *fixup; - - 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_BUFFER, - " Fixup table @ 0x%08X, count = %d:\n", - (unsigned int) fixup, fixup->count); - - for (i = 0; i < fixup->count; i += 1) { - GCDBG(GCZONE_BUFFER, " [%02d]" - " buffer offset = 0x%08X," - " surface offset = 0x%08X\n", - i, - fixup->fixup[i].dataoffset * 4, - fixup->fixup[i].surfoffset); - } +#define GCVERIFYBATCH(changeflags, prevrect, currrect) \ + verify_batch(changeflags, prevrect, currrect) - fixup = fixup->next; +static void dumpbatch(struct gcbatch *gcbatch) +{ + struct list_head *gcbufferhead; + struct gcbuffer *gcbuffer; + struct list_head *gcfixuphead; + struct gcfixup *gcfixup; + unsigned int i, size; + + if ((GCDBGFILTER.zone & (GCZONE_BUFFER)) == 0) + return; + + GCDBG(GCZONE_BUFFER, "BATCH DUMP (0x%08X)\n", + (unsigned int) gcbatch); + + list_for_each(gcbufferhead, &gcbatch->buffer) { + gcbuffer = list_entry(gcbufferhead, struct gcbuffer, link); + + list_for_each(gcfixuphead, &gcbuffer->fixup) { + gcfixup = list_entry(gcfixuphead, struct gcfixup, link); + + GCDBG(GCZONE_BUFFER, + " Fixup table @ 0x%08X, count = %d:\n", + (unsigned int) gcfixup, gcfixup->count); + + for (i = 0; i < gcfixup->count; i += 1) { + GCDBG(GCZONE_BUFFER, " [%02d]" + " buffer offset = 0x%08X," + " surface offset = 0x%08X\n", + i, + gcfixup->fixup[i].dataoffset * 4, + gcfixup->fixup[i].surfoffset); } + } - size = (unsigned char *) buffer->tail - - (unsigned char *) buffer->head; - GCDUMPBUFFER(GCZONE_BUFFER, buffer->head, 0, size); + size = (unsigned char *) gcbuffer->tail + - (unsigned char *) gcbuffer->head; + GCDUMPBUFFER(GCZONE_BUFFER, gcbuffer->head, 0, size); + } +} - buffer = buffer->next; +static void verify_batch(unsigned int changeflags, + struct bvrect *prevrect, + struct bvrect *currrect) +{ + if ((changeflags & 1) == 0) { + /* Origin did not change. */ + if ((prevrect->left != currrect->left) || + (prevrect->top != currrect->top)) { + GCERR("origin changed\n"); + GCERR(" previous = %d,%d\n", + prevrect->left, prevrect->top); + GCERR(" current = %d,%d\n", + currrect->left, currrect->top); + } + } + + if ((changeflags & 2) == 0) { + /* Size did not change. */ + if ((prevrect->width != currrect->width) || + (prevrect->height != currrect->height)) { + GCERR("size changed\n"); + GCERR(" previous = %dx%d\n", + prevrect->width, prevrect->height); + GCERR(" current = %dx%d\n", + currrect->width, currrect->height); } } + + prevrect->left = currrect->left; + prevrect->top = currrect->top; + prevrect->width = currrect->width; + prevrect->height = currrect->height; } #else -# define GCDUMPBATCH(...) +#define GCDUMPBATCH(...) +#define GCVERIFYBATCH(...) #endif @@ -438,29 +181,15 @@ static void dumpbatch(struct gcbatch *batch) * Error handling. */ -#define BVSETERROR(error, message, ...) \ -do { \ - snprintf(gccontext.bverrorstr, sizeof(gccontext.bverrorstr), \ - 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, ##__VA_ARGS__); \ - bltparams->errdesc = gccontext.bverrorstr; \ -} while (0) - #define BVSETBLTSURFERROR(errorid, errordesc) \ do { \ - snprintf(gccontext.bverrorstr, sizeof(gccontext.bverrorstr), \ + struct gccontext *tmpcontext = get_context(); \ + snprintf(tmpcontext->bverrorstr, sizeof(tmpcontext->bverrorstr), \ g_surferr[errorid].message, errordesc.id); \ GCDUMPSTRING("%s(%d): [ERROR] %s\n", __func__, __LINE__, \ - gccontext.bverrorstr); \ + tmpcontext->bverrorstr); \ bverror = errordesc.base + g_surferr[errorid].offset; \ - bltparams->errdesc = gccontext.bverrorstr; \ + bvbltparams->errdesc = tmpcontext->bverrorstr; \ } while (0) #define GCBVERR_DESC 0 @@ -522,27 +251,41 @@ static struct bvsurferrorid g_masksurferr = { "mask", BVERR_MASKDESC }; * Callback info management. */ +/* Callback information. */ +struct gccallbackinfo { + /* BLTsville callback function. */ + void (*callbackfn) (struct bvcallbackerror *err, + unsigned long callbackdata); + + /* Callback data. */ + unsigned long callbackdata; + + /* Previous/next callback information. */ + struct list_head link; +}; + static enum bverror get_callbackinfo(struct gccallbackinfo **gccallbackinfo) { enum bverror bverror; + struct gccontext *gccontext = get_context(); struct gccallbackinfo *temp; /* Lock access to callback info lists. */ - GCLOCK(&gccontext.callbacklock); + GCLOCK(&gccontext->callbacklock); - if (list_empty(&gccontext.callbackvac)) { + if (list_empty(&gccontext->callbackvac)) { temp = gcalloc(struct gccallbackinfo, sizeof(struct gccallbackinfo)); if (temp == NULL) { bverror = BVERR_OOM; goto exit; } - list_add(&temp->link, &gccontext.callbacklist); + list_add(&temp->link, &gccontext->callbacklist); } else { struct list_head *head; - head = gccontext.callbackvac.next; + head = gccontext->callbackvac.next; temp = list_entry(head, struct gccallbackinfo, link); - list_move(head, &gccontext.callbacklist); + list_move(head, &gccontext->callbacklist); } *gccallbackinfo = temp; @@ -550,20 +293,22 @@ static enum bverror get_callbackinfo(struct gccallbackinfo **gccallbackinfo) exit: /* Unlock access to callback info lists. */ - GCUNLOCK(&gccontext.callbacklock); + GCUNLOCK(&gccontext->callbacklock); return bverror; } static void free_callback(struct gccallbackinfo *gccallbackinfo) { + struct gccontext *gccontext = get_context(); + /* Lock access to callback info lists. */ - GCLOCK(&gccontext.callbacklock); + GCLOCK(&gccontext->callbacklock); - list_move(&gccallbackinfo->link, &gccontext.callbackvac); + list_move(&gccallbackinfo->link, &gccontext->callbackvac); /* Unlock access to callback info lists. */ - GCUNLOCK(&gccontext.callbacklock); + GCUNLOCK(&gccontext->callbacklock); } void gccallback(void *callbackparam) @@ -586,1780 +331,59 @@ void gccallback(void *callbackparam) /******************************************************************************* - * Memory management. + * Program the destination. */ -static enum bverror do_map(struct bvbuffdesc *bvbuffdesc, - struct gcbatch *batch, - struct bvbuffmap **map) -{ - static const int mapsize - = sizeof(struct bvbuffmap) - + sizeof(struct bvbuffmapinfo); - - enum bverror bverror; - struct bvbuffmap *bvbuffmap; - struct bvbuffmapinfo *bvbuffmapinfo; - struct bvphysdesc *bvphysdesc; - bool mappedbyothers; - struct gcmap gcmap; - struct gcschedunmap *gcschedunmap; - - GCENTERARG(GCZONE_MAPPING, "bvbuffdesc = 0x%08X\n", - (unsigned int) bvbuffdesc); - - /* Lock access to the mapping list. */ - GCLOCK(&gccontext.maplock); - - /* Try to find existing mapping. */ - bvbuffmap = bvbuffdesc->map; - while (bvbuffmap != NULL) { - if (bvbuffmap->bv_unmap == bv_unmap) - break; - bvbuffmap = bvbuffmap->nextmap; - } - - /* Not mapped yet? */ - if (bvbuffmap == NULL) { - /* New mapping, allocate a record. */ - if (gccontext.vac_buffmap == NULL) { - bvbuffmap = gcalloc(struct bvbuffmap, mapsize); - if (bvbuffmap == NULL) { - BVSETERROR(BVERR_OOM, - "failed to allocate mapping record"); - goto fail; - } - - bvbuffmap->structsize = sizeof(struct bvbuffmap); - bvbuffmap->bv_unmap = bv_unmap; - bvbuffmap->handle = (unsigned long) (bvbuffmap + 1); - } else { - bvbuffmap = gccontext.vac_buffmap; - gccontext.vac_buffmap = bvbuffmap->nextmap; - } - - /* Setup buffer mapping. Here we need to check and make sure - * that the buffer starts at a location that is supported by - * the hw. If it is not, offset is computed and the buffer is - * extended by the value of the offset. */ - gcmap.gcerror = GCERR_NONE; - gcmap.handle = 0; - - if (bvbuffdesc->auxtype == BVAT_PHYSDESC) { - bvphysdesc = (struct bvphysdesc *) bvbuffdesc->auxptr; - - if (bvphysdesc->structsize < - STRUCTSIZE(bvphysdesc, pageoffset)) { - BVSETERROR(BVERR_BUFFERDESC_VERS, - "unsupported bvphysdesc version"); - goto fail; - } - - 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"); - GCDBG(GCZONE_MAPPING, "pagesize = %lu\n", - bvphysdesc->pagesize); - GCDBG(GCZONE_MAPPING, "pagearray = 0x%08X\n", - (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 = bvbuffdesc->virtaddr; - gcmap.pagesize = 0; - gcmap.pagearray = NULL; - gcmap.size = bvbuffdesc->length; - - GCDBG(GCZONE_MAPPING, "new mapping (%s):\n", - (batch == NULL) ? "explicit" : "implicit"); - GCDBG(GCZONE_MAPPING, "specified virtaddr = 0x%08X\n", - (unsigned int) bvbuffdesc->virtaddr); - GCDBG(GCZONE_MAPPING, "aligned virtaddr = 0x%08X\n", - (unsigned int) gcmap.buf.logical); - 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"); - goto fail; - } - - /* Set map handle. */ - bvbuffmapinfo = (struct bvbuffmapinfo *) bvbuffmap->handle; - bvbuffmapinfo->handle = gcmap.handle; - - /* Initialize reference counters. */ - if (batch == NULL) { - /* Explicit mapping. */ - bvbuffmapinfo->usermap = 1; - bvbuffmapinfo->automap = 0; - } else { - /* Implicit mapping; if there are existing mappings - * from other implementations, mark this an explicit - * mapping as well. */ - mappedbyothers = (bvbuffdesc->map != NULL); - GCDBG(GCZONE_MAPPING, "%smapped by others.\n", - mappedbyothers ? "" : "not "); - - bvbuffmapinfo->usermap = mappedbyothers ? 1 : 0; - bvbuffmapinfo->automap = 1; - } - - /* Add the record to the list of mappings. */ - bvbuffmap->nextmap = bvbuffdesc->map; - bvbuffdesc->map = bvbuffmap; - } else { - /* Mapping already exists. */ - bvbuffmapinfo = (struct bvbuffmapinfo *) bvbuffmap->handle; - - /* Advance reference counters. */ - if (batch == NULL) { - /* Explicit mapping. */ - GCDBG(GCZONE_MAPPING, "explicit map.\n"); - bvbuffmapinfo->usermap += 1; - } else { - /* Implicit mapping. */ - GCDBG(GCZONE_MAPPING, "implicit map.\n"); - bvbuffmapinfo->automap += 1; - } - - GCDBG(GCZONE_MAPPING, "mapping exists.\n"); - } - - GCDBG(GCZONE_MAPPING, "bvbuffmap = 0x%08X\n", - (unsigned int) bvbuffmap); - GCDBG(GCZONE_MAPPING, "explicit count = %d\n", - bvbuffmapinfo->usermap); - GCDBG(GCZONE_MAPPING, "implicit count = %d\n", - bvbuffmapinfo->automap); - - /* Schedule for unmapping. */ - if (batch != NULL) { - if (list_empty(&gccontext.vac_unmap)) { - gcschedunmap = gcalloc(struct gcschedunmap, - sizeof(struct gcschedunmap)); - if (gcschedunmap == NULL) { - BVSETERROR(BVERR_OOM, - "failed to schedule unmapping"); - goto fail; - } - list_add(&gcschedunmap->link, &batch->unmap); - } else { - struct list_head *head; - head = gccontext.vac_unmap.next; - gcschedunmap = list_entry(head, - struct gcschedunmap, - link); - list_move(&gcschedunmap->link, &batch->unmap); - } - - gcschedunmap->handle = (unsigned long) bvbuffdesc; - - GCDBG(GCZONE_MAPPING, "scheduled for unmapping.\n"); - GCDUMPSCHEDULE(); - } - - /* Set the map pointer. */ - *map = bvbuffmap; - - /* Unlock access to the mapping list. */ - GCUNLOCK(&gccontext.maplock); - - GCEXITARG(GCZONE_MAPPING, "handle = 0x%08X\n", - bvbuffmapinfo->handle); - return BVERR_NONE; - -fail: - if (bvbuffmap != NULL) { - bvbuffmap->nextmap = gccontext.vac_buffmap; - gccontext.vac_buffmap = bvbuffmap; - } - - /* Unlock access to the mapping list. */ - GCUNLOCK(&gccontext.maplock); - - GCEXITARG(GCZONE_MAPPING, "bverror = %d\n", bverror); - return bverror; -} - -static void unmap_implicit(struct gcbatch *batch) -{ - struct list_head *head, *temphead; - struct gcschedunmap *gcschedunmap; - struct bvbuffdesc *bvbuffdesc; - struct bvbuffmap *prev, *bvbuffmap; - struct bvbuffmapinfo *bvbuffmapinfo; - - GCENTER(GCZONE_MAPPING); - - /* Lock access to the mapping list. */ - GCLOCK(&gccontext.maplock); - - /* Dump the schedule. */ - GCDUMPSCHEDULE(); - - /* Scan scheduled unmappings and remove the ones that are still - * in use. */ - list_for_each_safe(head, temphead, &batch->unmap) { - gcschedunmap = list_entry(head, struct gcschedunmap, link); - - /* Cast the handle. */ - bvbuffdesc = (struct bvbuffdesc *) gcschedunmap->handle; - - /* Find our mapping. */ - prev = NULL; - bvbuffmap = bvbuffdesc->map; - while (bvbuffmap != NULL) { - if (bvbuffmap->bv_unmap == bv_unmap) - break; - prev = bvbuffmap; - bvbuffmap = bvbuffmap->nextmap; - } - - /* Not found? */ - if (bvbuffmap == NULL) { - GCERR("mapping not found for bvbuffdesc 0x%08X.\n", - (unsigned int) bvbuffdesc); - - /* Remove scheduled unmapping. */ - list_move(head, &gccontext.vac_unmap); - continue; - } - - /* Get the info structure. */ - bvbuffmapinfo = (struct bvbuffmapinfo *) bvbuffmap->handle; - - GCDBG(GCZONE_MAPPING, "head = 0x%08X\n", - (unsigned int) gcschedunmap); - GCDBG(GCZONE_MAPPING, " bvbuffmap = 0x%08X\n", - (unsigned int) bvbuffmap); - GCDBG(GCZONE_MAPPING, " handle = 0x%08X\n", - bvbuffmapinfo->handle); - - /* Implicit unmapping. */ - bvbuffmapinfo->automap -= 1; - if (bvbuffmapinfo->automap < 0) { - GCERR("implicit count negative.\n"); - bvbuffmapinfo->automap = 0; - } - - GCDBG(GCZONE_MAPPING, " explicit count = %d\n", - bvbuffmapinfo->usermap); - GCDBG(GCZONE_MAPPING, " implicit count = %d\n", - bvbuffmapinfo->automap); - - /* Still referenced? */ - if (bvbuffmapinfo->usermap || bvbuffmapinfo->automap) { - GCDBG(GCZONE_MAPPING, " still referenced.\n"); - - /* Remove scheduled unmapping. */ - list_move(head, &gccontext.vac_unmap); - continue; - } - - GCDBG(GCZONE_MAPPING, " ready for unmapping.\n"); - - /* Set the handle. */ - gcschedunmap->handle = bvbuffmapinfo->handle; - - /* Remove from the buffer descriptor. */ - if (prev == NULL) - bvbuffdesc->map = bvbuffmap->nextmap; - else - prev->nextmap = bvbuffmap->nextmap; - - /* Add to the vacant list. */ - bvbuffmap->nextmap = gccontext.vac_buffmap; - gccontext.vac_buffmap = bvbuffmap; - } - - /* Dump the schedule. */ - GCDUMPSCHEDULE(); - - /* Unlock access to the mapping list. */ - GCUNLOCK(&gccontext.maplock); - - GCEXIT(GCZONE_MAPPING); -} - - -/******************************************************************************* - * Batch memory manager. - */ - -static enum bverror allocate_batch(struct bvbltparams *bltparams, - struct gcbatch **batch); -static void free_batch(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) -{ - return BVERR_NONE; -} - -static enum bverror allocate_batch(struct bvbltparams *bltparams, - struct gcbatch **batch) -{ - enum bverror bverror; - struct gcbatch *temp; - - GCENTER(GCZONE_BATCH); - - /* Lock access to batch management. */ - GCLOCK(&gccontext.batchlock); - - if (gccontext.vac_batches == NULL) { - temp = gcalloc(struct gcbatch, sizeof(struct gcbatch)); - if (temp == NULL) { - BVSETBLTERROR(BVERR_OOM, - "batch header allocation failed"); - goto exit; - } - - GCDBG(GCZONE_BATCH, "allocated new batch = 0x%08X\n", - (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); - } - - memset(temp, 0, sizeof(struct gcbatch)); - temp->structsize = sizeof(struct gcbatch); - temp->batchend = do_end; - INIT_LIST_HEAD(&temp->unmap); - - bverror = append_buffer(bltparams, temp); - if (bverror != BVERR_NONE) { - free_batch(temp); - goto exit; - } - - *batch = temp; - - GCDBG(GCZONE_BATCH, "batch allocated = 0x%08X\n", (unsigned int) temp); - -exit: - /* Unlock access to batch management. */ - GCUNLOCK(&gccontext.batchlock); - - GCEXITARG(GCZONE_BATCH, "bv%s = %d\n", - (bverror == BVERR_NONE) ? "result" : "error", bverror); - return bverror; -} - -static void free_batch(struct gcbatch *batch) -{ - struct gcbuffer *buffer; - - GCENTERARG(GCZONE_BATCH, "batch = 0x%08X\n", (unsigned int) batch); - - /* Lock access. */ - GCLOCK(&gccontext.batchlock); - GCLOCK(&gccontext.bufferlock); - GCLOCK(&gccontext.fixuplock); - GCLOCK(&gccontext.maplock); - - list_splice_init(&batch->unmap, &gccontext.vac_unmap); - - buffer = batch->bufhead; - if (buffer != NULL) { - do { - if (buffer->fixuphead != NULL) { - buffer->fixuptail->next = gccontext.vac_fixups; - gccontext.vac_fixups = buffer->fixuphead; - } - buffer = buffer->next; - } while (buffer != NULL); - - batch->buftail->next = gccontext.vac_buffers; - gccontext.vac_buffers = batch->bufhead; - } - - ((struct gcvacbatch *) batch)->next = gccontext.vac_batches; - gccontext.vac_batches = (struct gcvacbatch *) batch; - - /* Unlock access. */ - GCUNLOCK(&gccontext.maplock); - GCUNLOCK(&gccontext.fixuplock); - GCUNLOCK(&gccontext.bufferlock); - GCUNLOCK(&gccontext.batchlock); - - GCEXIT(GCZONE_BATCH); -} - -static enum bverror append_buffer(struct bvbltparams *bltparams, - struct gcbatch *batch) -{ - enum bverror bverror; - struct gcbuffer *temp; - - GCENTERARG(GCZONE_BUFFER_ALLOC, "batch = 0x%08X\n", - (unsigned int) batch); - - /* Lock access to buffer management. */ - GCLOCK(&gccontext.bufferlock); - - if (gccontext.vac_buffers == NULL) { - temp = gcalloc(struct gcbuffer, GC_BUFFER_SIZE); - if (temp == NULL) { - BVSETBLTERROR(BVERR_OOM, - "command buffer allocation failed"); - goto exit; - } - - 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_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); - - if (batch->bufhead == NULL) - batch->bufhead = temp; - else - batch->buftail->next = temp; - batch->buftail = temp; - - GCDBG(GCZONE_BUFFER_ALLOC, "new buffer appended = 0x%08X\n", - (unsigned int) temp); - - bverror = BVERR_NONE; - -exit: - /* Unlock access to buffer management. */ - GCUNLOCK(&gccontext.bufferlock); - - GCEXITARG(GCZONE_BUFFER_ALLOC, "bv%s = %d\n", - (bverror == BVERR_NONE) ? "result" : "error", bverror); - return bverror; -} - -static enum bverror add_fixup(struct bvbltparams *bltparams, - struct gcbatch *batch, - unsigned int *fixup, - unsigned int surfoffset) +enum bverror set_dst(struct bvbltparams *bltparams, + struct gcbatch *batch, + struct bvbuffmap *dstmap) { 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); - - /* Lock access to fixup management. */ - GCLOCK(&gccontext.fixuplock); - - buffer = batch->buftail; - temp = buffer->fixuptail; - - GCDBG(GCZONE_FIXUP, "buffer = 0x%08X, fixup struct = 0x%08X\n", - (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) { - BVSETBLTERROR(BVERR_OOM, - "fixup allocation failed"); - goto exit; - } - - GCDBG(GCZONE_FIXUP, - "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); - } - - temp->next = NULL; - temp->count = 0; - - if (buffer->fixuphead == NULL) - buffer->fixuphead = temp; - else - buffer->fixuptail->next = temp; - buffer->fixuptail = temp; - - GCDBG(GCZONE_FIXUP, "new fixup struct allocated = 0x%08X\n", - (unsigned int) temp); - - } else { - GCDBG(GCZONE_FIXUP, "fixups accumulated = %d\n", temp->count); - } - - temp->fixup[temp->count].dataoffset = fixup - buffer->head; - temp->fixup[temp->count].surfoffset = surfoffset; - temp->count += 1; - - GCDBG(GCZONE_FIXUP, "fixup offset = 0x%08X\n", fixup - buffer->head); - GCDBG(GCZONE_FIXUP, "surface offset = 0x%08X\n", surfoffset); - -exit: - /* Unlock access to fixup management. */ - GCUNLOCK(&gccontext.fixuplock); - - GCEXITARG(GCZONE_FIXUP, "bv%s = %d\n", - (bverror == BVERR_NONE) ? "result" : "error", bverror); - return bverror; -} - -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_ALLOC, "batch = 0x%08X, size = %d\n", - (unsigned int) batch, size); - - /* Get the current command buffer. */ - curbuf = batch->buftail; + struct gcmodst *gcmodst; - GCDBG(GCZONE_BUFFER_ALLOC, "buffer = 0x%08X, available = %d\n", - (unsigned int) curbuf, curbuf->available); + GCENTER(GCZONE_DEST); - if (curbuf->available < size) { - bverror = append_buffer(bltparams, batch); + /* Did destination surface change? */ + if ((batch->batchflags & BVBATCH_DST) != 0) { + /* Allocate command buffer. */ + bverror = claim_buffer(bltparams, batch, + sizeof(struct gcmodst), + (void **) &gcmodst); if (bverror != BVERR_NONE) goto exit; - curbuf = batch->buftail; - } - - if (curbuf->available < size) { - GCDBG(GCZONE_BUFFER_ALLOC, "requested size is too large.\n"); - BVSETBLTERROR(BVERR_OOM, - "command buffer allocation failed"); - goto exit; - } - - *buffer = curbuf->tail; - curbuf->tail = (unsigned int *) ((unsigned char *) curbuf->tail + size); - curbuf->available -= size; - batch->size += size; - bverror = BVERR_NONE; - -exit: - GCEXITARG(GCZONE_BUFFER_ALLOC, "bv%s = %d\n", - (bverror == BVERR_NONE) ? "result" : "error", bverror); - return bverror; -} - - -/******************************************************************************* - * Pixel format parser. - */ - -/* FIXME/TODO: change to use BLTsvile defines. */ - -#if defined(OCDFMTDEF_ALPHA_SHIFT) -# undef OCDFMTDEF_ALPHA_SHIFT -#endif - -#if defined(OCDFMTDEF_ALPHA_MASK) -# undef OCDFMTDEF_ALPHA_MASK -#endif - -#define OCDFMTDEF_ALPHA_SHIFT 18 -#define OCDFMTDEF_ALPHA_MASK (1 << OCDFMTDEF_ALPHA_SHIFT) - -#define OCDFMTDEF_PLACEMENT_SHIFT 9 -#define OCDFMTDEF_PLACEMENT_MASK (3 << OCDFMTDEF_PLACEMENT_SHIFT) - -#define OCDFMTDEF_BITS_SHIFT 3 -#define OCDFMTDEF_BITS_MASK (3 << OCDFMTDEF_BITS_SHIFT) - -#define OCDFMTDEF_BITS12 (0 << OCDFMTDEF_BITS_SHIFT) -#define OCDFMTDEF_BITS15 (1 << OCDFMTDEF_BITS_SHIFT) -#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; - unsigned int mask; -}; - -struct bvcsrgb { - struct bvcomponent r; - struct bvcomponent g; - struct bvcomponent b; - struct bvcomponent a; -}; - -struct bvformatxlate { - enum bvformattype type; - unsigned bitspp; - unsigned format; - unsigned swizzle; - struct bvcsrgb rgba; -}; - -#define BVFORMATRGBA(BPP, Format, Swizzle, R, G, B, A) \ -{ \ - BVFMT_RGB, \ - BPP, \ - GCREG_DE_FORMAT_ ## Format, \ - GCREG_DE_SWIZZLE_ ## Swizzle, \ - { R, G, B, A } \ -} - -#define BVCOMP(Shift, Size) \ - { Shift, Size, ((1 << Size) - 1) << Shift } - -#define BVRED(Shift, Size) \ - BVCOMP(Shift, Size) - -#define BVGREEN(Shift, Size) \ - BVCOMP(Shift, Size) - -#define BVBLUE(Shift, Size) \ - BVCOMP(Shift, Size) - -#define BVALPHA(Shift, Size) \ - BVCOMP(Shift, Size) - -#define BVFORMATINVALID \ - { 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, -}; -static struct bvformatxlate g_format_uyvy = { - .type = BVFMT_YUV, - .bitspp = 16, - .format = GCREG_DE_FORMAT_UYVY -}; -static struct bvformatxlate g_format_yuy2 = { - .type = BVFMT_YUV, - .bitspp = 16, - .format = GCREG_DE_FORMAT_YUY2 -}; - -static struct bvformatxlate formatxlate[] = { - /* #0: OCDFMT_xRGB12 - BITS=12 ALPHA=0 REVERSED=0 LEFT_JUSTIFIED=0 */ - BVFORMATRGBA(16, X4R4G4B4, ARGB, - BVRED(8, 4), BVGREEN(4, 4), BVBLUE(0, 4), BVALPHA(12, 0)), - - /* #1: OCDFMT_RGBx12 - BITS=12 ALPHA=0 REVERSED=0 LEFT_JUSTIFIED=1 */ - BVFORMATRGBA(16, X4R4G4B4, RGBA, - BVRED(12, 4), BVGREEN(8, 4), BVBLUE(4, 4), BVALPHA(0, 0)), - - /* #2: OCDFMT_xBGR12 - BITS=12 ALPHA=0 REVERSED=1 LEFT_JUSTIFIED=0 */ - BVFORMATRGBA(16, X4R4G4B4, ABGR, - BVRED(0, 4), BVGREEN(4, 4), BVBLUE(8, 4), BVALPHA(12, 0)), - - /* #3: OCDFMT_BGRx12 - BITS=12 ALPHA=0 REVERSED=1 LEFT_JUSTIFIED=1 */ - BVFORMATRGBA(16, X4R4G4B4, BGRA, - BVRED(4, 4), BVGREEN(8, 4), BVBLUE(12, 4), BVALPHA(0, 0)), - - /* #4: OCDFMT_ARGB12 - BITS=12 ALPHA=1 REVERSED=0 LEFT_JUSTIFIED=0 */ - BVFORMATRGBA(16, A4R4G4B4, ARGB, - BVRED(8, 4), BVGREEN(4, 4), BVBLUE(0, 4), BVALPHA(12, 4)), - - /* #5: OCDFMT_RGBA12 - BITS=12 ALPHA=1 REVERSED=0 LEFT_JUSTIFIED=1 */ - BVFORMATRGBA(16, A4R4G4B4, RGBA, - BVRED(12, 4), BVGREEN(8, 4), BVBLUE(4, 4), BVALPHA(0, 4)), - - /* #6: OCDFMT_ABGR12 - BITS=12 ALPHA=1 REVERSED=1 LEFT_JUSTIFIED=0 */ - BVFORMATRGBA(16, A4R4G4B4, ABGR, - BVRED(0, 4), BVGREEN(4, 4), BVBLUE(8, 4), BVALPHA(12, 4)), - - /* #7: OCDFMT_BGRA12 - BITS=12 ALPHA=1 REVERSED=1 LEFT_JUSTIFIED=1 */ - BVFORMATRGBA(16, A4R4G4B4, BGRA, - BVRED(4, 4), BVGREEN(8, 4), BVBLUE(12, 4), BVALPHA(0, 4)), - - /***********************************************/ - - /* #8: OCDFMT_xRGB15 - BITS=15 ALPHA=0 REVERSED=0 LEFT_JUSTIFIED=0 */ - BVFORMATRGBA(16, X1R5G5B5, ARGB, - BVRED(10, 5), BVGREEN(5, 5), BVBLUE(0, 5), BVALPHA(15, 0)), - - /* #9: OCDFMT_RGBx15 - BITS=15 ALPHA=0 REVERSED=0 LEFT_JUSTIFIED=1 */ - BVFORMATRGBA(16, X1R5G5B5, RGBA, - BVRED(11, 5), BVGREEN(6, 5), BVBLUE(1, 5), BVALPHA(0, 0)), - - /* #10: OCDFMT_xBGR15 - BITS=15 ALPHA=0 REVERSED=1 LEFT_JUSTIFIED=0 */ - BVFORMATRGBA(16, X1R5G5B5, ABGR, - BVRED(0, 5), BVGREEN(5, 5), BVBLUE(10, 5), BVALPHA(15, 0)), - - /* #11: OCDFMT_BGRx15 - BITS=15 ALPHA=0 REVERSED=1 LEFT_JUSTIFIED=1 */ - BVFORMATRGBA(16, X1R5G5B5, BGRA, - BVRED(1, 5), BVGREEN(6, 5), BVBLUE(11, 5), BVALPHA(0, 0)), - - /* #12: OCDFMT_ARGB15 - BITS=15 ALPHA=1 REVERSED=0 LEFT_JUSTIFIED=0 */ - BVFORMATRGBA(16, A1R5G5B5, ARGB, - BVRED(10, 5), BVGREEN(5, 5), BVBLUE(0, 5), BVALPHA(15, 1)), - - /* #13: OCDFMT_RGBA15 - BITS=15 ALPHA=1 REVERSED=0 LEFT_JUSTIFIED=1 */ - BVFORMATRGBA(16, A1R5G5B5, RGBA, - BVRED(11, 5), BVGREEN(6, 5), BVBLUE(1, 5), BVALPHA(0, 1)), - - /* #14: OCDFMT_ABGR15 - BITS=15 ALPHA=1 REVERSED=1 LEFT_JUSTIFIED=0 */ - BVFORMATRGBA(16, A1R5G5B5, ABGR, - BVRED(0, 5), BVGREEN(5, 5), BVBLUE(10, 5), BVALPHA(15, 1)), - - /* #15: OCDFMT_BGRA15 - BITS=15 ALPHA=1 REVERSED=1 LEFT_JUSTIFIED=1 */ - BVFORMATRGBA(16, A1R5G5B5, BGRA, - BVRED(1, 5), BVGREEN(6, 5), BVBLUE(11, 5), BVALPHA(0, 1)), - - /***********************************************/ - - /* #16: OCDFMT_RGB16 - BITS=16 ALPHA=0 REVERSED=0 LEFT_JUSTIFIED=0 */ - BVFORMATRGBA(16, R5G6B5, ARGB, - BVRED(11, 5), BVGREEN(5, 6), BVBLUE(0, 5), BVALPHA(0, 0)), - - /* #17: OCDFMT_RGB16 - BITS=16 ALPHA=0 REVERSED=0 LEFT_JUSTIFIED=1 */ - BVFORMATRGBA(16, R5G6B5, ARGB, - BVRED(11, 5), BVGREEN(5, 6), BVBLUE(0, 5), BVALPHA(0, 0)), - - /* #18: OCDFMT_BGR16 - BITS=16 ALPHA=0 REVERSED=1 LEFT_JUSTIFIED=0 */ - BVFORMATRGBA(16, R5G6B5, ABGR, - BVRED(0, 5), BVGREEN(5, 6), BVBLUE(11, 5), BVALPHA(0, 0)), - - /* #19: OCDFMT_BGR16 - BITS=16 ALPHA=0 REVERSED=1 LEFT_JUSTIFIED=1 */ - BVFORMATRGBA(16, R5G6B5, ABGR, - BVRED(0, 5), BVGREEN(5, 6), BVBLUE(11, 5), BVALPHA(0, 0)), - - /* #20 */ - BVFORMATINVALID, - - /* #21 */ - BVFORMATINVALID, - - /* #22 */ - BVFORMATINVALID, - - /* #23 */ - BVFORMATINVALID, - - /***********************************************/ - - /* #24: OCDFMT_xRGB24 - BITS=24 ALPHA=0 REVERSED=0 LEFT_JUSTIFIED=0 */ - BVFORMATRGBA(32, X8R8G8B8, BGRA, - BVRED(8, 8), BVGREEN(16, 8), BVBLUE(24, 8), BVALPHA(0, 0)), - - /* #25: OCDFMT_RGBx24 - BITS=24 ALPHA=0 REVERSED=0 LEFT_JUSTIFIED=1 */ - BVFORMATRGBA(32, X8R8G8B8, ABGR, - BVRED(0, 8), BVGREEN(8, 8), BVBLUE(16, 8), BVALPHA(24, 0)), - - /* #26: OCDFMT_xBGR24 - BITS=24 ALPHA=0 REVERSED=1 LEFT_JUSTIFIED=0 */ - BVFORMATRGBA(32, X8R8G8B8, RGBA, - BVRED(24, 8), BVGREEN(16, 8), BVBLUE(8, 8), BVALPHA(0, 0)), - - /* #27: OCDFMT_BGRx24 - BITS=24 ALPHA=0 REVERSED=1 LEFT_JUSTIFIED=1 */ - BVFORMATRGBA(32, X8R8G8B8, ARGB, - BVRED(16, 8), BVGREEN(8, 8), BVBLUE(0, 8), BVALPHA(24, 0)), - - /* #28: OCDFMT_ARGB24 - BITS=24 ALPHA=1 REVERSED=0 LEFT_JUSTIFIED=0 */ - BVFORMATRGBA(32, A8R8G8B8, BGRA, - BVRED(8, 8), BVGREEN(16, 8), BVBLUE(24, 8), BVALPHA(0, 8)), - - /* #29: OCDFMT_RGBA24 - BITS=24 ALPHA=1 REVERSED=0 LEFT_JUSTIFIED=1 */ - BVFORMATRGBA(32, A8R8G8B8, ABGR, - BVRED(0, 8), BVGREEN(8, 8), BVBLUE(16, 8), BVALPHA(24, 8)), - - /* #30: OCDFMT_ABGR24 - BITS=24 ALPHA=1 REVERSED=1 LEFT_JUSTIFIED=0 */ - BVFORMATRGBA(32, A8R8G8B8, RGBA, - BVRED(24, 8), BVGREEN(16, 8), BVBLUE(8, 8), BVALPHA(0, 8)), - - /* #31: OCDFMT_BGRA24 - BITS=24 ALPHA=1 REVERSED=1 LEFT_JUSTIFIED=1 */ - BVFORMATRGBA(32, A8R8G8B8, ARGB, - BVRED(16, 8), BVGREEN(8, 8), BVBLUE(0, 8), BVALPHA(24, 8)), -}; - -static int parse_format(enum ocdformat ocdformat, struct bvformatxlate **format) -{ - static unsigned int containers[] = { - 8, /* OCDFMTDEF_CONTAINER_8BIT */ - 16, /* OCDFMTDEF_CONTAINER_16BIT */ - 24, /* OCDFMTDEF_CONTAINER_24BIT */ - 32, /* OCDFMTDEF_CONTAINER_32BIT */ - ~0U, /* reserved */ - 48, /* OCDFMTDEF_CONTAINER_48BIT */ - ~0U, /* reserved */ - 64 /* OCDFMTDEF_CONTAINER_64BIT */ - }; - - unsigned int cs; - unsigned int bits; - unsigned int swizzle; - unsigned int alpha; - unsigned int index; - unsigned int cont; - - GCDBG(GCZONE_FORMAT, "ocdformat = 0x%08X\n", ocdformat); - - switch (ocdformat) { - case OCDFMT_NV12: - GCDBG(GCZONE_FORMAT, "OCDFMT_NV12\n"); - *format = &g_format_nv12; - return 1; - - case OCDFMT_UYVY: - GCDBG(GCZONE_FORMAT, "OCDFMT_UYVY\n"); - *format = &g_format_uyvy; - return 1; - - case OCDFMT_YUY2: - GCDBG(GCZONE_FORMAT, "OCDFMT_YUY2\n"); - *format = &g_format_yuy2; - return 1; - - default: - break; - } - - cs = (ocdformat & OCDFMTDEF_CS_MASK) >> OCDFMTDEF_CS_SHIFT; - bits = (ocdformat & OCDFMTDEF_COMPONENTSIZEMINUS1_MASK) - >> OCDFMTDEF_COMPONENTSIZEMINUS1_SHIFT; - cont = (ocdformat & OCDFMTDEF_CONTAINER_MASK) - >> OCDFMTDEF_CONTAINER_SHIFT; - - switch (cs) { - case (OCDFMTDEF_CS_RGB >> OCDFMTDEF_CS_SHIFT): - GCDBG(GCZONE_FORMAT, "OCDFMTDEF_CS_RGB\n"); - GCDBG(GCZONE_FORMAT, "bits = %d\n", bits); - GCDBG(GCZONE_FORMAT, "cont = %d\n", cont); - - if ((ocdformat & OCDFMTDEF_LAYOUT_MASK) != OCDFMTDEF_PACKED) - return 0; - - swizzle = (ocdformat & OCDFMTDEF_PLACEMENT_MASK) - >> OCDFMTDEF_PLACEMENT_SHIFT; - alpha = (ocdformat & OCDFMTDEF_ALPHA_MASK) - >> OCDFMTDEF_ALPHA_SHIFT; - - GCDBG(GCZONE_FORMAT, "swizzle = %d\n", swizzle); - GCDBG(GCZONE_FORMAT, "alpha = %d\n", alpha); - - index = swizzle | (alpha << 2); - - switch (bits) { - case 12 - 1: - index |= OCDFMTDEF_BITS12; - break; - - case 15 - 1: - index |= OCDFMTDEF_BITS15; - break; - - case 16 - 1: - index |= OCDFMTDEF_BITS16; - break; - - case 24 - 1: - index |= OCDFMTDEF_BITS24; - break; - - default: - return 0; - } - - GCDBG(GCZONE_FORMAT, "index = %d\n", index); - break; - - default: - return 0; - } - - if (formatxlate[index].bitspp != containers[cont]) - return 0; - - *format = &formatxlate[index]; - - GCDBG(GCZONE_FORMAT, "format record = 0x%08X\n", - (unsigned int) &formatxlate[index]); - GCDBG(GCZONE_FORMAT, " bpp = %d\n", formatxlate[index].bitspp); - GCDBG(GCZONE_FORMAT, " format = %d\n", formatxlate[index].format); - GCDBG(GCZONE_FORMAT, " swizzle = %d\n", formatxlate[index].swizzle); - - return 1; -} - -static inline unsigned int extract_component(unsigned int pixel, - struct bvcomponent *desc) -{ - unsigned int component; - unsigned int component8; - - component = (pixel & desc->mask) >> desc->shift; - GCDBG(GCZONE_COLOR, "mask=0x%08X, shift=%d, component=0x%08X\n", - desc->mask, desc->shift, component); - - switch (desc->size) { - case 0: - component8 = 0xFF; - GCDBG(GCZONE_COLOR, "component8=0x%08X\n", component8); - break; - - case 1: - component8 = component ? 0xFF : 0x00; - GCDBG(GCZONE_COLOR, "component8=0x%08X\n", component8); - break; - - case 4: - component8 = component | (component << 4); - GCDBG(GCZONE_COLOR, "component8=0x%08X\n", component8); - break; - - case 5: - component8 = (component << 3) | (component >> 2); - GCDBG(GCZONE_COLOR, "component8=0x%08X\n", component8); - break; - - case 6: - component8 = (component << 2) | (component >> 4); - GCDBG(GCZONE_COLOR, "component8=0x%08X\n", component8); - break; - - default: - component8 = component; - GCDBG(GCZONE_COLOR, "component8=0x%08X\n", component8); - } - - return component8; -} - -static unsigned int getinternalcolor(void *ptr, struct bvformatxlate *format) -{ - unsigned int srcpixel, dstpixel; - unsigned int r, g, b, a; - - switch (format->bitspp) { - case 16: - srcpixel = *(unsigned short *) ptr; - GCDBG(GCZONE_COLOR, "srcpixel=0x%08X\n", srcpixel); - break; - - case 32: - srcpixel = *(unsigned int *) ptr; - GCDBG(GCZONE_COLOR, "srcpixel=0x%08X\n", srcpixel); - break; - - default: - srcpixel = 0; - GCDBG(GCZONE_COLOR, "srcpixel=0x%08X\n", srcpixel); - } - - r = extract_component(srcpixel, &format->rgba.r); - g = extract_component(srcpixel, &format->rgba.g); - b = extract_component(srcpixel, &format->rgba.b); - a = extract_component(srcpixel, &format->rgba.a); - - GCDBG(GCZONE_COLOR, "(r,g,b,a)=0x%02X,0x%02X,0x%02X,0x%02X\n", - r, g, b, a); - - dstpixel = (a << 24) | (r << 16) | (g << 8) | b; - - GCDBG(GCZONE_COLOR, "dstpixel=0x%08X\n", dstpixel); - - return dstpixel; -} - - -/******************************************************************************* - * Alpha blending parser. - */ - -#define BVBLENDMATCH(Mode, Inverse, Normal) \ -( \ - BVBLENDDEF_ ## Mode | \ - BVBLENDDEF_ ## Inverse | \ - BVBLENDDEF_ ## Normal \ -) - -#define BVSRC1USE(Use) \ - Use - -#define BVSRC2USE(Use) \ - Use - -#define BVBLENDUNDEFINED() \ - { ~0, ~0, { 0, 0, 0, 0 }, { 0, 0, 0, 0 } } - -static struct bvblendxlate blendxlate[64] = { - /**********************************************************************/ - /* #0: color factor: 00 00 00 A:(1-C1,C1)=zero - alpha factor: zero ==> 00 00 00 */ - { - 0x00, - 0x00, - - { - GCREG_BLENDING_MODE_ZERO, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(0), BVSRC2USE(0), - }, - - { - GCREG_BLENDING_MODE_ZERO, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(0), BVSRC2USE(0) - } - }, - - /* #1: color factor: 00 00 01 A:(1-C1,A1)=A1 - alpha factor: A1 ==> 00 00 01 or 00 10 01 */ - { - BVBLENDMATCH(ONLY_A, INV_C1, NORM_A1), - BVBLENDMATCH(ONLY_A, INV_C2, NORM_A1), - - { - GCREG_BLENDING_MODE_NORMAL, - GCREG_FACTOR_INVERSE_ENABLE, - BVSRC1USE(1), BVSRC2USE(0), - }, - - { - GCREG_BLENDING_MODE_NORMAL, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1) - } - }, - - /* #2: color factor: 00 00 10 A:(1-C1,C2)=undefined - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #3: color factor: 00 00 11 A:(1-C1,A2)=A2 - alpha factor: A2 ==> 00 00 11 or 00 10 11 */ - { - BVBLENDMATCH(ONLY_A, INV_C1, NORM_A2), - BVBLENDMATCH(ONLY_A, INV_C2, NORM_A2), - - { - GCREG_BLENDING_MODE_NORMAL, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1), - }, - - { - GCREG_BLENDING_MODE_NORMAL, - GCREG_FACTOR_INVERSE_ENABLE, - BVSRC1USE(0), BVSRC2USE(1) - } - }, - - /* #4: color factor: 00 01 00 A:(1-A1,C1)=1-A1 - alpha factor: 1-A1 ==> 00 01 00 or 00 01 10 */ - { - BVBLENDMATCH(ONLY_A, INV_A1, NORM_C1), - BVBLENDMATCH(ONLY_A, INV_A1, NORM_C2), - - { - GCREG_BLENDING_MODE_INVERSED, - GCREG_FACTOR_INVERSE_ENABLE, - BVSRC1USE(1), BVSRC2USE(0), - }, - - { - GCREG_BLENDING_MODE_INVERSED, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1) - } - }, - - /* #5: color factor: 00 01 01 A:(1-A1,A1)=undefined - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #6: color factor: 00 01 10 A:(1-A1,C2)=1-A1 - alpha factor: 1-A1 ==> 00 01 00 or 00 01 10 */ - { - BVBLENDMATCH(ONLY_A, INV_A1, NORM_C1), - BVBLENDMATCH(ONLY_A, INV_A1, NORM_C2), - - { - GCREG_BLENDING_MODE_INVERSED, - GCREG_FACTOR_INVERSE_ENABLE, - BVSRC1USE(1), BVSRC2USE(0), - }, - - { - GCREG_BLENDING_MODE_INVERSED, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1) - } - }, - - /* #7: color factor: 00 01 11 A:(1-A1,A2)=undefined - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #8: color factor: 00 10 00 A:(1-C2,C1)=undefined - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #9: color factor: 00 10 01 A:(1-C2,A1)=A1 - alpha factor: A1 ==> 00 00 01 or 00 10 01 */ - { - BVBLENDMATCH(ONLY_A, INV_C1, NORM_A1), - BVBLENDMATCH(ONLY_A, INV_C2, NORM_A1), - - { - GCREG_BLENDING_MODE_NORMAL, - GCREG_FACTOR_INVERSE_ENABLE, - BVSRC1USE(1), BVSRC2USE(0), - }, - - { - GCREG_BLENDING_MODE_NORMAL, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1) - } - }, - - /* #10: color factor: 00 10 10 A:(1-C2,C2)=undefined - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #11: color factor: 00 10 11 A:(1-C2,A2)=A2 - alpha factor: A2 ==> 00 00 11 or 00 10 11 */ - { - BVBLENDMATCH(ONLY_A, INV_C1, NORM_A2), - BVBLENDMATCH(ONLY_A, INV_C2, NORM_A2), - - { - GCREG_BLENDING_MODE_NORMAL, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1), - }, - - { - GCREG_BLENDING_MODE_NORMAL, - GCREG_FACTOR_INVERSE_ENABLE, - BVSRC1USE(0), BVSRC2USE(1) - } - }, - - /* #12: color factor: 00 11 00 A:(1-A2,C1)=1-A2 - alpha factor: 1-A2 ==> 00 11 00 or 00 11 10 */ - { - BVBLENDMATCH(ONLY_A, INV_A2, NORM_C1), - BVBLENDMATCH(ONLY_A, INV_A2, NORM_C2), - - { - GCREG_BLENDING_MODE_INVERSED, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1), - }, - - { - GCREG_BLENDING_MODE_INVERSED, - GCREG_FACTOR_INVERSE_ENABLE, - BVSRC1USE(0), BVSRC2USE(1) - } - }, - - /* #13: color factor: 00 11 01 A:(1-A2,A1)=undefined - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #14: color factor: 00 11 10 A:(1-A2,C2)=1-A2 - alpha factor: 1-A2 ==> 00 11 00 or 00 11 10 */ - { - BVBLENDMATCH(ONLY_A, INV_A2, NORM_C1), - BVBLENDMATCH(ONLY_A, INV_A2, NORM_C2), - - { - GCREG_BLENDING_MODE_INVERSED, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1), - }, - - { - GCREG_BLENDING_MODE_INVERSED, - GCREG_FACTOR_INVERSE_ENABLE, - BVSRC1USE(0), BVSRC2USE(1) - } - }, - - /* #15: color factor: 00 11 11 A:(1-A2,A2)=undefined - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /**********************************************************************/ - /* #16: color factor: 01 00 00 MIN:(1-C1,C1) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #17: color factor: 01 00 01 MIN:(1-C1,A1) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #18: color factor: 01 00 10 MIN:(1-C1,C2) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #19: color factor: 01 00 11 MIN:(1-C1,A2) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #20: color factor: 01 01 00 MIN:(1-A1,C1) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #21: color factor: 01 01 01 MIN:(1-A1,A1) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #22: color factor: 01 01 10 MIN:(1-A1,C2) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #23: color factor: 01 01 11 MIN:(1-A1,A2) - alpha factor: one ==> 11 11 11 */ - { - 0x3F, - 0x3F, - - { - GCREG_BLENDING_MODE_SATURATED_DEST_ALPHA, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1), - }, - - { - GCREG_BLENDING_MODE_SATURATED_ALPHA, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1) - } - }, - - /* #24: color factor: 01 10 00 MIN:(1-C2,C1) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #25: color factor: 01 10 01 MIN:(1-C2,A1) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #26: color factor: 01 10 10 MIN:(1-C2,C2) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #27: color factor: 01 10 11 MIN:(1-C2,A2) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #28: color factor: 01 11 00 MIN:(1-A2,C1) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #29: color factor: 01 11 01 MIN:(1-A2,A1) - alpha factor: one ==> 11 11 11 */ - { - 0x3F, - 0x3F, - - { - GCREG_BLENDING_MODE_SATURATED_ALPHA, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1), - }, - - { - GCREG_BLENDING_MODE_SATURATED_DEST_ALPHA, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1) - } - }, - - /* #30: color factor: 01 11 10 MIN:(1-A2,C2) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #31: color factor: 01 11 11 MIN:(1-A2,A2) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /**********************************************************************/ - /* #32: color factor: 10 00 00 MAX:(1-C1,C1) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #33: color factor: 10 00 01 MAX:(1-C1,A1) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #34: color factor: 10 00 10 MAX:(1-C1,C2) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #35: color factor: 10 00 11 MAX:(1-C1,A2) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #36: color factor: 10 01 00 MAX:(1-A1,C1) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #37: color factor: 10 01 01 MAX:(1-A1,A1) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #38: color factor: 10 01 10 MAX:(1-A1,C2) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #39: color factor: 10 01 11 MAX:(1-A1,A2) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #40: color factor: 10 10 00 MAX:(1-C2,C1) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #41: color factor: 10 10 01 MAX:(1-C2,A1) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #42: color factor: 10 10 10 MAX:(1-C2,C2) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #43: color factor: 10 10 11 MAX:(1-C2,A2) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #44: color factor: 10 11 00 MAX:(1-A2,C1) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #45: color factor: 10 11 01 MAX:(1-A2,A1) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #46: color factor: 10 11 10 MAX:(1-A2,C2) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #47: color factor: 10 11 11 MAX:(1-A2,A2) ==> not supported - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /**********************************************************************/ - /* #48: color factor: 11 00 00 C:(1-C1,C1)=undefined - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #49: color factor: 11 00 01 C:(1-C1,A1)=1-C1 - alpha factor: 1-A1 ==> 00 01 00 or 00 01 10 */ - { - BVBLENDMATCH(ONLY_A, INV_A1, NORM_C1), - BVBLENDMATCH(ONLY_A, INV_A1, NORM_C2), - - { - GCREG_BLENDING_MODE_COLOR_INVERSED, - GCREG_FACTOR_INVERSE_ENABLE, - BVSRC1USE(1), BVSRC2USE(0), - }, - - { - GCREG_BLENDING_MODE_COLOR_INVERSED, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1) - } - }, - - /* #50: color factor: 11 00 10 C:(1-C1,C2)=undefined - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #51: color factor: 11 00 11 C:(1-C1,A2)=1-C1 - alpha factor: 1-A1 ==> 00 01 00 or 00 01 10 */ - { - BVBLENDMATCH(ONLY_A, INV_A1, NORM_C1), - BVBLENDMATCH(ONLY_A, INV_A1, NORM_C2), - - { - GCREG_BLENDING_MODE_COLOR_INVERSED, - GCREG_FACTOR_INVERSE_ENABLE, - BVSRC1USE(1), BVSRC2USE(0), - }, - - { - GCREG_BLENDING_MODE_COLOR_INVERSED, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1) - } - }, - - /* #52: color factor: 11 01 00 C:(1-A1,C1)=C1 - alpha factor: A1 ==> 00 00 01 or 00 10 01 */ - { - BVBLENDMATCH(ONLY_A, INV_C1, NORM_A1), - BVBLENDMATCH(ONLY_A, INV_C2, NORM_A1), - - { - GCREG_BLENDING_MODE_COLOR, - GCREG_FACTOR_INVERSE_ENABLE, - BVSRC1USE(1), BVSRC2USE(0), - }, - - { - GCREG_BLENDING_MODE_COLOR, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1) - } - }, - - /* #53: color factor: 11 01 01 C:(1-A1,A1)=undefined - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #54: color factor: 11 01 10 C:(1-A1,C2)=C2 - alpha factor: A2 ==> 00 00 11 or 00 10 11 */ - { - BVBLENDMATCH(ONLY_A, INV_C1, NORM_A2), - BVBLENDMATCH(ONLY_A, INV_C2, NORM_A2), - - { - GCREG_BLENDING_MODE_COLOR, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1), - }, - - { - GCREG_BLENDING_MODE_COLOR, - GCREG_FACTOR_INVERSE_ENABLE, - BVSRC1USE(0), BVSRC2USE(1) - } - }, - - /* #55: color factor: 11 01 11 C:(1-A1,A2)=undefined - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #56: color factor: 11 10 00 C:(1-C2,C1)=undefined - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #57: color factor: 11 10 01 C:(1-C2,A1)=1-C2 - alpha factor: 1-A2 ==> 00 11 00 or 00 11 10 */ - { - BVBLENDMATCH(ONLY_A, INV_A2, NORM_C1), - BVBLENDMATCH(ONLY_A, INV_A2, NORM_C2), - - { - GCREG_BLENDING_MODE_COLOR_INVERSED, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1), - }, - - { - GCREG_BLENDING_MODE_COLOR_INVERSED, - GCREG_FACTOR_INVERSE_ENABLE, - BVSRC1USE(0), BVSRC2USE(1) - } - }, - - /* #58: color factor: 11 10 10 C:(1-C2,C2)=undefined - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #59: color factor: 11 10 11 C:(1-C2,A2)=1-C2 - alpha factor: 1-A2 ==> 00 11 00 or 00 11 10 */ - { - BVBLENDMATCH(ONLY_A, INV_A2, NORM_C1), - BVBLENDMATCH(ONLY_A, INV_A2, NORM_C2), - - { - GCREG_BLENDING_MODE_COLOR_INVERSED, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1), - }, - - { - GCREG_BLENDING_MODE_COLOR_INVERSED, - GCREG_FACTOR_INVERSE_ENABLE, - BVSRC1USE(0), BVSRC2USE(0) - } - }, - - /* #60: color factor: 11 11 00 C:(1-A2,C1)=C1 - alpha factor: A1 ==> 00 00 01 or 00 10 01 */ - { - BVBLENDMATCH(ONLY_A, INV_C1, NORM_A1), - BVBLENDMATCH(ONLY_A, INV_C2, NORM_A1), - - { - GCREG_BLENDING_MODE_COLOR, - GCREG_FACTOR_INVERSE_ENABLE, - BVSRC1USE(1), BVSRC2USE(0), - }, - - { - GCREG_BLENDING_MODE_COLOR, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1) - } - }, - - /* #61: color factor: 11 11 01 C:(1-A2,A1)=undefined - alpha factor: N/A */ - BVBLENDUNDEFINED(), - - /* #62: color factor: 11 11 10 C:(1-A2,C2)=C2 - alpha factor: A2 ==> 00 00 11 or 00 10 11 */ - { - BVBLENDMATCH(ONLY_A, INV_C1, NORM_A2), - BVBLENDMATCH(ONLY_A, INV_C2, NORM_A2), - - { - GCREG_BLENDING_MODE_COLOR, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(1), - }, - - { - GCREG_BLENDING_MODE_COLOR, - GCREG_FACTOR_INVERSE_ENABLE, - BVSRC1USE(0), BVSRC2USE(1) - } - }, - - /* #63: color factor: 11 11 11 C:(1-A2,A2)=one - alpha factor: one ==> 11 11 11 */ - { - 0x3F, - 0x3F, - - { - GCREG_BLENDING_MODE_ONE, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(1), BVSRC2USE(0), - }, - - { - GCREG_BLENDING_MODE_ONE, - GCREG_FACTOR_INVERSE_DISABLE, - BVSRC1USE(0), BVSRC2USE(1) - } - } -}; - -static enum bverror parse_blend(struct bvbltparams *bltparams, - enum bvblend blend, struct gcalpha *gca) -{ - enum bverror bverror; - unsigned int global; - unsigned int k1, k2, k3, k4; - struct bvblendxlate *k1_xlate; - struct bvblendxlate *k2_xlate; - unsigned int alpha; - - GCDBG(GCZONE_BLEND, "blend = 0x%08X (%s)\n", - blend, gc_bvblend_name(blend)); - - if ((blend & BVBLENDDEF_REMOTE) != 0) { - BVSETBLTERROR(BVERR_BLEND, "remote alpha not supported"); - goto exit; - } - - global = (blend & BVBLENDDEF_GLOBAL_MASK) >> BVBLENDDEF_GLOBAL_SHIFT; - - switch (global) { - case (BVBLENDDEF_GLOBAL_NONE >> BVBLENDDEF_GLOBAL_SHIFT): - GCDBG(GCZONE_BLEND, "BVBLENDDEF_GLOBAL_NONE\n"); - - gca->src_global_color = - gca->dst_global_color = 0; - - gca->src_global_alpha_mode = GCREG_GLOBAL_ALPHA_MODE_NORMAL; - gca->dst_global_alpha_mode = GCREG_GLOBAL_ALPHA_MODE_NORMAL; - break; - - case (BVBLENDDEF_GLOBAL_UCHAR >> BVBLENDDEF_GLOBAL_SHIFT): - GCDBG(GCZONE_BLEND, "BVBLENDDEF_GLOBAL_UCHAR\n"); - - gca->src_global_color = - gca->dst_global_color = - ((unsigned int) bltparams->globalalpha.size8) << 24; - - gca->src_global_alpha_mode = GCREG_GLOBAL_ALPHA_MODE_GLOBAL; - gca->dst_global_alpha_mode = GCREG_GLOBAL_ALPHA_MODE_GLOBAL; - break; - - case (BVBLENDDEF_GLOBAL_FLOAT >> BVBLENDDEF_GLOBAL_SHIFT): - GCDBG(GCZONE_BLEND, "BVBLENDDEF_GLOBAL_FLOAT\n"); - - alpha = gcfp2norm8(bltparams->globalalpha.fp); - - gca->src_global_color = - gca->dst_global_color = alpha << 24; - - gca->src_global_alpha_mode = GCREG_GLOBAL_ALPHA_MODE_GLOBAL; - gca->dst_global_alpha_mode = GCREG_GLOBAL_ALPHA_MODE_GLOBAL; - break; - - default: - BVSETBLTERROR(BVERR_BLEND, "invalid global alpha mode"); - goto exit; - } - - /* - Co = k1 x C1 + k2 x C2 - Ao = k3 x A1 + k4 x A2 - */ - - k1 = (blend >> 18) & 0x3F; - k2 = (blend >> 12) & 0x3F; - k3 = (blend >> 6) & 0x3F; - k4 = blend & 0x3F; + /* Add the address fixup. */ + add_fixup(bltparams, batch, &gcmodst->address, + batch->dstbyteshift); - GCDBG(GCZONE_BLEND, "k1 = %d\n", k1); - GCDBG(GCZONE_BLEND, "k2 = %d\n", k2); - GCDBG(GCZONE_BLEND, "k3 = %d\n", k3); - GCDBG(GCZONE_BLEND, "k4 = %d\n", k4); + /* Set surface parameters. */ + gcmodst->address_ldst = gcmodst_address_ldst; + gcmodst->address = GET_MAP_HANDLE(dstmap); + gcmodst->stride = bltparams->dstgeom->virtstride; - k1_xlate = &blendxlate[k1]; - k2_xlate = &blendxlate[k2]; + /* 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; - if (((k3 != k1_xlate->match1) && (k3 != k1_xlate->match2)) || - ((k4 != k2_xlate->match1) && (k4 != k2_xlate->match2))) { - BVSETBLTERROR(BVERR_BLEND, - "not supported coefficient combination"); - goto exit; + /* Set clipping. */ + gcmodst->clip_ldst = gcmodst_clip_ldst; + gcmodst->cliplt.raw = 0; + gcmodst->cliprb.raw = 0; + gcmodst->cliprb.reg.right = GC_CLIP_RESET_RIGHT; + gcmodst->cliprb.reg.bottom = GC_CLIP_RESET_BOTTOM; } - gca->k1 = &k1_xlate->k1; - gca->k2 = &k2_xlate->k2; - - gca->src1used = gca->k1->src1used | gca->k2->src1used; - gca->src2used = gca->k1->src2used | gca->k2->src2used; - - bverror = BVERR_NONE; - exit: + GCEXITARG(GCZONE_DEST, "bv%s = %d\n", + (bverror == BVERR_NONE) ? "result" : "error", bverror); return bverror; } /******************************************************************************* - * 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. */ @@ -2428,32 +452,9 @@ static bool same_phys_area(struct bvbuffdesc *surf1, struct bvrect *rect1, return false; } -static bool valid_geom(struct bvbuffdesc *buffdesc, struct bvsurfgeom *geom, - struct bvformatxlate *format) -{ - unsigned int size; - - /* Compute the size of the surface. */ - size = (geom->width * geom->height * format->bitspp) / 8; - - /* Make sure the size is not greater then the surface. */ - if (size > buffdesc->length) { - GCERR("invalid geometry detected:\n"); - GCERR(" specified dimensions: %dx%d, %d bitspp\n", - geom->width, geom->height, format->bitspp); - GCERR(" surface size based on the dimensions: %d\n", - size); - GCERR(" specified surface size: %lu\n", - buffdesc->length); - return false; - } - - return true; -} - static int verify_surface(unsigned int tile, - union bvinbuff *surf, - struct bvsurfgeom *geom) + union bvinbuff *surf, + struct bvsurfgeom *geom) { if (tile) { if (surf->tileparams == NULL) @@ -2483,1337 +484,6 @@ static int verify_surface(unsigned int tile, return -1; } -#if GCDEBUG_ENABLE -# define VERIFYBATCH(changeflags, prevrect, currrect) \ - verify_batch(changeflags, prevrect, currrect) - -static void verify_batch(unsigned int changeflags, - struct bvrect *prevrect, - struct bvrect *currrect) -{ - if ((changeflags & 1) == 0) { - /* Origin did not change. */ - if ((prevrect->left != currrect->left) || - (prevrect->top != currrect->top)) { - GCERR("origin changed\n"); - GCERR(" previous = %d,%d\n", - prevrect->left, prevrect->top); - GCERR(" current = %d,%d\n", - currrect->left, currrect->top); - } - } - - if ((changeflags & 2) == 0) { - /* Size did not change. */ - if ((prevrect->width != currrect->width) || - (prevrect->height != currrect->height)) { - GCERR("size changed\n"); - GCERR(" previous = %dx%d\n", - prevrect->width, prevrect->height); - GCERR(" current = %dx%d\n", - currrect->width, currrect->height); - } - } - - prevrect->left = currrect->left; - prevrect->top = currrect->top; - prevrect->width = currrect->width; - prevrect->height = currrect->height; -} -#else -# 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; - } - - /* Compute the aligned offset. */ - alignedoffset = byteoffset & alignment; - - /* 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; - - GCENTER(GCZONE_DEST); - - /* 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, - " 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; - } - - /* 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; - } - - if (cliptop <= desttop) { - batch->deltatop = 0; - batch->clippedtop = desttop; - } else { - batch->deltatop = cliptop - desttop; - batch->clippedtop = cliptop; - } - - if (clipright >= destright) { - batch->deltaright = 0; - batch->clippedright = destright; - } else { - batch->deltaright = clipright - destright; - batch->clippedright = clipright; - } - - 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); - } - - /* Did the destination surface change? */ - if ((batch->batchflags & BVBATCH_DST) != 0) { - struct bvbuffdesc *dstdesc; - struct bvsurfgeom *dstgeom; - unsigned int stridealign; - - /* Make shortcuts to the destination objects. */ - dstdesc = bltparams->dstdesc; - dstgeom = bltparams->dstgeom; - - /* Check for unsupported dest formats. */ - switch (dstgeom->format) { - case OCDFMT_NV12: - BVSETBLTERROR(BVERR_DSTGEOM_FORMAT, - "destination format unsupported"); - goto exit; - - default: - break; - } - - /* 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; - } - - /* Validate geometry. */ - if (!valid_geom(dstdesc, dstgeom, batch->dstformat)) { - BVSETBLTERROR(BVERR_DSTGEOM, - "destination geom exceeds surface size"); - goto exit; - } - - /* Destination stride must be 8 pixel aligned. */ - stridealign = batch->dstformat->bitspp - 1; - if ((dstgeom->virtstride & stridealign) != 0) { - BVSETBLTERROR(BVERR_DSTGEOM_STRIDE, - "destination stride must be 8 pixel " - "aligned."); - 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); - return bverror; -} - -static enum bverror parse_source(struct bvbltparams *bltparams, - struct gcbatch *batch, - struct srcinfo *srcinfo) -{ - enum bverror bverror = BVERR_NONE; - struct bvbuffdesc *srcdesc; - struct bvsurfgeom *srcgeom; - struct bvrect *srcrect; - unsigned int stridealign; - - /* Make shortcuts to the source objects. */ - srcdesc = srcinfo->buf.desc; - srcgeom = srcinfo->geom; - srcrect = srcinfo->rect; - - /* 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; - } - - /* 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; - } - - /* Source must be 8 pixel aligned. */ - stridealign = srcinfo->format->bitspp - 1; - if ((srcgeom->virtstride & stridealign) != 0) { - BVSETBLTERROR((srcinfo->index == 0) - ? BVERR_SRC1GEOM_STRIDE - : BVERR_SRC2GEOM_STRIDE, - "source stride must be 8 pixel aligned."); - goto exit; - } - - /* 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); - } - - /* 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 { - GCDBG(GCZONE_DEST, " virtual address = 0x%08X\n", - (unsigned int) srcdesc->virtaddr); - } - - 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); - -exit: - return bverror; -} - - -/******************************************************************************* - * Primitive renderers. - */ - -static enum bverror set_dst(struct bvbltparams *bltparams, - struct gcbatch *batch, - struct bvbuffmap *dstmap) -{ - enum bverror bverror = BVERR_NONE; - struct gcmodst *gcmodst; - - GCENTER(GCZONE_DEST); - - /* Did destination surface change? */ - if ((batch->batchflags & BVBATCH_DST) != 0) { - /* 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; - - /* Set clipping. */ - gcmodst->clip_ldst = gcmodst_clip_ldst; - gcmodst->cliplt.raw = 0; - gcmodst->cliprb.raw = 0; - gcmodst->cliprb.reg.right = GC_CLIP_RESET_RIGHT; - gcmodst->cliprb.reg.bottom = GC_CLIP_RESET_BOTTOM; - } - -exit: - GCEXITARG(GCZONE_DEST, "bv%s = %d\n", - (bverror == BVERR_NONE) ? "result" : "error", bverror); - return bverror; -} - -static enum bverror do_fill(struct bvbltparams *bltparams, - struct gcbatch *batch, - struct srcinfo *srcinfo) -{ - enum bverror bverror; - int dstbyteshift; - struct gcmofill *gcmofill; - unsigned char *fillcolorptr; - struct bvbuffmap *dstmap = NULL; - - GCENTER(GCZONE_DO_FILL); - - /* Finish previous batch if any. */ - bverror = batch->batchend(bltparams, batch); - 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) { - bltparams->errdesc = gccontext.bverrorstr; - goto exit; - } - - /* Set the new destination. */ - bverror = set_dst(bltparams, batch, dstmap); - if (bverror != BVERR_NONE) - goto exit; - - /* Reset the modified flag. */ - batch->batchflags &= ~(BVBATCH_DST | - BVBATCH_CLIPRECT | - BVBATCH_DESTRECT); - - /*********************************************************************** - ** Allocate command buffer. - */ - - bverror = claim_buffer(bltparams, batch, - sizeof(struct gcmofill), - (void **) &gcmofill); - if (bverror != BVERR_NONE) - goto exit; - - /*********************************************************************** - ** 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 = 1; - gcmofill->src.config.raw = 0; - - gcmofill->src.rotationheight_ldst = gcmofillsrc_rotationheight_ldst; - 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; - gcmofill->src.alphacontrol.raw = 0; - gcmofill->src.alphacontrol.reg.enable = GCREG_ALPHA_CONTROL_ENABLE_OFF; - - /*********************************************************************** - ** Set fill color. - */ - - fillcolorptr - = (unsigned char *) srcinfo->buf.desc->virtaddr - + srcinfo->rect->top * srcinfo->geom->virtstride - + srcinfo->rect->left * srcinfo->format->bitspp / 8; - - gcmofill->clearcolor_ldst = gcmofill_clearcolor_ldst; - gcmofill->clearcolor.raw = getinternalcolor(fillcolorptr, - srcinfo->format); - - /*********************************************************************** - ** Configure and start fill. - */ - - /* Set destination configuration. */ - 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->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->startde.cmd.fld = gcfldstartde; - - /* Set destination rectangle. */ - 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); - return bverror; -} - -#define ENABLE_HW_BLOCK 1 -static enum bverror do_blit_end(struct bvbltparams *bltparams, - struct gcbatch *batch) -{ - enum bverror bverror; - struct gcmobltconfig *gcmobltconfig; - struct gcmostart *gcmostart; - - GCENTER(GCZONE_DO_BLIT); - - GCDBG(GCZONE_DO_BLIT, "finalizing the blit, scrcount = %d\n", - batch->gcblit.srccount); - - /*********************************************************************** - * Configure the operation. - */ - - /* Allocate command buffer. */ - bverror = claim_buffer(bltparams, batch, - sizeof(struct gcmobltconfig), - (void **) &gcmobltconfig); - if (bverror != BVERR_NONE) - goto exit; - - /* Configure multi-source control. */ - 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); - GCDBG(GCZONE_DO_BLIT, " swizzle code = %d\n", - batch->dstformat->swizzle); - GCDBG(GCZONE_DO_BLIT, " format code = %d\n", - batch->dstformat->format); - - 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. */ - 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->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->right - batch->left), - abs(batch->bottom - batch->top)); - -exit: - GCEXITARG(GCZONE_DO_BLIT, "bv%s = %d\n", - (bverror == BVERR_NONE) ? "result" : "error", bverror); - return bverror; -} - -static enum bverror do_blit(struct bvbltparams *bltparams, - struct gcbatch *batch, - struct srcinfo *srcinfo) -{ - enum bverror bverror = BVERR_NONE; - - struct gcmosrc *gcmosrc; - struct gcmosrcalpha *gcmosrcalpha; - - unsigned int index; - - struct bvbuffmap *dstmap; - struct bvbuffdesc *dstdesc; - struct bvsurfgeom *dstgeom; - struct bvformatxlate *dstformat; - int dstshiftX, dstshiftY; - int dstalign, dstbyteshift; - - struct bvbuffmap *srcmap; - struct gcalpha *srcgca; - struct bvrect *srcrect; - struct bvbuffdesc *srcdesc; - struct bvsurfgeom *srcgeom; - 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); - - /* Create source object shortcuts. */ - srcmap = NULL; - srcgca = srcinfo->gca; - srcrect = srcinfo->rect; - srcdesc = srcinfo->buf.desc; - srcgeom = srcinfo->geom; - srcformat = srcinfo->format; - - /* Create destination object shortcuts. */ - dstmap = NULL; - dstdesc = bltparams->dstdesc; - dstgeom = bltparams->dstgeom; - dstformat = batch->dstformat; - - /*********************************************************************** - * 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; - - case ROT_ANGLE_90: - srcshiftX = srctop - dsttop; - srcshiftY = (srcgeom->width - srcleft) - - (batch->dstwidth - dstleft); - break; - - case ROT_ANGLE_180: - srcshiftX = (srcgeom->width - srcleft) - - (batch->dstwidth - dstleft); - srcshiftY = (srcgeom->height - srctop) - - (batch->dstheight - dsttop); - break; - - case ROT_ANGLE_270: - srcshiftX = (srcgeom->height - srctop) - - (batch->dstheight - dsttop); - srcshiftY = srcleft - dstleft; - break; - - default: - srcshiftX = 0; - srcshiftY = 0; - } - - /* Compute the source surface offset in bytes. */ - srcbyteshift = srcshiftY * (int) srcgeom->virtstride - + srcshiftX * (int) srcformat->bitspp / 8; - - /* Compute the source offset in pixels needed to compensate - * for the surface base address misalignment if any. */ - srcalign = get_pixeloffset(srcdesc, srcformat, srcbyteshift); - - 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); - - /* Apply the source alignment. */ - srcbyteshift += srcalign * (int) srcformat->bitspp / 8; - srcshiftX += srcalign; - - GCDBG(GCZONE_SURF, " adjusted surface offset (pixels) = %d,%d\n", - srcshiftX, srcshiftY); - GCDBG(GCZONE_SURF, " adjusted surface offset (bytes) = 0x%08X\n", - srcbyteshift); - - /* Determine the destination surface shift. */ - dstshiftX = batch->dstalign; - dstshiftY = (((srcinfo->angle + 3) % 4) == batch->dstangle) - ? srcalign : 0; - - /* 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->format == GCREG_DE_FORMAT_NV12) || - (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. */ - 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; - - case ROT_ANGLE_90: - /* Adjust top coordinate. */ - srctop -= srcalign; - - /* Determine source size. */ - srcsurfwidth = srcgeom->height - srcalign; - srcsurfheight = srcgeom->width; - break; - - case ROT_ANGLE_180: - /* Determine source size. */ - srcsurfwidth = srcgeom->width - srcalign; - srcsurfheight = srcgeom->height; - break; - - case ROT_ANGLE_270: - /* Determine source size. */ - srcsurfwidth = srcgeom->height - srcalign; - srcsurfheight = srcgeom->width; - break; - - default: - srcsurfwidth = 0; - srcsurfheight = 0; - } - - GCDBG(GCZONE_SURF, "srcrect origin = %d,%d\n", - srcleft, srctop); - GCDBG(GCZONE_SURF, "source physical size = %dx%d\n", - srcsurfwidth, srcsurfheight); - - /* Set the physical destination size. */ - physwidth = batch->dstphyswidth; - physheight = batch->dstphysheight; - - /* 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; - - case ROT_ANGLE_90: - dsttop -= srcalign; - dstbottom -= srcalign; - - /* 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; - - 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; - - 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; - - default: - physwidth = 0; - physheight = 0; - } - - /* Source geometry is now the same as the destination. */ - if (orthogonal) { - srcsurfwidth = physheight; - srcsurfheight = physwidth; - } else { - srcsurfwidth = physwidth; - srcsurfheight = physheight; - } - - /* Enable multi-source. */ - multisrc = 1; - GCDBG(GCZONE_SURF, "multi-source enabled.\n"); - } - - /* 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; - - /* 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; - } - - /* Set destination coordinates. */ - batch->left = dstleft; - batch->top = dsttop; - batch->right = dstright; - batch->bottom = dstbottom; - - /* Map the destination. */ - bverror = do_map(dstdesc, batch, &dstmap); - if (bverror != BVERR_NONE) { - bltparams->errdesc = gccontext.bverrorstr; - goto exit; - } - - /* Set the new destination. */ - bverror = set_dst(bltparams, batch, dstmap); - if (bverror != BVERR_NONE) - goto exit; - - /* Reset the modified flag. */ - batch->batchflags &= ~(BVBATCH_DST | - BVBATCH_CLIPRECT | - BVBATCH_DESTRECT); - - /* Map the source. */ - bverror = do_map(srcdesc, batch, &srcmap); - if (bverror != BVERR_NONE) { - bltparams->errdesc = gccontext.bverrorstr; - goto exit; - } - - /*************************************************************** - ** Configure source. - */ - - /* We need to walk in blocks if the source and the destination - * surfaces are orthogonal to each other. */ - batch->blockenable = orthogonal; - - /* Allocate command buffer. */ - bverror = claim_buffer(bltparams, batch, - sizeof(struct gcmosrc), - (void **) &gcmosrc); - if (bverror != BVERR_NONE) - goto exit; - - /* Shortcut to the register index. */ - index = batch->gcblit.srccount; - - add_fixup(bltparams, batch, &gcmosrc->address, srcbyteshift); - - /* 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; - - if ((dstgeom->format & OCDFMTDEF_NON_PREMULT) != 0) { - gcmosrc->mult.reg.dstpremul - = GCREG_COLOR_MULTIPLY_MODES_SRC_PREMULTIPLY_ENABLE; - - gcmosrc->mult.reg.dstdemul - = GCREG_COLOR_MULTIPLY_MODES_DST_DEMULTIPLY_ENABLE; - } else { - gcmosrc->mult.reg.dstpremul - = GCREG_COLOR_MULTIPLY_MODES_SRC_PREMULTIPLY_DISABLE; - - gcmosrc->mult.reg.dstdemul - = GCREG_COLOR_MULTIPLY_MODES_DST_DEMULTIPLY_DISABLE; - } - - if (srcformat->format == GCREG_DE_FORMAT_NV12) { - struct gcmosrcplanaryuv *yuv; - int uvshift = srcbyteshift; - -#if 0 - /* TODO: needs rework */ - 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 - - bverror = claim_buffer(bltparams, batch, - sizeof(struct gcmosrcplanaryuv), - (void **) &yuv); - if (bverror != BVERR_NONE) - goto exit; - - 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 - /* TODO: needs rework */ - 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; - } -#endif - - GCDBG(GCZONE_SURF, " uvshift = 0x%08X (%d)\n", - uvshift, uvshift); - - /* add fixed offset from Y plane */ - uvshift += srcgeom->virtstride * srcgeom->height; - - GCDBG(GCZONE_SURF, " final uvshift = 0x%08X (%d)\n", - uvshift, uvshift); - - yuv->uplaneaddress = GET_MAP_HANDLE(srcmap); - add_fixup(bltparams, batch, &yuv->uplaneaddress, uvshift); - - yuv->uplanestride = srcgeom->virtstride; - - yuv->vplaneaddress = GET_MAP_HANDLE(srcmap); - add_fixup(bltparams, batch, &yuv->vplaneaddress, uvshift); - - yuv->vplanestride = srcgeom->virtstride; - } - - if (srcgca != NULL) { - gcmosrc->alphacontrol_ldst - = gcmosrc_alphacontrol_ldst[index]; - gcmosrc->alphacontrol.raw = 0; - gcmosrc->alphacontrol.reg.enable - = GCREG_ALPHA_CONTROL_ENABLE_ON; - - /* Allocate command buffer. */ - bverror = claim_buffer(bltparams, batch, - sizeof(struct gcmosrcalpha), - (void **) &gcmosrcalpha); - if (bverror != BVERR_NONE) - goto exit; - - 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"); - - 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); - return bverror; -} - -static enum bverror do_filter(struct bvbltparams *bltparams, - struct gcbatch *batch, - struct srcinfo *srcinfo) -{ - enum bverror bverror; - GCENTER(GCZONE_DO_FILTER); - - 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); - return bverror; -} - /******************************************************************************* * Library constructor and destructor. @@ -3821,55 +491,72 @@ static enum bverror do_filter(struct bvbltparams *bltparams, void bv_init(void) { + struct gccontext *gccontext = get_context(); + GCDBG_REGISTER(gcbv); - GCLOCK_INIT(&gccontext.batchlock); - GCLOCK_INIT(&gccontext.bufferlock); - GCLOCK_INIT(&gccontext.fixuplock); - GCLOCK_INIT(&gccontext.maplock); - GCLOCK_INIT(&gccontext.callbacklock); - INIT_LIST_HEAD(&gccontext.callbacklist); - INIT_LIST_HEAD(&gccontext.callbackvac); - INIT_LIST_HEAD(&gccontext.vac_unmap); + GCDBG_REGISTER(gcparser); + GCDBG_REGISTER(gcmap); + GCDBG_REGISTER(gcbuffer); + GCDBG_REGISTER(gcfill); + GCDBG_REGISTER(gcblit); + GCDBG_REGISTER(gcfilter); + + GCLOCK_INIT(&gccontext->batchlock); + GCLOCK_INIT(&gccontext->bufferlock); + GCLOCK_INIT(&gccontext->fixuplock); + GCLOCK_INIT(&gccontext->maplock); + GCLOCK_INIT(&gccontext->callbacklock); + + INIT_LIST_HEAD(&gccontext->unmapvac); + INIT_LIST_HEAD(&gccontext->buffervac); + INIT_LIST_HEAD(&gccontext->fixupvac); + INIT_LIST_HEAD(&gccontext->batchvac); + INIT_LIST_HEAD(&gccontext->callbacklist); + INIT_LIST_HEAD(&gccontext->callbackvac); } void bv_exit(void) { + struct gccontext *gccontext = get_context(); + struct bvbuffmap *bvbuffmap; struct list_head *head; - struct gcschedunmap *bufunmap; - struct bvbuffmap *bufmap; - struct gcbuffer *buffer; - struct gcfixup *fixup; - struct gcvacbatch *batch; - - while (!list_empty(&gccontext.vac_unmap)) { - head = gccontext.vac_unmap.next; - bufunmap = list_entry(head, struct gcschedunmap, link); - list_del(head); - gcfree(bufunmap); + struct gcschedunmap *gcschedunmap; + struct gcbuffer *gcbuffer; + struct gcfixup *gcfixup; + struct gcbatch *gcbatch; + + while (gccontext->buffmapvac != NULL) { + bvbuffmap = gccontext->buffmapvac; + gccontext->buffmapvac = bvbuffmap->nextmap; + gcfree(bvbuffmap); } - while (gccontext.vac_buffmap != NULL) { - bufmap = gccontext.vac_buffmap; - gccontext.vac_buffmap = bufmap->nextmap; - gcfree(bufmap); + while (!list_empty(&gccontext->unmapvac)) { + head = gccontext->unmapvac.next; + gcschedunmap = list_entry(head, struct gcschedunmap, link); + list_del(head); + gcfree(gcschedunmap); } - while (gccontext.vac_buffers != NULL) { - buffer = gccontext.vac_buffers; - gccontext.vac_buffers = buffer->next; - gcfree(buffer); + while (!list_empty(&gccontext->buffervac)) { + head = gccontext->buffervac.next; + gcbuffer = list_entry(head, struct gcbuffer, link); + list_del(head); + gcfree(gcbuffer); } - while (gccontext.vac_fixups != NULL) { - fixup = gccontext.vac_fixups; - gccontext.vac_fixups = fixup->next; - gcfree(fixup); + while (!list_empty(&gccontext->fixupvac)) { + head = gccontext->fixupvac.next; + gcfixup = list_entry(head, struct gcfixup, link); + list_del(head); + gcfree(gcfixup); } - while (gccontext.vac_batches != NULL) { - batch = gccontext.vac_batches; - gccontext.vac_batches = batch->next; - gcfree(batch); + while (!list_empty(&gccontext->batchvac)) { + head = gccontext->batchvac.next; + gcbatch = list_entry(head, struct gcbatch, link); + list_del(head); + gcfree(gcbatch); } } @@ -3878,25 +565,25 @@ void bv_exit(void) * Library API. */ -enum bverror bv_map(struct bvbuffdesc *buffdesc) +enum bverror bv_map(struct bvbuffdesc *bvbuffdesc) { enum bverror bverror; struct bvbuffmap *bvbuffmap; - GCENTERARG(GCZONE_MAPPING, "buffdesc = 0x%08X\n", - (unsigned int) buffdesc); + GCENTERARG(GCZONE_MAPPING, "bvbuffdesc = 0x%08X\n", + (unsigned int) bvbuffdesc); - if (buffdesc == NULL) { + if (bvbuffdesc == NULL) { BVSETERROR(BVERR_BUFFERDESC, "bvbuffdesc is NULL"); goto exit; } - if (buffdesc->structsize < STRUCTSIZE(buffdesc, map)) { + if (bvbuffdesc->structsize < STRUCTSIZE(bvbuffdesc, map)) { BVSETERROR(BVERR_BUFFERDESC_VERS, "argument has invalid size"); goto exit; } - bverror = do_map(buffdesc, NULL, &bvbuffmap); + bverror = do_map(bvbuffdesc, NULL, &bvbuffmap); exit: GCEXITARG(GCZONE_MAPPING, "bv%s = %d\n", @@ -3904,33 +591,34 @@ exit: return bverror; } -enum bverror bv_unmap(struct bvbuffdesc *buffdesc) +enum bverror bv_unmap(struct bvbuffdesc *bvbuffdesc) { enum bverror bverror = BVERR_NONE; enum bverror otherbverror = BVERR_NONE; + struct gccontext *gccontext = get_context(); struct bvbuffmap *prev = NULL; struct bvbuffmap *bvbuffmap; struct bvbuffmapinfo *bvbuffmapinfo; struct gcmap gcmap; - GCENTERARG(GCZONE_MAPPING, "buffdesc = 0x%08X\n", - (unsigned int) buffdesc); + GCENTERARG(GCZONE_MAPPING, "bvbuffdesc = 0x%08X\n", + (unsigned int) bvbuffdesc); /* Lock access to the mapping list. */ - GCLOCK(&gccontext.maplock); + GCLOCK(&gccontext->maplock); - if (buffdesc == NULL) { + if (bvbuffdesc == NULL) { BVSETERROR(BVERR_BUFFERDESC, "bvbuffdesc is NULL"); goto exit; } - if (buffdesc->structsize < STRUCTSIZE(buffdesc, map)) { + if (bvbuffdesc->structsize < STRUCTSIZE(bvbuffdesc, map)) { BVSETERROR(BVERR_BUFFERDESC_VERS, "argument has invalid size"); goto exit; } /* Is the buffer mapped? */ - bvbuffmap = buffdesc->map; + bvbuffmap = bvbuffdesc->map; if (bvbuffmap == NULL) { GCDBG(GCZONE_MAPPING, "buffer isn't mapped.\n"); goto exit; @@ -3955,7 +643,7 @@ enum bverror bv_unmap(struct bvbuffdesc *buffdesc) "no mapping from our implementation.\n"); /* No, call other implementations. */ - bverror = buffdesc->map->bv_unmap(buffdesc); + bverror = bvbuffdesc->map->bv_unmap(bvbuffdesc); goto exit; } @@ -3968,16 +656,16 @@ enum bverror bv_unmap(struct bvbuffdesc *buffdesc) /* Remove our mapping. */ if (prev == NULL) - buffdesc->map = bvbuffmap->nextmap; + bvbuffdesc->map = bvbuffmap->nextmap; else prev->nextmap = bvbuffmap->nextmap; /* Call other implementation. */ - otherbverror = buffdesc->map->bv_unmap(buffdesc); + otherbverror = bvbuffdesc->map->bv_unmap(bvbuffdesc); /* Add our mapping back. */ - bvbuffmap->nextmap = buffdesc->map; - buffdesc->map = bvbuffmap; + bvbuffmap->nextmap = bvbuffdesc->map; + bvbuffdesc->map = bvbuffmap; prev = NULL; } else { GCDBG(GCZONE_MAPPING, @@ -4017,7 +705,7 @@ enum bverror bv_unmap(struct bvbuffdesc *buffdesc) /* Remove from the buffer descriptor list. */ if (prev == NULL) - buffdesc->map = bvbuffmap->nextmap; + bvbuffdesc->map = bvbuffmap->nextmap; else prev->nextmap = bvbuffmap->nextmap; @@ -4025,27 +713,28 @@ enum bverror bv_unmap(struct bvbuffdesc *buffdesc) bvbuffmap->structsize = 0; /* Add to the vacant list. */ - bvbuffmap->nextmap = gccontext.vac_buffmap; - gccontext.vac_buffmap = bvbuffmap; + bvbuffmap->nextmap = gccontext->buffmapvac; + gccontext->buffmapvac = bvbuffmap; exit: /* Unlock access to the mapping list. */ - GCUNLOCK(&gccontext.maplock); + GCUNLOCK(&gccontext->maplock); GCEXITARG(GCZONE_MAPPING, "bv%s = %d\n", (bverror == BVERR_NONE) ? "result" : "error", bverror); return bverror; } -enum bverror bv_blt(struct bvbltparams *bltparams) +enum bverror bv_blt(struct bvbltparams *bvbltparams) { enum bverror bverror = BVERR_NONE; + struct gccontext *gccontext = get_context(); struct gcalpha *gca = NULL; struct gcalpha _gca; unsigned int op, type, blend, format; unsigned int batchexec = 0; bool nop = false; - struct gcbatch *batch; + struct gcbatch *gcbatch; struct bvrect *dstrect; int src1used, src2used, maskused; struct srcinfo srcinfo[2]; @@ -4053,106 +742,106 @@ enum bverror bv_blt(struct bvbltparams *bltparams) struct gccommit gccommit; int i, srccount, res; - GCENTERARG(GCZONE_BLIT, "bltparams = 0x%08X\n", - (unsigned int) bltparams); + GCENTERARG(GCZONE_BLIT, "bvbltparams = 0x%08X\n", + (unsigned int) bvbltparams); /* Verify blt parameters structure. */ - if (bltparams == NULL) { + if (bvbltparams == NULL) { BVSETERROR(BVERR_BLTPARAMS_VERS, "bvbltparams is NULL"); goto exit; } - if (bltparams->structsize < STRUCTSIZE(bltparams, callbackdata)) { + if (bvbltparams->structsize < STRUCTSIZE(bvbltparams, callbackdata)) { BVSETERROR(BVERR_BLTPARAMS_VERS, "argument has invalid size"); goto exit; } /* Reset the error message. */ - bltparams->errdesc = NULL; + bvbltparams->errdesc = NULL; /* Verify the destination parameters structure. */ - GCDBG(GCZONE_SURF, "verifying destination parameters.\n"); + GCDBG(GCZONE_DEST, "verifying destination parameters.\n"); - res = verify_surface(0, (union bvinbuff *) &bltparams->dstdesc, - bltparams->dstgeom); + res = verify_surface(0, (union bvinbuff *) &bvbltparams->dstdesc, + bvbltparams->dstgeom); if (res != -1) { BVSETBLTSURFERROR(res, g_destsurferr); goto exit; } /* Extract the operation flags. */ - op = (bltparams->flags & BVFLAG_OP_MASK) >> BVFLAG_OP_SHIFT; - type = (bltparams->flags & BVFLAG_BATCH_MASK) >> BVFLAG_BATCH_SHIFT; + op = (bvbltparams->flags & BVFLAG_OP_MASK) >> BVFLAG_OP_SHIFT; + type = (bvbltparams->flags & BVFLAG_BATCH_MASK) >> BVFLAG_BATCH_SHIFT; switch (type) { case (BVFLAG_BATCH_NONE >> BVFLAG_BATCH_SHIFT): - bverror = allocate_batch(bltparams, &batch); + bverror = allocate_batch(bvbltparams, &gcbatch); if (bverror != BVERR_NONE) { - bltparams->errdesc = gccontext.bverrorstr; + bvbltparams->errdesc = gccontext->bverrorstr; goto exit; } batchexec = 1; - batch->batchflags = 0x7FFFFFFF; + gcbatch->batchflags = 0x7FFFFFFF; GCDBG(GCZONE_BATCH, "BVFLAG_BATCH_NONE(0x%08X)\n", - (unsigned int) batch); + (unsigned int) gcbatch); break; case (BVFLAG_BATCH_BEGIN >> BVFLAG_BATCH_SHIFT): - bverror = allocate_batch(bltparams, &batch); + bverror = allocate_batch(bvbltparams, &gcbatch); if (bverror != BVERR_NONE) { - bltparams->errdesc = gccontext.bverrorstr; + bvbltparams->errdesc = gccontext->bverrorstr; goto exit; } - bltparams->batch = (struct bvbatch *) batch; + bvbltparams->batch = (struct bvbatch *) gcbatch; batchexec = 0; - bltparams->batchflags = - batch->batchflags = 0x7FFFFFFF; + bvbltparams->batchflags = + gcbatch->batchflags = 0x7FFFFFFF; GCDBG(GCZONE_BATCH, "BVFLAG_BATCH_BEGIN(0x%08X)\n", - (unsigned int) batch); + (unsigned int) gcbatch); break; case (BVFLAG_BATCH_CONTINUE >> BVFLAG_BATCH_SHIFT): - batch = (struct gcbatch *) bltparams->batch; - if (batch == NULL) { + gcbatch = (struct gcbatch *) bvbltparams->batch; + if (gcbatch == NULL) { BVSETBLTERROR(BVERR_BATCH, "batch is not initialized"); goto exit; } - if (batch->structsize < STRUCTSIZE(batch, unmap)) { + if (gcbatch->structsize < STRUCTSIZE(gcbatch, unmap)) { BVSETBLTERROR(BVERR_BATCH, "invalid batch"); goto exit; } batchexec = 0; - batch->batchflags = bltparams->batchflags; + gcbatch->batchflags = bvbltparams->batchflags; GCDBG(GCZONE_BATCH, "BVFLAG_BATCH_CONTINUE(0x%08X)\n", - (unsigned int) batch); + (unsigned int) gcbatch); break; case (BVFLAG_BATCH_END >> BVFLAG_BATCH_SHIFT): - batch = (struct gcbatch *) bltparams->batch; - if (batch == NULL) { + gcbatch = (struct gcbatch *) bvbltparams->batch; + if (gcbatch == NULL) { BVSETBLTERROR(BVERR_BATCH, "batch is not initialized"); goto exit; } - if (batch->structsize < STRUCTSIZE(batch, unmap)) { + if (gcbatch->structsize < STRUCTSIZE(gcbatch, unmap)) { BVSETBLTERROR(BVERR_BATCH, "invalid batch"); goto exit; } batchexec = 1; - nop = (bltparams->batchflags & BVBATCH_ENDNOP) != 0; - batch->batchflags = bltparams->batchflags; + nop = (bvbltparams->batchflags & BVBATCH_ENDNOP) != 0; + gcbatch->batchflags = bvbltparams->batchflags; GCDBG(GCZONE_BATCH, "BVFLAG_BATCH_END(0x%08X)\n", - (unsigned int) batch); + (unsigned int) gcbatch); break; default: @@ -4161,23 +850,21 @@ enum bverror bv_blt(struct bvbltparams *bltparams) } GCDBG(GCZONE_BATCH, "batchflags=0x%08X\n", - (unsigned int) batch->batchflags); - - GCDUMPSCHEDULE(); + (unsigned int) gcbatch->batchflags); if (!nop) { /* Get a shortcut to the destination rectangle. */ - dstrect = &bltparams->dstrect; + dstrect = &bvbltparams->dstrect; /* Verify the batch change flags. */ - VERIFYBATCH(batch->batchflags >> 12, - &batch->prevdstrect, dstrect); + GCVERIFYBATCH(gcbatch->batchflags >> 12, + &gcbatch->prevdstrect, dstrect); switch (op) { case (BVFLAG_ROP >> BVFLAG_OP_SHIFT): GCDBG(GCZONE_BLIT, "BVFLAG_ROP\n"); - rop = bltparams->op.rop; + rop = bvbltparams->op.rop; src1used = ((rop & 0xCCCC) >> 2) ^ (rop & 0x3333); src2used = ((rop & 0xF0F0) >> 4) @@ -4189,11 +876,11 @@ enum bverror bv_blt(struct bvbltparams *bltparams) case (BVFLAG_BLEND >> BVFLAG_OP_SHIFT): GCDBG(GCZONE_BLIT, "BVFLAG_BLEND\n"); - blend = bltparams->op.blend; + blend = bvbltparams->op.blend; format = (blend & BVBLENDDEF_FORMAT_MASK) >> BVBLENDDEF_FORMAT_SHIFT; - bverror = parse_blend(bltparams, blend, &_gca); + bverror = parse_blend(bvbltparams, blend, &_gca); if (bverror != BVERR_NONE) goto exit; @@ -4226,7 +913,7 @@ enum bverror bv_blt(struct bvbltparams *bltparams) } /* Parse destination parameters. */ - bverror = parse_destination(bltparams, batch); + bverror = parse_destination(bvbltparams, gcbatch); if (bverror != BVERR_NONE) goto exit; @@ -4234,35 +921,35 @@ enum bverror bv_blt(struct bvbltparams *bltparams) /* Verify the src1 parameters structure. */ if (src1used) { - GCDBG(GCZONE_BLIT, "src1used\n"); - GCDBG(GCZONE_SURF, "verifying source1 parameters.\n"); + GCDBG(GCZONE_SRC, "src1used\n"); + GCDBG(GCZONE_SRC, "verifying source1 parameters.\n"); res = verify_surface( - bltparams->flags & BVBATCH_TILE_SRC1, - &bltparams->src1, bltparams->src1geom); + bvbltparams->flags & BVBATCH_TILE_SRC1, + &bvbltparams->src1, bvbltparams->src1geom); if (res != -1) { BVSETBLTSURFERROR(res, g_src1surferr); goto exit; } /* Verify the batch change flags. */ - VERIFYBATCH(batch->batchflags >> 14, - &batch->prevsrc1rect, - &bltparams->src1rect); + GCVERIFYBATCH(gcbatch->batchflags >> 14, + &gcbatch->prevsrc1rect, + &bvbltparams->src1rect); /* Same as the destination? */ - if (same_phys_area(bltparams->src1.desc, - &bltparams->src1rect, - bltparams->dstdesc, + if (same_phys_area(bvbltparams->src1.desc, + &bvbltparams->src1rect, + bvbltparams->dstdesc, dstrect)) { GCDBG(GCZONE_BLIT, "src1 is the same as dst\n"); } else { srcinfo[srccount].index = 0; - srcinfo[srccount].buf = bltparams->src1; - srcinfo[srccount].geom = bltparams->src1geom; - srcinfo[srccount].rect = &bltparams->src1rect; + srcinfo[srccount].buf = bvbltparams->src1; + srcinfo[srccount].geom = bvbltparams->src1geom; + srcinfo[srccount].rect = &bvbltparams->src1rect; - bverror = parse_source(bltparams, batch, + bverror = parse_source(bvbltparams, gcbatch, &srcinfo[srccount]); if (bverror != BVERR_NONE) goto exit; @@ -4273,35 +960,35 @@ enum bverror bv_blt(struct bvbltparams *bltparams) /* Verify the src2 parameters structure. */ if (src2used) { - GCDBG(GCZONE_BLIT, "src2used\n"); - GCDBG(GCZONE_SURF, "verifying source2 parameters.\n"); + GCDBG(GCZONE_SRC, "src2used\n"); + GCDBG(GCZONE_SRC, "verifying source2 parameters.\n"); res = verify_surface( - bltparams->flags & BVBATCH_TILE_SRC2, - &bltparams->src2, bltparams->src2geom); + bvbltparams->flags & BVBATCH_TILE_SRC2, + &bvbltparams->src2, bvbltparams->src2geom); if (res != -1) { BVSETBLTSURFERROR(res, g_src2surferr); goto exit; } /* Verify the batch change flags. */ - VERIFYBATCH(batch->batchflags >> 16, - &batch->prevsrc2rect, - &bltparams->src2rect); + GCVERIFYBATCH(gcbatch->batchflags >> 16, + &gcbatch->prevsrc2rect, + &bvbltparams->src2rect); /* Same as the destination? */ - if (same_phys_area(bltparams->src2.desc, - &bltparams->src2rect, - bltparams->dstdesc, + if (same_phys_area(bvbltparams->src2.desc, + &bvbltparams->src2rect, + bvbltparams->dstdesc, dstrect)) { GCDBG(GCZONE_BLIT, "src2 is the same as dst\n"); } else { srcinfo[srccount].index = 1; - srcinfo[srccount].buf = bltparams->src2; - srcinfo[srccount].geom = bltparams->src2geom; - srcinfo[srccount].rect = &bltparams->src2rect; + srcinfo[srccount].buf = bvbltparams->src2; + srcinfo[srccount].geom = bvbltparams->src2geom; + srcinfo[srccount].rect = &bvbltparams->src2rect; - bverror = parse_source(bltparams, batch, + bverror = parse_source(bvbltparams, gcbatch, &srcinfo[srccount]); if (bverror != BVERR_NONE) goto exit; @@ -4312,21 +999,21 @@ enum bverror bv_blt(struct bvbltparams *bltparams) /* Verify the mask parameters structure. */ if (maskused) { - GCDBG(GCZONE_BLIT, "maskused\n"); - GCDBG(GCZONE_SURF, "verifying mask parameters.\n"); + GCDBG(GCZONE_MASK, "maskused\n"); + GCDBG(GCZONE_MASK, "verifying mask parameters.\n"); res = verify_surface( - bltparams->flags & BVBATCH_TILE_MASK, - &bltparams->mask, bltparams->maskgeom); + bvbltparams->flags & BVBATCH_TILE_MASK, + &bvbltparams->mask, bvbltparams->maskgeom); if (res != -1) { BVSETBLTSURFERROR(res, g_masksurferr); goto exit; } /* Verify the batch change flags. */ - VERIFYBATCH(batch->batchflags >> 18, - &batch->prevmaskrect, - &bltparams->maskrect); + GCVERIFYBATCH(gcbatch->batchflags >> 18, + &gcbatch->prevmaskrect, + &bvbltparams->maskrect); BVSETBLTERROR(BVERR_OP, "operation with mask not supported"); @@ -4342,7 +1029,7 @@ enum bverror bv_blt(struct bvbltparams *bltparams) } else { for (i = 0; i < srccount; i += 1) { if (gca == NULL) { - srcinfo[i].rop = bltparams->op.rop; + srcinfo[i].rop = bvbltparams->op.rop; srcinfo[i].gca = NULL; } else if ((i + 1) != srccount) { srcinfo[i].rop = 0xCC; @@ -4362,14 +1049,15 @@ enum bverror bv_blt(struct bvbltparams *bltparams) if ((srcinfo[i].rect->width == 1) && (srcinfo[i].rect->height == 1) && - (bltparams->src1.desc->virtaddr)) - bverror = do_fill(bltparams, batch, + (bvbltparams->src1.desc->virtaddr)) + bverror = do_fill(bvbltparams, gcbatch, &srcinfo[i]); else if (EQ_SIZE(srcinfo[i].rect, dstrect)) - bverror = do_blit(bltparams, batch, + bverror = do_blit(bvbltparams, gcbatch, &srcinfo[i]); else - bverror = do_filter(bltparams, batch, + bverror = do_filter(bvbltparams, + gcbatch, &srcinfo[i]); if (bverror != BVERR_NONE) @@ -4384,13 +1072,13 @@ enum bverror bv_blt(struct bvbltparams *bltparams) GCDBG(GCZONE_BLIT, "preparing to submit the batch.\n"); /* Finalize the current operation. */ - bverror = batch->batchend(bltparams, batch); + bverror = gcbatch->batchend(bvbltparams, gcbatch); if (bverror != BVERR_NONE) goto exit; /* Add PE flush. */ GCDBG(GCZONE_BLIT, "appending the flush.\n"); - bverror = claim_buffer(bltparams, batch, + bverror = claim_buffer(bvbltparams, gcbatch, sizeof(struct gcmoflush), (void **) &flush); if (bverror != BVERR_NONE) @@ -4400,7 +1088,7 @@ enum bverror bv_blt(struct bvbltparams *bltparams) flush->flush.reg = gcregflush_pe2D; /* Process asynchronous operation. */ - if ((bltparams->flags & BVFLAG_ASYNC) == 0) { + if ((bvbltparams->flags & BVFLAG_ASYNC) == 0) { GCDBG(GCZONE_BLIT, "synchronous batch.\n"); gccommit.callback = NULL; gccommit.callbackparam = NULL; @@ -4409,9 +1097,9 @@ enum bverror bv_blt(struct bvbltparams *bltparams) struct gccallbackinfo *gccallbackinfo; GCDBG(GCZONE_BLIT, "asynchronous batch (0x%08X):\n", - bltparams->flags); + bvbltparams->flags); - if (bltparams->callbackfn == NULL) { + if (bvbltparams->callbackfn == NULL) { GCDBG(GCZONE_BLIT, "no callback given.\n"); gccommit.callback = NULL; gccommit.callbackparam = NULL; @@ -4425,9 +1113,9 @@ enum bverror bv_blt(struct bvbltparams *bltparams) } gccallbackinfo->callbackfn - = bltparams->callbackfn; + = bvbltparams->callbackfn; gccallbackinfo->callbackdata - = bltparams->callbackdata; + = bvbltparams->callbackdata; gccommit.callback = gccallback; gccommit.callbackparam = gccallbackinfo; @@ -4452,24 +1140,27 @@ enum bverror bv_blt(struct bvbltparams *bltparams) } /* Process scheduled unmappings. */ - unmap_implicit(batch); + do_unmap_implicit(gcbatch); INIT_LIST_HEAD(&gccommit.unmap); - list_splice_init(&batch->unmap, &gccommit.unmap); + list_splice_init(&gcbatch->unmap, &gccommit.unmap); /* Pass the batch for execution. */ - GCDUMPBATCH(batch); + GCDUMPBATCH(gcbatch); gccommit.gcerror = GCERR_NONE; gccommit.entrypipe = GCPIPE_2D; gccommit.exitpipe = GCPIPE_2D; - gccommit.buffer = batch->bufhead; + + INIT_LIST_HEAD(&gccommit.buffer); + list_splice_init(&gcbatch->buffer, &gccommit.buffer); GCDBG(GCZONE_BLIT, "submitting the batch.\n"); gc_commit_wrapper(&gccommit); - /* Move the list of mappings back to batch. */ - list_splice_init(&gccommit.unmap, &batch->unmap); + /* Move the lists back to the batch. */ + list_splice_init(&gccommit.buffer, &gcbatch->buffer); + list_splice_init(&gccommit.unmap, &gcbatch->unmap); /* Error? */ if (gccommit.gcerror != GCERR_NONE) { @@ -4492,10 +1183,9 @@ enum bverror bv_blt(struct bvbltparams *bltparams) } exit: - if ((batch != NULL) && batchexec) { - GCDUMPSCHEDULE(); - free_batch(batch); - bltparams->batch = NULL; + if ((gcbatch != NULL) && batchexec) { + free_batch(gcbatch); + bvbltparams->batch = NULL; } GCEXITARG(GCZONE_BLIT, "bv%s = %d\n", diff --git a/drivers/misc/gcx/gcbv/gcbv.h b/drivers/misc/gcx/gcbv/gcbv.h new file mode 100644 index 0000000..789d710 --- /dev/null +++ b/drivers/misc/gcx/gcbv/gcbv.h @@ -0,0 +1,459 @@ +/* + * This file is provided under a dual BSD/GPLv2 license. When using or + * redistributing this file, you may do so under either license. + * + * GPL LICENSE SUMMARY + * + * Copyright(c) 2012 Vivante Corporation. All rights reserved. + * + * This program is free software; you can redistribute it and/or modify it + * under the terms and conditions of the GNU General Public License, + * version 2, as published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for + * more details. + * + * You should have received a copy of the GNU General Public License along with + * this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. + * + * The full GNU General Public License is included in this distribution in + * the file called "COPYING". + * + * BSD LICENSE + * + * Copyright(c) 2012 Vivante Corporation. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * * Neither the name of Vivante Corporation nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE + * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + */ + +#ifndef GCBV_H +#define GCBV_H + +#include "gcmain.h" + +/******************************************************************************* + * 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) \ +) + +#define STRUCTSIZE(structptr, lastmember) \ +( \ + (size_t) &structptr->lastmember + \ + sizeof(structptr->lastmember) - \ + (size_t) structptr \ +) + +#define GET_MAP_HANDLE(map) \ +( \ + ((struct bvbuffmapinfo *) map->handle)->handle \ +) + +#define GC_CLIP_RESET_LEFT ((unsigned short) 0) +#define GC_CLIP_RESET_TOP ((unsigned short) 0) +#define GC_CLIP_RESET_RIGHT ((unsigned short) ((1 << 15) - 1)) +#define GC_CLIP_RESET_BOTTOM ((unsigned short) ((1 << 15) - 1)) + +#define BVSETERROR(error, message, ...) \ +do { \ + struct gccontext *tmpcontext = get_context(); \ + snprintf(tmpcontext->bverrorstr, sizeof(tmpcontext->bverrorstr), \ + message, ##__VA_ARGS__); \ + GCDUMPSTRING("%s(%d): [ERROR] %s\n", __func__, __LINE__, \ + tmpcontext->bverrorstr); \ + bverror = error; \ +} while (0) + +#define BVSETBLTERROR(error, message, ...) \ +do { \ + struct gccontext *tmpcontext = get_context(); \ + snprintf(tmpcontext->bverrorstr, sizeof(tmpcontext->bverrorstr), \ + message, ##__VA_ARGS__); \ + GCDUMPSTRING("%s(%d): [ERROR] %s\n", __func__, __LINE__, \ + tmpcontext->bverrorstr); \ + bvbltparams->errdesc = tmpcontext->bverrorstr; \ + bverror = error; \ +} while (0) + + +/******************************************************************************* + * Global data structure. + */ + +struct gccontext { + /* Last generated error message. */ + char bverrorstr[128]; + + /* Dynamically allocated structure cache. */ + struct bvbuffmap *buffmapvac; /* bvbuffmap */ + struct list_head unmapvac; /* gcschedunmap */ + struct list_head buffervac; /* gcbuffer */ + struct list_head fixupvac; /* gcfixup */ + struct list_head batchvac; /* gcbatch */ + + /* Callback lists. */ + struct list_head callbacklist; /* gccallbackinfo */ + struct list_head callbackvac; /* gccallbackinfo */ + + /* Access locks. */ + GCLOCK_TYPE batchlock; + GCLOCK_TYPE bufferlock; + GCLOCK_TYPE fixuplock; + GCLOCK_TYPE maplock; + GCLOCK_TYPE callbacklock; +}; + + +/******************************************************************************* + * Mapping structures. + */ + +/* bvbuffmap struct attachment. */ +struct bvbuffmapinfo { + /* Mapped handle for the buffer. */ + unsigned long handle; + + /* Number of times the client explicitly mapped this buffer. */ + int usermap; + + /* Number of times implicit mapping happened. */ + int automap; +}; + + +/******************************************************************************* + * Color format. + */ + +enum bvformattype { + BVFMT_RGB, + BVFMT_YUV +}; + +struct bvcomponent { + unsigned int shift; + unsigned int size; + unsigned int mask; +}; + +struct bvcsrgb { + struct bvcomponent r; + struct bvcomponent g; + struct bvcomponent b; + struct bvcomponent a; +}; + +struct bvformatxlate { + enum bvformattype type; + unsigned bitspp; + unsigned format; + unsigned swizzle; + struct bvcsrgb rgba; +}; + + +/******************************************************************************* + * Alpha blending. + */ + +/* Alpha blending hardware configuration. */ +struct gcblendconfig { + unsigned char factor_mode; + unsigned char color_reverse; + + bool src1used; + bool src2used; +}; + +/* 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; + + bool src1used; + bool src2used; +}; + + +/******************************************************************************* + * Rotation and mirror defines. + */ + +#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 + +#define GCREG_MIRROR_NONE 0x0 +#define GCREG_MIRROR_X 0x1 +#define GCREG_MIRROR_Y 0x2 +#define GCREG_MIRROR_XY 0x3 + +extern const unsigned int rotencoding[]; + + +/******************************************************************************* + * Batch structures. + */ + +/* Operation finalization call. */ +struct gcbatch; +typedef enum bverror (*gcbatchend) (struct bvbltparams *bvbltparams, + struct gcbatch *gcbatch); + +/* Blit states. */ +struct gcblit { + unsigned int srccount; + unsigned int multisrc; + unsigned short rop; +}; + +/* Batch header. */ +struct gcbatch { + /* Used to ID structure version. */ + unsigned int structsize; + + /* Batch change flags. */ + unsigned long batchflags; + + /* Pointer to the function to finalize the current operation. */ + gcbatchend batchend; + + /* 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; + int deltatop; + int deltaright; + int deltabottom; + + /* 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; + + /* Physical size of the matched destination and source surfaces + * for multi-source setup. */ + unsigned int physwidth; + unsigned int physheight; + + /* 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. */ + struct bvrect prevdstrect; + struct bvrect prevsrc1rect; + struct bvrect prevsrc2rect; + struct bvrect prevmaskrect; +#endif + + /* Total size of the command buffer. */ + unsigned int size; + + /* Command buffer list (gcbuffer). */ + struct list_head buffer; + + /* Scheduled implicit unmappings (gcschedunmap). */ + struct list_head unmap; + + /* Batch linked list (gcbatch). */ + struct list_head link; +}; + + +/******************************************************************************* + * srcinfo is 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; +}; + + +/******************************************************************************* + * Internal API entries. + */ + +/* Get the pointer to the context. */ +struct gccontext *get_context(void); + +/* Parsers. */ +enum bverror parse_format(struct bvbltparams *bvbltparams, + enum ocdformat ocdformat, + struct bvformatxlate **format); +enum bverror parse_blend(struct bvbltparams *bvbltparams, + enum bvblend blend, + struct gcalpha *gca); +enum bverror parse_destination(struct bvbltparams *bvbltparams, + struct gcbatch *gcbatch); +enum bverror parse_source(struct bvbltparams *bvbltparams, + struct gcbatch *gcbatch, + struct srcinfo *srcinfo); + +/* Return surface alignment offset. */ +int get_pixel_offset(struct bvbuffdesc *bvbuffdesc, + struct bvformatxlate *format, + int offset); + +/* Buffer mapping. */ +enum bverror do_map(struct bvbuffdesc *bvbuffdesc, + struct gcbatch *gcbatch, + struct bvbuffmap **map); +void do_unmap_implicit(struct gcbatch *gcbatch); + +/* Batch/command buffer management. */ +enum bverror do_end(struct bvbltparams *bvbltparams, + struct gcbatch *gcbatch); +enum bverror allocate_batch(struct bvbltparams *bvbltparams, + struct gcbatch **gcbatch); +void free_batch(struct gcbatch *gcbatch); +enum bverror append_buffer(struct bvbltparams *bvbltparams, + struct gcbatch *gcbatch, + struct gcbuffer **gcbuffer); + +enum bverror add_fixup(struct bvbltparams *bvbltparams, + struct gcbatch *gcbatch, + unsigned int *fixup, + unsigned int surfoffset); +enum bverror claim_buffer(struct bvbltparams *bvbltparams, + struct gcbatch *gcbatch, + unsigned int size, + void **buffer); + +/* Program the destination. */ +enum bverror set_dst(struct bvbltparams *bltparams, + struct gcbatch *batch, + struct bvbuffmap *dstmap); + +/* Rendering entry points. */ +enum bverror do_fill(struct bvbltparams *bltparams, + struct gcbatch *gcbatch, + struct srcinfo *srcinfo); +enum bverror do_blit(struct bvbltparams *bltparams, + struct gcbatch *gcbatch, + struct srcinfo *srcinfo); +enum bverror do_filter(struct bvbltparams *bvbltparams, + struct gcbatch *gcbatch, + struct srcinfo *srcinfo); + +#endif diff --git a/drivers/misc/gcx/gcbv/gcfill.c b/drivers/misc/gcx/gcbv/gcfill.c new file mode 100644 index 0000000..475467d --- /dev/null +++ b/drivers/misc/gcx/gcbv/gcfill.c @@ -0,0 +1,272 @@ +/* + * This file is provided under a dual BSD/GPLv2 license. When using or + * redistributing this file, you may do so under either license. + * + * GPL LICENSE SUMMARY + * + * Copyright(c) 2012 Vivante Corporation. All rights reserved. + * + * This program is free software; you can redistribute it and/or modify it + * under the terms and conditions of the GNU General Public License, + * version 2, as published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for + * more details. + * + * You should have received a copy of the GNU General Public License along with + * this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. + * + * The full GNU General Public License is included in this distribution in + * the file called "COPYING". + * + * BSD LICENSE + * + * Copyright(c) 2012 Vivante Corporation. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * * Neither the name of Vivante Corporation nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE + * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + */ + +#include "gcbv.h" + +#define GCZONE_NONE 0 +#define GCZONE_ALL (~0U) +#define GCZONE_COLOR (1 << 0) +#define GCZONE_FILL (1 << 1) + +GCDBG_FILTERDEF(gcfill, GCZONE_NONE, + "color", + "fill") + + +static inline unsigned int extract_component(unsigned int pixel, + struct bvcomponent *desc) +{ + unsigned int component; + unsigned int component8; + + component = (pixel & desc->mask) >> desc->shift; + GCDBG(GCZONE_COLOR, "mask=0x%08X, shift=%d, component=0x%08X\n", + desc->mask, desc->shift, component); + + switch (desc->size) { + case 0: + component8 = 0xFF; + GCDBG(GCZONE_COLOR, "component8=0x%08X\n", component8); + break; + + case 1: + component8 = component ? 0xFF : 0x00; + GCDBG(GCZONE_COLOR, "component8=0x%08X\n", component8); + break; + + case 4: + component8 = component | (component << 4); + GCDBG(GCZONE_COLOR, "component8=0x%08X\n", component8); + break; + + case 5: + component8 = (component << 3) | (component >> 2); + GCDBG(GCZONE_COLOR, "component8=0x%08X\n", component8); + break; + + case 6: + component8 = (component << 2) | (component >> 4); + GCDBG(GCZONE_COLOR, "component8=0x%08X\n", component8); + break; + + default: + component8 = component; + GCDBG(GCZONE_COLOR, "component8=0x%08X\n", component8); + } + + return component8; +} + +static unsigned int getinternalcolor(void *ptr, struct bvformatxlate *format) +{ + unsigned int srcpixel, dstpixel; + unsigned int r, g, b, a; + + switch (format->bitspp) { + case 16: + srcpixel = *(unsigned short *) ptr; + GCDBG(GCZONE_COLOR, "srcpixel=0x%08X\n", srcpixel); + break; + + case 32: + srcpixel = *(unsigned int *) ptr; + GCDBG(GCZONE_COLOR, "srcpixel=0x%08X\n", srcpixel); + break; + + default: + srcpixel = 0; + GCDBG(GCZONE_COLOR, "srcpixel=0x%08X\n", srcpixel); + } + + r = extract_component(srcpixel, &format->rgba.r); + g = extract_component(srcpixel, &format->rgba.g); + b = extract_component(srcpixel, &format->rgba.b); + a = extract_component(srcpixel, &format->rgba.a); + + GCDBG(GCZONE_COLOR, "(r,g,b,a)=0x%02X,0x%02X,0x%02X,0x%02X\n", + r, g, b, a); + + dstpixel = (a << 24) | (r << 16) | (g << 8) | b; + + GCDBG(GCZONE_COLOR, "dstpixel=0x%08X\n", dstpixel); + + return dstpixel; +} + +enum bverror do_fill(struct bvbltparams *bltparams, + struct gcbatch *batch, + struct srcinfo *srcinfo) +{ + enum bverror bverror; + struct gccontext *gccontext = get_context(); + int dstbyteshift; + struct gcmofill *gcmofill; + unsigned char *fillcolorptr; + struct bvbuffmap *dstmap = NULL; + + GCENTER(GCZONE_FILL); + + /* Finish previous batch if any. */ + bverror = batch->batchend(bltparams, batch); + 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) { + bltparams->errdesc = gccontext->bverrorstr; + goto exit; + } + + /* Set the new destination. */ + bverror = set_dst(bltparams, batch, dstmap); + if (bverror != BVERR_NONE) + goto exit; + + /* Reset the modified flag. */ + batch->batchflags &= ~(BVBATCH_DST | + BVBATCH_CLIPRECT | + BVBATCH_DESTRECT); + + /*********************************************************************** + ** Allocate command buffer. + */ + + bverror = claim_buffer(bltparams, batch, + sizeof(struct gcmofill), + (void **) &gcmofill); + if (bverror != BVERR_NONE) + goto exit; + + /*********************************************************************** + ** 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 = 1; + gcmofill->src.config.raw = 0; + + gcmofill->src.rotationheight_ldst = gcmofillsrc_rotationheight_ldst; + 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; + gcmofill->src.alphacontrol.raw = 0; + gcmofill->src.alphacontrol.reg.enable = GCREG_ALPHA_CONTROL_ENABLE_OFF; + + /*********************************************************************** + ** Set fill color. + */ + + fillcolorptr + = (unsigned char *) srcinfo->buf.desc->virtaddr + + srcinfo->rect->top * srcinfo->geom->virtstride + + srcinfo->rect->left * srcinfo->format->bitspp / 8; + + gcmofill->clearcolor_ldst = gcmofill_clearcolor_ldst; + gcmofill->clearcolor.raw = getinternalcolor(fillcolorptr, + srcinfo->format); + + /*********************************************************************** + ** Configure and start fill. + */ + + /* Set destination configuration. */ + 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->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->startde.cmd.fld = gcfldstartde; + + /* Set destination rectangle. */ + 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_FILL, "bv%s = %d\n", + (bverror == BVERR_NONE) ? "result" : "error", bverror); + return bverror; +} diff --git a/drivers/misc/gcx/gcbv/gcfilter.c b/drivers/misc/gcx/gcbv/gcfilter.c new file mode 100644 index 0000000..aa1c11e --- /dev/null +++ b/drivers/misc/gcx/gcbv/gcfilter.c @@ -0,0 +1,128 @@ +/* + * This file is provided under a dual BSD/GPLv2 license. When using or + * redistributing this file, you may do so under either license. + * + * GPL LICENSE SUMMARY + * + * Copyright(c) 2012 Vivante Corporation. All rights reserved. + * + * This program is free software; you can redistribute it and/or modify it + * under the terms and conditions of the GNU General Public License, + * version 2, as published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for + * more details. + * + * You should have received a copy of the GNU General Public License along with + * this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. + * + * The full GNU General Public License is included in this distribution in + * the file called "COPYING". + * + * BSD LICENSE + * + * Copyright(c) 2012 Vivante Corporation. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * * Neither the name of Vivante Corporation nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE + * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + */ + +#include "gcbv.h" + +#define GCZONE_NONE 0 +#define GCZONE_ALL (~0U) +#define GCZONE_FILTER (1 << 11) + +GCDBG_FILTERDEF(gcfilter, GCZONE_NONE, + "filter") + +enum bverror do_filter(struct bvbltparams *bltparams, + struct gcbatch *batch, + struct srcinfo *srcinfo) +{ +/* enum bverror bverror = BVERR_NONE;*/ + int verpass; + + GCENTER(GCZONE_FILTER); + + GCDBG(GCZONE_FILTER, "source rectangle:\n"); + GCDBG(GCZONE_FILTER, " stride = %d, geom = %dx%d\n", + srcinfo->geom->virtstride, + srcinfo->geom->width, srcinfo->geom->height); + GCDBG(GCZONE_FILTER, " rotation = %d\n", + srcinfo->angle); + GCDBG(GCZONE_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_FILTER, "destination rectangle:\n"); + GCDBG(GCZONE_FILTER, " stride = %d, geom size = %dx%d\n", + bltparams->dstgeom->virtstride, + bltparams->dstgeom->width, bltparams->dstgeom->height); + GCDBG(GCZONE_FILTER, " rotaton = %d\n", + batch->dstangle); + GCDBG(GCZONE_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); + + /*********************************************************************** + * Update kernel arrays. + */ + + /* Do we need the vertical pass? */ + verpass = (srcinfo->rect->height != bltparams->dstrect.height); + +#if 0 + /* Recompute the table if necessary. */ + gcmONERROR(_CalculateSyncTable( + hardware, + State->newHorKernelSize, + srcRectSize.x, + destRectSize.x, + horKernel + )); + + gcmONERROR(_CalculateSyncTable( + hardware, + State->newVerKernelSize, + srcRectSize.y, + destRectSize.y, + verKernel + )); +#endif + + GCEXITARG(GCZONE_FILTER, "bv%s = %d\n", + (bverror == BVERR_NONE) ? "result" : "error", bverror); + + /* Not implemented yet */ + return BVERR_OP; +} diff --git a/drivers/misc/gcx/gcbv/gcmap.c b/drivers/misc/gcx/gcbv/gcmap.c new file mode 100644 index 0000000..eac76f8 --- /dev/null +++ b/drivers/misc/gcx/gcbv/gcmap.c @@ -0,0 +1,364 @@ +/* + * This file is provided under a dual BSD/GPLv2 license. When using or + * redistributing this file, you may do so under either license. + * + * GPL LICENSE SUMMARY + * + * Copyright(c) 2012 Vivante Corporation. All rights reserved. + * + * This program is free software; you can redistribute it and/or modify it + * under the terms and conditions of the GNU General Public License, + * version 2, as published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for + * more details. + * + * You should have received a copy of the GNU General Public License along with + * this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. + * + * The full GNU General Public License is included in this distribution in + * the file called "COPYING". + * + * BSD LICENSE + * + * Copyright(c) 2012 Vivante Corporation. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * * Neither the name of Vivante Corporation nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE + * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + */ + +#include "gcbv.h" + +#define GCZONE_NONE 0 +#define GCZONE_ALL (~0U) +#define GCZONE_MAPPING (1 << 0) + +GCDBG_FILTERDEF(gcmap, GCZONE_NONE, + "mapping") + + +/******************************************************************************* + * Memory management. + */ + +enum bverror do_map(struct bvbuffdesc *bvbuffdesc, + struct gcbatch *batch, + struct bvbuffmap **map) +{ + static const int mapsize + = sizeof(struct bvbuffmap) + + sizeof(struct bvbuffmapinfo); + + enum bverror bverror; + struct gccontext *gccontext = get_context(); + struct bvbuffmap *bvbuffmap; + struct bvbuffmapinfo *bvbuffmapinfo; + struct bvphysdesc *bvphysdesc; + bool mappedbyothers; + struct gcmap gcmap; + struct gcschedunmap *gcschedunmap; + + GCENTERARG(GCZONE_MAPPING, "bvbuffdesc = 0x%08X\n", + (unsigned int) bvbuffdesc); + + /* Lock access to the mapping list. */ + GCLOCK(&gccontext->maplock); + + /* Try to find existing mapping. */ + bvbuffmap = bvbuffdesc->map; + while (bvbuffmap != NULL) { + if (bvbuffmap->bv_unmap == bv_unmap) + break; + bvbuffmap = bvbuffmap->nextmap; + } + + /* Not mapped yet? */ + if (bvbuffmap == NULL) { + /* New mapping, allocate a record. */ + if (gccontext->buffmapvac == NULL) { + bvbuffmap = gcalloc(struct bvbuffmap, mapsize); + if (bvbuffmap == NULL) { + BVSETERROR(BVERR_OOM, + "failed to allocate mapping record"); + goto fail; + } + + bvbuffmap->structsize = sizeof(struct bvbuffmap); + bvbuffmap->bv_unmap = bv_unmap; + bvbuffmap->handle = (unsigned long) (bvbuffmap + 1); + } else { + bvbuffmap = gccontext->buffmapvac; + gccontext->buffmapvac = bvbuffmap->nextmap; + } + + /* Setup buffer mapping. Here we need to check and make sure + * that the buffer starts at a location that is supported by + * the hw. If it is not, offset is computed and the buffer is + * extended by the value of the offset. */ + gcmap.gcerror = GCERR_NONE; + gcmap.handle = 0; + + if (bvbuffdesc->auxtype == BVAT_PHYSDESC) { + bvphysdesc = (struct bvphysdesc *) bvbuffdesc->auxptr; + + if (bvphysdesc->structsize < + STRUCTSIZE(bvphysdesc, pageoffset)) { + BVSETERROR(BVERR_BUFFERDESC_VERS, + "unsupported bvphysdesc version"); + goto fail; + } + + 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"); + GCDBG(GCZONE_MAPPING, "pagesize = %lu\n", + bvphysdesc->pagesize); + GCDBG(GCZONE_MAPPING, "pagearray = 0x%08X\n", + (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 = bvbuffdesc->virtaddr; + gcmap.pagesize = 0; + gcmap.pagearray = NULL; + gcmap.size = bvbuffdesc->length; + + GCDBG(GCZONE_MAPPING, "new mapping (%s):\n", + (batch == NULL) ? "explicit" : "implicit"); + GCDBG(GCZONE_MAPPING, "specified virtaddr = 0x%08X\n", + (unsigned int) bvbuffdesc->virtaddr); + GCDBG(GCZONE_MAPPING, "aligned virtaddr = 0x%08X\n", + (unsigned int) gcmap.buf.logical); + 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"); + goto fail; + } + + /* Set map handle. */ + bvbuffmapinfo = (struct bvbuffmapinfo *) bvbuffmap->handle; + bvbuffmapinfo->handle = gcmap.handle; + + /* Initialize reference counters. */ + if (batch == NULL) { + /* Explicit mapping. */ + bvbuffmapinfo->usermap = 1; + bvbuffmapinfo->automap = 0; + } else { + /* Implicit mapping; if there are existing mappings + * from other implementations, mark this an explicit + * mapping as well. */ + mappedbyothers = (bvbuffdesc->map != NULL); + GCDBG(GCZONE_MAPPING, "%smapped by others.\n", + mappedbyothers ? "" : "not "); + + bvbuffmapinfo->usermap = mappedbyothers ? 1 : 0; + bvbuffmapinfo->automap = 1; + } + + /* Add the record to the list of mappings. */ + bvbuffmap->nextmap = bvbuffdesc->map; + bvbuffdesc->map = bvbuffmap; + } else { + /* Mapping already exists. */ + bvbuffmapinfo = (struct bvbuffmapinfo *) bvbuffmap->handle; + + /* Advance reference counters. */ + if (batch == NULL) { + /* Explicit mapping. */ + GCDBG(GCZONE_MAPPING, "explicit map.\n"); + bvbuffmapinfo->usermap += 1; + } else { + /* Implicit mapping. */ + GCDBG(GCZONE_MAPPING, "implicit map.\n"); + bvbuffmapinfo->automap += 1; + } + + GCDBG(GCZONE_MAPPING, "mapping exists.\n"); + } + + GCDBG(GCZONE_MAPPING, "bvbuffmap = 0x%08X\n", + (unsigned int) bvbuffmap); + GCDBG(GCZONE_MAPPING, "explicit count = %d\n", + bvbuffmapinfo->usermap); + GCDBG(GCZONE_MAPPING, "implicit count = %d\n", + bvbuffmapinfo->automap); + + /* Schedule for unmapping. */ + if (batch != NULL) { + if (list_empty(&gccontext->unmapvac)) { + gcschedunmap = gcalloc(struct gcschedunmap, + sizeof(struct gcschedunmap)); + if (gcschedunmap == NULL) { + BVSETERROR(BVERR_OOM, + "failed to schedule unmapping"); + goto fail; + } + list_add(&gcschedunmap->link, &batch->unmap); + } else { + struct list_head *head; + head = gccontext->unmapvac.next; + gcschedunmap = list_entry(head, + struct gcschedunmap, + link); + list_move(&gcschedunmap->link, &batch->unmap); + } + + gcschedunmap->handle = (unsigned long) bvbuffdesc; + + GCDBG(GCZONE_MAPPING, "scheduled for unmapping.\n"); + } + + /* Set the map pointer. */ + *map = bvbuffmap; + + /* Unlock access to the mapping list. */ + GCUNLOCK(&gccontext->maplock); + + GCEXITARG(GCZONE_MAPPING, "handle = 0x%08X\n", + bvbuffmapinfo->handle); + return BVERR_NONE; + +fail: + if (bvbuffmap != NULL) { + bvbuffmap->nextmap = gccontext->buffmapvac; + gccontext->buffmapvac = bvbuffmap; + } + + /* Unlock access to the mapping list. */ + GCUNLOCK(&gccontext->maplock); + + GCEXITARG(GCZONE_MAPPING, "bverror = %d\n", bverror); + return bverror; +} + +void do_unmap_implicit(struct gcbatch *batch) +{ + struct gccontext *gccontext = get_context(); + struct list_head *head, *temphead; + struct gcschedunmap *gcschedunmap; + struct bvbuffdesc *bvbuffdesc; + struct bvbuffmap *prev, *bvbuffmap; + struct bvbuffmapinfo *bvbuffmapinfo; + + GCENTER(GCZONE_MAPPING); + + /* Lock access to the mapping list. */ + GCLOCK(&gccontext->maplock); + + /* Scan scheduled unmappings and remove the ones that are still + * in use. */ + list_for_each_safe(head, temphead, &batch->unmap) { + gcschedunmap = list_entry(head, struct gcschedunmap, link); + + /* Cast the handle. */ + bvbuffdesc = (struct bvbuffdesc *) gcschedunmap->handle; + + /* Find our mapping. */ + prev = NULL; + bvbuffmap = bvbuffdesc->map; + while (bvbuffmap != NULL) { + if (bvbuffmap->bv_unmap == bv_unmap) + break; + prev = bvbuffmap; + bvbuffmap = bvbuffmap->nextmap; + } + + /* Not found? */ + if (bvbuffmap == NULL) { + GCERR("mapping not found for bvbuffdesc 0x%08X.\n", + (unsigned int) bvbuffdesc); + + /* Remove scheduled unmapping. */ + list_move(head, &gccontext->unmapvac); + continue; + } + + /* Get the info structure. */ + bvbuffmapinfo = (struct bvbuffmapinfo *) bvbuffmap->handle; + + GCDBG(GCZONE_MAPPING, "head = 0x%08X\n", + (unsigned int) gcschedunmap); + GCDBG(GCZONE_MAPPING, " bvbuffmap = 0x%08X\n", + (unsigned int) bvbuffmap); + GCDBG(GCZONE_MAPPING, " handle = 0x%08X\n", + bvbuffmapinfo->handle); + + /* Implicit unmapping. */ + bvbuffmapinfo->automap -= 1; + if (bvbuffmapinfo->automap < 0) { + GCERR("implicit count negative.\n"); + bvbuffmapinfo->automap = 0; + } + + GCDBG(GCZONE_MAPPING, " explicit count = %d\n", + bvbuffmapinfo->usermap); + GCDBG(GCZONE_MAPPING, " implicit count = %d\n", + bvbuffmapinfo->automap); + + /* Still referenced? */ + if (bvbuffmapinfo->usermap || bvbuffmapinfo->automap) { + GCDBG(GCZONE_MAPPING, " still referenced.\n"); + + /* Remove scheduled unmapping. */ + list_move(head, &gccontext->unmapvac); + continue; + } + + GCDBG(GCZONE_MAPPING, " ready for unmapping.\n"); + + /* Set the handle. */ + gcschedunmap->handle = bvbuffmapinfo->handle; + + /* Remove from the buffer descriptor. */ + if (prev == NULL) + bvbuffdesc->map = bvbuffmap->nextmap; + else + prev->nextmap = bvbuffmap->nextmap; + + /* Add to the vacant list. */ + bvbuffmap->nextmap = gccontext->buffmapvac; + gccontext->buffmapvac = bvbuffmap; + } + + /* Unlock access to the mapping list. */ + GCUNLOCK(&gccontext->maplock); + + GCEXIT(GCZONE_MAPPING); +} diff --git a/drivers/misc/gcx/gcbv/gcparser.c b/drivers/misc/gcx/gcbv/gcparser.c new file mode 100644 index 0000000..654fc76 --- /dev/null +++ b/drivers/misc/gcx/gcbv/gcparser.c @@ -0,0 +1,1602 @@ +/* + * This file is provided under a dual BSD/GPLv2 license. When using or + * redistributing this file, you may do so under either license. + * + * GPL LICENSE SUMMARY + * + * Copyright(c) 2012 Vivante Corporation. All rights reserved. + * + * This program is free software; you can redistribute it and/or modify it + * under the terms and conditions of the GNU General Public License, + * version 2, as published by the Free Software Foundation. + * + * This program is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for + * more details. + * + * You should have received a copy of the GNU General Public License along with + * this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. + * + * The full GNU General Public License is included in this distribution in + * the file called "COPYING". + * + * BSD LICENSE + * + * Copyright(c) 2012 Vivante Corporation. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * * Neither the name of Vivante Corporation nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE + * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + */ + +#include "gcbv.h" + +#define GCZONE_NONE 0 +#define GCZONE_ALL (~0U) +#define GCZONE_FORMAT (1 << 0) +#define GCZONE_BLEND (1 << 1) +#define GCZONE_DEST (1 << 2) +#define GCZONE_SRC (1 << 3) + +GCDBG_FILTERDEF(gcparser, GCZONE_NONE, + "format", + "blend", + "dest", + "src") + + +/******************************************************************************* + * Pixel format parser. + */ + +/* FIXME/TODO: change to use BLTsvile defines. */ + +#if defined(OCDFMTDEF_ALPHA_SHIFT) +# undef OCDFMTDEF_ALPHA_SHIFT +#endif + +#if defined(OCDFMTDEF_ALPHA_MASK) +# undef OCDFMTDEF_ALPHA_MASK +#endif + +#define OCDFMTDEF_ALPHA_SHIFT 18 +#define OCDFMTDEF_ALPHA_MASK (1 << OCDFMTDEF_ALPHA_SHIFT) + +#define OCDFMTDEF_PLACEMENT_SHIFT 9 +#define OCDFMTDEF_PLACEMENT_MASK (3 << OCDFMTDEF_PLACEMENT_SHIFT) + +#define OCDFMTDEF_BITS_SHIFT 3 +#define OCDFMTDEF_BITS_MASK (3 << OCDFMTDEF_BITS_SHIFT) + +#define OCDFMTDEF_BITS12 (0 << OCDFMTDEF_BITS_SHIFT) +#define OCDFMTDEF_BITS15 (1 << OCDFMTDEF_BITS_SHIFT) +#define OCDFMTDEF_BITS16 (2 << OCDFMTDEF_BITS_SHIFT) +#define OCDFMTDEF_BITS24 (3 << OCDFMTDEF_BITS_SHIFT) + +#define BVFORMATRGBA(BPP, Format, Swizzle, R, G, B, A) \ +{ \ + BVFMT_RGB, \ + BPP, \ + GCREG_DE_FORMAT_ ## Format, \ + GCREG_DE_SWIZZLE_ ## Swizzle, \ + { R, G, B, A } \ +} + +#define BVCOMP(Shift, Size) \ + { Shift, Size, ((1 << Size) - 1) << Shift } + +#define BVRED(Shift, Size) \ + BVCOMP(Shift, Size) + +#define BVGREEN(Shift, Size) \ + BVCOMP(Shift, Size) + +#define BVBLUE(Shift, Size) \ + BVCOMP(Shift, Size) + +#define BVALPHA(Shift, Size) \ + BVCOMP(Shift, Size) + +#define BVFORMATINVALID \ + { 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, +}; +static struct bvformatxlate g_format_uyvy = { + .type = BVFMT_YUV, + .bitspp = 16, + .format = GCREG_DE_FORMAT_UYVY +}; +static struct bvformatxlate g_format_yuy2 = { + .type = BVFMT_YUV, + .bitspp = 16, + .format = GCREG_DE_FORMAT_YUY2 +}; + +static struct bvformatxlate formatxlate[] = { + /* #0: OCDFMT_xRGB12 + BITS=12 ALPHA=0 REVERSED=0 LEFT_JUSTIFIED=0 */ + BVFORMATRGBA(16, X4R4G4B4, ARGB, + BVRED(8, 4), BVGREEN(4, 4), BVBLUE(0, 4), BVALPHA(12, 0)), + + /* #1: OCDFMT_RGBx12 + BITS=12 ALPHA=0 REVERSED=0 LEFT_JUSTIFIED=1 */ + BVFORMATRGBA(16, X4R4G4B4, RGBA, + BVRED(12, 4), BVGREEN(8, 4), BVBLUE(4, 4), BVALPHA(0, 0)), + + /* #2: OCDFMT_xBGR12 + BITS=12 ALPHA=0 REVERSED=1 LEFT_JUSTIFIED=0 */ + BVFORMATRGBA(16, X4R4G4B4, ABGR, + BVRED(0, 4), BVGREEN(4, 4), BVBLUE(8, 4), BVALPHA(12, 0)), + + /* #3: OCDFMT_BGRx12 + BITS=12 ALPHA=0 REVERSED=1 LEFT_JUSTIFIED=1 */ + BVFORMATRGBA(16, X4R4G4B4, BGRA, + BVRED(4, 4), BVGREEN(8, 4), BVBLUE(12, 4), BVALPHA(0, 0)), + + /* #4: OCDFMT_ARGB12 + BITS=12 ALPHA=1 REVERSED=0 LEFT_JUSTIFIED=0 */ + BVFORMATRGBA(16, A4R4G4B4, ARGB, + BVRED(8, 4), BVGREEN(4, 4), BVBLUE(0, 4), BVALPHA(12, 4)), + + /* #5: OCDFMT_RGBA12 + BITS=12 ALPHA=1 REVERSED=0 LEFT_JUSTIFIED=1 */ + BVFORMATRGBA(16, A4R4G4B4, RGBA, + BVRED(12, 4), BVGREEN(8, 4), BVBLUE(4, 4), BVALPHA(0, 4)), + + /* #6: OCDFMT_ABGR12 + BITS=12 ALPHA=1 REVERSED=1 LEFT_JUSTIFIED=0 */ + BVFORMATRGBA(16, A4R4G4B4, ABGR, + BVRED(0, 4), BVGREEN(4, 4), BVBLUE(8, 4), BVALPHA(12, 4)), + + /* #7: OCDFMT_BGRA12 + BITS=12 ALPHA=1 REVERSED=1 LEFT_JUSTIFIED=1 */ + BVFORMATRGBA(16, A4R4G4B4, BGRA, + BVRED(4, 4), BVGREEN(8, 4), BVBLUE(12, 4), BVALPHA(0, 4)), + + /***********************************************/ + + /* #8: OCDFMT_xRGB15 + BITS=15 ALPHA=0 REVERSED=0 LEFT_JUSTIFIED=0 */ + BVFORMATRGBA(16, X1R5G5B5, ARGB, + BVRED(10, 5), BVGREEN(5, 5), BVBLUE(0, 5), BVALPHA(15, 0)), + + /* #9: OCDFMT_RGBx15 + BITS=15 ALPHA=0 REVERSED=0 LEFT_JUSTIFIED=1 */ + BVFORMATRGBA(16, X1R5G5B5, RGBA, + BVRED(11, 5), BVGREEN(6, 5), BVBLUE(1, 5), BVALPHA(0, 0)), + + /* #10: OCDFMT_xBGR15 + BITS=15 ALPHA=0 REVERSED=1 LEFT_JUSTIFIED=0 */ + BVFORMATRGBA(16, X1R5G5B5, ABGR, + BVRED(0, 5), BVGREEN(5, 5), BVBLUE(10, 5), BVALPHA(15, 0)), + + /* #11: OCDFMT_BGRx15 + BITS=15 ALPHA=0 REVERSED=1 LEFT_JUSTIFIED=1 */ + BVFORMATRGBA(16, X1R5G5B5, BGRA, + BVRED(1, 5), BVGREEN(6, 5), BVBLUE(11, 5), BVALPHA(0, 0)), + + /* #12: OCDFMT_ARGB15 + BITS=15 ALPHA=1 REVERSED=0 LEFT_JUSTIFIED=0 */ + BVFORMATRGBA(16, A1R5G5B5, ARGB, + BVRED(10, 5), BVGREEN(5, 5), BVBLUE(0, 5), BVALPHA(15, 1)), + + /* #13: OCDFMT_RGBA15 + BITS=15 ALPHA=1 REVERSED=0 LEFT_JUSTIFIED=1 */ + BVFORMATRGBA(16, A1R5G5B5, RGBA, + BVRED(11, 5), BVGREEN(6, 5), BVBLUE(1, 5), BVALPHA(0, 1)), + + /* #14: OCDFMT_ABGR15 + BITS=15 ALPHA=1 REVERSED=1 LEFT_JUSTIFIED=0 */ + BVFORMATRGBA(16, A1R5G5B5, ABGR, + BVRED(0, 5), BVGREEN(5, 5), BVBLUE(10, 5), BVALPHA(15, 1)), + + /* #15: OCDFMT_BGRA15 + BITS=15 ALPHA=1 REVERSED=1 LEFT_JUSTIFIED=1 */ + BVFORMATRGBA(16, A1R5G5B5, BGRA, + BVRED(1, 5), BVGREEN(6, 5), BVBLUE(11, 5), BVALPHA(0, 1)), + + /***********************************************/ + + /* #16: OCDFMT_RGB16 + BITS=16 ALPHA=0 REVERSED=0 LEFT_JUSTIFIED=0 */ + BVFORMATRGBA(16, R5G6B5, ARGB, + BVRED(11, 5), BVGREEN(5, 6), BVBLUE(0, 5), BVALPHA(0, 0)), + + /* #17: OCDFMT_RGB16 + BITS=16 ALPHA=0 REVERSED=0 LEFT_JUSTIFIED=1 */ + BVFORMATRGBA(16, R5G6B5, ARGB, + BVRED(11, 5), BVGREEN(5, 6), BVBLUE(0, 5), BVALPHA(0, 0)), + + /* #18: OCDFMT_BGR16 + BITS=16 ALPHA=0 REVERSED=1 LEFT_JUSTIFIED=0 */ + BVFORMATRGBA(16, R5G6B5, ABGR, + BVRED(0, 5), BVGREEN(5, 6), BVBLUE(11, 5), BVALPHA(0, 0)), + + /* #19: OCDFMT_BGR16 + BITS=16 ALPHA=0 REVERSED=1 LEFT_JUSTIFIED=1 */ + BVFORMATRGBA(16, R5G6B5, ABGR, + BVRED(0, 5), BVGREEN(5, 6), BVBLUE(11, 5), BVALPHA(0, 0)), + + /* #20 */ + BVFORMATINVALID, + + /* #21 */ + BVFORMATINVALID, + + /* #22 */ + BVFORMATINVALID, + + /* #23 */ + BVFORMATINVALID, + + /***********************************************/ + + /* #24: OCDFMT_xRGB24 + BITS=24 ALPHA=0 REVERSED=0 LEFT_JUSTIFIED=0 */ + BVFORMATRGBA(32, X8R8G8B8, BGRA, + BVRED(8, 8), BVGREEN(16, 8), BVBLUE(24, 8), BVALPHA(0, 0)), + + /* #25: OCDFMT_RGBx24 + BITS=24 ALPHA=0 REVERSED=0 LEFT_JUSTIFIED=1 */ + BVFORMATRGBA(32, X8R8G8B8, ABGR, + BVRED(0, 8), BVGREEN(8, 8), BVBLUE(16, 8), BVALPHA(24, 0)), + + /* #26: OCDFMT_xBGR24 + BITS=24 ALPHA=0 REVERSED=1 LEFT_JUSTIFIED=0 */ + BVFORMATRGBA(32, X8R8G8B8, RGBA, + BVRED(24, 8), BVGREEN(16, 8), BVBLUE(8, 8), BVALPHA(0, 0)), + + /* #27: OCDFMT_BGRx24 + BITS=24 ALPHA=0 REVERSED=1 LEFT_JUSTIFIED=1 */ + BVFORMATRGBA(32, X8R8G8B8, ARGB, + BVRED(16, 8), BVGREEN(8, 8), BVBLUE(0, 8), BVALPHA(24, 0)), + + /* #28: OCDFMT_ARGB24 + BITS=24 ALPHA=1 REVERSED=0 LEFT_JUSTIFIED=0 */ + BVFORMATRGBA(32, A8R8G8B8, BGRA, + BVRED(8, 8), BVGREEN(16, 8), BVBLUE(24, 8), BVALPHA(0, 8)), + + /* #29: OCDFMT_RGBA24 + BITS=24 ALPHA=1 REVERSED=0 LEFT_JUSTIFIED=1 */ + BVFORMATRGBA(32, A8R8G8B8, ABGR, + BVRED(0, 8), BVGREEN(8, 8), BVBLUE(16, 8), BVALPHA(24, 8)), + + /* #30: OCDFMT_ABGR24 + BITS=24 ALPHA=1 REVERSED=1 LEFT_JUSTIFIED=0 */ + BVFORMATRGBA(32, A8R8G8B8, RGBA, + BVRED(24, 8), BVGREEN(16, 8), BVBLUE(8, 8), BVALPHA(0, 8)), + + /* #31: OCDFMT_BGRA24 + BITS=24 ALPHA=1 REVERSED=1 LEFT_JUSTIFIED=1 */ + BVFORMATRGBA(32, A8R8G8B8, ARGB, + BVRED(16, 8), BVGREEN(8, 8), BVBLUE(0, 8), BVALPHA(24, 8)), +}; + +enum bverror parse_format(struct bvbltparams *bvbltparams, + enum ocdformat ocdformat, + struct bvformatxlate **format) +{ + static unsigned int containers[] = { + 8, /* OCDFMTDEF_CONTAINER_8BIT */ + 16, /* OCDFMTDEF_CONTAINER_16BIT */ + 24, /* OCDFMTDEF_CONTAINER_24BIT */ + 32, /* OCDFMTDEF_CONTAINER_32BIT */ + ~0U, /* reserved */ + 48, /* OCDFMTDEF_CONTAINER_48BIT */ + ~0U, /* reserved */ + 64 /* OCDFMTDEF_CONTAINER_64BIT */ + }; + + enum bverror bverror = BVERR_NONE; + unsigned int cs; + unsigned int bits; + unsigned int swizzle; + unsigned int alpha; + unsigned int index; + unsigned int cont; + + GCENTERARG(GCZONE_FORMAT, "ocdformat = 0x%08X\n", ocdformat); + + cs = (ocdformat & OCDFMTDEF_CS_MASK) >> OCDFMTDEF_CS_SHIFT; + bits = (ocdformat & OCDFMTDEF_COMPONENTSIZEMINUS1_MASK) + >> OCDFMTDEF_COMPONENTSIZEMINUS1_SHIFT; + cont = (ocdformat & OCDFMTDEF_CONTAINER_MASK) + >> OCDFMTDEF_CONTAINER_SHIFT; + GCDBG(GCZONE_FORMAT, "cs = %d\n", cs); + GCDBG(GCZONE_FORMAT, "bits = %d\n", bits); + GCDBG(GCZONE_FORMAT, "cont = %d\n", cont); + + switch (cs) { + case (OCDFMTDEF_CS_RGB >> OCDFMTDEF_CS_SHIFT): + GCDBG(GCZONE_FORMAT, "OCDFMTDEF_CS_RGB\n"); + + if ((ocdformat & OCDFMTDEF_LAYOUT_MASK) != OCDFMTDEF_PACKED) { + BVSETBLTERROR(BVERR_UNK, + "only packed RGBA formats are supported"); + goto exit; + } + + swizzle = (ocdformat & OCDFMTDEF_PLACEMENT_MASK) + >> OCDFMTDEF_PLACEMENT_SHIFT; + alpha = (ocdformat & OCDFMTDEF_ALPHA_MASK) + >> OCDFMTDEF_ALPHA_SHIFT; + + GCDBG(GCZONE_FORMAT, "swizzle = %d\n", swizzle); + GCDBG(GCZONE_FORMAT, "alpha = %d\n", alpha); + + index = swizzle | (alpha << 2); + + switch (bits) { + case 12 - 1: + index |= OCDFMTDEF_BITS12; + break; + + case 15 - 1: + index |= OCDFMTDEF_BITS15; + break; + + case 16 - 1: + index |= OCDFMTDEF_BITS16; + break; + + case 24 - 1: + index |= OCDFMTDEF_BITS24; + break; + + default: + BVSETBLTERROR(BVERR_UNK, + "unsupported bit width %d", bits); + goto exit; + } + + GCDBG(GCZONE_FORMAT, "index = %d\n", index); + break; + + case (OCDFMTDEF_CS_YCbCr >> OCDFMTDEF_CS_SHIFT): + GCDBG(GCZONE_FORMAT, "OCDFMTDEF_CS_YCbCr\n"); + + /* FIXME/TODO: add proper YUV parsing. */ + switch (ocdformat) { + case OCDFMT_NV12: + GCDBG(GCZONE_FORMAT, "OCDFMT_NV12\n"); + *format = &g_format_nv12; + goto exit; + + case OCDFMT_UYVY: + GCDBG(GCZONE_FORMAT, "OCDFMT_UYVY\n"); + *format = &g_format_uyvy; + goto exit; + + case OCDFMT_YUY2: + GCDBG(GCZONE_FORMAT, "OCDFMT_YUY2\n"); + *format = &g_format_yuy2; + goto exit; + + default: + BVSETBLTERROR(BVERR_UNK, + "unsupported YUV format %d", ocdformat); + goto exit; + } + + default: + BVSETBLTERROR(BVERR_UNK, + "unsupported color space %d", cs); + goto exit; + } + + if (formatxlate[index].bitspp != containers[cont]) { + BVSETBLTERROR(BVERR_UNK, + "unsupported bit width %d", bits); + goto exit; + } + + *format = &formatxlate[index]; + + GCDBG(GCZONE_FORMAT, "format record = 0x%08X\n", + (unsigned int) &formatxlate[index]); + GCDBG(GCZONE_FORMAT, " bpp = %d\n", formatxlate[index].bitspp); + GCDBG(GCZONE_FORMAT, " format = %d\n", formatxlate[index].format); + GCDBG(GCZONE_FORMAT, " swizzle = %d\n", formatxlate[index].swizzle); + + bverror = BVERR_NONE; + +exit: + GCEXITARG(GCZONE_FORMAT, "bv%s = %d\n", + (bverror == BVERR_NONE) ? "result" : "error", bverror); + return bverror; +} + + +/******************************************************************************* + * Alpha blending parser. + */ + +#define BVBLENDMATCH(Mode, Inverse, Normal) \ +( \ + BVBLENDDEF_ ## Mode | \ + BVBLENDDEF_ ## Inverse | \ + BVBLENDDEF_ ## Normal \ +) + +#define BVSRC1USE(Use) \ + Use + +#define BVSRC2USE(Use) \ + Use + +#define BVBLENDUNDEFINED() \ + { ~0, ~0, { 0, 0, 0, 0 }, { 0, 0, 0, 0 } } + +struct bvblendxlate { + unsigned char match1; + unsigned char match2; + + struct gcblendconfig k1; + struct gcblendconfig k2; +}; + +static struct bvblendxlate blendxlate[64] = { + /**********************************************************************/ + /* #0: color factor: 00 00 00 A:(1-C1,C1)=zero + alpha factor: zero ==> 00 00 00 */ + { + 0x00, + 0x00, + + { + GCREG_BLENDING_MODE_ZERO, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(false), BVSRC2USE(false) + }, + + { + GCREG_BLENDING_MODE_ZERO, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(false), BVSRC2USE(false) + } + }, + + /* #1: color factor: 00 00 01 A:(1-C1,A1)=A1 + alpha factor: A1 ==> 00 00 01 or 00 10 01 */ + { + BVBLENDMATCH(ONLY_A, INV_C1, NORM_A1), + BVBLENDMATCH(ONLY_A, INV_C2, NORM_A1), + + { + GCREG_BLENDING_MODE_NORMAL, + GCREG_FACTOR_INVERSE_ENABLE, + BVSRC1USE(true), BVSRC2USE(false) + }, + + { + GCREG_BLENDING_MODE_NORMAL, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + } + }, + + /* #2: color factor: 00 00 10 A:(1-C1,C2)=undefined + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #3: color factor: 00 00 11 A:(1-C1,A2)=A2 + alpha factor: A2 ==> 00 00 11 or 00 10 11 */ + { + BVBLENDMATCH(ONLY_A, INV_C1, NORM_A2), + BVBLENDMATCH(ONLY_A, INV_C2, NORM_A2), + + { + GCREG_BLENDING_MODE_NORMAL, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + }, + + { + GCREG_BLENDING_MODE_NORMAL, + GCREG_FACTOR_INVERSE_ENABLE, + BVSRC1USE(false), BVSRC2USE(true) + } + }, + + /* #4: color factor: 00 01 00 A:(1-A1,C1)=1-A1 + alpha factor: 1-A1 ==> 00 01 00 or 00 01 10 */ + { + BVBLENDMATCH(ONLY_A, INV_A1, NORM_C1), + BVBLENDMATCH(ONLY_A, INV_A1, NORM_C2), + + { + GCREG_BLENDING_MODE_INVERSED, + GCREG_FACTOR_INVERSE_ENABLE, + BVSRC1USE(true), BVSRC2USE(false) + }, + + { + GCREG_BLENDING_MODE_INVERSED, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + } + }, + + /* #5: color factor: 00 01 01 A:(1-A1,A1)=undefined + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #6: color factor: 00 01 10 A:(1-A1,C2)=1-A1 + alpha factor: 1-A1 ==> 00 01 00 or 00 01 10 */ + { + BVBLENDMATCH(ONLY_A, INV_A1, NORM_C1), + BVBLENDMATCH(ONLY_A, INV_A1, NORM_C2), + + { + GCREG_BLENDING_MODE_INVERSED, + GCREG_FACTOR_INVERSE_ENABLE, + BVSRC1USE(true), BVSRC2USE(false) + }, + + { + GCREG_BLENDING_MODE_INVERSED, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + } + }, + + /* #7: color factor: 00 01 11 A:(1-A1,A2)=undefined + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #8: color factor: 00 10 00 A:(1-C2,C1)=undefined + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #9: color factor: 00 10 01 A:(1-C2,A1)=A1 + alpha factor: A1 ==> 00 00 01 or 00 10 01 */ + { + BVBLENDMATCH(ONLY_A, INV_C1, NORM_A1), + BVBLENDMATCH(ONLY_A, INV_C2, NORM_A1), + + { + GCREG_BLENDING_MODE_NORMAL, + GCREG_FACTOR_INVERSE_ENABLE, + BVSRC1USE(true), BVSRC2USE(false) + }, + + { + GCREG_BLENDING_MODE_NORMAL, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + } + }, + + /* #10: color factor: 00 10 10 A:(1-C2,C2)=undefined + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #11: color factor: 00 10 11 A:(1-C2,A2)=A2 + alpha factor: A2 ==> 00 00 11 or 00 10 11 */ + { + BVBLENDMATCH(ONLY_A, INV_C1, NORM_A2), + BVBLENDMATCH(ONLY_A, INV_C2, NORM_A2), + + { + GCREG_BLENDING_MODE_NORMAL, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + }, + + { + GCREG_BLENDING_MODE_NORMAL, + GCREG_FACTOR_INVERSE_ENABLE, + BVSRC1USE(false), BVSRC2USE(true) + } + }, + + /* #12: color factor: 00 11 00 A:(1-A2,C1)=1-A2 + alpha factor: 1-A2 ==> 00 11 00 or 00 11 10 */ + { + BVBLENDMATCH(ONLY_A, INV_A2, NORM_C1), + BVBLENDMATCH(ONLY_A, INV_A2, NORM_C2), + + { + GCREG_BLENDING_MODE_INVERSED, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + }, + + { + GCREG_BLENDING_MODE_INVERSED, + GCREG_FACTOR_INVERSE_ENABLE, + BVSRC1USE(false), BVSRC2USE(true) + } + }, + + /* #13: color factor: 00 11 01 A:(1-A2,A1)=undefined + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #14: color factor: 00 11 10 A:(1-A2,C2)=1-A2 + alpha factor: 1-A2 ==> 00 11 00 or 00 11 10 */ + { + BVBLENDMATCH(ONLY_A, INV_A2, NORM_C1), + BVBLENDMATCH(ONLY_A, INV_A2, NORM_C2), + + { + GCREG_BLENDING_MODE_INVERSED, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + }, + + { + GCREG_BLENDING_MODE_INVERSED, + GCREG_FACTOR_INVERSE_ENABLE, + BVSRC1USE(false), BVSRC2USE(true) + } + }, + + /* #15: color factor: 00 11 11 A:(1-A2,A2)=undefined + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /**********************************************************************/ + /* #16: color factor: 01 00 00 MIN:(1-C1,C1) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #17: color factor: 01 00 01 MIN:(1-C1,A1) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #18: color factor: 01 00 10 MIN:(1-C1,C2) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #19: color factor: 01 00 11 MIN:(1-C1,A2) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #20: color factor: 01 01 00 MIN:(1-A1,C1) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #21: color factor: 01 01 01 MIN:(1-A1,A1) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #22: color factor: 01 01 10 MIN:(1-A1,C2) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #23: color factor: 01 01 11 MIN:(1-A1,A2) + alpha factor: one ==> 11 11 11 */ + { + 0x3F, + 0x3F, + + { + GCREG_BLENDING_MODE_SATURATED_DEST_ALPHA, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + }, + + { + GCREG_BLENDING_MODE_SATURATED_ALPHA, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + } + }, + + /* #24: color factor: 01 10 00 MIN:(1-C2,C1) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #25: color factor: 01 10 01 MIN:(1-C2,A1) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #26: color factor: 01 10 10 MIN:(1-C2,C2) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #27: color factor: 01 10 11 MIN:(1-C2,A2) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #28: color factor: 01 11 00 MIN:(1-A2,C1) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #29: color factor: 01 11 01 MIN:(1-A2,A1) + alpha factor: one ==> 11 11 11 */ + { + 0x3F, + 0x3F, + + { + GCREG_BLENDING_MODE_SATURATED_ALPHA, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + }, + + { + GCREG_BLENDING_MODE_SATURATED_DEST_ALPHA, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + } + }, + + /* #30: color factor: 01 11 10 MIN:(1-A2,C2) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #31: color factor: 01 11 11 MIN:(1-A2,A2) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /**********************************************************************/ + /* #32: color factor: 10 00 00 MAX:(1-C1,C1) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #33: color factor: 10 00 01 MAX:(1-C1,A1) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #34: color factor: 10 00 10 MAX:(1-C1,C2) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #35: color factor: 10 00 11 MAX:(1-C1,A2) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #36: color factor: 10 01 00 MAX:(1-A1,C1) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #37: color factor: 10 01 01 MAX:(1-A1,A1) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #38: color factor: 10 01 10 MAX:(1-A1,C2) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #39: color factor: 10 01 11 MAX:(1-A1,A2) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #40: color factor: 10 10 00 MAX:(1-C2,C1) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #41: color factor: 10 10 01 MAX:(1-C2,A1) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #42: color factor: 10 10 10 MAX:(1-C2,C2) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #43: color factor: 10 10 11 MAX:(1-C2,A2) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #44: color factor: 10 11 00 MAX:(1-A2,C1) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #45: color factor: 10 11 01 MAX:(1-A2,A1) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #46: color factor: 10 11 10 MAX:(1-A2,C2) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #47: color factor: 10 11 11 MAX:(1-A2,A2) ==> not supported + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /**********************************************************************/ + /* #48: color factor: 11 00 00 C:(1-C1,C1)=undefined + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #49: color factor: 11 00 01 C:(1-C1,A1)=1-C1 + alpha factor: 1-A1 ==> 00 01 00 or 00 01 10 */ + { + BVBLENDMATCH(ONLY_A, INV_A1, NORM_C1), + BVBLENDMATCH(ONLY_A, INV_A1, NORM_C2), + + { + GCREG_BLENDING_MODE_COLOR_INVERSED, + GCREG_FACTOR_INVERSE_ENABLE, + BVSRC1USE(true), BVSRC2USE(false) + }, + + { + GCREG_BLENDING_MODE_COLOR_INVERSED, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + } + }, + + /* #50: color factor: 11 00 10 C:(1-C1,C2)=undefined + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #51: color factor: 11 00 11 C:(1-C1,A2)=1-C1 + alpha factor: 1-A1 ==> 00 01 00 or 00 01 10 */ + { + BVBLENDMATCH(ONLY_A, INV_A1, NORM_C1), + BVBLENDMATCH(ONLY_A, INV_A1, NORM_C2), + + { + GCREG_BLENDING_MODE_COLOR_INVERSED, + GCREG_FACTOR_INVERSE_ENABLE, + BVSRC1USE(true), BVSRC2USE(false) + }, + + { + GCREG_BLENDING_MODE_COLOR_INVERSED, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + } + }, + + /* #52: color factor: 11 01 00 C:(1-A1,C1)=C1 + alpha factor: A1 ==> 00 00 01 or 00 10 01 */ + { + BVBLENDMATCH(ONLY_A, INV_C1, NORM_A1), + BVBLENDMATCH(ONLY_A, INV_C2, NORM_A1), + + { + GCREG_BLENDING_MODE_COLOR, + GCREG_FACTOR_INVERSE_ENABLE, + BVSRC1USE(true), BVSRC2USE(false) + }, + + { + GCREG_BLENDING_MODE_COLOR, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + } + }, + + /* #53: color factor: 11 01 01 C:(1-A1,A1)=undefined + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #54: color factor: 11 01 10 C:(1-A1,C2)=C2 + alpha factor: A2 ==> 00 00 11 or 00 10 11 */ + { + BVBLENDMATCH(ONLY_A, INV_C1, NORM_A2), + BVBLENDMATCH(ONLY_A, INV_C2, NORM_A2), + + { + GCREG_BLENDING_MODE_COLOR, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + }, + + { + GCREG_BLENDING_MODE_COLOR, + GCREG_FACTOR_INVERSE_ENABLE, + BVSRC1USE(false), BVSRC2USE(true) + } + }, + + /* #55: color factor: 11 01 11 C:(1-A1,A2)=undefined + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #56: color factor: 11 10 00 C:(1-C2,C1)=undefined + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #57: color factor: 11 10 01 C:(1-C2,A1)=1-C2 + alpha factor: 1-A2 ==> 00 11 00 or 00 11 10 */ + { + BVBLENDMATCH(ONLY_A, INV_A2, NORM_C1), + BVBLENDMATCH(ONLY_A, INV_A2, NORM_C2), + + { + GCREG_BLENDING_MODE_COLOR_INVERSED, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + }, + + { + GCREG_BLENDING_MODE_COLOR_INVERSED, + GCREG_FACTOR_INVERSE_ENABLE, + BVSRC1USE(false), BVSRC2USE(true) + } + }, + + /* #58: color factor: 11 10 10 C:(1-C2,C2)=undefined + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #59: color factor: 11 10 11 C:(1-C2,A2)=1-C2 + alpha factor: 1-A2 ==> 00 11 00 or 00 11 10 */ + { + BVBLENDMATCH(ONLY_A, INV_A2, NORM_C1), + BVBLENDMATCH(ONLY_A, INV_A2, NORM_C2), + + { + GCREG_BLENDING_MODE_COLOR_INVERSED, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + }, + + { + GCREG_BLENDING_MODE_COLOR_INVERSED, + GCREG_FACTOR_INVERSE_ENABLE, + BVSRC1USE(false), BVSRC2USE(false) + } + }, + + /* #60: color factor: 11 11 00 C:(1-A2,C1)=C1 + alpha factor: A1 ==> 00 00 01 or 00 10 01 */ + { + BVBLENDMATCH(ONLY_A, INV_C1, NORM_A1), + BVBLENDMATCH(ONLY_A, INV_C2, NORM_A1), + + { + GCREG_BLENDING_MODE_COLOR, + GCREG_FACTOR_INVERSE_ENABLE, + BVSRC1USE(true), BVSRC2USE(false) + }, + + { + GCREG_BLENDING_MODE_COLOR, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + } + }, + + /* #61: color factor: 11 11 01 C:(1-A2,A1)=undefined + alpha factor: N/A */ + BVBLENDUNDEFINED(), + + /* #62: color factor: 11 11 10 C:(1-A2,C2)=C2 + alpha factor: A2 ==> 00 00 11 or 00 10 11 */ + { + BVBLENDMATCH(ONLY_A, INV_C1, NORM_A2), + BVBLENDMATCH(ONLY_A, INV_C2, NORM_A2), + + { + GCREG_BLENDING_MODE_COLOR, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(true) + }, + + { + GCREG_BLENDING_MODE_COLOR, + GCREG_FACTOR_INVERSE_ENABLE, + BVSRC1USE(false), BVSRC2USE(true) + } + }, + + /* #63: color factor: 11 11 11 C:(1-A2,A2)=one + alpha factor: one ==> 11 11 11 */ + { + 0x3F, + 0x3F, + + { + GCREG_BLENDING_MODE_ONE, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(true), BVSRC2USE(false) + }, + + { + GCREG_BLENDING_MODE_ONE, + GCREG_FACTOR_INVERSE_DISABLE, + BVSRC1USE(false), BVSRC2USE(true) + } + } +}; + +enum bverror parse_blend(struct bvbltparams *bvbltparams, + enum bvblend blend, + struct gcalpha *gca) +{ + enum bverror bverror; + unsigned int global; + unsigned int k1, k2, k3, k4; + struct bvblendxlate *k1_xlate; + struct bvblendxlate *k2_xlate; + unsigned int alpha; + + GCENTERARG(GCZONE_BLEND, "blend = 0x%08X (%s)\n", + blend, gc_bvblend_name(blend)); + + if ((blend & BVBLENDDEF_REMOTE) != 0) { + BVSETBLTERROR(BVERR_BLEND, "remote alpha not supported"); + goto exit; + } + + global = (blend & BVBLENDDEF_GLOBAL_MASK) >> BVBLENDDEF_GLOBAL_SHIFT; + + switch (global) { + case (BVBLENDDEF_GLOBAL_NONE >> BVBLENDDEF_GLOBAL_SHIFT): + GCDBG(GCZONE_BLEND, "BVBLENDDEF_GLOBAL_NONE\n"); + + gca->src_global_color = + gca->dst_global_color = 0; + + gca->src_global_alpha_mode = GCREG_GLOBAL_ALPHA_MODE_NORMAL; + gca->dst_global_alpha_mode = GCREG_GLOBAL_ALPHA_MODE_NORMAL; + break; + + case (BVBLENDDEF_GLOBAL_UCHAR >> BVBLENDDEF_GLOBAL_SHIFT): + GCDBG(GCZONE_BLEND, "BVBLENDDEF_GLOBAL_UCHAR\n"); + + gca->src_global_color = + gca->dst_global_color = + ((unsigned int) bvbltparams->globalalpha.size8) << 24; + + gca->src_global_alpha_mode = GCREG_GLOBAL_ALPHA_MODE_GLOBAL; + gca->dst_global_alpha_mode = GCREG_GLOBAL_ALPHA_MODE_GLOBAL; + break; + + case (BVBLENDDEF_GLOBAL_FLOAT >> BVBLENDDEF_GLOBAL_SHIFT): + GCDBG(GCZONE_BLEND, "BVBLENDDEF_GLOBAL_FLOAT\n"); + + alpha = gcfp2norm8(bvbltparams->globalalpha.fp); + + gca->src_global_color = + gca->dst_global_color = alpha << 24; + + gca->src_global_alpha_mode = GCREG_GLOBAL_ALPHA_MODE_GLOBAL; + gca->dst_global_alpha_mode = GCREG_GLOBAL_ALPHA_MODE_GLOBAL; + break; + + default: + BVSETBLTERROR(BVERR_BLEND, "invalid global alpha mode"); + goto exit; + } + + /* + Co = k1 x C1 + k2 x C2 + Ao = k3 x A1 + k4 x A2 + */ + + k1 = (blend >> 18) & 0x3F; + k2 = (blend >> 12) & 0x3F; + k3 = (blend >> 6) & 0x3F; + k4 = blend & 0x3F; + + GCDBG(GCZONE_BLEND, "k1 = %d\n", k1); + GCDBG(GCZONE_BLEND, "k2 = %d\n", k2); + GCDBG(GCZONE_BLEND, "k3 = %d\n", k3); + GCDBG(GCZONE_BLEND, "k4 = %d\n", k4); + + k1_xlate = &blendxlate[k1]; + k2_xlate = &blendxlate[k2]; + + if (((k3 != k1_xlate->match1) && (k3 != k1_xlate->match2)) || + ((k4 != k2_xlate->match1) && (k4 != k2_xlate->match2))) { + BVSETBLTERROR(BVERR_BLEND, + "not supported coefficient combination"); + goto exit; + } + + gca->k1 = &k1_xlate->k1; + gca->k2 = &k2_xlate->k2; + + gca->src1used = gca->k1->src1used | gca->k2->src1used; + gca->src2used = gca->k1->src2used | gca->k2->src2used; + + bverror = BVERR_NONE; + +exit: + GCEXITARG(BVERR_BLEND, "bv%s = %d\n", + (bverror == BVERR_NONE) ? "result" : "error", bverror); + return bverror; +} + + +/******************************************************************************* + * 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. */ +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. + */ + +static bool valid_geom(struct bvbuffdesc *buffdesc, + struct bvsurfgeom *geom, + struct bvformatxlate *format) +{ + unsigned int size; + + /* Compute the size of the surface. */ + size = (geom->width * geom->height * format->bitspp) / 8; + + /* Make sure the size is not greater then the surface. */ + if (size > buffdesc->length) { + GCERR("invalid geometry detected:\n"); + GCERR(" specified dimensions: %dx%d, %d bitspp\n", + geom->width, geom->height, format->bitspp); + GCERR(" surface size based on the dimensions: %d\n", + size); + GCERR(" specified surface size: %lu\n", + buffdesc->length); + return false; + } + + return true; +} + +int get_pixel_offset(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; + } + + /* Compute the aligned offset. */ + alignedoffset = byteoffset & alignment; + + /* Convert to pixels. */ + pixeloffset = alignedoffset * 8 / format->bitspp; + return -pixeloffset; +} + +enum bverror parse_destination(struct bvbltparams *bvbltparams, + struct gcbatch *batch) +{ + enum bverror bverror = BVERR_NONE; + + GCENTER(GCZONE_DEST); + + /* 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; + unsigned short clippedleft, clippedtop; + unsigned short clippedright, clippedbottom; + + /* Make shortcuts to the destination objects. */ + dstrect = &bvbltparams->dstrect; + cliprect = &bvbltparams->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 ((bvbltparams->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, + " 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; + } + + /* Compute clipping deltas and the adjusted destination rect. */ + if (clipleft <= destleft) { + batch->deltaleft = 0; + clippedleft = destleft; + } else { + batch->deltaleft = clipleft - destleft; + clippedleft = clipleft; + } + + if (cliptop <= desttop) { + batch->deltatop = 0; + clippedtop = desttop; + } else { + batch->deltatop = cliptop - desttop; + clippedtop = cliptop; + } + + if (clipright >= destright) { + batch->deltaright = 0; + clippedright = destright; + } else { + batch->deltaright = clipright - destright; + clippedright = clipright; + } + + if (clipbottom >= destbottom) { + batch->deltabottom = 0; + clippedbottom = destbottom; + } else { + batch->deltabottom = clipbottom - destbottom; + clippedbottom = clipbottom; + } + + /* Validate the rectangle. */ + if ((clippedright > (int) bvbltparams->dstgeom->width) || + (clippedbottom > (int) bvbltparams->dstgeom->height)) { + BVSETBLTERROR(BVERR_DSTRECT, + "destination rect exceeds surface size"); + goto exit; + } + + /* Set clipped coordinates. */ + batch->clippedleft = clippedleft; + batch->clippedtop = clippedtop; + batch->clippedright = clippedright; + batch->clippedbottom = clippedbottom; + + GCDBG(GCZONE_DEST, + " clipped dstrect = (%d,%d)-(%d,%d), %dx%d\n", + clippedleft, clippedtop, clippedright, clippedbottom, + clippedright - clippedleft, clippedbottom - clippedtop); + GCDBG(GCZONE_DEST, + " clipping delta = (%d,%d)-(%d,%d)\n", + batch->deltaleft, batch->deltatop, + batch->deltaright, batch->deltabottom); + } + + /* Did the destination surface change? */ + if ((batch->batchflags & BVBATCH_DST) != 0) { + struct bvbuffdesc *dstdesc; + struct bvsurfgeom *dstgeom; + unsigned int stridealign; + + /* Make shortcuts to the destination objects. */ + dstdesc = bvbltparams->dstdesc; + dstgeom = bvbltparams->dstgeom; + + /* Check for unsupported dest formats. */ + switch (dstgeom->format) { + case OCDFMT_NV12: + BVSETBLTERROR(BVERR_DSTGEOM_FORMAT, + "destination format unsupported"); + goto exit; + + default: + break; + } + + /* Parse the destination format. */ + GCDBG(GCZONE_FORMAT, "parsing destination format.\n"); + if (parse_format(bvbltparams, dstgeom->format, + &batch->dstformat) != BVERR_NONE) { + bverror = BVERR_DSTGEOM_FORMAT; + goto exit; + } + + /* Validate geometry. */ + if (!valid_geom(dstdesc, dstgeom, batch->dstformat)) { + BVSETBLTERROR(BVERR_DSTGEOM, + "destination geom exceeds surface size"); + goto exit; + } + + /* Destination stride must be 8 pixel aligned. */ + stridealign = batch->dstformat->bitspp - 1; + if ((dstgeom->virtstride & stridealign) != 0) { + BVSETBLTERROR(BVERR_DSTGEOM_STRIDE, + "destination stride must be 8 pixel " + "aligned."); + 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_pixel_offset(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_DEST, " 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); + return bverror; +} + +enum bverror parse_source(struct bvbltparams *bvbltparams, + struct gcbatch *batch, + struct srcinfo *srcinfo) +{ + enum bverror bverror = BVERR_NONE; + struct bvbuffdesc *srcdesc; + struct bvsurfgeom *srcgeom; + struct bvrect *srcrect; + unsigned int stridealign; + + /* Make shortcuts to the source objects. */ + srcdesc = srcinfo->buf.desc; + srcgeom = srcinfo->geom; + srcrect = srcinfo->rect; + + /* Parse the source format. */ + GCDBG(GCZONE_FORMAT, "parsing source%d format.\n", + srcinfo->index + 1); + if (parse_format(bvbltparams, srcgeom->format, + &srcinfo->format) != BVERR_NONE) { + bverror = (srcinfo->index == 0) + ? BVERR_SRC1GEOM_FORMAT + : BVERR_SRC2GEOM_FORMAT; + goto exit; + } + + /* 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; + } + + /* Source must be 8 pixel aligned. */ + stridealign = srcinfo->format->bitspp - 1; + if ((srcgeom->virtstride & stridealign) != 0) { + BVSETBLTERROR((srcinfo->index == 0) + ? BVERR_SRC1GEOM_STRIDE + : BVERR_SRC2GEOM_STRIDE, + "source stride must be 8 pixel aligned."); + goto exit; + } + + /* 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); + } + + /* Determine source mirror. */ + srcinfo->mirror = (srcinfo->index == 0) + ? (bvbltparams->flags >> BVFLAG_FLIP_SRC1_SHIFT) + & BVFLAG_FLIP_MASK + : (bvbltparams->flags >> BVFLAG_FLIP_SRC2_SHIFT) + & BVFLAG_FLIP_MASK; + + GCDBG(GCZONE_SRC, "source surface %d:\n", srcinfo->index + 1); + GCDBG(GCZONE_SRC, " 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 { + GCDBG(GCZONE_DEST, " virtual address = 0x%08X\n", + (unsigned int) srcdesc->virtaddr); + } + + GCDBG(GCZONE_SRC, " stride = %ld\n", + srcgeom->virtstride); + GCDBG(GCZONE_SRC, " geometry size = %dx%d\n", + srcgeom->width, srcgeom->height); + GCDBG(GCZONE_SRC, " 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_SRC, " mirror = %d\n", srcinfo->mirror); + +exit: + return bverror; +} diff --git a/drivers/misc/gcx/gccore/gcdbglog.c b/drivers/misc/gcx/gccore/gcdbglog.c index 95f2a14..93c020c 100644 --- a/drivers/misc/gcx/gccore/gcdbglog.c +++ b/drivers/misc/gcx/gccore/gcdbglog.c @@ -66,6 +66,9 @@ /* Specifies spacing for thread messages. */ #define GC_THREAD_INDENT 0 +/* Print the timestamp. */ +#define GC_SHOW_TIME 1 + /* When set to non-zero, specifies how many prints are accumulated in the * buffer before the buffer is flushed. */ #define GC_SHOW_DUMP_LINE 1 @@ -120,6 +123,10 @@ (g_initdone && ((filter == NULL) || ((filter->zone & zone) != 0))) #endif +#if GC_SHOW_TIME +#define GC_TIME_FORMAT "[%5ld.%ld] " +#endif + #if GC_SHOW_DUMP_LINE #define GC_DUMPLINE_FORMAT "[%12d] " #endif @@ -155,14 +162,18 @@ struct itemstring { enum itemtype itemtype; int indent; -#if GC_SHOW_PID - struct task_struct *task; +#if GC_SHOW_TIME + struct timespec timestamp; #endif #if GC_SHOW_DUMP_LINE unsigned int dumpline; #endif +#if GC_SHOW_PID + struct task_struct *task; +#endif + const char *message; va_list messagedata; unsigned int datasize; @@ -376,6 +387,13 @@ static void gc_print_string(struct seq_file *s, struct itemstring *str) int len = 0; char buffer[GC_MAXSTR_LENGTH]; +#if GC_SHOW_TIME + len += snprintf(buffer + len, sizeof(buffer) - len - GC_EOL_RESERVE, + GC_TIME_FORMAT, + str->timestamp.tv_sec, + str->timestamp.tv_nsec); +#endif + #if GC_SHOW_DUMP_LINE len += snprintf(buffer + len, sizeof(buffer) - len - GC_EOL_RESERVE, GC_DUMPLINE_FORMAT, str->dumpline); @@ -1148,14 +1166,18 @@ static void gc_append_string(struct buffout *buffout, item->messagedata = *(va_list *) &messagedata; item->datasize = itemstring->datasize; -#if GC_SHOW_PID - item->task = itemstring->task; +#if GC_SHOW_TIME + item->timestamp = itemstring->timestamp; #endif #if GC_SHOW_DUMP_LINE item->dumpline = itemstring->dumpline; #endif +#if GC_SHOW_PID + item->task = itemstring->task; +#endif + /* Copy argument value. */ if (itemstring->datasize != 0) { GC_DEBUGMSG("copying %d bytes of messagedata.\n", @@ -1258,14 +1280,18 @@ static void gc_print(struct buffout *buffout, unsigned int argsize, itemstring.messagedata = args; itemstring.datasize = argsize; -#if GC_SHOW_PID - itemstring.task = threadinfo->task; +#if GC_SHOW_TIME + ktime_get_ts(&itemstring.timestamp); #endif #if GC_SHOW_DUMP_LINE itemstring.dumpline = ++buffout->dumpline; #endif +#if GC_SHOW_PID + itemstring.task = threadinfo->task; +#endif + /* Print the message. */ #if GC_BUFFERED_OUTPUT gc_append_string(buffout, &itemstring); diff --git a/drivers/misc/gcx/gccore/gcmain.c b/drivers/misc/gcx/gccore/gcmain.c index 03c01eb..6c7f0db 100644 --- a/drivers/misc/gcx/gccore/gcmain.c +++ b/drivers/misc/gcx/gccore/gcmain.c @@ -515,8 +515,8 @@ void gc_commit(struct gccommit *gccommit, bool fromuser) gccorecontext->gcpipe = gccommit->exitpipe; /* Go through all buffers one at a time. */ - gcbuffer = gccommit->buffer; - while (gcbuffer != NULL) { + list_for_each(head, &gccommit->buffer) { + gcbuffer = list_entry(head, struct gcbuffer, link); GCDBG(GCZONE_COMMIT, "gcbuffer = 0x%08X\n", (unsigned int) gcbuffer); @@ -552,12 +552,9 @@ void gc_commit(struct gccommit *gccommit, bool fromuser) } /* Process fixups. */ - gccommit->gcerror = gcmmu_fixup(gcbuffer->fixuphead, logical); + gccommit->gcerror = gcmmu_fixup(&gcbuffer->fixup, logical); if (gccommit->gcerror != GCERR_NONE) goto exit; - - /* Get the next buffer. */ - gcbuffer = gcbuffer->next; } /* Add the callback. */ diff --git a/drivers/misc/gcx/gccore/gcmmu.c b/drivers/misc/gcx/gccore/gcmmu.c index 7c5a671..98136ca 100644 --- a/drivers/misc/gcx/gccore/gcmmu.c +++ b/drivers/misc/gcx/gccore/gcmmu.c @@ -1194,9 +1194,12 @@ void gcmmu_flush_finalize(struct gccmdbuf *gccmdbuf, GCEXIT(GCZONE_FLUSH); } -enum gcerror gcmmu_fixup(struct gcfixup *fixup, unsigned int *data) +enum gcerror gcmmu_fixup(struct list_head *fixuplist, + unsigned int *data) { enum gcerror gcerror = GCERR_NONE; + struct list_head *head; + struct gcfixup *gcfixup; struct gcfixupentry *table; struct gcmmuarena *arena; unsigned int dataoffset; @@ -1205,20 +1208,15 @@ enum gcerror gcmmu_fixup(struct gcfixup *fixup, unsigned int *data) GCENTER(GCZONE_FIXUP); /* Process fixups. */ - while (fixup != NULL) { - /* Verify fixup pointer. */ - if (!virt_addr_valid(fixup)) { - GCERR("bad fixup @ 0x%08X\n", (unsigned int) fixup); - gcerror = GCERR_OODM; - goto exit; - } + list_for_each(head, fixuplist) { + gcfixup = list_entry(head, struct gcfixup, link); GCDBG(GCZONE_FIXUP, "%d fixup(s) @ 0x%08X\n", - fixup->count, (unsigned int) fixup); + gcfixup->count, (unsigned int) gcfixup); /* Apply fixups. */ - table = fixup->fixup; - for (i = 0; i < fixup->count; i += 1) { + table = gcfixup->fixup; + for (i = 0; i < gcfixup->count; i += 1) { GCDBG(GCZONE_FIXUP, "#%d\n", i); GCDBG(GCZONE_FIXUP, " buffer offset = 0x%08X\n", table->dataoffset * 4); @@ -1249,9 +1247,6 @@ enum gcerror gcmmu_fixup(struct gcfixup *fixup, unsigned int *data) table += 1; } - - /* Get the next fixup. */ - fixup = fixup->next; } exit: diff --git a/drivers/misc/gcx/gccore/gcmmu.h b/drivers/misc/gcx/gccore/gcmmu.h index 226f435..445f7fd 100644 --- a/drivers/misc/gcx/gccore/gcmmu.h +++ b/drivers/misc/gcx/gccore/gcmmu.h @@ -265,6 +265,7 @@ void gcmmu_flush_finalize(struct gccmdbuf *openentry, struct gcmommuflush *flushlogical, unsigned int flushaddress); -enum gcerror gcmmu_fixup(struct gcfixup *fixup, unsigned int *data); +enum gcerror gcmmu_fixup(struct list_head *fixuplist, + unsigned int *data); #endif diff --git a/drivers/misc/gcx/gcioctl/gcif.c b/drivers/misc/gcx/gcioctl/gcif.c index 55a2bf6..b0d2ed9 100644 --- a/drivers/misc/gcx/gcioctl/gcif.c +++ b/drivers/misc/gcx/gcioctl/gcif.c @@ -46,105 +46,147 @@ GCDBG_FILTERDEF(ioctl, GCZONE_NONE, "ioctl", "callback") +static GCDEFINE_LOCK(g_fixuplock); +static GCDEFINE_LOCK(g_bufferlock); +static GCDEFINE_LOCK(g_unmaplock); + +static LIST_HEAD(g_buffervac); /* gcbuffer */ +static LIST_HEAD(g_fixupvac); /* gcfixup */ +static LIST_HEAD(g_unmapvac); /* gcschedunmap */ + /******************************************************************************* * Command buffer copy management. */ -static GCLOCK_TYPE g_lock; -struct gcbuffer *g_buffervacant; -struct gcfixup *g_fixupvacant; - -static enum gcerror alloc_buffer(struct gcbuffer **gcbuffer) +static enum gcerror alloc_fixup(struct gcfixup **gcfixup) { enum gcerror gcerror = GCERR_NONE; - struct gcbuffer *temp; + struct gcfixup *temp; - if (g_buffervacant == NULL) { - temp = kmalloc(sizeof(struct gcbuffer), GFP_KERNEL); + GCLOCK(&g_fixuplock); + + if (list_empty(&g_fixupvac)) { + temp = kmalloc(sizeof(struct gcfixup), GFP_KERNEL); if (temp == NULL) { GCERR("out of memory.\n"); gcerror = GCERR_SETGRP(GCERR_OODM, - GCERR_IOCTL_BUF_ALLOC); + GCERR_IOCTL_FIXUP_ALLOC); goto exit; } } else { - temp = g_buffervacant; - g_buffervacant = g_buffervacant->next; + struct list_head *head; + head = g_fixupvac.next; + temp = list_entry(head, struct gcfixup, link); + list_del(head); } - temp->fixuphead = NULL; - temp->next = NULL; + GCUNLOCK(&g_fixuplock); - *gcbuffer = temp; + INIT_LIST_HEAD(&temp->link); + *gcfixup = temp; exit: return gcerror; } -static enum gcerror alloc_fixup(struct gcfixup **gcfixup) +static void free_fixup(struct gcfixup *gcfixup) +{ + GCLOCK(&g_fixuplock); + list_move(&gcfixup->link, &g_fixupvac); + GCUNLOCK(&g_fixuplock); +} + +static void free_fixup_list(struct list_head *fixuplist) +{ + GCLOCK(&g_fixuplock); + list_splice_init(fixuplist, &g_fixupvac); + GCUNLOCK(&g_fixuplock); +} + +static void free_vacant_fixups(void) +{ + struct list_head *head; + struct gcfixup *gcfixup; + + GCLOCK(&g_fixuplock); + while (!list_empty(&g_fixupvac)) { + head = g_fixupvac.next; + gcfixup = list_entry(head, struct gcfixup, link); + list_del(head); + kfree(gcfixup); + } + GCUNLOCK(&g_fixuplock); +} + +static enum gcerror alloc_buffer(struct gcbuffer **gcbuffer) { enum gcerror gcerror = GCERR_NONE; - struct gcfixup *temp; + struct gcbuffer *temp; - if (g_fixupvacant == NULL) { - temp = kmalloc(sizeof(struct gcfixup), GFP_KERNEL); + GCLOCK(&g_bufferlock); + + if (list_empty(&g_buffervac)) { + temp = kmalloc(sizeof(struct gcbuffer), GFP_KERNEL); if (temp == NULL) { GCERR("out of memory.\n"); gcerror = GCERR_SETGRP(GCERR_OODM, - GCERR_IOCTL_FIXUP_ALLOC); + GCERR_IOCTL_BUF_ALLOC); goto exit; } } else { - temp = g_fixupvacant; - g_fixupvacant = g_fixupvacant->next; + struct list_head *head; + head = g_buffervac.next; + temp = list_entry(head, struct gcbuffer, link); + list_del(head); } - *gcfixup = temp; + GCUNLOCK(&g_bufferlock); + + INIT_LIST_HEAD(&temp->fixup); + INIT_LIST_HEAD(&temp->link); + *gcbuffer = temp; exit: return gcerror; } -static void free_buffer_tree(struct gcbuffer *gcbuffer) +static void free_buffer(struct gcbuffer *gcbuffer) { - struct gcbuffer *prev; - struct gcbuffer *curr; - - prev = NULL; - curr = gcbuffer; - while (curr != NULL) { - if (curr->fixuphead != NULL) { - curr->fixuptail->next = g_fixupvacant; - g_fixupvacant = curr->fixuphead; - } + /* Free fixups. */ + free_fixup_list(&gcbuffer->fixup); - prev = curr; - curr = curr->next; - } - - prev->next = g_buffervacant; - g_buffervacant = gcbuffer; + /* Free the buffer. */ + GCLOCK(&g_bufferlock); + list_move(&gcbuffer->link, &g_buffervac); + GCUNLOCK(&g_bufferlock); } -static void destroy_buffer(void) +static void free_buffer_list(struct list_head *bufferlist) { - struct gcbuffer *currbuffer, *tempbuffer; - struct gcfixup *currfixup, *tempfixup; + struct list_head *head; + struct gcbuffer *gcbuffer; - currbuffer = g_buffervacant; - while (currbuffer != NULL) { - tempbuffer = currbuffer; - currbuffer = currbuffer->next; - kfree(tempbuffer); + while (!list_empty(bufferlist)) { + head = bufferlist->next; + gcbuffer = list_entry(head, struct gcbuffer, link); + free_buffer(gcbuffer); } +} - currfixup = g_fixupvacant; - while (currfixup != NULL) { - tempfixup = currfixup; - currfixup = currfixup->next; - kfree(tempfixup); +static void free_vacant_buffers(void) +{ + struct list_head *head; + struct gcbuffer *gcbuffer; + + GCLOCK(&g_bufferlock); + while (!list_empty(&g_buffervac)) { + head = g_buffervac.next; + gcbuffer = list_entry(head, struct gcbuffer, link); + list_del(head); + kfree(gcbuffer); } + GCUNLOCK(&g_bufferlock); } @@ -152,13 +194,13 @@ static void destroy_buffer(void) * Unmap list management. */ -static LIST_HEAD(g_unmapvac); - static enum gcerror alloc_schedunmap(struct gcschedunmap **gcschedunmap) { enum gcerror gcerror = GCERR_NONE; struct gcschedunmap *temp; + GCLOCK(&g_unmaplock); + if (list_empty(&g_unmapvac)) { temp = kmalloc(sizeof(struct gcschedunmap), GFP_KERNEL); if (temp == NULL) { @@ -174,23 +216,42 @@ static enum gcerror alloc_schedunmap(struct gcschedunmap **gcschedunmap) list_del(head); } + GCUNLOCK(&g_unmaplock); + + INIT_LIST_HEAD(&temp->link); *gcschedunmap = temp; exit: return gcerror; } -static void destroy_unmap(void) +static void free_schedunmap(struct gcschedunmap *gcschedunmap) +{ + GCLOCK(&g_unmaplock); + list_move(&gcschedunmap->link, &g_unmapvac); + GCUNLOCK(&g_unmaplock); +} + +static void free_schedunmap_list(struct list_head *schedunmaplist) +{ + GCLOCK(&g_unmaplock); + list_splice_init(schedunmaplist, &g_unmapvac); + GCUNLOCK(&g_unmaplock); +} + +static void free_vacant_unmap(void) { struct list_head *head; struct gcschedunmap *gcschedunmap; + GCLOCK(&g_unmaplock); while (!list_empty(&g_unmapvac)) { head = g_unmapvac.next; gcschedunmap = list_entry(head, struct gcschedunmap, link); list_del(head); kfree(gcschedunmap); } + GCUNLOCK(&g_unmaplock); } @@ -385,19 +446,24 @@ static void destroy_callback(void) static int gc_commit_wrapper(struct gccommit *gccommit) { int ret = 0; - bool locked = false; + bool buffercopied = false; bool unmapcopied = false; struct gccommit cpcommit; - struct gcbuffer *cpbuffer = NULL, *gcbuffer; - struct gcbuffer *headbuffer = NULL, *tailbuffer; - struct gcfixup *cpfixup = NULL, *gcfixup; - struct gcfixupentry *gcfixupentry; - struct gcschedunmap *cpschedunmap, *gcschedunmap; struct gccallbackinfo *gccallbackinfo; - struct list_head *head; - LIST_HEAD(cpunmap); + + struct list_head *gcbufferhead; + struct gcbuffer *cpbuffer, *gcbuffer; + LIST_HEAD(cpbufferlist); + + struct list_head *gcfixuphead; + struct gcfixup *cpfixup, *gcfixup; + struct gcfixupentry *gcfixupentry; int tablesize; + struct list_head *gcschedunmaphead; + struct gcschedunmap *cpschedunmap, *gcschedunmap; + LIST_HEAD(cpunmaplist); + GCENTER(GCZONE_COMMIT); /* Get IOCTL parameters. */ @@ -407,12 +473,10 @@ static int gc_commit_wrapper(struct gccommit *gccommit) goto exit; } - GCLOCK(&g_lock); - locked = true; - - /* Make a copy of the user buffer structures. */ - gcbuffer = cpcommit.buffer; - while (gcbuffer != NULL) { + /* Make a copy of the user buffer list. */ + gcbufferhead = cpcommit.buffer.next; + while (gcbufferhead != &gccommit->buffer) { + gcbuffer = list_entry(gcbufferhead, struct gcbuffer, link); GCDBG(GCZONE_COMMIT, "copying buffer 0x%08X.\n", (unsigned int) gcbuffer); @@ -421,31 +485,26 @@ static int gc_commit_wrapper(struct gccommit *gccommit) if (cpcommit.gcerror != GCERR_NONE) goto exit; - /* Add to the list. */ - if (headbuffer == NULL) - headbuffer = cpbuffer; - else - tailbuffer->next = cpbuffer; - tailbuffer = cpbuffer; - /* Get the data from the user. */ if (copy_from_user(cpbuffer, gcbuffer, sizeof(struct gcbuffer))) { + free_buffer(cpbuffer); GCERR("failed to read data.\n"); cpcommit.gcerror = GCERR_USER_READ; goto exit; } /* Get the next user buffer. */ - gcbuffer = cpbuffer->next; - cpbuffer->next = NULL; + gcbufferhead = cpbuffer->link.next; - /* Get fixups and reset them in the kernel copy. */ - gcfixup = cpbuffer->fixuphead; - cpbuffer->fixuphead = NULL; + /* Add the buffer to the list. */ + list_add_tail(&cpbuffer->link, &cpbufferlist); /* Copy all fixups. */ - while (gcfixup != NULL) { + gcfixuphead = cpbuffer->fixup.next; + INIT_LIST_HEAD(&cpbuffer->fixup); + while (gcfixuphead != &gcbuffer->fixup) { + gcfixup = list_entry(gcfixuphead, struct gcfixup, link); GCDBG(GCZONE_COMMIT, "copying fixup 0x%08X.\n", (unsigned int) gcfixup); @@ -454,28 +513,24 @@ static int gc_commit_wrapper(struct gccommit *gccommit) if (cpcommit.gcerror != GCERR_NONE) goto exit; - /* Add to the list. */ - if (cpbuffer->fixuphead == NULL) - cpbuffer->fixuphead = cpfixup; - else - cpbuffer->fixuptail->next = cpfixup; - cpbuffer->fixuptail = cpfixup; - /* Get the size of the fixup array. */ if (copy_from_user(cpfixup, gcfixup, offsetof(struct gcfixup, fixup))) { + free_fixup(cpfixup); GCERR("failed to read data.\n"); cpcommit.gcerror = GCERR_USER_READ; goto exit; } + /* Get the next fixup. */ + gcfixuphead = cpfixup->link.next; + + /* Add to the list. */ + list_add_tail(&cpfixup->link, &cpbuffer->fixup); + /* Get the fixup array. */ gcfixupentry = gcfixup->fixup; - /* Get the next fixup. */ - gcfixup = cpfixup->next; - cpfixup->next = NULL; - /* Compute the size of the fixup table. */ tablesize = cpfixup->count * sizeof(struct gcfixupentry); @@ -490,12 +545,19 @@ static int gc_commit_wrapper(struct gccommit *gccommit) } } + /* Move the local list to the commit structure. */ + INIT_LIST_HEAD(&cpcommit.buffer); + list_splice_init(&cpbufferlist, &cpcommit.buffer); + buffercopied = true; + /* Copy scheduled unmappings to the local list. */ GCDBG(GCZONE_COMMIT, "copying unmaps.\n"); - head = cpcommit.unmap.next; - while (head != &gccommit->unmap) { + gcschedunmaphead = cpcommit.unmap.next; + while (gcschedunmaphead != &gccommit->unmap) { /* Get a pointer to the user structure. */ - gcschedunmap = list_entry(head, struct gcschedunmap, link); + gcschedunmap = list_entry(gcschedunmaphead, + struct gcschedunmap, + link); GCDBG(GCZONE_COMMIT, "unmap 0x%08X.\n", (unsigned int) gcschedunmap); @@ -507,27 +569,24 @@ static int gc_commit_wrapper(struct gccommit *gccommit) /* Copy unmap record from user. */ if (copy_from_user(cpschedunmap, gcschedunmap, sizeof(struct gcschedunmap))) { - list_add(&cpschedunmap->link, &cpunmap); + free_schedunmap(cpschedunmap); GCERR("failed to read data.\n"); cpcommit.gcerror = GCERR_USER_READ; goto exit; } /* Get the next record. */ - head = cpschedunmap->link.next; + gcschedunmaphead = cpschedunmap->link.next; /* Append to the list. */ - list_add_tail(&cpschedunmap->link, &cpunmap); + list_add_tail(&cpschedunmap->link, &cpunmaplist); } /* Move the local list to the commit structure. */ INIT_LIST_HEAD(&cpcommit.unmap); - list_splice_init(&cpunmap, &cpcommit.unmap); + list_splice_init(&cpunmaplist, &cpcommit.unmap); unmapcopied = true; - GCUNLOCK(&g_lock); - locked = false; - /* Setup callback. */ if (cpcommit.callback != NULL) { GCDBG(GCZONE_COMMIT, "setting up callback.\n"); @@ -550,7 +609,6 @@ static int gc_commit_wrapper(struct gccommit *gccommit) } /* Call the core driver. */ - cpcommit.buffer = headbuffer; gc_commit(&cpcommit, true); exit: @@ -561,12 +619,8 @@ exit: } /* Free temporary resources. */ - if (!locked) - GCLOCK(&g_lock); - if (headbuffer != NULL) - free_buffer_tree(headbuffer); - list_splice(unmapcopied ? &cpcommit.unmap : &cpunmap, &g_unmapvac); - GCUNLOCK(&g_lock); + free_buffer_list(buffercopied ? &cpcommit.buffer : &cpbufferlist); + free_schedunmap_list(unmapcopied ? &cpcommit.unmap : &cpunmaplist); GCEXIT(GCZONE_COMMIT); return ret; @@ -959,7 +1013,6 @@ static int mod_init(void) goto failed; } - GCLOCK_INIT(&g_lock); GCDBG_REGISTER(ioctl); GCDBG(GCZONE_INIT, "device number = %d\n", dev_major); @@ -997,8 +1050,9 @@ static void mod_exit(void) platform_driver_unregister(&gcx_drv); driver_remove_file(&gcx_drv.driver, &driver_attr_version); - destroy_buffer(); - destroy_unmap(); + free_vacant_buffers(); + free_vacant_fixups(); + free_vacant_unmap(); destroy_callback(); } |