boost fast_pool_allocator иногда запрашивает огромное выделение
У меня есть многопоточное приложение, которое использует fast_pool_allocator boost (версия 1.55) под quickfix (1.13.3). Приложение распределяет большое количество объектов в течение дня, увеличиваясь более или менее линейно до тех пор, пока мы не выключим его, используя к концу дня порядка 32 ГБ виртуальной памяти. Большинство выделенных ресурсов составляют порядка 200 МБ, так как мы наблюдаем рост виртуальной памяти приложения. Но затем, в какой-то момент, обычно позже в тот же день, boost решает выделить 6 ГБ, хотя поток транзакций через приложение существенно не меняется.
Глядя на код распределителя, первое, что делает повышение, после распределения выделяет новый блок на куски. Функция simple_segregated_storage<SizeType>::segregate
at simple_segregated_storage.hpp:280. Мы подключили к процессу отладчик и заметили, что когда происходит гигантское распределение, выполнение этой функции (что неудивительно) занимает много времени, особенно цикл for в строке 302. Это занимает 20-30 секунд, и этот код защищен мьютексом, поэтому каждый другой поток пытается что-то сделать в блоках распределителя. Это злит наших клиентов.
Вопросы:
- Зачем ему внезапно выделять 6 ГБ, если до этого он постоянно запрашивал блоки по ~200 МБ?
- Можно ли как-то ограничить распределение? Я бы предпочел, чтобы он попросил меньшие кусочки чаще.
- Это неправильный распределитель? Я полагаю, что это вопрос для разработчиков быстрых исправлений, но это, кажется, их предпочтительный путь. Объекты, которые используют распределитель, в основном
std::map
а такжеstd::multimap
,
1 ответ
Как предложено выше Yakk, оказывается, что пул наддува позволяет вам указать MaxSize в шаблоне. Немного странно, что он работает в единицах "чанков" - концепции, внутренней для реализации пула. ИМХО байты были бы более естественными, но так и будет.
fast_pool_allocator
определяется значениями по умолчанию для всех аргументов шаблона (кроме первого). Я скопировал эти значения по умолчанию и изменил последний. Для этого приложения размер фрагмента составляет 88 байтов, но я предполагаю, что это зависит от класса на карте.
typedef std::map <int, std::vector <FieldMap*>, std::less<int>,
boost::fast_pool_allocator<std::pair<const int, std::vector<FieldMap*>>,
boost::default_user_allocator_new_delete,
boost::details::pool::default_mutex,
32, // NextSize (default from boost)
1500000 // MaxSize, in 88 byte chunks.
>> Groups;
В этом случае 1500000 * 88 = 132M, что примерно столько, сколько я хочу, чтобы распределение было на данный момент.