mirror of https://github.com/mgba-emu/mgba.git
DS GX: Switch from scanline-based to all-at-once
This commit is contained in:
parent
dc5f626683
commit
3dd18ecc2f
|
@ -17,12 +17,15 @@ CXX_GUARD_START
|
||||||
|
|
||||||
struct DSGXSoftwarePolygon {
|
struct DSGXSoftwarePolygon {
|
||||||
struct DSGXPolygon* poly;
|
struct DSGXPolygon* poly;
|
||||||
|
unsigned polyId;
|
||||||
uint16_t* texBase;
|
uint16_t* texBase;
|
||||||
uint16_t* palBase;
|
uint16_t* palBase;
|
||||||
int texFormat;
|
int texFormat;
|
||||||
int blendFormat;
|
int blendFormat;
|
||||||
int texW;
|
int texW;
|
||||||
int texH;
|
int texH;
|
||||||
|
int minY;
|
||||||
|
int maxY;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct DSGXSoftwareEdge {
|
struct DSGXSoftwareEdge {
|
||||||
|
@ -75,20 +78,16 @@ struct DSGXSoftwareSpan {
|
||||||
|
|
||||||
DECLARE_VECTOR(DSGXSoftwarePolygonList, struct DSGXSoftwarePolygon);
|
DECLARE_VECTOR(DSGXSoftwarePolygonList, struct DSGXSoftwarePolygon);
|
||||||
DECLARE_VECTOR(DSGXSoftwareEdgeList, struct DSGXSoftwareEdge);
|
DECLARE_VECTOR(DSGXSoftwareEdgeList, struct DSGXSoftwareEdge);
|
||||||
DECLARE_VECTOR(DSGXSoftwareSpanList, struct DSGXSoftwareSpan);
|
|
||||||
|
|
||||||
struct DSGXSoftwareRenderer {
|
struct DSGXSoftwareRenderer {
|
||||||
struct DSGXRenderer d;
|
struct DSGXRenderer d;
|
||||||
|
|
||||||
struct DSGXSoftwarePolygonList activePolys;
|
struct DSGXSoftwarePolygonList activePolys;
|
||||||
struct DSGXSoftwareEdgeList activeEdges;
|
struct DSGXSoftwareEdgeList activeEdges;
|
||||||
struct DSGXSoftwareSpanList activeSpans;
|
|
||||||
struct DSGXSoftwareSpan** bucket;
|
|
||||||
|
|
||||||
int32_t depthBuffer[DS_VIDEO_HORIZONTAL_PIXELS];
|
int32_t* depthBuffer;
|
||||||
uint8_t stencilBuffer[DS_VIDEO_HORIZONTAL_PIXELS];
|
uint8_t* stencilBuffer;
|
||||||
color_t* scanlineCache;
|
color_t* scanlineCache;
|
||||||
bool flushPending;
|
|
||||||
int sort;
|
int sort;
|
||||||
|
|
||||||
struct DSGXVertex* verts;
|
struct DSGXVertex* verts;
|
||||||
|
|
|
@ -10,7 +10,6 @@
|
||||||
|
|
||||||
DEFINE_VECTOR(DSGXSoftwarePolygonList, struct DSGXSoftwarePolygon);
|
DEFINE_VECTOR(DSGXSoftwarePolygonList, struct DSGXSoftwarePolygon);
|
||||||
DEFINE_VECTOR(DSGXSoftwareEdgeList, struct DSGXSoftwareEdge);
|
DEFINE_VECTOR(DSGXSoftwareEdgeList, struct DSGXSoftwareEdge);
|
||||||
DEFINE_VECTOR(DSGXSoftwareSpanList, struct DSGXSoftwareSpan);
|
|
||||||
|
|
||||||
static void DSGXSoftwareRendererInit(struct DSGXRenderer* renderer);
|
static void DSGXSoftwareRendererInit(struct DSGXRenderer* renderer);
|
||||||
static void DSGXSoftwareRendererReset(struct DSGXRenderer* renderer);
|
static void DSGXSoftwareRendererReset(struct DSGXRenderer* renderer);
|
||||||
|
@ -415,24 +414,23 @@ void DSGXSoftwareRendererCreate(struct DSGXSoftwareRenderer* renderer) {
|
||||||
static void DSGXSoftwareRendererInit(struct DSGXRenderer* renderer) {
|
static void DSGXSoftwareRendererInit(struct DSGXRenderer* renderer) {
|
||||||
struct DSGXSoftwareRenderer* softwareRenderer = (struct DSGXSoftwareRenderer*) renderer;
|
struct DSGXSoftwareRenderer* softwareRenderer = (struct DSGXSoftwareRenderer*) renderer;
|
||||||
DSGXSoftwarePolygonListInit(&softwareRenderer->activePolys, DS_GX_POLYGON_BUFFER_SIZE / 4);
|
DSGXSoftwarePolygonListInit(&softwareRenderer->activePolys, DS_GX_POLYGON_BUFFER_SIZE / 4);
|
||||||
DSGXSoftwareEdgeListInit(&softwareRenderer->activeEdges, DS_GX_POLYGON_BUFFER_SIZE);
|
DSGXSoftwareEdgeListInit(&softwareRenderer->activeEdges, 16);
|
||||||
DSGXSoftwareSpanListInit(&softwareRenderer->activeSpans, DS_GX_POLYGON_BUFFER_SIZE / 2);
|
|
||||||
softwareRenderer->bucket = anonymousMemoryMap(sizeof(*softwareRenderer->bucket) * DS_GX_POLYGON_BUFFER_SIZE);
|
|
||||||
softwareRenderer->scanlineCache = anonymousMemoryMap(sizeof(color_t) * DS_VIDEO_VERTICAL_PIXELS * DS_VIDEO_HORIZONTAL_PIXELS);
|
softwareRenderer->scanlineCache = anonymousMemoryMap(sizeof(color_t) * DS_VIDEO_VERTICAL_PIXELS * DS_VIDEO_HORIZONTAL_PIXELS);
|
||||||
|
softwareRenderer->depthBuffer = anonymousMemoryMap(sizeof(int32_t) * DS_VIDEO_VERTICAL_PIXELS * DS_VIDEO_HORIZONTAL_PIXELS);
|
||||||
|
softwareRenderer->stencilBuffer = anonymousMemoryMap(sizeof(uint8_t) * DS_VIDEO_VERTICAL_PIXELS * DS_VIDEO_HORIZONTAL_PIXELS);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void DSGXSoftwareRendererReset(struct DSGXRenderer* renderer) {
|
static void DSGXSoftwareRendererReset(struct DSGXRenderer* renderer) {
|
||||||
struct DSGXSoftwareRenderer* softwareRenderer = (struct DSGXSoftwareRenderer*) renderer;
|
struct DSGXSoftwareRenderer* softwareRenderer = (struct DSGXSoftwareRenderer*) renderer;
|
||||||
softwareRenderer->flushPending = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void DSGXSoftwareRendererDeinit(struct DSGXRenderer* renderer) {
|
static void DSGXSoftwareRendererDeinit(struct DSGXRenderer* renderer) {
|
||||||
struct DSGXSoftwareRenderer* softwareRenderer = (struct DSGXSoftwareRenderer*) renderer;
|
struct DSGXSoftwareRenderer* softwareRenderer = (struct DSGXSoftwareRenderer*) renderer;
|
||||||
DSGXSoftwarePolygonListDeinit(&softwareRenderer->activePolys);
|
DSGXSoftwarePolygonListDeinit(&softwareRenderer->activePolys);
|
||||||
DSGXSoftwareEdgeListDeinit(&softwareRenderer->activeEdges);
|
DSGXSoftwareEdgeListDeinit(&softwareRenderer->activeEdges);
|
||||||
DSGXSoftwareSpanListDeinit(&softwareRenderer->activeSpans);
|
|
||||||
mappedMemoryFree(softwareRenderer->bucket, sizeof(*softwareRenderer->bucket) * DS_GX_POLYGON_BUFFER_SIZE);
|
|
||||||
mappedMemoryFree(softwareRenderer->scanlineCache, sizeof(color_t) * DS_VIDEO_VERTICAL_PIXELS * DS_VIDEO_HORIZONTAL_PIXELS);
|
mappedMemoryFree(softwareRenderer->scanlineCache, sizeof(color_t) * DS_VIDEO_VERTICAL_PIXELS * DS_VIDEO_HORIZONTAL_PIXELS);
|
||||||
|
mappedMemoryFree(softwareRenderer->depthBuffer, sizeof(int32_t) * DS_VIDEO_VERTICAL_PIXELS * DS_VIDEO_HORIZONTAL_PIXELS);
|
||||||
|
mappedMemoryFree(softwareRenderer->stencilBuffer, sizeof(uint8_t) * DS_VIDEO_VERTICAL_PIXELS * DS_VIDEO_HORIZONTAL_PIXELS);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void DSGXSoftwareRendererInvalidateTex(struct DSGXRenderer* renderer, int slot) {
|
static void DSGXSoftwareRendererInvalidateTex(struct DSGXRenderer* renderer, int slot) {
|
||||||
|
@ -440,7 +438,7 @@ static void DSGXSoftwareRendererInvalidateTex(struct DSGXRenderer* renderer, int
|
||||||
// TODO
|
// TODO
|
||||||
}
|
}
|
||||||
|
|
||||||
static void _preparePoly(struct DSGXRenderer* renderer, struct DSGXVertex* verts, struct DSGXSoftwarePolygon* poly, int polyId) {
|
static void _preparePoly(struct DSGXRenderer* renderer, struct DSGXVertex* verts, struct DSGXSoftwarePolygon* poly) {
|
||||||
struct DSGXSoftwareRenderer* softwareRenderer = (struct DSGXSoftwareRenderer*) renderer;
|
struct DSGXSoftwareRenderer* softwareRenderer = (struct DSGXSoftwareRenderer*) renderer;
|
||||||
struct DSGXSoftwareEdge* edge = DSGXSoftwareEdgeListAppend(&softwareRenderer->activeEdges);
|
struct DSGXSoftwareEdge* edge = DSGXSoftwareEdgeListAppend(&softwareRenderer->activeEdges);
|
||||||
poly->texFormat = DSGXTexParamsGetFormat(poly->poly->texParams);
|
poly->texFormat = DSGXTexParamsGetFormat(poly->poly->texParams);
|
||||||
|
@ -467,19 +465,33 @@ static void _preparePoly(struct DSGXRenderer* renderer, struct DSGXVertex* verts
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
edge->polyId = polyId;
|
edge->polyId = poly->polyId;
|
||||||
|
poly->minY = DS_VIDEO_VERTICAL_PIXELS - 1;
|
||||||
|
poly->maxY = 0;
|
||||||
|
|
||||||
struct DSGXVertex* v0 = &verts[poly->poly->vertIds[0]];
|
struct DSGXVertex* v0 = &verts[poly->poly->vertIds[0]];
|
||||||
struct DSGXVertex* v1;
|
struct DSGXVertex* v1;
|
||||||
|
|
||||||
int32_t v0x = (v0->viewCoord[0] + v0->viewCoord[3]) * (int64_t) (renderer->viewportWidth << 12) / (v0->viewCoord[3] * 2) + (renderer->viewportX << 12);
|
int32_t v0x = (v0->viewCoord[0] + v0->viewCoord[3]) * (int64_t) (renderer->viewportWidth << 12) / (v0->viewCoord[3] * 2) + (renderer->viewportX << 12);
|
||||||
int32_t v0y = (-v0->viewCoord[1] + v0->viewCoord[3]) * (int64_t) (renderer->viewportHeight << 12) / (v0->viewCoord[3] * 2) + (renderer->viewportY << 12);
|
int32_t v0y = (-v0->viewCoord[1] + v0->viewCoord[3]) * (int64_t) (renderer->viewportHeight << 12) / (v0->viewCoord[3] * 2) + (renderer->viewportY << 12);
|
||||||
|
if (poly->minY > v0y >> 12) {
|
||||||
|
poly->minY = v0y >> 12;
|
||||||
|
}
|
||||||
|
if (poly->maxY < v0y >> 12) {
|
||||||
|
poly->maxY = v0y >> 12;
|
||||||
|
}
|
||||||
|
|
||||||
int v;
|
int v;
|
||||||
for (v = 1; v < poly->poly->verts; ++v) {
|
for (v = 1; v < poly->poly->verts; ++v) {
|
||||||
v1 = &verts[poly->poly->vertIds[v]];
|
v1 = &verts[poly->poly->vertIds[v]];
|
||||||
int32_t v1x = (v1->viewCoord[0] + v1->viewCoord[3]) * (int64_t) (renderer->viewportWidth << 12) / (v1->viewCoord[3] * 2) + (renderer->viewportX << 12);
|
int32_t v1x = (v1->viewCoord[0] + v1->viewCoord[3]) * (int64_t) (renderer->viewportWidth << 12) / (v1->viewCoord[3] * 2) + (renderer->viewportX << 12);
|
||||||
int32_t v1y = (-v1->viewCoord[1] + v1->viewCoord[3]) * (int64_t) (renderer->viewportHeight << 12) / (v1->viewCoord[3] * 2) + (renderer->viewportY << 12);
|
int32_t v1y = (-v1->viewCoord[1] + v1->viewCoord[3]) * (int64_t) (renderer->viewportHeight << 12) / (v1->viewCoord[3] * 2) + (renderer->viewportY << 12);
|
||||||
|
if (poly->minY > v1y >> 12) {
|
||||||
|
poly->minY = v1y >> 12;
|
||||||
|
}
|
||||||
|
if (poly->maxY < v1y >> 12) {
|
||||||
|
poly->maxY = v1y >> 12;
|
||||||
|
}
|
||||||
|
|
||||||
if (v0y <= v1y) {
|
if (v0y <= v1y) {
|
||||||
edge->y0 = v0y;
|
edge->y0 = v0y;
|
||||||
|
@ -516,7 +528,7 @@ static void _preparePoly(struct DSGXRenderer* renderer, struct DSGXVertex* verts
|
||||||
}
|
}
|
||||||
|
|
||||||
edge = DSGXSoftwareEdgeListAppend(&softwareRenderer->activeEdges);
|
edge = DSGXSoftwareEdgeListAppend(&softwareRenderer->activeEdges);
|
||||||
edge->polyId = polyId;
|
edge->polyId = poly->polyId;
|
||||||
v0 = v1;
|
v0 = v1;
|
||||||
v0x = v1x;
|
v0x = v1x;
|
||||||
v0y = v1y;
|
v0y = v1y;
|
||||||
|
@ -526,6 +538,13 @@ static void _preparePoly(struct DSGXRenderer* renderer, struct DSGXVertex* verts
|
||||||
int32_t v1x = (v1->viewCoord[0] + v1->viewCoord[3]) * (int64_t) (renderer->viewportWidth << 12) / (v1->viewCoord[3] * 2) + (renderer->viewportX << 12);
|
int32_t v1x = (v1->viewCoord[0] + v1->viewCoord[3]) * (int64_t) (renderer->viewportWidth << 12) / (v1->viewCoord[3] * 2) + (renderer->viewportX << 12);
|
||||||
int32_t v1y = (-v1->viewCoord[1] + v1->viewCoord[3]) * (int64_t) (renderer->viewportHeight << 12) / (v1->viewCoord[3] * 2) + (renderer->viewportY << 12);
|
int32_t v1y = (-v1->viewCoord[1] + v1->viewCoord[3]) * (int64_t) (renderer->viewportHeight << 12) / (v1->viewCoord[3] * 2) + (renderer->viewportY << 12);
|
||||||
|
|
||||||
|
if (poly->minY > v1y >> 12) {
|
||||||
|
poly->minY = v1y >> 12;
|
||||||
|
}
|
||||||
|
if (poly->maxY < v1y >> 12) {
|
||||||
|
poly->maxY = v1y >> 12;
|
||||||
|
}
|
||||||
|
|
||||||
if (v0y <= v1y) {
|
if (v0y <= v1y) {
|
||||||
edge->y0 = v0y;
|
edge->y0 = v0y;
|
||||||
edge->x0 = v0x;
|
edge->x0 = v0x;
|
||||||
|
@ -559,17 +578,75 @@ static void _preparePoly(struct DSGXRenderer* renderer, struct DSGXVertex* verts
|
||||||
edge->s1 = v0->vs;
|
edge->s1 = v0->vs;
|
||||||
edge->t1 = v0->vt;
|
edge->t1 = v0->vt;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (poly->maxY >= DS_VIDEO_VERTICAL_PIXELS) {
|
||||||
|
poly->maxY = DS_VIDEO_VERTICAL_PIXELS - 1;
|
||||||
|
}
|
||||||
|
if (poly->minY < 0) {
|
||||||
|
poly->minY = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void _drawSpan(struct DSGXSoftwareRenderer* softwareRenderer, struct DSGXSoftwareSpan* span, int y) {
|
||||||
|
color_t* scanline = &softwareRenderer->scanlineCache[DS_VIDEO_HORIZONTAL_PIXELS * y];
|
||||||
|
int32_t* depth = &softwareRenderer->depthBuffer[DS_VIDEO_HORIZONTAL_PIXELS * y];
|
||||||
|
uint8_t* stencil = &softwareRenderer->stencilBuffer[DS_VIDEO_HORIZONTAL_PIXELS * y];
|
||||||
|
int32_t x = span->ep[0].coord[0] >> 12;
|
||||||
|
if (x < 0) {
|
||||||
|
x = 0;
|
||||||
|
}
|
||||||
|
unsigned stencilValue = span->polyId;
|
||||||
|
if (span->poly->blendFormat == 3) {
|
||||||
|
stencilValue |= 0x40;
|
||||||
|
}
|
||||||
|
for (; x < (span->ep[1].coord[0] >> 12) && x < DS_VIDEO_HORIZONTAL_PIXELS; ++x) {
|
||||||
|
if (span->ep[0].coord[softwareRenderer->sort] < depth[x]) {
|
||||||
|
_resolveEndpoint(span);
|
||||||
|
color_t color = _lookupColor(softwareRenderer, &span->ep[0], span->poly);
|
||||||
|
unsigned a = color >> 27;
|
||||||
|
unsigned current = scanline[x];
|
||||||
|
unsigned b = current >> 27;
|
||||||
|
unsigned ab = a;
|
||||||
|
unsigned s = stencilValue;
|
||||||
|
if (b > ab) {
|
||||||
|
ab = b;
|
||||||
|
}
|
||||||
|
if (a == 0x1F) {
|
||||||
|
if (!(s == 0x40 || (stencil[x] & 0x40))) {
|
||||||
|
depth[x] = span->ep[0].coord[softwareRenderer->sort];
|
||||||
|
scanline[x] = color;
|
||||||
|
s &= ~0x40;
|
||||||
|
}
|
||||||
|
stencil[x] = s;
|
||||||
|
} else if (a) {
|
||||||
|
// TODO: Disable alpha?
|
||||||
|
if (b) {
|
||||||
|
color = _mix32(a, color, 0x1F - a, current);
|
||||||
|
color |= ab << 27;
|
||||||
|
}
|
||||||
|
if (stencil[x] != s) {
|
||||||
|
if (!(s == 0x40 || (stencil[x] & 0x40))) {
|
||||||
|
if (DSGXPolygonAttrsIsUpdateDepth(span->poly->poly->polyParams)) {
|
||||||
|
depth[x] = span->ep[0].coord[softwareRenderer->sort];
|
||||||
|
}
|
||||||
|
scanline[x] = color;
|
||||||
|
s &= ~0x40;
|
||||||
|
}
|
||||||
|
stencil[x] = s;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_stepEndpoint(span);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void DSGXSoftwareRendererSetRAM(struct DSGXRenderer* renderer, struct DSGXVertex* verts, struct DSGXPolygon* polys, unsigned polyCount, bool wSort) {
|
static void DSGXSoftwareRendererSetRAM(struct DSGXRenderer* renderer, struct DSGXVertex* verts, struct DSGXPolygon* polys, unsigned polyCount, bool wSort) {
|
||||||
struct DSGXSoftwareRenderer* softwareRenderer = (struct DSGXSoftwareRenderer*) renderer;
|
struct DSGXSoftwareRenderer* softwareRenderer = (struct DSGXSoftwareRenderer*) renderer;
|
||||||
|
|
||||||
softwareRenderer->flushPending = true;
|
|
||||||
softwareRenderer->sort = wSort ? 3 : 2;
|
softwareRenderer->sort = wSort ? 3 : 2;
|
||||||
softwareRenderer->verts = verts;
|
softwareRenderer->verts = verts;
|
||||||
DSGXSoftwarePolygonListClear(&softwareRenderer->activePolys);
|
DSGXSoftwarePolygonListClear(&softwareRenderer->activePolys);
|
||||||
DSGXSoftwareEdgeListClear(&softwareRenderer->activeEdges);
|
size_t i;
|
||||||
unsigned i;
|
|
||||||
// Pass 1: Opaque
|
// Pass 1: Opaque
|
||||||
for (i = 0; i < polyCount; ++i) {
|
for (i = 0; i < polyCount; ++i) {
|
||||||
struct DSGXSoftwarePolygon* poly = NULL;
|
struct DSGXSoftwarePolygon* poly = NULL;
|
||||||
|
@ -587,7 +664,7 @@ static void DSGXSoftwareRendererSetRAM(struct DSGXRenderer* renderer, struct DSG
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
poly->poly = &polys[i];
|
poly->poly = &polys[i];
|
||||||
_preparePoly(renderer, verts, poly, DSGXSoftwarePolygonListSize(&softwareRenderer->activePolys) - 1);
|
poly->polyId = DSGXSoftwarePolygonListSize(&softwareRenderer->activePolys) - 1;
|
||||||
}
|
}
|
||||||
// Pass 2: Translucent
|
// Pass 2: Translucent
|
||||||
for (i = 0; i < polyCount; ++i) {
|
for (i = 0; i < polyCount; ++i) {
|
||||||
|
@ -606,112 +683,53 @@ static void DSGXSoftwareRendererSetRAM(struct DSGXRenderer* renderer, struct DSG
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
poly->poly = &polys[i];
|
poly->poly = &polys[i];
|
||||||
_preparePoly(renderer, verts, poly, DSGXSoftwarePolygonListSize(&softwareRenderer->activePolys) - 1);
|
poly->polyId = DSGXSoftwarePolygonListSize(&softwareRenderer->activePolys) - 1;
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void DSGXSoftwareRendererDrawScanline(struct DSGXRenderer* renderer, int y) {
|
|
||||||
struct DSGXSoftwareRenderer* softwareRenderer = (struct DSGXSoftwareRenderer*) renderer;
|
|
||||||
if (!softwareRenderer->flushPending) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
DSGXSoftwareSpanListClear(&softwareRenderer->activeSpans);
|
|
||||||
memset(softwareRenderer->bucket, 0, sizeof(*softwareRenderer->bucket) * DS_GX_POLYGON_BUFFER_SIZE);
|
|
||||||
size_t i;
|
|
||||||
for (i = 0; i < DSGXSoftwareEdgeListSize(&softwareRenderer->activeEdges); ++i) {
|
|
||||||
struct DSGXSoftwareEdge* edge = DSGXSoftwareEdgeListGetPointer(&softwareRenderer->activeEdges, i);
|
|
||||||
if (edge->y1 < (y << 12)) {
|
|
||||||
continue;
|
|
||||||
} else if (edge->y0 > (y << 12)) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
unsigned poly = edge->polyId;
|
|
||||||
struct DSGXSoftwareSpan* span = softwareRenderer->bucket[poly];
|
|
||||||
if (span && !span->ep[1].coord[3]) {
|
|
||||||
if (_edgeToSpan(span, edge, 1, y << 12)) {
|
|
||||||
_createStep(span);
|
|
||||||
softwareRenderer->bucket[poly] = NULL;
|
|
||||||
}
|
|
||||||
} else if (!span) {
|
|
||||||
span = DSGXSoftwareSpanListAppend(&softwareRenderer->activeSpans);
|
|
||||||
memset(&span->ep[1], 0, sizeof(span->ep[1]));
|
|
||||||
span->poly = DSGXSoftwarePolygonListGetPointer(&softwareRenderer->activePolys, poly);
|
|
||||||
span->polyId = DSGXPolygonAttrsGetId(span->poly->poly->polyParams);
|
|
||||||
if (!_edgeToSpan(span, edge, 0, y << 12)) {
|
|
||||||
// Horizontal line
|
|
||||||
DSGXSoftwareSpanListShift(&softwareRenderer->activeSpans, DSGXSoftwareSpanListSize(&softwareRenderer->activeSpans) - 1, 1);
|
|
||||||
} else {
|
|
||||||
softwareRenderer->bucket[poly] = span;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
color_t* scanline = &softwareRenderer->scanlineCache[DS_VIDEO_HORIZONTAL_PIXELS * y];
|
memset(softwareRenderer->scanlineCache, 0, sizeof(color_t) * DS_VIDEO_VERTICAL_PIXELS * DS_VIDEO_HORIZONTAL_PIXELS);
|
||||||
memset(scanline, 0, sizeof(color_t) * DS_VIDEO_HORIZONTAL_PIXELS);
|
memset(softwareRenderer->stencilBuffer, 0, sizeof(uint8_t) * DS_VIDEO_VERTICAL_PIXELS * DS_VIDEO_HORIZONTAL_PIXELS);
|
||||||
memset(softwareRenderer->stencilBuffer, 0, sizeof(softwareRenderer->stencilBuffer[0]) * DS_VIDEO_HORIZONTAL_PIXELS);
|
for (i = 0; i < DS_VIDEO_VERTICAL_PIXELS * DS_VIDEO_HORIZONTAL_PIXELS ; i += 4) {
|
||||||
for (i = 0; i < DS_VIDEO_HORIZONTAL_PIXELS; i += 4) {
|
|
||||||
softwareRenderer->depthBuffer[i] = INT32_MAX;
|
softwareRenderer->depthBuffer[i] = INT32_MAX;
|
||||||
softwareRenderer->depthBuffer[i + 1] = INT32_MAX;
|
softwareRenderer->depthBuffer[i + 1] = INT32_MAX;
|
||||||
softwareRenderer->depthBuffer[i + 2] = INT32_MAX;
|
softwareRenderer->depthBuffer[i + 2] = INT32_MAX;
|
||||||
softwareRenderer->depthBuffer[i + 3] = INT32_MAX;
|
softwareRenderer->depthBuffer[i + 3] = INT32_MAX;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (i = 0; i < DSGXSoftwareSpanListSize(&softwareRenderer->activeSpans); ++i) {
|
size_t p;
|
||||||
struct DSGXSoftwareSpan* span = DSGXSoftwareSpanListGetPointer(&softwareRenderer->activeSpans, i);
|
for (p = 0; p < DSGXSoftwarePolygonListSize(&softwareRenderer->activePolys); ++p) {
|
||||||
|
struct DSGXSoftwarePolygon* poly = DSGXSoftwarePolygonListGetPointer(&softwareRenderer->activePolys, p);
|
||||||
int32_t x = span->ep[0].coord[0] >> 12;
|
DSGXSoftwareEdgeListClear(&softwareRenderer->activeEdges);
|
||||||
if (x < 0) {
|
_preparePoly(renderer, verts, poly);
|
||||||
x = 0;
|
int y;
|
||||||
}
|
for (y = poly->minY; y <= poly->maxY; ++y) {
|
||||||
unsigned stencilValue = span->polyId;
|
struct DSGXSoftwareSpan span = {
|
||||||
if (span->poly->blendFormat == 3) {
|
.poly = poly,
|
||||||
stencilValue |= 0x40;
|
.polyId = DSGXPolygonAttrsGetId(poly->poly->polyParams),
|
||||||
}
|
};
|
||||||
for (; x < (span->ep[1].coord[0] >> 12) && x < DS_VIDEO_HORIZONTAL_PIXELS; ++x) {
|
for (i = 0; i < DSGXSoftwareEdgeListSize(&softwareRenderer->activeEdges); ++i) {
|
||||||
if (span->ep[0].coord[softwareRenderer->sort] < softwareRenderer->depthBuffer[x]) {
|
struct DSGXSoftwareEdge* edge = DSGXSoftwareEdgeListGetPointer(&softwareRenderer->activeEdges, i);
|
||||||
_resolveEndpoint(span);
|
if (edge->y1 < (y << 12)) {
|
||||||
color_t color = _lookupColor(softwareRenderer, &span->ep[0], span->poly);
|
continue;
|
||||||
unsigned a = color >> 27;
|
} else if (edge->y0 > (y << 12)) {
|
||||||
unsigned current = scanline[x];
|
continue;
|
||||||
unsigned b = current >> 27;
|
|
||||||
unsigned ab = a;
|
|
||||||
unsigned s = stencilValue;
|
|
||||||
if (b > ab) {
|
|
||||||
ab = b;
|
|
||||||
}
|
}
|
||||||
if (a == 0x1F) {
|
|
||||||
if (!(s == 0x40 || (softwareRenderer->stencilBuffer[x] & 0x40))) {
|
if (span.ep[0].coord[3]) {
|
||||||
softwareRenderer->depthBuffer[x] = span->ep[0].coord[softwareRenderer->sort];
|
if (_edgeToSpan(&span, edge, 1, y << 12)) {
|
||||||
scanline[x] = color;
|
_createStep(&span);
|
||||||
s &= ~0x40;
|
_drawSpan(softwareRenderer, &span, y);
|
||||||
}
|
|
||||||
softwareRenderer->stencilBuffer[x] = s;
|
|
||||||
} else if (a) {
|
|
||||||
// TODO: Disable alpha?
|
|
||||||
if (b) {
|
|
||||||
color = _mix32(a, color, 0x1F - a, current);
|
|
||||||
color |= ab << 27;
|
|
||||||
}
|
|
||||||
if (softwareRenderer->stencilBuffer[x] != s) {
|
|
||||||
if (!(s == 0x40 || (softwareRenderer->stencilBuffer[x] & 0x40))) {
|
|
||||||
if (DSGXPolygonAttrsIsUpdateDepth(span->poly->poly->polyParams)) {
|
|
||||||
softwareRenderer->depthBuffer[x] = span->ep[0].coord[softwareRenderer->sort];
|
|
||||||
}
|
|
||||||
scanline[x] = color;
|
|
||||||
s &= ~0x40;
|
|
||||||
}
|
|
||||||
softwareRenderer->stencilBuffer[x] = s;
|
|
||||||
}
|
}
|
||||||
|
} else if (!_edgeToSpan(&span, edge, 0, y << 12)) {
|
||||||
|
// Horizontal line
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
_stepEndpoint(span);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (y == DS_VIDEO_VERTICAL_PIXELS - 1) {
|
static void DSGXSoftwareRendererDrawScanline(struct DSGXRenderer* renderer, int y) {
|
||||||
softwareRenderer->flushPending = false;
|
struct DSGXSoftwareRenderer* softwareRenderer = (struct DSGXSoftwareRenderer*) renderer;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void DSGXSoftwareRendererGetScanline(struct DSGXRenderer* renderer, int y, const color_t** output) {
|
static void DSGXSoftwareRendererGetScanline(struct DSGXRenderer* renderer, int y, const color_t** output) {
|
||||||
|
|
Loading…
Reference in New Issue