浏览代码

totalValue: fix overflow

Fixes #1715
Konstantin 2 年之前
父节点
当前提交
54b1b49f5a
共有 1 个文件被更改,包括 9 次插入19 次删除
  1. 9 19
      lib/HeroBonus.cpp

+ 9 - 19
lib/HeroBonus.cpp

@@ -455,7 +455,9 @@ int BonusList::totalValue() const
 		int indepMax = std::numeric_limits<int>::min();
 		int indepMax = std::numeric_limits<int>::min();
 	};
 	};
 
 
-	auto percent = [](int base, int percent) -> int {return (base * (100 + percent)) / 100;};
+	auto percent = [](int64_t base, int64_t percent) -> int {
+		return static_cast<int>(vstd::clamp((base * (100 + percent)) / 100, std::numeric_limits<int>::min(), std::numeric_limits<int>::max()));
+	};
 	std::array <BonusCollection, Bonus::BonusSource::NUM_BONUS_SOURCE> sources = {};
 	std::array <BonusCollection, Bonus::BonusSource::NUM_BONUS_SOURCE> sources = {};
 	BonusCollection any;
 	BonusCollection any;
 	bool hasIndepMax = false;
 	bool hasIndepMax = false;
@@ -508,30 +510,18 @@ int BonusList::totalValue() const
 	any.base += any.additive;
 	any.base += any.additive;
 	auto valFirst = percent(any.base ,any.percentToAll);
 	auto valFirst = percent(any.base ,any.percentToAll);
 
 
-	if(hasIndepMin && hasIndepMax)
-		assert(any.indepMin < any.indepMax);
+	if(hasIndepMin && hasIndepMax && any.indepMin < any.indepMax)
+		any.indepMax = any.indepMin;
 
 
 	const int notIndepBonuses = static_cast<int>(std::count_if(bonuses.cbegin(), bonuses.cend(), [](const std::shared_ptr<Bonus>& b)
 	const int notIndepBonuses = static_cast<int>(std::count_if(bonuses.cbegin(), bonuses.cend(), [](const std::shared_ptr<Bonus>& b)
 	{
 	{
 		return b->valType != Bonus::INDEPENDENT_MAX && b->valType != Bonus::INDEPENDENT_MIN;
 		return b->valType != Bonus::INDEPENDENT_MAX && b->valType != Bonus::INDEPENDENT_MIN;
 	}));
 	}));
 
 
-	if (hasIndepMax)
-	{
-		if(notIndepBonuses)
-			vstd::amax(valFirst, any.indepMax);
-		else
-			valFirst = any.indepMax;
-	}
-	if (hasIndepMin)
-	{
-		if(notIndepBonuses)
-			vstd::amin(valFirst, any.indepMin);
-		else
-			valFirst = any.indepMin;
-	}
-
-	return valFirst;
+	if(notIndepBonuses)
+		return vstd::clamp(valFirst, any.indepMax, any.indepMin);
+	
+	return hasIndepMin ? any.indepMin : hasIndepMax ? any.indepMax : 0;
 }
 }
 
 
 std::shared_ptr<Bonus> BonusList::getFirst(const CSelector &select)
 std::shared_ptr<Bonus> BonusList::getFirst(const CSelector &select)