#define ALLOWED_LOOSE_TURFS 500 /** * Responsible for managing the sizes of area.contained_turfs and area.turfs_to_uncontain * These lists do not check for duplicates, which is fine, but it also means they can balloon in size over time * as a consequence of repeated changes in area in a space * They additionally may not always resolve often enough to avoid memory leaks * This is annoying, so lets keep an eye on them and cut them down to size if needed */ SUBSYSTEM_DEF(area_contents) name = "Area Contents" flags = SS_NO_INIT runlevels = RUNLEVEL_LOBBY|RUNLEVELS_DEFAULT var/list/currentrun var/list/area/marked_for_clearing = list() /datum/controller/subsystem/area_contents/stat_entry(msg) var/total_clearing_from = 0 var/total_to_clear = 0 for(var/area/to_clear as anything in marked_for_clearing) total_to_clear += length(to_clear.turfs_to_uncontain) total_clearing_from += length(to_clear.contained_turfs) msg = "A:[length(currentrun)] MR:[length(marked_for_clearing)] TC:[total_to_clear] CF:[total_clearing_from]" return ..() /datum/controller/subsystem/area_contents/get_metrics() . = ..() var/total_clearing_from = 0 var/total_to_clear = 0 for(var/area/to_clear as anything in marked_for_clearing) total_to_clear += length(to_clear.turfs_to_uncontain) total_clearing_from += length(to_clear.contained_turfs) .["areas"] = length(currentrun) .["marked_for_clearing"] = length(marked_for_clearing) .["total_to_clear"] = total_to_clear .["total_clearing_from"] = total_clearing_from /datum/controller/subsystem/area_contents/fire(resumed) if(!resumed) currentrun = GLOB.areas.Copy() while(length(currentrun)) var/area/test = currentrun[length(currentrun)] if(length(test.turfs_to_uncontain) > ALLOWED_LOOSE_TURFS) marked_for_clearing |= test currentrun.len-- if(MC_TICK_CHECK) return // Alright, if we've done a scan on all our areas, it's time to knock the existing ones down to size while(length(marked_for_clearing)) var/area/clear = marked_for_clearing[length(marked_for_clearing)] // The operation of cutting large lists can be expensive // It scales almost directly with the size of the list we're cutting with // Because of this, we're gonna stick to cutting 1 entry at a time // There's no reason to batch it I promise, this is faster. No overtime too var/amount_cut = 0 var/list/cut_from = clear.turfs_to_uncontain for(amount_cut in 1 to length(cut_from)) clear.contained_turfs -= cut_from[amount_cut] if(MC_TICK_CHECK) cut_from.Cut(1, amount_cut + 1) return clear.turfs_to_uncontain = list() marked_for_clearing.len--