diff --git a/src/bind_unit.c b/src/bind_unit.c index ddd7b742f..4e3047367 100644 --- a/src/bind_unit.c +++ b/src/bind_unit.c @@ -62,7 +62,7 @@ static int tolua_bufunit(lua_State * L) { if (f) { char buf[8192]; int mode = (int)tolua_tonumber(L, 3, (int)seen_unit); - bufunit(f, u, mode, buf, sizeof(buf)); + bufunit_depr(f, u, mode, buf, sizeof(buf)); tolua_pushstring(L, buf); return 1; } @@ -199,7 +199,7 @@ static int tolua_unit_set_id(lua_State * L) static int tolua_unit_get_auramax(lua_State * L) { unit *self = (unit *)tolua_tousertype(L, 1, 0); - lua_pushinteger(L, max_spellpoints(self->region, self)); + lua_pushinteger(L, max_spellpoints_depr(self->region, self)); return 1; } diff --git a/src/bindings.c b/src/bindings.c index e14c13ea5..c46c0e80e 100755 --- a/src/bindings.c +++ b/src/bindings.c @@ -817,7 +817,7 @@ static int tolua_report_unit(lua_State * L) char buffer[512]; unit *u = (unit *)tolua_tousertype(L, 1, 0); faction *f = (faction *)tolua_tousertype(L, 2, 0); - bufunit(f, u, seen_unit, buffer, sizeof(buffer)); + bufunit_depr(f, u, seen_unit, buffer, sizeof(buffer)); tolua_pushstring(L, buffer); return 1; } diff --git a/src/creport.c b/src/creport.c index db3df6c7e..3ac9dffe9 100644 --- a/src/creport.c +++ b/src/creport.c @@ -917,7 +917,7 @@ void cr_output_unit(stream *out, const faction * f, } if (is_mage(u)) { stream_printf(out, "%d;Aura\n", get_spellpoints(u)); - stream_printf(out, "%d;Auramax\n", max_spellpoints(u->region, u)); + stream_printf(out, "%d;Auramax\n", max_spellpoints_depr(u->region, u)); } /* default commands */ stream_printf(out, "COMMANDS\n"); diff --git a/src/items/xerewards.c b/src/items/xerewards.c index 71746bbf3..d039a19a1 100644 --- a/src/items/xerewards.c +++ b/src/items/xerewards.c @@ -74,7 +74,7 @@ use_manacrystal(struct unit *u, const struct item_type *itype, int amount, return -1; } - msp = max_spellpoints(u->region, u) / 2; + msp = max_spellpoints_depr(u->region, u) / 2; for (i = 0; i != amount; ++i) { sp += MAX(25, msp); change_spellpoints(u, sp); diff --git a/src/kernel/pool.c b/src/kernel/pool.c index 52e445c94..70e02cb96 100644 --- a/src/kernel/pool.c +++ b/src/kernel/pool.c @@ -62,7 +62,7 @@ int get_resource(const unit * u, const resource_type * rtype) return get_spellpoints(u); } if (rtype == get_resourcetype(R_PERMAURA)) { - return max_spellpoints(u->region, u); + return max_spellpoints_depr(u->region, u); } log_error("trying to get unknown resource '%s'.\n", rtype->_name); return 0; diff --git a/src/laws.c b/src/laws.c index fdf763741..feacca8a1 100644 --- a/src/laws.c +++ b/src/laws.c @@ -326,7 +326,7 @@ static void calculate_emigration(region * r) static double peasant_growth_factor(void) { - return config_get_flt("rules.peasants.growth.factor", 0.0001F * PEASANTGROWTH); + return config_get_flt("rules.peasants.growth.factor", 0.0001 * (double)PEASANTGROWTH); } static double peasant_luck_factor(void) @@ -424,8 +424,10 @@ static migration *get_migrants(region * r) /* Es gibt noch keine Migration. Also eine erzeugen */ m = free_migrants; - if (!m) + if (!m) { m = calloc(1, sizeof(migration)); + if (!m) abort(); + } else { free_migrants = free_migrants->next; m->horses = 0; @@ -476,8 +478,8 @@ static void horses(region * r) } else if (maxhorses > 0) { double growth = - (RESOURCE_QUANTITY * HORSEGROWTH * 200 * (maxhorses - - horses)) / maxhorses; + (RESOURCE_QUANTITY * (HORSEGROWTH * 200.0 * ((double)maxhorses - + horses))) / (double)maxhorses; if (growth > 0) { int i; @@ -1233,6 +1235,7 @@ static void remove_idle_players(void) i = turn + 1; if (i < 4) i = 4; age = calloc(i, sizeof(int)); + if (!age) abort(); for (fp = &factions; *fp;) { faction *f = *fp; if (!is_monsters(f)) { @@ -1393,8 +1396,10 @@ static void init_prefixnames(void) } in = in->next; } - if (in == NULL) + if (in == NULL) { in = calloc(sizeof(local_names), 1); + if (!in) abort(); + } in->next = pnames; in->lang = lang; @@ -1432,7 +1437,8 @@ int prefix_cmd(unit * u, struct order *ord) } if (in == NULL) { init_prefixnames(); - for (in = pnames; in->lang != lang; in = in->next); + for (in = pnames; in && in->lang != lang; in = in->next); + if (!in) return 0; } init_order_depr(ord); @@ -2173,8 +2179,10 @@ static void display_item(unit * u, const item_type * itype) static void display_race(faction * f, const race * rc) { char buf[2048]; + sbstring sbs; - report_raceinfo(rc, f->locale, buf, sizeof(buf)); + sbs_init(&sbs, buf, sizeof(buf)); + report_raceinfo(rc, f->locale, &sbs); addmessage(0, f, buf, MSG_EVENT, ML_IMPORTANT); } @@ -3070,9 +3078,11 @@ void monthly_healing(void) /* hp über Maximum bauen sich ab. Wird zb durch Elixier der Macht * oder verändertes Ausdauertalent verursacht */ if (u->hp > umhp) { - u->hp -= (int)ceil((u->hp - umhp) / 2.0); - if (u->hp < umhp) + int diff = u->hp - umhp; + u->hp -= (int)ceil(diff / 2.0); + if (u->hp < umhp) { u->hp = umhp; + } continue; } @@ -3509,6 +3519,7 @@ static processor *add_proc(int priority, const char *name, processor_t type) } proc = (processor *)malloc(sizeof(processor)); + if (!proc) abort(); proc->priority = priority; proc->type = type; proc->name = name; diff --git a/src/laws.test.c b/src/laws.test.c index 5a4f85b85..a2eca7ab5 100644 --- a/src/laws.test.c +++ b/src/laws.test.c @@ -1483,6 +1483,34 @@ static void test_show_without_item(CuTest *tc) test_teardown(); } +static struct locale *setup_locale(void) { + struct locale *loc; + + loc = get_or_create_locale("de"); + locale_setstring(loc, "elvenhorse", "Elfenpferd"); + locale_setstring(loc, "elvenhorse_p", "Elfenpferde"); + locale_setstring(loc, "iteminfo::elvenhorse", "Elfenpferd Informationen"); + locale_setstring(loc, "race::elf_p", "Elfen"); + locale_setstring(loc, "race::elf", "Elf"); + locale_setstring(loc, "race::human_p", "Menschen"); + locale_setstring(loc, "race::human", "Mensch"); + locale_setstring(loc, "stat_hitpoints", "Trefferpunkte"); + locale_setstring(loc, "stat_attack", "Angriff"); + locale_setstring(loc, "stat_attacks", "Angriffe"); + locale_setstring(loc, "stat_defense", "Verteidigung"); + locale_setstring(loc, "stat_armor", "Ruestung"); + locale_setstring(loc, "stat_equipment", "Gegenstaende"); + locale_setstring(loc, "stat_pierce", "Pieks"); + locale_setstring(loc, "stat_cut", "Schnipp"); + locale_setstring(loc, "stat_bash", "Boink"); + locale_setstring(loc, "attack_magical", "magisch"); + locale_setstring(loc, "attack_standard", "normal"); + locale_setstring(loc, "attack_natural", "natuerlich"); + locale_setstring(loc, "attack_structural", "strukturell"); + init_locale(loc); + return loc; +} + static void test_show_race(CuTest *tc) { order *ord; race * rc; @@ -1495,12 +1523,7 @@ static void test_show_race(CuTest *tc) { test_create_race("human"); rc = test_create_race("elf"); - loc = get_or_create_locale("de"); - locale_setstring(loc, "race::elf_p", "Elfen"); - locale_setstring(loc, "race::elf", "Elf"); - locale_setstring(loc, "race::human_p", "Menschen"); - locale_setstring(loc, "race::human", "Mensch"); - init_locale(loc); + loc = setup_locale(); u = test_create_unit(test_create_faction(rc), test_create_region(0, 0, NULL)); u->faction->locale = loc; @@ -1536,13 +1559,7 @@ static void test_show_both(CuTest *tc) { rc = test_create_race("elf"); test_create_itemtype("elvenhorse"); - loc = get_or_create_locale("de"); - locale_setstring(loc, "elvenhorse", "Elfenpferd"); - locale_setstring(loc, "elvenhorse_p", "Elfenpferde"); - locale_setstring(loc, "iteminfo::elvenhorse", "Hiyaa!"); - locale_setstring(loc, "race::elf_p", "Elfen"); - locale_setstring(loc, "race::elf", "Elf"); - init_locale(loc); + loc = setup_locale(); CuAssertPtrNotNull(tc, finditemtype("elf", loc)); CuAssertPtrNotNull(tc, findrace("elf", loc)); @@ -1558,7 +1575,7 @@ static void test_show_both(CuTest *tc) { CuAssertTrue(tc, memcmp("Elf:", msg->parameters[0].v, 4) == 0); msg = test_find_messagetype(u->faction->msgs, "displayitem"); CuAssertPtrNotNull(tc, msg); - CuAssertTrue(tc, memcmp("Hiyaa!", msg->parameters[2].v, 4) == 0); + CuAssertTrue(tc, memcmp("Elfenpferd Informationen", msg->parameters[2].v, 4) == 0); test_clear_messages(u->faction); free_order(ord); test_teardown(); diff --git a/src/magic.c b/src/magic.c index c13785a2c..ebaac3583 100644 --- a/src/magic.c +++ b/src/magic.c @@ -646,7 +646,7 @@ static int use_item_aura(const region * r, const unit * u) return n; } -int max_spellpoints(const region * r, const unit * u) +int max_spellpoints(const struct unit *u, const region * r) { int sk; double n, msp = 0; @@ -654,6 +654,9 @@ int max_spellpoints(const region * r, const unit * u) double divisor = 1.2; const struct resource_type *rtype; + assert(u); + if (!r) r = u->region; + sk = effskill(u, SK_MAGIC, r); msp = rc_maxaura(u_race(u)) * (pow(sk, potenz) / divisor + 1) + get_spchange(u); @@ -668,6 +671,11 @@ int max_spellpoints(const region * r, const unit * u) return (msp > 0) ? (int)msp : 0; } +int max_spellpoints_depr(const struct region *r, const struct unit *u) +{ + return max_spellpoints(u, r); +} + int change_maxspellpoints(unit * u, int csp) { sc_mage *m = get_mage(u); @@ -675,7 +683,7 @@ int change_maxspellpoints(unit * u, int csp) return 0; } m->spchange += csp; - return max_spellpoints(u->region, u); + return max_spellpoints_depr(u->region, u); } /* ------------------------------------------------------------- */ @@ -1456,7 +1464,7 @@ void regenerate_aura(void) for (u = r->units; u; u = u->next) { if (u->number && is_mage(u)) { aura = get_spellpoints(u); - auramax = max_spellpoints(r, u); + auramax = max_spellpoints_depr(r, u); if (aura < auramax) { struct building *b = inside_building(u); const struct building_type *btype = building_is_active(b) ? b->type : NULL; @@ -2979,6 +2987,11 @@ int cast_spell(struct castorder *co) return fun(co); } +const char *magic_name(magic_t mtype, const struct locale *lang) +{ + return LOC(lang, mkname("school", magic_school[mtype])); +} + static critbit_tree cb_spellbooks; #define SBNAMELEN 16 diff --git a/src/magic.h b/src/magic.h index 23e4692f7..4a184fe9e 100644 --- a/src/magic.h +++ b/src/magic.h @@ -252,7 +252,8 @@ extern "C" { /* setzt die Magiepunkte auf sp */ int change_spellpoints(struct unit *u, int mp); /* verändert die Anzahl der Magiepunkte der Einheit um +mp */ - int max_spellpoints(const struct region *r, const struct unit *u); + int max_spellpoints_depr(const struct region *r, const struct unit *u); + int max_spellpoints(const struct unit *u, const struct region *r); /* gibt die aktuell maximal möglichen Magiepunkte der Einheit zurück */ int change_maxspellpoints(struct unit *u, int csp); /* verändert die maximalen Magiepunkte einer Einheit */ @@ -316,6 +317,7 @@ extern "C" { * widersteht */ extern struct spell * unit_getspell(struct unit *u, const char *s, const struct locale *lang); + const char *magic_name(magic_t mtype, const struct locale *lang); /* Sprüche in der struct region */ /* (sind in curse) */ diff --git a/src/magic.test.c b/src/magic.test.c index 9979caf6d..577e88a37 100644 --- a/src/magic.test.c +++ b/src/magic.test.c @@ -446,19 +446,19 @@ static void test_max_spellpoints(CuTest *tc) { test_setup(); rc = test_create_race("human"); u = test_create_unit(test_create_faction(rc), test_create_region(0, 0, NULL)); - CuAssertIntEquals(tc, 1, max_spellpoints(u->region, u)); + CuAssertIntEquals(tc, 1, max_spellpoints_depr(u->region, u)); rc->maxaura = 100; - CuAssertIntEquals(tc, 1, max_spellpoints(u->region, u)); + CuAssertIntEquals(tc, 1, max_spellpoints_depr(u->region, u)); rc->maxaura = 200; - CuAssertIntEquals(tc, 2, max_spellpoints(u->region, u)); + CuAssertIntEquals(tc, 2, max_spellpoints_depr(u->region, u)); create_mage(u, M_GWYRRD); set_level(u, SK_MAGIC, 1); - CuAssertIntEquals(tc, 3, max_spellpoints(u->region, u)); + CuAssertIntEquals(tc, 3, max_spellpoints_depr(u->region, u)); set_level(u, SK_MAGIC, 2); - CuAssertIntEquals(tc, 9, max_spellpoints(u->region, u)); + CuAssertIntEquals(tc, 9, max_spellpoints_depr(u->region, u)); /* permanent aura loss: */ CuAssertIntEquals(tc, 7, change_maxspellpoints(u, -2)); - CuAssertIntEquals(tc, 7, max_spellpoints(u->region, u)); + CuAssertIntEquals(tc, 7, max_spellpoints_depr(u->region, u)); test_teardown(); } diff --git a/src/renumber.c b/src/renumber.c index a4653e1a0..e562a0285 100644 --- a/src/renumber.c +++ b/src/renumber.c @@ -167,7 +167,7 @@ int renumber_cmd(unit * u, order * ord) break; } s = gettoken(token, sizeof(token)); - if (*s == 0) { + if (s == NULL || *s == 0) { i = newcontainerid(); } else { diff --git a/src/renumber.test.c b/src/renumber.test.c index e5f9c5c33..f65011cf7 100644 --- a/src/renumber.test.c +++ b/src/renumber.test.c @@ -101,7 +101,12 @@ static void test_renumber_building(CuTest *tc) { no = u->building->no; uno = (no > 1) ? no - 1 : no + 1; lang = u->faction->locale; - u->thisorder = create_order(K_NUMBER, lang, "%s %s", LOC(lang, parameters[P_BUILDING]), itoa36(uno)); + u->thisorder = create_order(K_NUMBER, lang, LOC(lang, parameters[P_BUILDING])); + renumber_cmd(u, u->thisorder); + CuAssertTrue(tc, no != u->building->no); + free_order(u->thisorder); + + u->thisorder = create_order(K_NUMBER, lang, "%s %i", LOC(lang, parameters[P_BUILDING]), uno); renumber_cmd(u, u->thisorder); CuAssertIntEquals(tc, uno, u->building->no); test_teardown(); @@ -120,7 +125,7 @@ static void test_renumber_building_duplicate(CuTest *tc) { u->building = test_create_building(u->region, NULL); no = u->building->no; lang = f->locale; - u->thisorder = create_order(K_NUMBER, lang, "%s %s", LOC(lang, parameters[P_BUILDING]), itoa36(uno)); + u->thisorder = create_order(K_NUMBER, lang, "%s %i", LOC(lang, parameters[P_BUILDING]), uno); renumber_cmd(u, u->thisorder); CuAssertPtrNotNull(tc, test_find_messagetype(f->msgs, "error115")); CuAssertIntEquals(tc, no, u->building->no); diff --git a/src/report.c b/src/report.c index b6ed3f736..04e5408ad 100644 --- a/src/report.c +++ b/src/report.c @@ -676,7 +676,7 @@ nr_unit(struct stream *out, const faction * f, const unit * u, int indent, seen_ return; newline(out); - dh = bufunit(f, u, mode, buf, sizeof(buf)); + dh = bufunit_depr(f, u, mode, buf, sizeof(buf)); if (u->faction == f) { marker = '*'; @@ -1982,8 +1982,6 @@ void report_travelthru(struct stream *out, region *r, const faction *f) } } -#include "util/bsdstring.h" - int report_plaintext(const char *filename, report_context * ctx, const char *bom) @@ -2026,7 +2024,7 @@ report_plaintext(const char *filename, report_context * ctx, newline(out); sprintf(buf, "%s, %s/%s (%s)", factionname(f), LOC(f->locale, rc_name_s(f->race, NAME_PLURAL)), - LOC(f->locale, mkname("school", magic_school[f->magiegebiet])), faction_getemail(f)); + magic_name(f->magiegebiet, f->locale), faction_getemail(f)); centre(out, buf, true); if (f_get_alliance(f)) { centre(out, alliancename(f->alliance), true); diff --git a/src/reports.c b/src/reports.c index 3ce83a7de..56c6fd2b1 100644 --- a/src/reports.c +++ b/src/reports.c @@ -64,7 +64,6 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. /* util includes */ #include "kernel/attrib.h" #include "util/base36.h" -#include "util/bsdstring.h" #include "util/functions.h" #include "util/goodies.h" #include "util/language.h" @@ -162,29 +161,37 @@ const char *combatstatus[] = { "status_avoid", "status_flee" }; -size_t report_status(const unit * u, const struct locale *lang, char *fsbuf, size_t buflen) +void report_status(const unit * u, const struct locale *lang, struct sbstring *sbp) { const char * status = LOC(lang, combatstatus[u->status]); - size_t len = 0; if (!status) { const char *lname = locale_name(lang); struct locale *wloc = get_locale(lname); + log_warning("no translation for combat status %s in %s", combatstatus[u->status], lname); locale_setstring(wloc, combatstatus[u->status], combatstatus[u->status] + 7); - len = str_strlcpy(fsbuf, combatstatus[u->status] + 7, buflen); + sbs_strcat(sbp, combatstatus[u->status] + 7); } else { - len = str_strlcpy(fsbuf, status, buflen); + sbs_strcat(sbp, status); } if (fval(u, UFL_NOAID)) { - len += str_strlcat(fsbuf + len, ", ", buflen - len); - len += str_strlcat(fsbuf + len, LOC(lang, "status_noaid"), buflen - len); + sbs_strcat(sbp, ", "); + sbs_strcat(sbp, LOC(lang, "status_noaid")); } - - return len; } +size_t report_status_depr(const unit * u, const struct locale *lang, char *fsbuf, size_t buflen) +{ + sbstring sbs; + + sbs_init(&sbs, fsbuf, buflen); + report_status(u, lang, &sbs); + return sbs_length(&sbs); +} + + const char *hp_status(const unit * u) { double p; @@ -274,29 +281,19 @@ report_item(const unit * owner, const item * i, const faction * viewer, } #define ORDERS_IN_NR 1 -static size_t buforder(char *buffer, size_t size, const order * ord, const struct locale *lang, int mode) +static void buforder(sbstring *sbp, const order * ord, const struct locale *lang, int mode) { - char *bufp = buffer; - - bufp = STRLCPY(bufp, ", \"", size); + sbs_strcat(sbp, ", \""); if (mode < ORDERS_IN_NR) { char cmd[ORDERSIZE]; get_command(ord, lang, cmd, sizeof(cmd)); - bufp = STRLCPY(bufp, cmd, size); + sbs_strcat(sbp, cmd); } else { - bufp = STRLCPY(bufp, "...", size); + sbs_strcat(sbp, "..."); } - if (size > 1) { - *bufp++ = '\"'; - --size; - } - else { - WARN_STATIC_BUFFER(); - } - - return bufp - buffer; + sbs_strcat(sbp, "\""); } /** create a report of a list of items to a non-owner. @@ -365,21 +362,24 @@ static void report_resource(resource_report * result, const resource_type *rtype result->level = level; } -void report_raceinfo(const struct race *rc, const struct locale *lang, char *buf, size_t length) +static void bufattack(struct sbstring *sbp, const struct locale *lang, const char *name, const char *dmg) { + sbs_strcat(sbp, LOC(lang, name)); + if (dmg) { + sbs_strcat(sbp, " ("); + sbs_strcat(sbp, dmg); + sbs_strcat(sbp, ")"); + } +} + +void report_raceinfo(const struct race *rc, const struct locale *lang, struct sbstring *sbp) { const char *info; int a, at_count; const char *name, *key; - char *bufp = buf; - size_t size = length - 1; - size_t bytes; name = rc_name_s(rc, NAME_SINGULAR); - - bytes = slprintf(bufp, size, "%s: ", LOC(lang, name)); - assert(bytes <= INT_MAX); - if (wrptr(&bufp, &size, (int)bytes) != 0) - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, LOC(lang, name)); + sbs_strcat(sbp, ": "); key = mkname("raceinfo", rc->_name); info = locale_getstring(lang, key); @@ -387,47 +387,37 @@ void report_raceinfo(const struct race *rc, const struct locale *lang, char *buf info = LOC(lang, mkname("raceinfo", "no_info")); } - if (info) bufp = STRLCPY(bufp, info, size); + if (info) { + sbs_strcat(sbp, info); + } /* hp_p : Trefferpunkte */ - bytes = - slprintf(bufp, size, " %d %s", rc->hitpoints, LOC(lang, - "stat_hitpoints")); - assert(bytes <= INT_MAX); - if (wrptr(&bufp, &size, (int)bytes) != 0) - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, " "); + sbs_strcat(sbp, str_itoa(rc->hitpoints)); + sbs_strcat(sbp, " "); + sbs_strcat(sbp, LOC(lang, "stat_hitpoints")); /* b_attacke : Angriff */ - bytes = - slprintf(bufp, size, ", %s: %d", LOC(lang, "stat_attack"), - (rc->at_default + rc->at_bonus)); - assert(bytes <= INT_MAX); - if (wrptr(&bufp, &size, (int)bytes) != 0) - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, ", "); + sbs_strcat(sbp, LOC(lang, "stat_attack")); + sbs_strcat(sbp, ": "); + sbs_strcat(sbp, str_itoa(rc->at_default + rc->at_bonus)); /* b_defense : Verteidigung */ - bytes = - slprintf(bufp, size, ", %s: %d", LOC(lang, "stat_defense"), - (rc->df_default + rc->df_bonus)); - assert(bytes <= INT_MAX); - if (wrptr(&bufp, &size, (int)bytes) != 0) - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, ", "); + sbs_strcat(sbp, LOC(lang, "stat_defense")); + sbs_strcat(sbp, ": "); + sbs_strcat(sbp, str_itoa(rc->df_default + rc->df_bonus)); /* b_armor : Rüstung */ if (rc->armor > 0) { - bytes = - slprintf(bufp, size, ", %s: %d", LOC(lang, "stat_armor"), rc->armor); - assert(bytes <= INT_MAX); - if (wrptr(&bufp, &size, (int)bytes) != 0) - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, ", "); + sbs_strcat(sbp, LOC(lang, "stat_armor")); + sbs_strcat(sbp, ": "); + sbs_strcat(sbp, str_itoa(rc->armor)); } - if (size > 1) { - *bufp++ = '.'; - --size; - } - else - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, "."); /* b_damage : Schaden */ at_count = 0; @@ -437,73 +427,53 @@ void report_raceinfo(const struct race *rc, const struct locale *lang, char *buf } } if (rc->battle_flags & BF_EQUIPMENT) { - if (wrptr(&bufp, &size, snprintf(bufp, size, " %s", LOC(lang, "stat_equipment"))) != 0) - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, " "); + sbs_strcat(sbp, LOC(lang, "stat_equipment")); } if (rc->battle_flags & BF_RES_PIERCE) { - if (wrptr(&bufp, &size, snprintf(bufp, size, " %s", LOC(lang, "stat_pierce"))) != 0) - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, " "); + sbs_strcat(sbp, LOC(lang, "stat_pierce")); } if (rc->battle_flags & BF_RES_CUT) { - if (wrptr(&bufp, &size, snprintf(bufp, size, " %s", LOC(lang, "stat_cut"))) != 0) - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, " "); + sbs_strcat(sbp, LOC(lang, "stat_cut")); } if (rc->battle_flags & BF_RES_BASH) { - if (wrptr(&bufp, &size, snprintf(bufp, size, " %s", LOC(lang, "stat_bash"))) != 0) - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, " "); + sbs_strcat(sbp, LOC(lang, "stat_bash")); } - if (wrptr(&bufp, &size, snprintf(bufp, size, " %d %s", at_count, LOC(lang, (at_count == 1) ? "stat_attack" : "stat_attacks"))) != 0) - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, " "); + sbs_strcat(sbp, str_itoa(at_count)); + sbs_strcat(sbp, " "); + sbs_strcat(sbp, LOC(lang, (at_count == 1) ? "stat_attack" : "stat_attacks")); for (a = 0; a < RACE_ATTACKS; a++) { if (rc->attack[a].type != AT_NONE) { - if (a != 0) - bufp = STRLCPY(bufp, ", ", size); - else - bufp = STRLCPY(bufp, ": ", size); + sbs_strcat(sbp, (a == 0) ? ": " : ", "); switch (rc->attack[a].type) { case AT_STANDARD: - bytes = - (size_t)snprintf(bufp, size, "%s (%s)", - LOC(lang, "attack_standard"), rc->def_damage); + bufattack(sbp, lang, "attack_standard", rc->def_damage); break; case AT_NATURAL: - bytes = - (size_t)snprintf(bufp, size, "%s (%s)", - LOC(lang, "attack_natural"), rc->attack[a].data.dice); + bufattack(sbp, lang, "attack_natural", rc->attack[a].data.dice); break; case AT_SPELL: case AT_COMBATSPELL: case AT_DRAIN_ST: case AT_DRAIN_EXP: case AT_DAZZLE: - bytes = (size_t)snprintf(bufp, size, "%s", LOC(lang, "attack_magical")); + bufattack(sbp, lang, "attack_magical", NULL); break; case AT_STRUCTURAL: - bytes = - (size_t)snprintf(bufp, size, "%s (%s)", - LOC(lang, "attack_structural"), rc->attack[a].data.dice); + bufattack(sbp, lang, "attack_structural", rc->attack[a].data.dice); break; - default: - bytes = 0; } - - assert(bytes <= INT_MAX); - if (bytes && wrptr(&bufp, &size, (int)bytes) != 0) - WARN_STATIC_BUFFER(); } } - if (size > 1) { - *bufp++ = '.'; - --size; - } - else - WARN_STATIC_BUFFER(); - - *bufp = 0; + sbs_strcat(sbp, "."); } void @@ -612,50 +582,47 @@ report_resources(const region * r, resource_report * result, int size, return n; } -static size_t spskill(char *buffer, size_t size, const struct locale * lang, +static void spskill(sbstring *sbp, const struct locale * lang, const struct unit * u, struct skill * sv, int *dh) { - char *bufp = buffer; int effsk; - if (!u->number) - return 0; + if (!u->number) { + return; + } if (sv->level <= 0) { if (sv->old <= 0 || (u->faction->options & WANT_OPTION(O_SHOWSKCHANGE)) == 0) { - return 0; + return; } } - - bufp = STRLCPY(bufp, ", ", size); + sbs_strcat(sbp, ", "); if (!*dh) { - bufp = STRLCPY(bufp, LOC(lang, "nr_skills"), size); - bufp = STRLCPY(bufp, ": ", size); + sbs_strcat(sbp, LOC(lang, "nr_skills")); + sbs_strcat(sbp, ": "); *dh = 1; } - bufp = STRLCPY(bufp, skillname(sv->id, lang), size); - bufp = STRLCPY(bufp, " ", size); + sbs_strcat(sbp, skillname(sv->id, lang)); + sbs_strcat(sbp, " "); if (sv->id == SK_MAGIC) { magic_t mtype = unit_get_magic(u); if (mtype != M_GRAY) { - bufp = STRLCPY(bufp, - LOC(lang, mkname("school", magic_school[mtype])), size); - bufp = STRLCPY(bufp, " ", size); + sbs_strcat(sbp, magic_name(mtype, lang)); + sbs_strcat(sbp, " "); } } if (sv->id == SK_STEALTH && fval(u, UFL_STEALTH)) { int i = u_geteffstealth(u); if (i >= 0) { - if (wrptr(&bufp, &size, snprintf(bufp, size, "%d/", i)) != 0) - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, str_itoa(i)); + sbs_strcat(sbp, "/"); } } effsk = eff_skill(u, sv, 0); - if (wrptr(&bufp, &size, snprintf(bufp, size, "%d", effsk)) != 0) - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, str_itoa(effsk)); if (u->faction->options & WANT_OPTION(O_SHOWSKCHANGE)) { int oldeff = 0; @@ -669,150 +636,137 @@ static size_t spskill(char *buffer, size_t size, const struct locale * lang, diff = effsk - oldeff; if (diff != 0) { - if (wrptr(&bufp, &size, snprintf(bufp, size, " (%s%d)", (diff > 0) ? "+" : "", diff)) != 0) - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, " ("); + sbs_strcat(sbp, (diff > 0) ? "+" : ""); + sbs_strcat(sbp, str_itoa(diff)); + sbs_strcat(sbp, ")"); } } - return bufp - buffer; } -int -bufunit(const faction * f, const unit * u, seen_mode mode, char *buf, - size_t size) +void bufunit(const faction * f, const unit * u, const faction *fv, + seen_mode mode, int getarnt, struct sbstring *sbp) { int i, dh; - int getarnt = fval(u, UFL_ANON_FACTION); const char *pzTmp, *str; bool isbattle = (mode == seen_battle); item *itm, *show = NULL; - faction *fv; - char *bufp = buf; - int result = 0; item results[MAX_INVENTORY]; const struct locale *lang = f->locale; + if (!fv) { + fv = visible_faction(f, u); + } assert(f); - bufp = STRLCPY(bufp, unitname(u), size); - fv = visible_faction(f, u); + sbs_strcat(sbp, unitname(u)); if (!isbattle) { if (u->faction == f) { if (fval(u, UFL_GROUP)) { group *g = get_group(u); if (g) { - bufp = STRLCPY(bufp, ", ", size); - bufp = STRLCPY(bufp, groupid(g, f), size); + sbs_strcat(sbp, ", "); + sbs_strcat(sbp, groupid(g, f)); } } if (getarnt) { - bufp = STRLCPY(bufp, ", ", size); - bufp = STRLCPY(bufp, LOC(lang, "anonymous"), size); + sbs_strcat(sbp, ", "); + sbs_strcat(sbp, LOC(lang, "anonymous")); } else if (u->attribs) { faction *otherf = get_otherfaction(u); if (otherf) { - bufp = STRLCPY(bufp, ", ", size); - bufp = STRLCPY(bufp, factionname(otherf), size); + sbs_strcat(sbp, ", "); + sbs_strcat(sbp, factionname(otherf)); } } } else { if (getarnt) { - bufp = STRLCPY(bufp, ", ", size); - bufp = STRLCPY(bufp, LOC(lang, "anonymous"), size); + sbs_strcat(sbp, ", "); + sbs_strcat(sbp, LOC(lang, "anonymous")); } else { if (u->attribs && alliedunit(u, f, HELP_FSTEALTH)) { faction *otherf = get_otherfaction(u); if (otherf) { - int n = snprintf(bufp, size, ", %s (%s)", - factionname(otherf), factionname(u->faction)); - if (wrptr(&bufp, &size, n) != 0) - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, ", "); + sbs_strcat(sbp, factionname(otherf)); + sbs_strcat(sbp, " ("); + sbs_strcat(sbp, factionname(u->faction)); + sbs_strcat(sbp, ")"); } else { - bufp = STRLCPY(bufp, ", ", size); - bufp = STRLCPY(bufp, factionname(fv), size); + sbs_strcat(sbp, ", "); + sbs_strcat(sbp, factionname(fv)); } } else { - bufp = STRLCPY(bufp, ", ", size); - bufp = STRLCPY(bufp, factionname(fv), size); + sbs_strcat(sbp, ", "); + sbs_strcat(sbp, factionname(fv)); } } } } - bufp = STRLCPY(bufp, ", ", size); - - if (wrptr(&bufp, &size, snprintf(bufp, size, "%d ", u->number))) - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, ", "); + sbs_strcat(sbp, str_itoa(u->number)); + sbs_strcat(sbp, " "); pzTmp = get_racename(u->attribs); if (pzTmp) { - bufp = STRLCPY(bufp, pzTmp, size); + sbs_strcat(sbp, pzTmp); if (u->faction == f && fval(u_race(u), RCF_SHAPESHIFTANY)) { - bufp = STRLCPY(bufp, " (", size); - bufp = STRLCPY(bufp, racename(lang, u, u_race(u)), size); - if (size > 1) { - strcpy(bufp++, ")"); - --size; - } + sbs_strcat(sbp, " ("); + sbs_strcat(sbp, racename(lang, u, u_race(u))); + sbs_strcat(sbp, ")"); } } else { const race *irace = u_irace(u); - bufp = STRLCPY(bufp, racename(lang, u, irace), size); - if (u->faction == f && irace != u_race(u)) { - bufp = STRLCPY(bufp, " (", size); - bufp = STRLCPY(bufp, racename(lang, u, u_race(u)), size); - if (size > 1) { - strcpy(bufp++, ")"); - --size; - } + const race *urace = u_race(u); + sbs_strcat(sbp, racename(lang, u, irace)); + if (u->faction == f && irace != urace) { + sbs_strcat(sbp, " ("); + sbs_strcat(sbp, racename(lang, u, urace)); + sbs_strcat(sbp, ")"); } } if (fval(u, UFL_HERO) && (u->faction == f || omniscient(f))) { - bufp = STRLCPY(bufp, ", ", size); - bufp = STRLCPY(bufp, LOC(lang, "hero"), size); + sbs_strcat(sbp, ", "); + sbs_strcat(sbp, LOC(lang, "hero")); } /* status */ if (u->number && (u->faction == f || isbattle)) { const char *c = hp_status(u); c = c ? LOC(lang, c) : 0; - bufp = STRLCPY(bufp, ", ", size); - bufp += report_status(u, lang, bufp, size); + sbs_strcat(sbp, ", "); + report_status(u, lang, sbp); if (c || fval(u, UFL_HUNGER)) { - bufp = STRLCPY(bufp, " (", size); + sbs_strcat(sbp, " ("); if (c) { - bufp = STRLCPY(bufp, c, size); + sbs_strcat(sbp, c); } if (fval(u, UFL_HUNGER)) { if (c) { - bufp = STRLCPY(bufp, ", ", size); + sbs_strcat(sbp, ", "); } - bufp = STRLCPY(bufp, LOC(lang, "unit_hungers"), size); - } - if (size > 1) { - strcpy(bufp++, ")"); - --size; + sbs_strcat(sbp, LOC(lang, "unit_hungers")); } + sbs_strcat(sbp, ")"); } } if (is_guard(u)) { - bufp = STRLCPY(bufp, ", ", size); - bufp = STRLCPY(bufp, LOC(lang, "unit_guards"), size); + sbs_strcat(sbp, ", "); + sbs_strcat(sbp, LOC(lang, "unit_guards")); } dh = 0; if (u->faction == f) { skill *sv; for (sv = u->skills; sv != u->skills + u->skill_size; ++sv) { - size_t bytes = spskill(bufp, size, lang, u, sv, &dh); - assert(bytes <= INT_MAX); - if (wrptr(&bufp, &size, (int)bytes) != 0) - WARN_STATIC_BUFFER(); + spskill(sbp, lang, u, sv, &dh); } } @@ -831,22 +785,23 @@ bufunit(const faction * f, const unit * u, seen_mode mode, char *buf, const char *ic; int in; report_item(u, itm, f, &ic, NULL, &in, false); - if (in == 0 || ic == NULL) + if (in == 0 || ic == NULL) { continue; - bufp = STRLCPY(bufp, ", ", size); + } + sbs_strcat(sbp, ", "); if (!dh) { - result = snprintf(bufp, size, "%s: ", LOC(lang, "nr_inventory")); - if (wrptr(&bufp, &size, result) != 0) - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, LOC(lang, "nr_inventory")); + sbs_strcat(sbp, ": "); dh = 1; } if (in == 1) { - bufp = STRLCPY(bufp, ic, size); + sbs_strcat(sbp, ic); } else { - if (wrptr(&bufp, &size, snprintf(bufp, size, "%d %s", in, ic))) - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, str_itoa(in)); + sbs_strcat(sbp, " "); + sbs_strcat(sbp, ic); } } @@ -856,27 +811,26 @@ bufunit(const faction * f, const unit * u, seen_mode mode, char *buf, if (book) { selist *ql = book->spells; int qi, header, maxlevel = effskill(u, SK_MAGIC, 0); - int n = snprintf(bufp, size, ". Aura %d/%d", get_spellpoints(u), max_spellpoints(u->region, u)); - if (wrptr(&bufp, &size, n) != 0) { - WARN_STATIC_BUFFER(); - } + sbs_strcat(sbp, ". Aura "); + sbs_strcat(sbp, str_itoa(get_spellpoints(u))); + sbs_strcat(sbp, "/"); + sbs_strcat(sbp, str_itoa(max_spellpoints(u, NULL))); for (header = 0, qi = 0; ql; selist_advance(&ql, &qi, 1)) { spellbook_entry * sbe = (spellbook_entry *)selist_get(ql, qi); const spell *sp = spellref_get(&sbe->spref); if (sbe->level <= maxlevel) { if (!header) { - n = snprintf(bufp, size, ", %s: ", LOC(lang, "nr_spells")); + sbs_strcat(sbp, ", "); + sbs_strcat(sbp, LOC(lang, "nr_spells")); + sbs_strcat(sbp, ": "); header = 1; } else { - n = (int)str_strlcpy(bufp, ", ", size); - } - if (wrptr(&bufp, &size, n) != 0) { - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, ", "); } /* TODO: no need to deref the spellref here (spref->name is good) */ - bufp = STRLCPY(bufp, spell_name(sp, lang), size); + sbs_strcat(sbp, spell_name(sp, lang)); } } @@ -885,10 +839,9 @@ bufunit(const faction * f, const unit * u, seen_mode mode, char *buf, break; } if (i != MAXCOMBATSPELLS) { - n = snprintf(bufp, size, ", %s: ", LOC(lang, "nr_combatspells")); - if (wrptr(&bufp, &size, n) != 0) - WARN_STATIC_BUFFER(); - + sbs_strcat(sbp, ", "); + sbs_strcat(sbp, LOC(lang, "nr_combatspells")); + sbs_strcat(sbp, ": "); dh = 0; for (i = 0; i < MAXCOMBATSPELLS; i++) { const spell *sp; @@ -896,20 +849,20 @@ bufunit(const faction * f, const unit * u, seen_mode mode, char *buf, dh = 1; } else { - bufp = STRLCPY(bufp, ", ", size); + sbs_strcat(sbp, ", "); } sp = get_combatspell(u, i); if (sp) { int sl = get_combatspelllevel(u, i); - bufp = STRLCPY(bufp, spell_name(sp, lang), size); + sbs_strcat(sbp, spell_name(sp, lang)); if (sl > 0) { - n = snprintf(bufp, size, " (%d)", sl); - if (wrptr(&bufp, &size, n) != 0) - WARN_STATIC_BUFFER(); + sbs_strcat(sbp, "( "); + sbs_strcat(sbp, str_itoa(sl)); + sbs_strcat(sbp, ")"); } } else { - bufp = STRLCPY(bufp, LOC(lang, "nr_nospells"), size); + sbs_strcat(sbp, LOC(lang, "nr_nospells")); } } } @@ -920,62 +873,59 @@ bufunit(const faction * f, const unit * u, seen_mode mode, char *buf, for (ord = u->old_orders; ord; ord = ord->next) { keyword_t kwd = getkeyword(ord); if (is_repeated(kwd)) { - if (printed < ORDERS_IN_NR) { - int n = (int)buforder(bufp, size, ord, u->faction->locale, printed++); - if (wrptr(&bufp, &size, n) != 0) - WARN_STATIC_BUFFER(); - } - else + if (printed >= ORDERS_IN_NR) { break; + } + buforder(sbp, ord, u->faction->locale, printed++); } } - if (printed < ORDERS_IN_NR) + if (printed < ORDERS_IN_NR) { for (ord = u->orders; ord; ord = ord->next) { keyword_t kwd = getkeyword(ord); if (is_repeated(kwd)) { - if (printed < ORDERS_IN_NR) { - int n = (int)buforder(bufp, size, ord, lang, printed++); - if (wrptr(&bufp, &size, n) != 0) - WARN_STATIC_BUFFER(); - } - else { + if (printed >= ORDERS_IN_NR) { break; } + buforder(sbp, ord, lang, printed++); } } + } } } i = 0; str = u_description(u, lang); if (str) { - bufp = STRLCPY(bufp, "; ", size); - bufp = STRLCPY(bufp, str, size); + sbs_strcat(sbp, "; "); + sbs_strcat(sbp, str); i = str[strlen(str) - 1]; } if (i != '!' && i != '?' && i != '.') { - if (size > 1) { - strcpy(bufp++, "."); - --size; - } + sbs_strcat(sbp, "."); } pzTmp = uprivate(u); if (u->faction == f && pzTmp) { - bufp = STRLCPY(bufp, " (Bem: ", size); - bufp = STRLCPY(bufp, pzTmp, size); - bufp = STRLCPY(bufp, ")", size); + sbs_strcat(sbp, " (Bem: "); + sbs_strcat(sbp, pzTmp); + sbs_strcat(sbp, ")"); } +} - dh = 0; +int bufunit_depr(const faction * f, const unit * u, seen_mode mode, + char *buf, size_t size) +{ + int getarnt = fval(u, UFL_ANON_FACTION); + const faction * fv = visible_faction(f, u); + sbstring sbs; + + sbs_init(&sbs, buf, size); + bufunit(f, u, fv, mode, getarnt, &sbs); if (!getarnt) { if (alliedfaction(f, fv, HELP_ALL)) { - dh = 1; + return 1; } } - if (size <= 1) { - log_warning("bufunit ran out of space after writing %u bytes.\n", (bufp - buf)); - } - return dh; + return 0; } void split_paragraph(strlist ** SP, const char *s, unsigned int indent, unsigned int width, char mark) @@ -1049,7 +999,7 @@ spunit(struct strlist **SP, const struct faction *f, const unit * u, unsigned in seen_mode mode) { char buf[DISPLAYSIZE]; - int dh = bufunit(f, u, mode, buf, sizeof(buf)); + int dh = bufunit_depr(f, u, mode, buf, sizeof(buf)); lparagraph(SP, buf, indent, (char)((u->faction == f) ? '*' : (dh ? '+' : '-'))); } @@ -2198,28 +2148,23 @@ static void eval_resources(struct opstack **stack, const void *userdata) const struct locale *lang = f ? f->locale : default_locale; const struct resource *res = (const struct resource *)opop(stack).v; char buf[1024]; /* but we only use about half of this */ - size_t size = sizeof(buf) - 1; variant var; - - char *bufp = buf; - while (res != NULL && size > 4) { + sbstring sbs; + + sbs_init(&sbs, buf, sizeof(buf)); + while (res != NULL) { const char *rname = resourcename(res->type, (res->number != 1) ? NMF_PLURAL : 0); - int result = snprintf(bufp, size, "%d %s", res->number, LOC(lang, rname)); - if (wrptr(&bufp, &size, result) != 0 || size < sizeof(buf) / 2) { - WARN_STATIC_BUFFER(); - break; - } + sbs_strcat(&sbs, str_itoa(res->number)); + sbs_strcat(&sbs, ""); + sbs_strcat(&sbs, LOC(lang, rname)); res = res->next; - if (res != NULL && size > 2) { - strcat(bufp, ", "); - bufp += 2; - size -= 2; + if (res != NULL) { + sbs_strcat(&sbs, ", "); } } - *bufp = 0; - var.v = strcpy(balloc((size_t)(bufp - buf + 1)), buf); + var.v = strcpy(balloc(sbs_length(&sbs)), buf); opush(stack, var); } @@ -2230,31 +2175,29 @@ static void eval_regions(struct opstack **stack, const void *userdata) int handle_end, begin = opop(stack).i; const arg_regions *aregs = (const arg_regions *)opop(stack).v; char buf[256]; - size_t size = sizeof(buf) - 1; variant var; - char *bufp = buf; + sbstring sbs; + sbs_init(&sbs, buf, sizeof(buf)); if (aregs == NULL) { handle_end = begin; } - else { - if (i >= 0) - handle_end = begin + i; - else - handle_end = aregs->nregions + i; + else if (i >= 0) { + handle_end = begin + i; } + else { + handle_end = aregs->nregions + i; + } + for (i = begin; i < handle_end; ++i) { const char *rname = (const char *)regionname(aregs->regions[i], report); - bufp = STRLCPY(bufp, rname, size); + sbs_strcat(&sbs, rname); - if (i + 1 < handle_end && size > 2) { - strcat(bufp, ", "); - bufp += 2; - size -= 2; + if (i + 1 < handle_end) { + sbs_strcat(&sbs, ", "); } } - *bufp = 0; - var.v = strcpy(balloc((size_t)(bufp - buf + 1)), buf); + var.v = strcpy(balloc(sbs_length(&sbs)), buf); opush(stack, var); } @@ -2265,41 +2208,49 @@ const char *get_mailcmd(const struct locale *loc) return result; } +static void print_trail(const faction *f, const region *r, + const struct locale *lang, struct sbstring *sbp) +{ + char buf[64]; + const char *trail = trailinto(r, lang); + const char *rn = f_regionid_s(r, f); + if (snprintf(buf, sizeof(buf), trail, rn) != 0) { + sbs_strcat(sbp, buf); + } +} + static void eval_trail(struct opstack **stack, const void *userdata) { /* order -> string */ const faction *report = (const faction *)userdata; const struct locale *lang = report ? report->locale : default_locale; const arg_regions *aregs = (const arg_regions *)opop(stack).v; char buf[512]; - size_t size = sizeof(buf) - 1; variant var; - char *bufp = buf; + sbstring sbs; #ifdef _SECURECRT_ERRCODE_VALUES_DEFINED - /* stupid MS broke snprintf */ + /* MSVC touches errno in snprintf */ int eold = errno; #endif + sbs_init(&sbs, buf, sizeof(buf)); if (aregs != NULL) { int i, handle_end = 0, begin = 0; handle_end = aregs->nregions; for (i = begin; i < handle_end; ++i) { region *r = aregs->regions[i]; - const char *trail = trailinto(r, lang); - const char *rn = f_regionid_s(r, report); + sbs_strcat(&sbs, ", "); - if (wrptr(&bufp, &size, snprintf(bufp, size, trail, rn)) != 0) - WARN_STATIC_BUFFER(); + print_trail(report, r, lang, &sbs); if (i + 2 < handle_end) { - bufp = STRLCPY(bufp, ", ", size); + sbs_strcat(&sbs, ", "); } else if (i + 1 < handle_end) { - bufp = STRLCPY(bufp, LOC(lang, "list_and"), size); + sbs_strcat(&sbs, LOC(lang, "list_and")); } } } - *bufp = 0; - var.v = strcpy(balloc((size_t)(bufp - buf + 1)), buf); + var.v = strcpy(balloc(sbs_length(&sbs)), buf); opush(stack, var); #ifdef _SECURECRT_ERRCODE_VALUES_DEFINED if (errno == ERANGE) { @@ -2308,11 +2259,9 @@ static void eval_trail(struct opstack **stack, const void *userdata) #endif } -void report_race_skills(const race *rc, char *zText, size_t length, const struct locale *lang) +void report_race_skills(const race *rc, const struct locale *lang, sbstring *sbp) { - size_t size = length - 1; int dh = 0, dh1 = 0, sk; - char *bufp = zText; for (sk = 0; sk < MAXSKILLS; ++sk) { if (skill_enabled(sk) && rc->bonus[sk] > -5) @@ -2321,29 +2270,30 @@ void report_race_skills(const race *rc, char *zText, size_t length, const struct for (sk = 0; sk < MAXSKILLS; sk++) { if (skill_enabled(sk) && rc->bonus[sk] > -5) { - size_t bytes; dh--; if (dh1 == 0) { dh1 = 1; } else { if (dh == 0) { - bytes = str_strlcpy(bufp, LOC(lang, "list_and"), size); + sbs_strcat(sbp, LOC(lang, "list_and")); } else { - bytes = str_strlcpy(bufp, ", ", size); + sbs_strcat(sbp, ", "); } - assert(bytes <= INT_MAX); - BUFFER_STRCAT(bufp, size, bytes); } - bytes = str_strlcpy(bufp, skillname((skill_t)sk, lang), - size); - assert(bytes <= INT_MAX); - BUFFER_STRCAT(bufp, size, (int)bytes); + sbs_strcat(sbp, skillname((skill_t)sk, lang)); } } } +void report_race_skills_depr(const race *rc, char *zText, size_t length, const struct locale *lang) +{ + sbstring sbs; + sbs_init(&sbs, zText, length); + report_race_skills(rc, lang, &sbs); +} + static void eval_direction(struct opstack **stack, const void *userdata) { const faction *report = (const faction *)userdata; diff --git a/src/reports.h b/src/reports.h index a6dddb1b4..91af51bc2 100644 --- a/src/reports.h +++ b/src/reports.h @@ -31,6 +31,7 @@ extern "C" { struct battle; struct gamedate; + struct sbstring; struct selist; struct stream; struct seen_region; @@ -87,12 +88,15 @@ extern "C" { void register_reporttype(const char *extension, report_fun write, int flag); - int bufunit(const struct faction *f, const struct unit *u, seen_mode mode, + int bufunit_depr(const struct faction *f, const struct unit *u, seen_mode mode, char *buf, size_t size); + void bufunit(const struct faction * f, const struct unit * u, + const struct faction *fv, seen_mode mode, int getarnt, + struct sbstring *sbp); const char *trailinto(const struct region *r, const struct locale *lang); - size_t report_status(const struct unit *u, + size_t report_status_depr(const struct unit *u, const struct locale *lang, char *buf, size_t siz); void report_battle_start(struct battle * b); @@ -116,8 +120,8 @@ extern "C" { int report_items(const struct unit *u, struct item *result, int size, const struct unit *owner, const struct faction *viewer); void report_warnings(struct faction *f, int now); - void report_raceinfo(const struct race *rc, const struct locale *lang, char *buf, size_t length); - void report_race_skills(const struct race *rc, char *zText, size_t length, const struct locale *lang); + void report_raceinfo(const struct race *rc, const struct locale *lang, struct sbstring *sbp); + void report_race_skills_depr(const struct race *rc, char *zText, size_t length, const struct locale *lang); void report_item(const struct unit *owner, const struct item *i, const struct faction *viewer, const char **name, const char **basename, int *number, bool singular); diff --git a/src/reports.test.c b/src/reports.test.c index b1cf36d99..86f69b0fd 100644 --- a/src/reports.test.c +++ b/src/reports.test.c @@ -199,55 +199,55 @@ static void test_bufunit_fstealth(CuTest *tc) { key_set(&u->attribs, 42, 42); /* report to ourselves */ - bufunit(f1, u, seen_unit, buf, sizeof(buf)); + bufunit_depr(f1, u, seen_unit, buf, sizeof(buf)); CuAssertStrEquals(tc, "Hodor (1), 1 human, aggressive.", buf); /* ... also when we are anonymous */ u->flags |= UFL_ANON_FACTION; - bufunit(f1, u, seen_unit, buf, sizeof(buf)); + bufunit_depr(f1, u, seen_unit, buf, sizeof(buf)); CuAssertStrEquals(tc, "Hodor (1), anonymous, 1 human, aggressive.", buf); u->flags &= ~UFL_ANON_FACTION; /* we see that our unit is cloaked */ set_factionstealth(u, f2); CuAssertPtrNotNull(tc, u->attribs); - bufunit(f1, u, seen_unit, buf, sizeof(buf)); + bufunit_depr(f1, u, seen_unit, buf, sizeof(buf)); CuAssertStrEquals(tc, "Hodor (1), TWW (2), 1 human, aggressive.", buf); /* ... also when we are anonymous */ u->flags |= UFL_ANON_FACTION; - bufunit(f1, u, seen_unit, buf, sizeof(buf)); + bufunit_depr(f1, u, seen_unit, buf, sizeof(buf)); CuAssertStrEquals(tc, "Hodor (1), anonymous, 1 human, aggressive.", buf); u->flags &= ~UFL_ANON_FACTION; /* we can see that someone is presenting as us */ - bufunit(f2, u, seen_unit, buf, sizeof(buf)); + bufunit_depr(f2, u, seen_unit, buf, sizeof(buf)); CuAssertStrEquals(tc, "Hodor (1), TWW (2), 1 human.", buf); /* ... but not if they are anonymous */ u->flags |= UFL_ANON_FACTION; - bufunit(f2, u, seen_unit, buf, sizeof(buf)); + bufunit_depr(f2, u, seen_unit, buf, sizeof(buf)); CuAssertStrEquals(tc, "Hodor (1), anonymous, 1 human.", buf); u->flags &= ~UFL_ANON_FACTION; /* we see the same thing as them when we are an ally */ ally_set(&f1->allies, f2, HELP_FSTEALTH); - bufunit(f2, u, seen_unit, buf, sizeof(buf)); + bufunit_depr(f2, u, seen_unit, buf, sizeof(buf)); CuAssertStrEquals(tc, "Hodor (1), TWW (2) (UFO (1)), 1 human.", buf); /* ... also when they are anonymous */ u->flags |= UFL_ANON_FACTION; - bufunit(f2, u, seen_unit, buf, sizeof(buf)); + bufunit_depr(f2, u, seen_unit, buf, sizeof(buf)); CuAssertStrEquals(tc, "Hodor (1), anonymous, 1 human.", buf); u->flags &= ~UFL_ANON_FACTION; /* fstealth has no influence when we are allies, same results again */ set_factionstealth(u, NULL); - bufunit(f2, u, seen_unit, buf, sizeof(buf)); + bufunit_depr(f2, u, seen_unit, buf, sizeof(buf)); CuAssertStrEquals(tc, "Hodor (1), UFO (1), 1 human.", buf); u->flags |= UFL_ANON_FACTION; - bufunit(f2, u, seen_unit, buf, sizeof(buf)); + bufunit_depr(f2, u, seen_unit, buf, sizeof(buf)); CuAssertStrEquals(tc, "Hodor (1), anonymous, 1 human.", buf); u->flags &= ~UFL_ANON_FACTION; @@ -277,20 +277,20 @@ static void test_bufunit(CuTest *tc) { unit_setname(u, "Hodor"); unit_setid(u, 1); - bufunit(u->faction, u, 0, buffer, sizeof(buffer)); + bufunit_depr(u->faction, u, 0, buffer, sizeof(buffer)); CuAssertStrEquals(tc, "Hodor (1), 1 human, aggressiv.", buffer); set_level(u, SK_ALCHEMY, 1); - bufunit(u->faction, u, 0, buffer, sizeof(buffer)); + bufunit_depr(u->faction, u, 0, buffer, sizeof(buffer)); CuAssertStrEquals(tc, "Hodor (1), 1 human, aggressiv, Talente: Alchemie 2.", buffer); set_level(u, SK_SAILING, 1); - bufunit(u->faction, u, 0, buffer, sizeof(buffer)); + bufunit_depr(u->faction, u, 0, buffer, sizeof(buffer)); CuAssertStrEquals(tc, "Hodor (1), 1 human, aggressiv, Talente: Alchemie 2, Segeln 1.", buffer); f = test_create_faction(NULL); f->locale = get_or_create_locale("de"); - bufunit(f, u, 0, buffer, sizeof(buffer)); + bufunit_depr(f, u, 0, buffer, sizeof(buffer)); CuAssertStrEquals(tc, "Hodor (1), UFO (1), 1 human.", buffer); test_teardown(); diff --git a/src/spells.c b/src/spells.c index 7707a05b7..b29e86ea9 100644 --- a/src/spells.c +++ b/src/spells.c @@ -592,7 +592,7 @@ static int sp_summon_familiar(castorder * co) msg_release(msg); make_familiar(caster, r, rc, zText); - report_race_skills(rc, zText, sizeof(zText), caster->faction->locale); + report_race_skills_depr(rc, zText, sizeof(zText), caster->faction->locale); ADDMSG(&caster->faction->msgs, msg_message("familiar_describe", "mage race skills", caster, rc, zText)); return cast_level; @@ -3790,7 +3790,7 @@ static int sp_migranten(castorder * co) return 0; } /* maximal Stufe Personen */ - if (target->number > cast_level || target->number > max_spellpoints(r, mage)) { + if (target->number > cast_level || target->number > max_spellpoints_depr(r, mage)) { ADDMSG(&mage->faction->msgs, msg_feedback(mage, co->order, "spellfail_toomanytargets", "")); return 0; diff --git a/src/spy.c b/src/spy.c index 02590bd77..c0d4d614c 100644 --- a/src/spy.c +++ b/src/spy.c @@ -67,7 +67,7 @@ void spy_message(int spy, const unit * u, const unit * target) { char status[32]; - report_status(target, u->faction->locale, status, sizeof(status)); + report_status_depr(target, u->faction->locale, status, sizeof(status)); ADDMSG(&u->faction->msgs, msg_message("spyreport", "spy target status", u, target, status)); diff --git a/src/triggers/shock.c b/src/triggers/shock.c index 9a9d1c74b..0d0177025 100644 --- a/src/triggers/shock.c +++ b/src/triggers/shock.c @@ -67,7 +67,7 @@ static void do_shock(unit * u, const char *reason) /* Aura - Verlust */ if (is_mage(u)) { - int aura = max_spellpoints(u->region, u) / 10; + int aura = max_spellpoints_depr(u->region, u) / 10; int now = get_spellpoints(u); if (now > aura) { set_spellpoints(u, aura); diff --git a/src/util/strings.c b/src/util/strings.c index e9434fb25..4eac7856e 100644 --- a/src/util/strings.c +++ b/src/util/strings.c @@ -18,6 +18,7 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #ifdef _MSC_VER #include +#define HAVE__ITOA #endif #include "strings.h" @@ -34,6 +35,30 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. #include #endif +const char* str_itoab(int val, int base) +{ + static char buf[32] = { 0 }; +#ifdef HAVE__ITOAB + return _itoa(val, buf, base); +#else + int i = 30; + for (; val && i; --i, val /= base) { + buf[i] = "0123456789abcdefghijklmnopqrstuvwxyz"[val % base]; + } + return &buf[i + 1]; +#endif +} + +const char *str_itoa(int n) +{ +#ifdef HAVE__ITOA + static char buf[32] = { 0 }; + return _itoa(n, buf, 10); +#else + return str_itoab(n, 10); +#endif +} + size_t str_strlcpy(char *dst, const char *src, size_t len) { #ifdef HAVE_BSDSTRING diff --git a/src/util/strings.h b/src/util/strings.h index bd58c0eb2..bf6299332 100644 --- a/src/util/strings.h +++ b/src/util/strings.h @@ -28,6 +28,8 @@ extern "C" { void str_replace(char *buffer, size_t size, const char *tmpl, const char *var, const char *value); int str_hash(const char *s); + const char *str_itoa(int i); + const char *str_itoab(int i, int base); size_t str_slprintf(char * dst, size_t size, const char * format, ...); size_t str_strlcpy(char *dst, const char *src, size_t len); size_t str_strlcat(char *dst, const char *src, size_t len);