X-Git-Url: http://git.mutantstargoat.com/user/nuclear/?a=blobdiff_plain;f=src%2F3dgfx.c;h=a90da7adfe042af9d3251c1604b4722bc4a4579c;hb=0165ec15f868a16a70b56ada2d42db0cb69ea193;hp=d7e2e41704067282c433d3ddbd467c9bc82ea27c;hpb=ee4a19d1405ea1ee7fbbeef1cdf71292d21f353a;p=dosdemo diff --git a/src/3dgfx.c b/src/3dgfx.c index d7e2e41..a90da7a 100644 --- a/src/3dgfx.c +++ b/src/3dgfx.c @@ -5,6 +5,7 @@ #include #include "3dgfx.h" #include "polyfill.h" +#include "polyclip.h" #include "inttypes.h" #include "util.h" @@ -42,6 +43,8 @@ struct g3d_state { int width, height; void *pixels; + + int vport[4]; }; static void xform4_vec3(const float *mat, float *vec); @@ -91,9 +94,19 @@ void g3d_framebuffer(int width, int height, void *pixels) st->height = height; st->pixels = pixels; - pimg_fb.pixels = pixels; - pimg_fb.width = width; - pimg_fb.height = height; + pfill_fb.pixels = pixels; + pfill_fb.width = width; + pfill_fb.height = height; + + g3d_viewport(0, 0, width, height); +} + +void g3d_viewport(int x, int y, int w, int h) +{ + st->vport[0] = x; + st->vport[1] = y; + st->vport[2] = w; + st->vport[3] = h; } void g3d_enable(unsigned int opt) @@ -340,6 +353,33 @@ void g3d_mtl_shininess(float shin) st->mtl.shin = shin; } +static INLINE int calc_shift(unsigned int x) +{ + int res = -1; + while(x) { + x >>= 1; + ++res; + } + return res; +} + +static INLINE int calc_mask(unsigned int x) +{ + return x - 1; +} + +void g3d_set_texture(int xsz, int ysz, void *pixels) +{ + pfill_tex.pixels = pixels; + pfill_tex.width = xsz; + pfill_tex.height = ysz; + + pfill_tex.xshift = calc_shift(xsz); + pfill_tex.yshift = calc_shift(ysz); + pfill_tex.xmask = calc_mask(xsz); + pfill_tex.ymask = calc_mask(ysz); +} + void g3d_draw(int prim, const struct g3d_vertex *varr, int varr_size) { g3d_draw_indexed(prim, varr, varr_size, 0, 0); @@ -349,8 +389,8 @@ void g3d_draw_indexed(int prim, const struct g3d_vertex *varr, int varr_size, const int16_t *iarr, int iarr_size) { int i, j, nfaces; - struct pvertex pv[4]; - struct g3d_vertex v[4]; + struct pvertex pv[16]; + struct g3d_vertex v[16]; int vnum = prim; /* primitive vertex counts correspond to enum values */ int mvtop = st->mtop[G3D_MODELVIEW]; int ptop = st->mtop[G3D_PROJECTION]; @@ -362,6 +402,7 @@ void g3d_draw_indexed(int prim, const struct g3d_vertex *varr, int varr_size, nfaces = (iarr ? iarr_size : varr_size) / vnum; for(j=0; jmat[G3D_PROJECTION][ptop], &v[i].x); } - /* TODO clipping */ + /* clipping */ + for(i=0; i<6; i++) { + struct g3d_vertex tmpv[16]; + memcpy(tmpv, v, vnum * sizeof *v); + + if(clip_frustum(v, &vnum, tmpv, vnum, i) < 0) { + /* polygon completely outside of view volume. discard */ + vnum = 0; + break; + } + } + + if(!vnum) continue; for(i=0; iwidth; - v[i].y = (0.5f - v[i].y * 0.5f) * (float)st->height; + v[i].x = (v[i].x * 0.5f + 0.5f) * (float)st->vport[2] + st->vport[0]; + v[i].y = (0.5f - v[i].y * 0.5f) * (float)st->vport[3] + st->vport[1]; /* convert pos to 24.8 fixed point */ pv[i].x = cround64(v[i].x * 256.0f);