Skip to content

Commit

Permalink
Fix heap use after free in components/lua/storage.cpp
Browse files Browse the repository at this point in the history
  • Loading branch information
petrmikheev committed Jan 23, 2022
1 parent c263bbf commit 067d71f
Show file tree
Hide file tree
Showing 3 changed files with 16 additions and 13 deletions.
2 changes: 1 addition & 1 deletion apps/openmw_test_suite/lua/test_storage.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -91,10 +91,10 @@ namespace
mLua.safe_script("permanent:set('z', 4)");

LuaUtil::LuaStorage storage2(mLua);
storage2.load(tmpFile);
mLua["permanent"] = storage2.getMutableSection("permanent");
mLua["temporary"] = storage2.getMutableSection("temporary");

storage2.load(tmpFile);
EXPECT_EQ(get<int>(mLua, "permanent:get('x')"), 1);
EXPECT_TRUE(get<bool>(mLua, "permanent:get('z') == nil"));
EXPECT_TRUE(get<bool>(mLua, "temporary:get('y') == nil"));
Expand Down
19 changes: 11 additions & 8 deletions components/lua/storage.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -121,15 +121,18 @@ namespace LuaUtil
while (it != mData.end())
{
if (!it->second->mPermanent)
{
it->second->mValues.clear();
it = mData.erase(it);
}
else
++it;
}
}

void LuaStorage::load(const std::string& path)
{
mData.clear();
assert(mData.empty()); // Shouldn't be used before loading
try
{
Log(Debug::Info) << "Loading Lua storage \"" << path << "\" (" << std::filesystem::file_size(path) << " bytes)";
Expand All @@ -138,7 +141,7 @@ namespace LuaUtil
sol::table data = deserialize(mLua, serializedData);
for (const auto& [sectionName, sectionTable] : data)
{
Section* section = getSection(sectionName.as<std::string_view>());
const std::shared_ptr<Section>& section = getSection(sectionName.as<std::string_view>());
for (const auto& [key, value] : sol::table(sectionTable))
section->set(key.as<std::string_view>(), value);
}
Expand All @@ -164,26 +167,26 @@ namespace LuaUtil
fout.close();
}

LuaStorage::Section* LuaStorage::getSection(std::string_view sectionName)
const std::shared_ptr<LuaStorage::Section>& LuaStorage::getSection(std::string_view sectionName)
{
auto it = mData.find(sectionName);
if (it != mData.end())
return it->second.get();
auto section = std::make_unique<Section>(this, std::string(sectionName));
return it->second;
auto section = std::make_shared<Section>(this, std::string(sectionName));
sectionName = section->mSectionName;
auto [newIt, _] = mData.emplace(sectionName, std::move(section));
return newIt->second.get();
return newIt->second;
}

sol::object LuaStorage::getReadOnlySection(std::string_view sectionName)
{
Section* section = getSection(sectionName);
const std::shared_ptr<Section>& section = getSection(sectionName);
return sol::make_object<SectionReadOnlyView>(mLua, SectionReadOnlyView{section, section->mChangeCounter});
}

sol::object LuaStorage::getMutableSection(std::string_view sectionName)
{
Section* section = getSection(sectionName);
const std::shared_ptr<Section>& section = getSection(sectionName);
return sol::make_object<SectionMutableView>(mLua, SectionMutableView{section, section->mChangeCounter});
}

Expand Down
8 changes: 4 additions & 4 deletions components/lua/storage.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -60,19 +60,19 @@ namespace LuaUtil
};
struct SectionMutableView
{
Section* mSection = nullptr;
std::shared_ptr<Section> mSection = nullptr;
int64_t mLastCheck = 0;
};
struct SectionReadOnlyView
{
Section* mSection = nullptr;
std::shared_ptr<Section> mSection = nullptr;
int64_t mLastCheck = 0;
};

Section* getSection(std::string_view sectionName);
const std::shared_ptr<Section>& getSection(std::string_view sectionName);

lua_State* mLua;
std::map<std::string_view, std::unique_ptr<Section>> mData;
std::map<std::string_view, std::shared_ptr<Section>> mData;
std::optional<ListenerFn> mListener;
};

Expand Down

0 comments on commit 067d71f

Please sign in to comment.