/* * Descent 3 * Copyright (C) 2024 Parallax Software * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * 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, see . */ #include "3d.h" #include "texture.h" #include "gametexture.h" #include "erooms.h" #include "editor_lighting.h" #include "descent.h" #include "room.h" #include "lightmap.h" #include "polymodel.h" #include "objinfo.h" #include #include #include "terrain.h" #include "radiosity.h" #include "lighting.h" #include "findintersection.h" #include "lightmap_info.h" #include "object_lighting.h" #include "mem.h" void ComputeObjectSurfaceRes (rad_surface *surf,object *obj,int subnum,int facenum) { int i; float left=1.1f,right=-1,top=1.1f,bottom=-1; lightmap_object_face *lfp=&obj->lm_object.lightmap_faces[subnum][facenum]; int lw=lmi_w(lfp->lmi_handle); int lh=lmi_h(lfp->lmi_handle); for (i=0;inum_verts;i++) { if (lfp->u2[i]u2[i]; if (lfp->u2[i]>right) right=lfp->u2[i]; if (lfp->v2[i]v2[i]; if (lfp->v2[i]>bottom) bottom=lfp->v2[i]; } float left_result=(left*lw)+.0001; float right_result=(right*lw)+.0001; float top_result=(top*lh)+.0001; float bottom_result=(bottom*lh)+.0001; surf->x1=floor (left_result); surf->x2=floor (right_result); surf->y1=floor(top_result); surf->y2=floor(bottom_result); surf->xresolution=(surf->x2-surf->x1); surf->yresolution=(surf->y2-surf->y1); // Adjust for a accuracy errors if ( ( (right_result)-(float)surf->x2)>.005) surf->xresolution++; if ( ( (bottom_result)-(float)surf->y2)>.005) surf->yresolution++; if ( ( (top_result)-(float)surf->y1)>.99) surf->y1++; if ( ( (left_result)-(float)surf->x1)>.99) surf->x1++; } void ApplyLightmapToObjectSurface (object *obj,int subnum,int facenum,rad_surface *sp) { lightmap_object_face *fp=&obj->lm_object.lightmap_faces[subnum][facenum]; int i,t,lmi_handle; int xres,yres; int lw,lh; int x1=sp->x1; int y1=sp->y1; xres=sp->xresolution; yres=sp->yresolution; ASSERT (fp->lmi_handle!=BAD_LMI_INDEX); lmi_handle=fp->lmi_handle; lw=lmi_w(lmi_handle); lh=lmi_h(lmi_handle); ASSERT ((xres+x1)<=lw); ASSERT ((yres+y1)<=lh); ASSERT (lw>=2); ASSERT (lh>=2); ushort *dest_data=lm_data (LightmapInfo[lmi_handle].lm_handle); for (i=0;ielements[i*xres+t].flags & EF_IGNORE)) { ddgr_color color=GR_16_TO_COLOR(dest_data[(i+y1)*lw+(t+x1)]); int red=GR_COLOR_RED(color); int green=GR_COLOR_GREEN(color); int blue=GR_COLOR_BLUE(color); float fr,fg,fb; if (!(dest_data[(i+y1)*lw+(t+x1)] & OPAQUE_FLAG)) { red=green=blue=0; } fr=min(1.0,sp->elements[i*xres+t].exitance.r+Ambient_red); fg=min(1.0,sp->elements[i*xres+t].exitance.g+Ambient_green); fb=min(1.0,sp->elements[i*xres+t].exitance.b+Ambient_blue); fr=(fr*255)+.5; fg=(fg*255)+.5; fb=(fb*255)+.5; red+=(int)fr; green+=(int)fg; blue+=(int)fb; if (dest_data[(i+y1)*lw+(t+x1)] & OPAQUE_FLAG) { red/=2; green/=2; blue/=2; } red=min(red,255); green=min(green,255); blue=min(blue,255); dest_data[(i+y1)*lw+(t+x1)]=OPAQUE_FLAG|GR_RGB16(red,green,blue); } } } } void GetPointInObjectSpace (vector *dest,vector *pos,object *obj,int subnum,int world) { poly_model *pm=&Poly_models[obj->rtype.pobj_info.model_num]; bsp_info *sm=&pm->submodel[subnum]; float normalized_time[MAX_SUBOBJECTS]; int i; int rotate_list[MAX_SUBOBJECTS]; int num_to_rotate=0; if (!pm->new_style) return; for (i=0;isubmodel[mn].parent; } // Subtract and rotate position if (world) tpnt = pnt - obj->pos; else tpnt=pnt; pnt = tpnt * obj->orient; for (i=num_to_rotate-1;i>=0;i--) { // Subtract and rotate position for this submodel mn=rotate_list[i]; if (world) tpnt = pnt - pm->submodel[mn].offset; else tpnt=pnt; vm_AnglesToMatrix(&m, pm->submodel[mn].angs.p,pm->submodel[mn].angs.h, pm->submodel[mn].angs.b); pnt = tpnt * m; } *dest=pnt; } // Goes through all objects and fills in the lightmap data for them void AssignLightmapsToObjectSurfaces (int surface_index,int terrain) { int i,t,j; uint8_t rotated[MAX_LIGHTMAP_INFOS]; memset (rotated,0,MAX_LIGHTMAP_INFOS); for (i=0;i<=Highest_object_index;i++) { if ((terrain != 0) != (OBJECT_OUTSIDE(&Objects[i]) != 0)) continue; if (Objects[i].type!=OBJ_NONE && Objects[i].lighting_render_type==LRT_LIGHTMAPS) { poly_model *po=&Poly_models[Objects[i].rtype.pobj_info.model_num]; if (!po->new_style) continue; for (t=0;tn_models;t++) { bsp_info *sm=&po->submodel[t]; if (IsNonRenderableSubmodel (po,t)) continue; for (j=0;jnum_faces;j++,surface_index++) { ApplyLightmapToObjectSurface (&Objects[i],t,j,&Light_surfaces[surface_index]); // Rotate the lightmap upper left object *obj=&Objects[i]; lightmap_object_face *fp=&obj->lm_object.lightmap_faces[t][j]; lightmap_info *lmi_ptr=&LightmapInfo[fp->lmi_handle]; if (!rotated[fp->lmi_handle]) { vector uleft,rvec,uvec,norm; GetPointInObjectSpace (&uleft,&lmi_ptr->upper_left,obj,t,1); GetPointInObjectSpace (&norm,&lmi_ptr->normal,obj,t,0); lmi_ptr->normal=norm; lmi_ptr->upper_left=uleft; GetPointInObjectSpace (&rvec,&ScratchRVecs[fp->lmi_handle],obj,t,0); GetPointInObjectSpace (&uvec,&ScratchUVecs[fp->lmi_handle],obj,t,0); rotated[fp->lmi_handle]=1; // Find all the faces in this submodel that have this lightmap info handle for (int k=0;knum_faces;k++) { lightmap_object_face *this_fp=&obj->lm_object.lightmap_faces[t][k]; if (fp->lmi_handle==this_fp->lmi_handle) { this_fp->rvec=rvec; this_fp->uvec=uvec; } } } } } } } } // Goes through all objects int a room and fills in the lightmap data for them void AssignLightmapsToObjectSurfacesForSingleRoom (int surface_index,int roomnum) { int i,t,j; for (i=0;i<=Highest_object_index;i++) { if (Objects[i].type!=OBJ_NONE && Objects[i].lighting_render_type==LRT_LIGHTMAPS && Objects[i].roomnum==roomnum) { poly_model *po=&Poly_models[Objects[i].rtype.pobj_info.model_num]; if (!po->new_style) continue; for (t=0;tn_models;t++) { bsp_info *sm=&po->submodel[t]; if (IsNonRenderableSubmodel (po,t)) continue; for (j=0;jnum_faces;j++,surface_index++) ApplyLightmapToObjectSurface (&Objects[i],t,j,&Light_surfaces[surface_index]); } } } } // Sets up radiosity surfaces for objects in the mine // Returns the number of new surfaces int ComputeSurfacesForObjects (int surface_index,int terrain) { int i,t,j; for (i=0;i<=Highest_object_index;i++) { if ((terrain != 0) != (OBJECT_OUTSIDE(&Objects[i]) != 0)) continue; if (Objects[i].type!=OBJ_NONE && Objects[i].lighting_render_type==LRT_LIGHTMAPS) { poly_model *po=&Poly_models[Objects[i].rtype.pobj_info.model_num]; if (!po->new_style) continue; SetupObjectLightmapMemory (&Objects[i]); if (terrain) CombineObjectLightmapUVs (&Objects[i],LMI_TERRAIN_OBJECT); else CombineObjectLightmapUVs (&Objects[i],LMI_ROOM_OBJECT); for (t=0;tn_models;t++) { bsp_info *sm=&po->submodel[t]; if (IsNonRenderableSubmodel (po,t)) continue; for (j=0;jnum_faces;j++,surface_index++) { ComputeObjectSurfaceRes (&Light_surfaces[surface_index],&Objects[i],t,j); if (sm->faces[j].nverts>0) { Light_surfaces[surface_index].verts=(vector *)mem_malloc (sm->faces[j].nverts*sizeof(vector)); ASSERT (Light_surfaces[surface_index].verts!=NULL); } else Light_surfaces[surface_index].verts=NULL; if (Light_surfaces[surface_index].xresolution*Light_surfaces[surface_index].yresolution>0) { Light_surfaces[surface_index].elements=(rad_element *)mem_malloc (Light_surfaces[surface_index].xresolution*Light_surfaces[surface_index].yresolution*sizeof(rad_element)); ASSERT (Light_surfaces[surface_index].elements!=NULL); } else Light_surfaces[surface_index].elements=NULL; Light_surfaces[surface_index].flags=0; if (sm->faces[j].texnum==-1) { Light_surfaces[surface_index].emittance.r=0; Light_surfaces[surface_index].emittance.g=0; Light_surfaces[surface_index].emittance.b=0; Light_surfaces[surface_index].reflectivity=.5; } else { Light_surfaces[surface_index].emittance.r=(float)GameTextures[po->textures[sm->faces[j].texnum]].r; Light_surfaces[surface_index].emittance.g=(float)GameTextures[po->textures[sm->faces[j].texnum]].g; Light_surfaces[surface_index].emittance.b=(float)GameTextures[po->textures[sm->faces[j].texnum]].b; Light_surfaces[surface_index].reflectivity=GameTextures[po->textures[sm->faces[j].texnum]].reflectivity; if ((GetMaxColor (&Light_surfaces[surface_index].emittance))>.005) Light_surfaces[surface_index].flags|=SF_LIGHTSOURCE; } if (terrain) Light_surfaces[surface_index].surface_type=ST_TERRAIN_OBJECT; else Light_surfaces[surface_index].surface_type=ST_ROOM_OBJECT; Light_surfaces[surface_index].normal=LightmapInfo[Objects[i].lm_object.lightmap_faces[t][j].lmi_handle].normal; Light_surfaces[surface_index].roomnum=Objects[i].roomnum; if (Light_surfaces[surface_index].surface_type==ST_ROOM_OBJECT) { if (Rooms[Objects[i].roomnum].flags & RF_TOUCHES_TERRAIN) Light_surfaces[surface_index].flags|=SF_TOUCHES_TERRAIN; for (int k=0;knew_style) continue; SetupObjectLightmapMemory (&Objects[i]); CombineObjectLightmapUVs (&Objects[i],LMI_ROOM_OBJECT); for (t=0;tn_models;t++) { bsp_info *sm=&po->submodel[t]; if (IsNonRenderableSubmodel (po,t)) continue; for (j=0;jnum_faces;j++,surface_index++) { ComputeObjectSurfaceRes (&Light_surfaces[surface_index],&Objects[i],t,j); if (sm->faces[j].nverts>0) { Light_surfaces[surface_index].verts=(vector *)mem_malloc (sm->faces[j].nverts*sizeof(vector)); ASSERT (Light_surfaces[surface_index].verts!=NULL); } else Light_surfaces[surface_index].verts=NULL; if (Light_surfaces[surface_index].xresolution*Light_surfaces[surface_index].yresolution>0) { Light_surfaces[surface_index].elements=(rad_element *)mem_malloc (Light_surfaces[surface_index].xresolution*Light_surfaces[surface_index].yresolution*sizeof(rad_element)); ASSERT (Light_surfaces[surface_index].elements!=NULL); } else Light_surfaces[surface_index].elements=NULL; if (sm->faces[j].texnum==-1) { Light_surfaces[surface_index].emittance.r=0; Light_surfaces[surface_index].emittance.g=0; Light_surfaces[surface_index].emittance.b=0; Light_surfaces[surface_index].reflectivity=.5; } else { Light_surfaces[surface_index].emittance.r=(float)GameTextures[po->textures[sm->faces[j].texnum]].r; Light_surfaces[surface_index].emittance.g=(float)GameTextures[po->textures[sm->faces[j].texnum]].g; Light_surfaces[surface_index].emittance.b=(float)GameTextures[po->textures[sm->faces[j].texnum]].b; Light_surfaces[surface_index].reflectivity=GameTextures[po->textures[sm->faces[j].texnum]].reflectivity; } Light_surfaces[surface_index].surface_type=ST_ROOM_OBJECT; Light_surfaces[surface_index].normal=LightmapInfo[Objects[i].lm_object.lightmap_faces[t][j].lmi_handle].normal; Light_surfaces[surface_index].roomnum=Objects[i].roomnum; // Set the vertices for each element BuildElementListForObjectFace (i,t,j,&Light_surfaces[surface_index]); } } } } return 0; } // Gets the total number of object faces that exist in a mine int GetTotalObjectFaces (int terrain) { int i; int facecount=0; for (i=0;i<=Highest_object_index;i++) { if (Objects[i].type!=OBJ_NONE) { if ((terrain != 0) != (OBJECT_OUTSIDE(&Objects[i]) != 0)) continue; if (Objects[i].lighting_render_type==LRT_LIGHTMAPS) { poly_model *po=&Poly_models[Objects[i].rtype.pobj_info.model_num]; if (!po->new_style) continue; facecount+=CountFacesInPolymodel (po); } } } return facecount; } // Gets the total number of object faces that exist in a room int GetTotalObjectFacesForSingleRoom (int roomnum) { int i; int facecount=0; for (i=0;i<=Highest_object_index;i++) { if (Objects[i].type!=OBJ_NONE) { if (Objects[i].roomnum!=roomnum) continue; if (Objects[i].lighting_render_type==LRT_LIGHTMAPS) { poly_model *po=&Poly_models[Objects[i].rtype.pobj_info.model_num]; if (!po->new_style) continue; facecount+=CountFacesInPolymodel (po); } } } return facecount; } void BuildObjectLightmapUVs (object *obj,int *sublist,int *facelist,int count,vector *lightmap_poly,int nv,int lm_type) { matrix face_matrix,trans_matrix; vector fvec; vector avg_vert; vector verts[MAX_VERTS_PER_FACE*5]; vector facevert; vector rot_vert; int i,t; int lmi_handle; vector world_verts[32]; poly_model *pm=&Poly_models[obj->rtype.pobj_info.model_num]; for (i=0;isubmodel[sublist[0]].faces[facelist[0]].nverts;i++) GetObjectPointInWorld (&world_verts[i],obj,sublist[0],pm->submodel[sublist[0]].faces[facelist[0]].vertnums[i]); // find the center point of this face vm_MakeZero (&avg_vert); for (i=0;itopmost_y) { topmost_point=i; topmost_y=verts[i].y; } } ASSERT (topmost_point!=-1); // Find right most point int rightmost_point=-1; float rightmost_x=-900000.00f; // a big number for (i=0;irightmost_x) { rightmost_point=i; rightmost_x=verts[i].x; } } ASSERT (rightmost_point!=-1); // Find bottom most point int bottommost_point=-1; float bottommost_y=900000.0f; // a big number for (i=0;i0) res++; res++; if (res>126) xspacing+=1; else done_spacing=1; } // Set a mininum, at least if (res<2) res=2; lightmap_x_res=res; done_spacing=0; while (!done_spacing) { res=(ydiff/yspacing); if (((ydiff/yspacing)-res)>0) res++; res++; if (res>126) yspacing+=1; else done_spacing=1; } // Set a mininum, at least if (res<2) res=2; lightmap_y_res=res; /* // Find power of 2 number for (i=0;i<=7;i++) { int low_num=1low_num) { lightmap_res=hi_num; break; } }*/ lmi_handle=AllocLightmapInfo (lightmap_x_res,lightmap_y_res,lm_type); ASSERT (lmi_handle!=BAD_LMI_INDEX); // Now do best fit spacing if (BestFit) { xspace_int=(xdiff/lightmap_x_res); if ((xdiff-(lightmap_x_res*xspace_int))>0) xspace_int++; yspace_int=(ydiff/lightmap_y_res); if ((ydiff-(lightmap_y_res*yspace_int))>0) yspace_int++; } else { xspace_int=xspacing; yspace_int=yspacing; } // Figure out lightmap uvs // Rotate all the face points for (i=0;ilm_object.lightmap_faces[sublist[i]][facelist[i]].lmi_handle=lmi_handle; bsp_info *sm=&pm->submodel[sublist[i]]; polyface *fp=&sm->faces[facelist[i]]; lightmap_object_face *lfp=&obj->lm_object.lightmap_faces[sublist[i]][facelist[i]]; for (t=0;tnverts;t++) GetObjectPointInWorld (&world_verts[t],obj,sublist[i],fp->vertnums[t]); for (t=0;tnverts;t++) { vector vert=world_verts[t]; vert-=avg_vert; vm_MatrixMulVector (&rot_vert,&vert,&trans_matrix); facevert=rot_vert; // Find uv2s for this vertex lfp->u2[t]=(facevert.x-verts[leftmost_point].x)/(float)(lightmap_x_res*xspace_int); lfp->v2[t]=fabs((verts[topmost_point].y-facevert.y))/(float)(lightmap_y_res*yspace_int); ASSERT (lfp->u2[t]>=0 && lfp->u2[t]<=1.0); ASSERT (lfp->v2[t]>=0 && lfp->v2[t]<=1.0); } } // Find upper left corner vm_TransposeMatrix (&trans_matrix); vm_MatrixMulVector (&rot_vert,&base_vector,&trans_matrix); LightmapInfo[lmi_handle].upper_left=rot_vert+avg_vert; LightmapInfo[lmi_handle].xspacing=xspace_int; LightmapInfo[lmi_handle].yspacing=yspace_int; LightmapInfo[lmi_handle].normal=-fvec; ScratchCenters[lmi_handle]=avg_vert; } // Important - vertnum is the index into the face_verts[] array in the face structure, // not an index into the verts[] array of the room structure void BuildElementListForObjectFace (int objnum,int subnum,int facenum,rad_surface *surf) { matrix face_matrix,trans_matrix; vector fvec; vector avg_vert; vector verts[MAX_VERTS_PER_FACE*5]; vector rot_vert; vector vert; vector world_verts[32]; int i,t; int xres,yres; int lmi_handle; int x1=surf->x1,y1=surf->y1; poly_model *pm=&Poly_models[Objects[objnum].rtype.pobj_info.model_num]; bsp_info *sm=&pm->submodel[subnum]; polyface *fp=&sm->faces[facenum]; xres=surf->xresolution; yres=surf->yresolution; ASSERT (pm->used); ASSERT (fp->nverts>=3); ASSERT (Objects[objnum].lm_object.lightmap_faces[subnum][facenum].lmi_handle!=BAD_LMI_INDEX); ASSERT (fp->nverts<32); for (i=0;inverts;i++) GetObjectPointInWorld (&world_verts[i],&Objects[objnum],subnum,fp->vertnums[i]); lmi_handle=Objects[objnum].lm_object.lightmap_faces[subnum][facenum].lmi_handle; avg_vert=ScratchCenters[lmi_handle]; // Make the orientation matrix // Reverse the normal because we're looking "at" the face, not from it fvec=-LightmapInfo[lmi_handle].normal; if ((vm_NormalizeVector(&fvec))!=0) vm_VectorToMatrix(&face_matrix,&fvec,NULL,NULL); else vm_MakeIdentity (&face_matrix); ScratchRVecs[lmi_handle]=face_matrix.rvec; ScratchUVecs[lmi_handle]=face_matrix.uvec; // Make the transformation matrix angvec avec; vm_ExtractAnglesFromMatrix(&avec,&face_matrix); vm_AnglesToMatrix (&trans_matrix,avec.p,avec.h,avec.b); // Rotate all the points for (i=0;inverts;i++) { vert=world_verts[i]; vert-=avg_vert; vm_MatrixMulVector (&rot_vert,&vert,&trans_matrix); verts[i]=rot_vert; } // Find a base vector vector base_vector; vector xdiff,ydiff; vm_MakeZero (&xdiff); vm_MakeZero (&ydiff); // Rotate our upper left point into our 2d space vert=LightmapInfo[lmi_handle].upper_left-avg_vert; vm_MatrixMulVector (&base_vector,&vert,&trans_matrix); vm_TransposeMatrix (&trans_matrix); xdiff.x=LightmapInfo[lmi_handle].xspacing; ydiff.y=LightmapInfo[lmi_handle].yspacing; for (i=0;ielements[element_index]; clip_verts[0]=base_vector+(xdiff*(t+x1))-(ydiff*(i+y1)); clip_verts[1]=base_vector+(xdiff*(t+x1+1))-(ydiff*(i+y1)); clip_verts[2]=base_vector+(xdiff*(t+x1+1))-(ydiff*(i+y1+1)); clip_verts[3]=base_vector+(xdiff*(t+x1))-(ydiff*(i+y1+1)); ClipSurfaceElement (verts,ep,clip_verts,fp->nverts); for (int k=0;knum_verts;k++) { vm_MatrixMulVector (&rot_vert,&ep->verts[k],&trans_matrix); ep->verts[k]=rot_vert+avg_vert; } } } if (Square_surfaces) { surf->verts[0]=base_vector; surf->verts[1]=base_vector+(xdiff*xres); surf->verts[2]=base_vector+(xdiff*xres)-(ydiff*yres); surf->verts[3]=base_vector-(ydiff*yres); for (int k=0;k<4;k++) { vm_MatrixMulVector (&rot_vert,&surf->verts[k],&trans_matrix); surf->verts[k]=rot_vert+avg_vert; } surf->num_verts=4; } else { surf->num_verts=fp->nverts; for (int k=0;knum_verts;k++) { surf->verts[k]=world_verts[k]; } } } #define MAX_COMBINES 50 #define LM_ADJACENT_FACE_THRESHOLD .95 uint8_t *ObjectsAlreadyCombined[MAX_OBJECTS]; // Given a submodel and a face, goes through the entire object and checks to see // if this face can share a lightmap with any other face int TestObjectLightAdjacency (object *obj,int subnum,int facenum,int lmi_type) { int i,t,k; poly_model *pm=&Poly_models[obj->rtype.pobj_info.model_num]; bsp_info *a_sm=&pm->submodel[subnum]; polyface *afp=&a_sm->faces[facenum]; vector anormal; vector averts[MAX_VERTS_PER_FACE*5]; vector bverts[MAX_VERTS_PER_FACE*5]; vector dest_verts[MAX_VERTS_PER_FACE*5]; int face_combine_list[MAX_COMBINES]; int submodel_combine_list[MAX_COMBINES]; if (afp->texnum==-1) return 0; int tex=pm->textures[afp->texnum]; if (GameTextures[tex].r>0 || GameTextures[tex].g>0 || GameTextures[tex].b>0) return 0; // Setup our 'base' face int anv=afp->nverts; int total_faces=1; submodel_combine_list[0]=subnum; face_combine_list[0]=facenum; for (i=0;inverts;i++) GetObjectPointInWorld (&averts[i],obj,subnum,afp->vertnums[i]); vm_GetNormal (&anormal,&averts[0],&averts[1],&averts[2]); StartOver: // Go through each room and find an adjacent face for (i=0;in_models;i++) { bsp_info *bsm=&pm->submodel[i]; if (IsNonRenderableSubmodel (pm,i)) continue; if (bsm!=a_sm) // only combine faces in the same submodel continue; for (t=0;tnum_faces;t++) { if (total_faces>=MAX_COMBINES-1) continue; if (bsm==a_sm && t==facenum) continue; // don't do self // Don't do if already spoken fore if (ObjectsAlreadyCombined[i][t]) continue; polyface *bfp=&bsm->faces[t]; vector bnormal; // Don't do combine light sources tex=pm->textures[bfp->texnum]; if (GameTextures[tex].r>0 || GameTextures[tex].g>0 || GameTextures[tex].b>0) continue; for (k=0;knverts;k++) GetObjectPointInWorld (&bverts[k],obj,i,bfp->vertnums[k]); vm_GetNormal (&bnormal,&bverts[0],&bverts[1],&bverts[2]); int nv=CombineLightFaces (dest_verts,averts,anv,&anormal,bverts,bfp->nverts,&bnormal); // We have a combine! Mark this face in the appropriate list // And update our new polygon if (nv>0) { submodel_combine_list[total_faces]=i; face_combine_list[total_faces]=t; total_faces++; ObjectsAlreadyCombined[subnum][facenum]=1; ObjectsAlreadyCombined[i][t]=1; for (k=0;k1) { BuildObjectLightmapUVs (obj,submodel_combine_list,face_combine_list,total_faces,averts,anv,lmi_type); } return 1; } // Computes the the mines UVs // Faces can now share one lightmap, so this routine goes through and tries to // combine as many faces as it can into one lightmap void CombineObjectLightmapUVs (object *obj,int lmi_type) { int i,t,k; int not_combined=0; poly_model *pm=&Poly_models[obj->rtype.pobj_info.model_num]; ASSERT (obj->lm_object.used); for (i=0;in_models;i++) { bsp_info *sm=&pm->submodel[i]; if (IsNonRenderableSubmodel (pm,i)) continue; ObjectsAlreadyCombined[i]=(uint8_t *)mem_malloc (sm->num_faces); ASSERT (ObjectsAlreadyCombined[i]); for (k=0;knum_faces;k++) ObjectsAlreadyCombined[i][k]=0; } for (i=0;in_models;i++) { bsp_info *sm=&pm->submodel[i]; if (IsNonRenderableSubmodel (pm,i)) continue; for (t=0;tnum_faces;t++) { if (*(ObjectsAlreadyCombined[i]+t)==0) TestObjectLightAdjacency (obj,i,t,lmi_type); } } // Now build lightmaps for any faces that couldn't be combined for (i=0;in_models;i++) { bsp_info *sm=&pm->submodel[i]; if (IsNonRenderableSubmodel (pm,i)) continue; for (t=0;tnum_faces;t++) { if (!ObjectsAlreadyCombined[i][t]) { vector verts[MAX_VERTS_PER_FACE*5]; int submodel_list[2],face_list[2]; for (k=0;kfaces[t].nverts;k++) { GetObjectPointInWorld (&verts[k],obj,i,sm->faces[t].vertnums[k]); } submodel_list[0]=i; face_list[0]=t; BuildObjectLightmapUVs (obj,submodel_list,face_list,1,verts,sm->faces[t].nverts,lmi_type); not_combined++; } } } mprintf ((0,"%d %s faces couldn't be combined!\n",not_combined,pm->name)); // Free memory for (i=0;in_models;i++) { bsp_info *sm=&pm->submodel[i]; if (IsNonRenderableSubmodel (pm,i)) continue; mem_free (ObjectsAlreadyCombined[i]); } }