From 877b3a3dcd35c6a30e90dfc0b4a736ced5e2da9c Mon Sep 17 00:00:00 2001 From: Santiago Gimeno Date: Thu, 2 Nov 2023 11:04:29 +0100 Subject: [PATCH] src: handle bad allocation errors PR-URL: https://github.com/nodesource/nsolid/pull/20 Reviewed-by: Trevor Norris --- src/nsolid.h | 167 ++++++++++++++++++++++++++++++++++----------------- 1 file changed, 111 insertions(+), 56 deletions(-) diff --git a/src/nsolid.h b/src/nsolid.h index 914269d7cf..25411b66f1 100644 --- a/src/nsolid.h +++ b/src/nsolid.h @@ -989,9 +989,13 @@ int ThreadMetrics::Update(Cb&& cb, Data&&... data) { } // _1 - ThreadMetrics* - UserData* user_data = new UserData( + UserData* user_data = new (std::nothrow) UserData( std::bind(std::forward(cb), _1, std::forward(data)...)); + if (user_data == nullptr) { + return UV_ENOMEM; + } + user_data_ = user_data; proxy_ = thread_metrics_proxy_; stor_.thread_id = thread_id_; @@ -1032,8 +1036,13 @@ MetricsStream* MetricsStream::CreateInstance(uint32_t flags, std::forward(cb), _1, _2, std::forward(data)...)); // _1 - MetricsStream* metrics_stream // _2 - const metrics_stream_bucket& bucket - UserData* user_data = new UserData(std::bind( - std::forward(cb), _1, _2, std::forward(data)...)); + UserData* user_data = new (std::nothrow) UserData( + std::bind(std::forward(cb), _1, _2, std::forward(data)...)); + + if (user_data == nullptr) { + return nullptr; + } + stream->DoSetup(flags, metrics_stream_proxy_, internal::delete_proxy_, @@ -1062,8 +1071,13 @@ Tracer* Tracer::CreateInstance(uint32_t flags, Cb&& cb, Data&&... data) { std::forward(cb), _1, _2, std::forward(data)...)); // _1 - Tracer* // _2 - const SpanStor& - UserData* user_data = new UserData(std::bind( - std::forward(cb), _1, _2, std::forward(data)...)); + UserData* user_data = new (std::nothrow) UserData( + std::bind(std::forward(cb), _1, _2, std::forward(data)...)); + + if (user_data == nullptr) { + return nullptr; + } + tracer->DoSetup(flags, trace_proxy_, internal::delete_proxy_, @@ -1092,16 +1106,22 @@ int CpuProfiler::TakeProfile(SharedEnvInst envinst, // _1 - int status // _2 - std::string json - std::unique_ptr user_data = std::make_unique(std::bind( - std::forward(cb), _1, _2, std::forward(data)...)); + UserData* user_data = new (std::nothrow) UserData( + std::bind(std::forward(cb), _1, _2, std::forward(data)...)); + + if (user_data == nullptr) { + return UV_ENOMEM; + } int er = get_cpu_profile_(envinst, duration, - user_data.get(), + user_data, cpu_profiler_proxy_, internal::delete_proxy_); - if (!er) - user_data.release(); + if (er) { + delete user_data; + } + return er; } @@ -1126,16 +1146,23 @@ int Snapshot::TakeSnapshot(SharedEnvInst envinst, // _1 - int status // _2 - std::string json - std::unique_ptr user_data = std::make_unique(std::bind( - std::forward(cb), _1, _2, std::forward(data)...)); + UserData* user_data = new (std::nothrow) UserData( + std::bind(std::forward(cb), _1, _2, std::forward(data)...)); + + if (user_data == nullptr) { + return UV_ENOMEM; + } int er = get_snapshot_(envinst, redacted, - user_data.get(), + user_data, snapshot_proxy_, internal::delete_proxy_); - if (!er) - user_data.release(); + + if (er) { + delete user_data; + } + return er; } @@ -1151,14 +1178,20 @@ int QueueCallback(Cb&& cb, Data&&... data) { using UserData = decltype(std::bind( std::forward(cb), std::forward(data)...)); - std::unique_ptr user_data = std::make_unique(std::bind( - std::forward(cb), std::forward(data)...)); + UserData* user_data = new (std::nothrow) + UserData(std::bind(std::forward(cb), std::forward(data)...)); + + if (user_data == nullptr) { + return UV_ENOMEM; + } + + int er = internal::queue_callback_(user_data, + internal::queue_callback_proxy_); + + if (er) { + delete user_data; + } - int er = internal::queue_callback_( - user_data.get(), - internal::queue_callback_proxy_); - if (!er) - user_data.release(); return er; } @@ -1168,15 +1201,20 @@ int QueueCallback(uint64_t timeout, Cb&& cb, Data&&... data) { using UserData = decltype(std::bind( std::forward(cb), std::forward(data)...)); - std::unique_ptr user_data = std::make_unique(std::bind( - std::forward(cb), std::forward(data)...)); + UserData* user_data = new (std::nothrow) + UserData(std::bind(std::forward(cb), std::forward(data)...)); + + if (user_data == nullptr) { + return UV_ENOMEM; + } int er = internal::queue_callback_( - timeout, - user_data.get(), - internal::queue_callback_proxy_); - if (!er) - user_data.release(); + timeout, user_data, internal::queue_callback_proxy_); + + if (er) { + delete user_data; + } + return er; } @@ -1191,17 +1229,20 @@ int RunCommand(SharedEnvInst envinst, using UserData = decltype(std::bind( std::forward(cb), _1, std::forward(data)...)); // _1 - SharedEnvInst - std::unique_ptr user_data = std::make_unique(std::bind( - std::forward(cb), _1, std::forward(data)...)); + UserData* user_data = new (std::nothrow) UserData( + std::bind(std::forward(cb), _1, std::forward(data)...)); + + if (user_data == nullptr) { + return UV_ENOMEM; + } int er = internal::run_command_( - envinst, - type, - user_data.get(), - internal::run_command_proxy_); + envinst, type, user_data, internal::run_command_proxy_); + + if (er) { + delete user_data; + } - if (!er) - user_data.release(); return 0; } @@ -1223,17 +1264,24 @@ int CustomCommand(SharedEnvInst envinst, // _3 - int status // _4 - std::pair error // _5 - std::pair value - std::unique_ptr user_data = std::make_unique(std::bind( - std::forward(cb), _1, _2, _3, _4, _5, std::forward(data)...)); - int er = internal::custom_command_( - envinst, - req_id, - command, - args, - user_data.get(), - internal::custom_command_proxy_); - if (!er) - user_data.release(); + UserData* user_data = new (std::nothrow) UserData(std::bind( + std::forward(cb), _1, _2, _3, _4, _5, std::forward(data)...)); + + if (user_data == nullptr) { + return UV_ENOMEM; + } + + int er = internal::custom_command_(envinst, + req_id, + command, + args, + user_data, + internal::custom_command_proxy_); + + if (er) { + delete user_data; + } + return er; } @@ -1247,15 +1295,21 @@ int AtExitHook(Cb&& cb, Data&&... data) { // _1 - bool on_signal // _2 - bool profile_stopped - std::unique_ptr user_data = std::make_unique(std::bind( - std::forward(cb), _1, _2, std::forward(data)...)); + UserData* user_data = new (std::nothrow) UserData( + std::bind(std::forward(cb), _1, _2, std::forward(data)...)); + + if (user_data == nullptr) { + return UV_ENOMEM; + } + + int er = internal::at_exit_hook_(user_data, + internal::at_exit_hook_proxy_, + internal::delete_proxy_); + + if (er) { + delete user_data; + } - int er = internal::at_exit_hook_( - user_data.get(), - internal::at_exit_hook_proxy_, - internal::delete_proxy_); - if (!er) - user_data.release(); return er; } @@ -1271,6 +1325,7 @@ int OnBlockedLoopHook(uint64_t threshold, Cb&& cb, Data&&... data) { // _2 - std::string info UserData* user_data = new (std::nothrow) UserData(std::bind( std::forward(cb), _1, _2, std::forward(data)...)); + if (user_data == nullptr) { return UV_ENOMEM; }