/* vi: set ts=2: * * * Eressea PB(E)M host Copyright (C) 1998-2000 * Christian Schlittchen (corwin@amber.kn-bremen.de) * Katja Zedel (katze@felidae.kn-bremen.de) * Henning Peters (faroul@beyond.kn-bremen.de) * Enno Rehling (enno@eressea-pbem.de) * Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de) * * This program may not be used, modified or distributed without * prior permission by the authors of Eressea. */ #include #include "eressea.h" #include "building.h" /* kernel includes */ #include "item.h" #include "curse.h" /* für C_NOCOST */ #include "unit.h" #include "region.h" #include "skill.h" #include "save.h" /* util includes */ #include #include #include #include #include #include /* libc includes */ #include #include #include /* attributes includes */ #include building_typelist *buildingtypes; const building_type * bt_find(const char* name) { const struct building_typelist * btl = buildingtypes; if (global.data_version < RELEASE_VERSION) { const char * translation[3][2] = { { "illusion", "illusioncastle" }, { "generic", "genericbuilding" }, { NULL, NULL } }; int i; for (i=0;translation[i][0];++i) { /* calling a building "illusion" was a bad idea" */ if (strcmp(translation[i][0], name)==0) { name = translation[i][1]; break; } } } while (btl && strcasecmp(btl->type->_name, name)) btl = btl->next; if (!btl) { btl = buildingtypes; while (btl && strncasecmp(btl->type->_name, name, strlen(name))) btl = btl->next; } assert(btl); return btl?btl->type:NULL; } void bt_register(building_type * type) { struct building_typelist * btl = malloc(sizeof(building_type)); if (type->init) type->init(type); btl->type = type; btl->next = buildingtypes; buildingtypes = btl; } int buildingcapacity(const building * b) { if (b->type->capacity>=0) { if (b->type->maxcapacity>=0) return min(b->type->maxcapacity, b->size * b->type->capacity); return b->size * b->type->capacity; } if (b->size>=b->type->maxsize) return b->type->maxcapacity; return 0; } attrib_type at_building_generic_type = { "building_generic_type", NULL, NULL, NULL, a_writestring, a_readstring, ATF_UNIQUE }; const char * buildingtype(const building * b, int bsize) { const char * s = NULL; const building_type * btype = b->type; static const struct building_type * bt_generic; if (!bt_generic) bt_generic = bt_find("generic"); assert(bt_generic); if (btype == bt_generic) { const attrib *a = a_find(b->attribs, &at_building_generic_type); if (a) s = (const char*)a->data.v; } if (btype->name) s = btype->name(bsize); if (s==NULL) s = btype->_name; return s; } int buildingmaintenance(const building * b, resource_t rtype) { const building_type * bt = b->type; int c, cost=0; if (is_cursed(b->attribs, C_NOCOST, 0)) { return 0; } for (c=0;bt->maintenance && bt->maintenance[c].number;++c) { const maintenance * m = bt->maintenance + c; if (m->type==rtype) { if (fval(m, MTF_VARIABLE)) cost += (b->size * m->number); else cost += m->number; } } return cost; } #define BMAXHASH 8191 static building *buildhash[BMAXHASH]; void bhash(building * b) { building *old = buildhash[b->no % BMAXHASH]; buildhash[b->no % BMAXHASH] = b; b->nexthash = old; } void bunhash(building * b) { building **show; for (show = &buildhash[b->no % BMAXHASH]; *show; show = &(*show)->nexthash) { if ((*show)->no == b->no) break; } if (*show) { assert(*show == b); *show = (*show)->nexthash; b->nexthash = 0; } } static building * bfindhash(int i) { building *old; for (old = buildhash[i % BMAXHASH]; old; old = old->nexthash) if (old->no == i) return old; return 0; } building * findbuilding(int i) { return bfindhash(i); } /* ** old building types ** */ /** Building: Fortification */ enum { B_SITE, #if LARGE_CASTLES B_TRADEPOST, #endif B_FORTIFICATION, B_TOWER, B_CASTLE, B_FORTRESS, B_CITADEL, MAXBUILDINGS }; static int sm_smithy(const unit * u, const region * r, skill_t sk, int value) /* skillmod */ { if (sk==SK_WEAPONSMITH || sk==SK_ARMORER) { if (u->region == r) return value + 1; } return value; } static int mm_smithy(const unit * u, const resource_type * rtype, int value) /* material-mod */ { if (rtype == oldresourcetype[R_IRON]) return value * 2; return value; } static void init_smithy(struct building_type * bt) { a_add(&bt->attribs, make_skillmod(NOSKILL, SMF_PRODUCTION, sm_smithy, 0, 0)); a_add(&bt->attribs, make_matmod(mm_smithy)); } static const char * castle_name(int bsize) { const char * fname[MAXBUILDINGS] = { "site", #if LARGE_CASTLES "tradepost", #endif "fortification", "tower", "castle", "fortress", "citadel" }; const construction * ctype; static const struct building_type * bt_castle; int i = 0; if (!bt_castle) bt_castle = bt_find("castle"); assert(bt_castle); ctype = bt_castle->construction; while (ctype && ctype->maxsize != -1 && ctype->maxsize<=bsize) { bsize-=ctype->maxsize; ctype=ctype->improvement; ++i; } return fname[i]; } static requirement castle_req[] = { { R_STONE, 1, 0.5 }, { NORESOURCE, 0, 0.0 }, }; #if LARGE_CASTLES static construction castle_bld[MAXBUILDINGS] = { { SK_BUILDING, 1, 2, 1, castle_req, &castle_bld[1] }, { SK_BUILDING, 1, 8, 1, castle_req, &castle_bld[2] }, { SK_BUILDING, 2, 40, 1, castle_req, &castle_bld[3] }, { SK_BUILDING, 3, 200, 1, castle_req, &castle_bld[4] }, { SK_BUILDING, 4, 1000, 1, castle_req, &castle_bld[5] }, { SK_BUILDING, 5, 5000, 1, castle_req, &castle_bld[6] }, { SK_BUILDING, 6, -1, 1, castle_req, NULL } }; #else static construction castle_bld[MAXBUILDINGS] = { { SK_BUILDING, 1, 2, 1, castle_req, &castle_bld[1] }, { SK_BUILDING, 2, 8, 1, castle_req, &castle_bld[2] }, { SK_BUILDING, 3, 40, 1, castle_req, &castle_bld[3] }, { SK_BUILDING, 4, 200, 1, castle_req, &castle_bld[4] }, { SK_BUILDING, 5, 1000, 1, castle_req, &castle_bld[5] }, { SK_BUILDING, 6, -1, 1, castle_req, NULL } }; #endif building_type bt_castle = { "castle", BFL_NONE, 1, 4, -1, 0, 0, 0, 0.0, NULL, &castle_bld[0], castle_name }; /* for finding out what was meant by a particular building string */ static local_names * bnames; const building_type * findbuildingtype(const char * name, const locale * lang) { local_names * bn = bnames; void * i; while (bn) { if (bn->lang==lang) break; bn=bn->next; } if (!bn) { struct building_typelist * btl = buildingtypes; bn = calloc(sizeof(local_names), 1); bn->next = bnames; bn->lang = lang; while (btl) { const char * n = locale_string(lang, btl->type->_name); addtoken(&bn->names, n, (void*)btl->type); btl=btl->next; } bnames = bn; } if (findtoken(&bn->names, name, &i)==E_TOK_NOMATCH) return NULL; return (const building_type*)i; } static int tagend(struct xml_stack * stack) { const xml_tag * tag = stack->tag; if (strcmp(tag->name, "building")==0) { bt_register((building_type*)stack->state); stack->state = 0; } return XML_OK; } static int tagbegin(struct xml_stack * stack) { building_type * bt = (building_type *)stack->state; const xml_tag * tag = stack->tag; if (strcmp(tag->name, "building")==0) { const char * name = xml_value(tag, "name"); if (name!=NULL) { const char * x; bt = stack->state = calloc(sizeof(building_type), 1); bt->_name = strdup(name); bt->magres = xml_ivalue(tag, "magres"); bt->magresbonus = xml_ivalue(tag, "magresbonus"); bt->fumblebonus = xml_ivalue(tag, "fumblebonus"); bt->auraregen = xml_fvalue(tag, "auraregen"); if ((x = xml_value(tag, "capacity"))!=0) bt->capacity = atoi(x); else bt->capacity = -1; if ((x = xml_value(tag, "maxcapacity"))!=0) bt->maxcapacity = atoi(x); else bt->maxcapacity = -1; if ((x = xml_value(tag, "maxsize"))!=0) bt->maxsize = atoi(x); else bt->maxsize = -1; if (xml_bvalue(tag, "nodestroy")) bt->flags |= BTF_INDESTRUCTIBLE; if (xml_bvalue(tag, "nobuild")) bt->flags |= BTF_NOBUILD; if (xml_bvalue(tag, "unique")) bt->flags |= BTF_UNIQUE; if (xml_bvalue(tag, "decay")) bt->flags |= BTF_DECAY; if (xml_bvalue(tag, "magic")) bt->flags |= BTF_MAGIC; if (xml_bvalue(tag, "protection")) bt->flags |= BTF_PROTECTION; } } else if (bt!=NULL) { if (strcmp(tag->name, "construction")==0) { const char * x; construction * con = calloc(sizeof(construction), 1); if ((x=xml_value(tag, "maxsize"))!=0) con->maxsize = atoi(x); else con->maxsize = -1; if ((x=xml_value(tag, "minskill"))!=0) con->minskill = atoi(x); else con->minskill = -1; if ((x=xml_value(tag, "reqsize"))!=0) con->reqsize = atoi(x); else con->reqsize = -1; con->skill = sk_find(xml_value(tag, "skill")); bt->construction = con; } else if (strcmp(tag->name, "function")==0) { const char * name = xml_value(tag, "name"); const char * value = xml_value(tag, "value"); if (name && value) { pf_generic fun = get_function(value); if (fun==NULL) { log_error(("unknown function value '%s=%s' for building %s\n", name, value, bt->_name)); } else { if (strcmp(name, "name")==0) { bt->name = (const char * (*)(int size))fun; } else if (strcmp(name, "init")==0) { bt->init = (void (*)(struct building_type*))fun; } else { log_error(("unknown function type '%s=%s' for building %s\n", name, value, bt->_name)); } } } } else if (strcmp(tag->name, "maintenance")==0) { size_t len = 0; const resource_type * rtype = NULL; maintenance * mt = bt->maintenance; resource_t type = NORESOURCE; if (mt==NULL) { mt = bt->maintenance = calloc(sizeof(struct maintenance), 2); len = 0; } else { while (mt[len].number) ++len; mt = bt->maintenance = realloc(mt, sizeof(struct maintenance)*(len+2)); } mt[len+1].number = 0; mt[len].number = xml_ivalue(tag, "amount"); rtype = rt_find(xml_value(tag, "type")); for (type=0;type!=MAX_RESOURCES;++type) { if (oldresourcetype[type]==rtype) { mt[len].type = type; break; } } if (xml_bvalue(tag, "variable")) mt[len].flags |= MTF_VARIABLE; if (xml_bvalue(tag, "vital")) mt[len].flags |= MTF_VITAL; } else if (strcmp(tag->name, "requirement")==0) { xml_readrequirement(tag, bt->construction); } } return XML_OK; } static xml_callbacks xml_buildings = { tagbegin, tagend, NULL }; void register_buildings(void) { xml_register(&xml_buildings, "eressea building", 0); register_function((pf_generic)init_smithy, "init_smithy"); register_function((pf_generic)castle_name, "castle_name"); bt_register(&bt_castle); } void bt_write(FILE * F, const building_type * bt) { fprintf(F, "BUILDINGTYPE %s\n", bt->_name); a_write(F, bt->attribs); /* scheisse, weil nicht CR-Format */ fputs("\n", F); fprintf(F, "\"%s\";name\n", bt->_name); fprintf(F, "%d;flags\n", bt->flags); fprintf(F, "%d;capacity\n", bt->capacity); fprintf(F, "%d;maxcapacity\n", bt->maxcapacity); fprintf(F, "%d;maxsize\n", bt->maxsize); if (bt->maintenance!=NULL) assert(!"not implemented"); if (bt->construction!=NULL) assert(!"not implemented"); if (bt->construction!=NULL) assert(!"not implemented"); if (bt->name!=NULL) assert(!"not implemented"); fputs("END BUILDINGTYPE\n", F); } building_type * bt_read(FILE * F) /* this function is pretty picky */ { building_type * bt = calloc(sizeof(building_type), 1); int i = fscanf(F, "%s\n", buf); if (i==0 || i==EOF) { free(bt); return NULL; } bt->_name = strdup(buf); a_read(F, &bt->attribs); /* scheisse, weil nicht CR. */ for (;;) { char * semi = buf; fgets(buf, sizeof(buf), F); if (strlen(buf)==1) continue; buf[strlen(buf)-1]=0; for(;;) { char * s = strchr(semi, ';'); if (s==NULL) break; semi = s + 1; } if (semi==buf) { assert(!strcmp(buf, "END BUILDINGTYPE")); break; } *(semi-1)=0; if (buf[0]=='\"') { char * s = buf+1; assert(*(semi-2)=='\"'); *(semi-2)=0; if (!strcmp(semi, "name") && !bt->_name) bt->_name = strdup(s); } else { int j = atoi(buf); switch (semi[0]) { case 'c': if (!strcmp(semi, "capacity")) bt->capacity=j; break; case 'f': if (!strcmp(semi, "flags")) bt->flags=j; break; case 'm': if (!strcmp(semi, "maxcapacity")) bt->maxcapacity=j; else if (!strcmp(semi, "maxsize")) bt->maxsize=j; break; } } } bt_register(bt); return bt; } building_type * bt_make(const char * name, int flags, int capacity, int maxcapacity, int maxsize) { building_type * btype = calloc(sizeof(building_type), 1); btype->_name = name; btype->flags = flags | BTF_DYNAMIC; btype->capacity = capacity; btype->maxcapacity = maxcapacity; btype->maxsize = maxsize; bt_register(btype); return btype; } void * resolve_building(void * id) { return findbuilding((int)id); } void write_building_reference(const struct building * b, FILE * F) { fprintf(F, "%s ", b?itoa36(b->no):"0"); } int read_building_reference(struct building ** b, FILE * F) { int id; char zText[10]; fscanf(F, "%s ", zText); id = atoi36(zText); if (id==0) { *b = NULL; return AT_READ_FAIL; } else { *b = findbuilding(id); if (*b==NULL) ur_add((void*)id, (void**)b, resolve_building); return AT_READ_OK; } } building * new_building(const struct building_type * btype, region * r, const struct locale * lang) { building *b = (building *) calloc(1, sizeof(building)); b->no = newcontainerid(); bhash(b); b->type = btype; set_string(&b->display, ""); fset(b, FL_UNNAMED); b->region = r; addlist(&r->buildings, b); { static char buffer[IDSIZE + 1 + NAMESIZE + 1]; if (b->type->name) sprintf(buffer, "%s", locale_string(lang, btype->_name)); else sprintf(buffer, "%s", LOC(lang, buildingtype(b, 0))); set_string(&b->name, buffer); } return b; } void destroy_building(building * b) { unit *u; if(!bfindhash(b->no)) return; for(u=b->region->units; u; u=u->next) { if(u->building == b) leave(b->region, u); } b->size = 0; update_lighthouse(b); bunhash(b); #if 0 /* Memoryleak. Aber ohne klappt das Rendern nicht! */ removelist(&b->region->buildings, b); #endif /* Stattdessen nur aus Liste entfernen, aber im Speicher halten. */ choplist(&b->region->buildings, b); handle_event(&b->attribs, "destroy", b); } extern attrib_type at_icastle; int buildingeffsize(const building * b, boolean img) { int i = b->size, n = 0; const construction * cons; static const struct building_type * bt_castle; if (!bt_castle) bt_castle = bt_find("castle"); assert(bt_castle); if (b==NULL) return 0; if (b->type!=bt_castle) { if (img) { const attrib * a = a_find(b->attribs, &at_icastle); if (!a || a->data.v != bt_castle) return 0; } else return 0; } cons = bt_castle->construction; assert(cons); while (cons && cons->maxsize != -1 && i>=cons->maxsize) { i-=cons->maxsize; cons = cons->improvement; ++n; } return n; } unit * buildingowner(const region * r, const building * b) { unit *u = NULL; unit *first = NULL; #ifndef BROKEN_OWNERS assert(r == b->region); #endif /* Prüfen ob Eigentümer am leben. */ for (u = r->units; u; u = u->next) { if (u->building == b) { if (!first && u->number > 0) first = u; if (fval(u, FL_OWNER) && u->number > 0) return u; if (u->number == 0) freset(u, FL_OWNER); } } /* Eigentümer tot oder kein Eigentümer vorhanden. Erste lebende Einheit * nehmen. */ if (first) fset(first, FL_OWNER); return first; } #ifdef BETA_CODE void xml_writebuildings(void) { FILE * F = fopen("buildings.xml", "w"); building_typelist *btl= buildingtypes; while (btl) { int i; const building_type * bt = btl->type; fprintf(F, "_name); if (bt->capacity>=0) fprintf(F, " capacity=\"%d\"", bt->capacity); if (bt->maxcapacity>=0) fprintf(F, " maxcapacity=\"%d\"", bt->maxcapacity); if (bt->maxsize>=0) fprintf(F, " maxsize=\"%d\"", bt->maxsize); if (bt->flags & BTF_INDESTRUCTIBLE) fputs(" nodestroy", F); if (bt->flags & BTF_NOBUILD) fputs(" nobuild", F); if (bt->flags & BTF_UNIQUE) fputs(" unique", F); if (bt->flags & BTF_DECAY) fputs(" decay", F); if (bt->flags & BTF_PROTECTION) fputs(" protection", F); if (bt->flags & BTF_MAGIC) fputs(" magic", F); fputs(">\n", F); if (bt->name) { const char * name = get_functionname((pf_generic)bt->name); assert(name); fprintf(F, "\t\n", name); } if (bt->init) { const char * name = get_functionname((pf_generic)bt->init); assert(name); fprintf(F, "\t\n", name); } for (i=0;bt->maintenance && bt->maintenance[i].number;++i) { const maintenance * m = bt->maintenance + i; fprintf(F, "\ttype]->_name[0], m->number); if (m->flags & MTF_VARIABLE) fputs(" variable", F); if (m->flags & MTF_VITAL) fputs(" vital", F); fputs(">\n", F); } if (bt->construction) { fprintf(F, "\tconstruction->skill, NULL), bt->construction->minskill, bt->construction->reqsize); if (bt->construction->maxsize>=0) fprintf(F, " maxsize=\"%d\"", bt->construction->maxsize); fputs(">\n", F); for (i=0;bt->construction->materials[i].number!=0;++i) { fprintf(F, "\t\t\n", oldresourcetype[bt->construction->materials[i].type]->_name[0], bt->construction->materials[i].number); } fputs("\t\n", F); } fputs("\n\n", F); btl=btl->next; } fclose(F); } #endif