DS GX: Improve clipping interpolation

This commit is contained in:
Vicki Pfau 2017-03-13 18:41:18 -07:00
parent 6d4236bd29
commit b16932dbc6
1 changed files with 35 additions and 59 deletions

View File

@ -187,12 +187,14 @@ static void _updateClipMatrix(struct DSGX* gx) {
gx->p->memory.io9[DS9_REG_CLIPMTX_RESULT_1F >> 1] = gx->clipMatrix.m[15] >> 16; gx->p->memory.io9[DS9_REG_CLIPMTX_RESULT_1F >> 1] = gx->clipMatrix.m[15] >> 16;
} }
static inline int32_t _lerp(int32_t x0, int32_t x1, int32_t w0, int32_t w1, int32_t q) { static inline int32_t _lerp(int32_t x0, int32_t x1, int32_t q, int64_t r, int point) {
int64_t x = (x1 * (int64_t) w1 - x0 * (int64_t) w0) >> 16; int64_t x = x1 - x0;
x *= q; x *= q;
x >>= 15; x >>= point >> 1;
x += x0 * (int64_t) w0; x *= r;
return x >> 16; x >>= 27 + (point >> 1);
x += x0;
return x + 1;
} }
static int _cohenSutherlandCode(const struct DSGX* gx, const struct DSGXVertex* v) { static int _cohenSutherlandCode(const struct DSGX* gx, const struct DSGXVertex* v) {
@ -215,68 +217,42 @@ static int _cohenSutherlandCode(const struct DSGX* gx, const struct DSGXVertex*
return code; return code;
} }
static void _lerpVertex(const struct DSGXVertex* v0, const struct DSGXVertex* v1, struct DSGXVertex* out, int32_t xw) { static bool _lerpVertex(const struct DSGXVertex* v0, const struct DSGXVertex* v1, struct DSGXVertex* out, int32_t q, int64_t r) {
int32_t w0 = v0->vw; if (!r) {
int32_t w1 = v1->vw; return false;
int64_t w = (((int64_t) (w1 - w0) * xw) >> 32) + w0; }
int64_t wRecip;// = 0x1000000000000LL / w; if (q < 0) {
return false;
}
if (q > r) {
return false;
}
r = (INT64_MAX >> 24) / r;
int32_t w = _lerp(v0->vw, v1->vw, q, r, 12);
out->vw = w; out->vw = w;
// XXX: Disable perspective correction until I figure out how to fix it
wRecip = 0x100000000;
w0 = 0x10000;
w1 = 0x10000;
out->color = v0->color; // TODO out->color = v0->color; // TODO
int32_t vx = _lerp(v0->vx, v1->vx, w0, w1, xw); out->vx = _lerp(v0->vx, v1->vx, q, r, 12);
int32_t vy = _lerp(v0->vy, v1->vy, w0, w1, xw); out->vy = _lerp(v0->vy, v1->vy, q, r, 12);
int32_t vz = _lerp(v0->vz, v1->vz, w0, w1, xw); out->vz = _lerp(v0->vz, v1->vz, q, r, 12);
out->vx = ((vx * wRecip) >> 32) + 1; out->vw = _lerp(v0->vw, v1->vw, q, r, 12);
out->vy = ((vy * wRecip) >> 32) + 1; out->vs = _lerp(v0->vs, v1->vs, q, r, 12);
out->vz = ((vz * wRecip) >> 32) + 1; out->vt = _lerp(v0->vt, v1->vt, q, r, 12);
int32_t s = _lerp(v0->vs, v1->vs, w0, w1, xw);
int32_t t = _lerp(v0->vt, v1->vt, w0, w1, xw);
out->vs = (s * wRecip) >> 32;
out->vt = (t * wRecip) >> 32;
}
static bool _lerpVertexX(const struct DSGXVertex* v0, const struct DSGXVertex* v1, struct DSGXVertex* out, int32_t x) {
int64_t width = v1->vx - v0->vx;
int64_t xw = x - v0->vx;
if (!width) {
return false;
}
if (xw < 0) {
return false;
} else if (xw > width) {
return false;
}
xw *= 0x80000000LL;
xw /= width;
_lerpVertex(v0, v1, out, xw);
return true; return true;
} }
static bool _lerpVertexY(const struct DSGXVertex* v0, const struct DSGXVertex* v1, struct DSGXVertex* out, int32_t y) { static bool _lerpVertexX(const struct DSGXVertex* v0, const struct DSGXVertex* v1, struct DSGXVertex* out, int32_t xw) {
int64_t height = v1->vy - v0->vy; int32_t q = xw - v0->vx;
int64_t yw = y - v0->vy; int64_t r = v1->vx - v0->vx;
if (!height) { return _lerpVertex(v0, v1, out, q, r);
return false;
}
if (yw < 0) {
return false;
} else if (yw > height) {
return false;
} }
yw *= 0x80000000LL; static bool _lerpVertexY(const struct DSGXVertex* v0, const struct DSGXVertex* v1, struct DSGXVertex* out, int32_t yw) {
yw /= height; int32_t q = yw - v0->vy;
int64_t r = v1->vy - v0->vy;
_lerpVertex(v0, v1, out, yw); return _lerpVertex(v0, v1, out, q, r);
return true;
} }
static bool _clipPolygon(struct DSGX* gx, struct DSGXPolygon* poly) { static bool _clipPolygon(struct DSGX* gx, struct DSGXPolygon* poly) {