summaryrefslogtreecommitdiff
path: root/grfspecial.c
diff options
context:
space:
mode:
authordarkvater <darkvater@openttd.org>2004-11-14 23:14:44 +0000
committerdarkvater <darkvater@openttd.org>2004-11-14 23:14:44 +0000
commitafca207cf085599ce543ca0d68cfd2ba5e5453ae (patch)
tree68b0c17d0e97a93d344bbb679b8bece49a1a43a9 /grfspecial.c
parent5914888ad6d7ba7a732790089f458bb703b0c4fd (diff)
downloadopenttd-afca207cf085599ce543ca0d68cfd2ba5e5453ae.tar.xz
(svn r616) -newgrf: Fixed crashing when trying to resolve references to unknown or callback result spritegroups in variational spritegroup handler of NewSpriteGroup() (pasky).
Diffstat (limited to 'grfspecial.c')
-rw-r--r--grfspecial.c24
1 files changed, 20 insertions, 4 deletions
diff --git a/grfspecial.c b/grfspecial.c
index a1b73bedc..6e0f723d2 100644
--- a/grfspecial.c
+++ b/grfspecial.c
@@ -1068,6 +1068,7 @@ static void NewSpriteGroup(byte *buf, int len)
if (numloaded == 0x81 || numloaded == 0x82) {
struct DeterministicSpriteGroup *dg;
+ uint16 groupid;
int i;
// Ok, this is gonna get a little wild, so hold your breath...
@@ -1104,20 +1105,35 @@ static void NewSpriteGroup(byte *buf, int len)
dg->divmod_val = grf_load_byte(&buf);
}
+ /* (groupid & 0x8000) means this is callback result; we happily
+ * ignore that for now. */
+
dg->num_ranges = grf_load_byte(&buf);
dg->ranges = calloc(dg->num_ranges, sizeof(*dg->ranges));
for (i = 0; i < dg->num_ranges; i++) {
- uint16 setid = grf_load_word(&buf);
-
+ groupid = grf_load_word(&buf);
+ if (groupid & 0x8000 || groupid >= _cur_grffile->spritegroups_count) {
+ /* This doesn't exist for us. */
+ i--; dg->num_ranges--;
+ continue;
+ }
/* XXX: If multiple surreal sets attach a surreal
* set this way, we are in trouble. */
- dg->ranges[i].group = _cur_grffile->spritegroups[setid];
+ dg->ranges[i].group = _cur_grffile->spritegroups[groupid];
dg->ranges[i].low = grf_load_byte(&buf);
dg->ranges[i].high = grf_load_byte(&buf);
}
+ groupid = grf_load_word(&buf);
+ if (groupid & 0x8000 || groupid >= _cur_grffile->spritegroups_count) {
+ /* This spritegroup stinks. */
+ free(dg->ranges);
+ grfmsg(GMS_WARN, "NewSpriteGroup(%02x:0x%x): Default groupid %04x is cargo callback or unknown, ignoring spritegroup.", setid, numloaded, groupid);
+ return;
+ }
+
dg->default_group = malloc(sizeof(*dg->default_group));
- memcpy(dg->default_group, &_cur_grffile->spritegroups[grf_load_word(&buf)], sizeof(*dg->default_group));
+ memcpy(dg->default_group, &_cur_grffile->spritegroups[groupid], sizeof(*dg->default_group));
return;