mirror of
https://github.com/boostorg/beast.git
synced 2025-08-01 05:44:38 +02:00
Replace static_string in parser
close #1574 This change yields a modest performance improment of 1-2% by replacing the exception-based handling of buffer overflow with one based on regular conditional checks. Signed-off-by: Damian Jarek <damian.jarek93@gmail.com>
This commit is contained in:
committed by
Vinnie Falco
parent
b7a8fb5178
commit
2cfe3ba1b8
@@ -4,6 +4,7 @@ Version 251:
|
|||||||
* detect_ssl uses bool
|
* detect_ssl uses bool
|
||||||
* launder pointers
|
* launder pointers
|
||||||
* Fix compilation on MSVC with std::string_view
|
* Fix compilation on MSVC with std::string_view
|
||||||
|
* Replace static_string in parser
|
||||||
|
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
@@ -16,8 +16,7 @@
|
|||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
#include <initializer_list>
|
#include <initializer_list>
|
||||||
#include <iterator>
|
#include <iosfwd>
|
||||||
#include <ostream>
|
|
||||||
#include <stdexcept>
|
#include <stdexcept>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <type_traits>
|
#include <type_traits>
|
||||||
|
@@ -10,7 +10,6 @@
|
|||||||
#ifndef BOOST_BEAST_HTTP_DETAIL_BASIC_PARSER_HPP
|
#ifndef BOOST_BEAST_HTTP_DETAIL_BASIC_PARSER_HPP
|
||||||
#define BOOST_BEAST_HTTP_DETAIL_BASIC_PARSER_HPP
|
#define BOOST_BEAST_HTTP_DETAIL_BASIC_PARSER_HPP
|
||||||
|
|
||||||
#include <boost/beast/core/static_string.hpp>
|
|
||||||
#include <boost/beast/core/string.hpp>
|
#include <boost/beast/core/string.hpp>
|
||||||
#include <boost/beast/http/error.hpp>
|
#include <boost/beast/http/error.hpp>
|
||||||
#include <boost/beast/http/detail/rfc7230.hpp>
|
#include <boost/beast/http/detail/rfc7230.hpp>
|
||||||
@@ -24,6 +23,59 @@ namespace beast {
|
|||||||
namespace http {
|
namespace http {
|
||||||
namespace detail {
|
namespace detail {
|
||||||
|
|
||||||
|
template <std::size_t N>
|
||||||
|
class char_buffer
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
bool try_push_back(char c)
|
||||||
|
{
|
||||||
|
if (size_ == N)
|
||||||
|
return false;
|
||||||
|
buf_[size_++] = c;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool try_append(char const* first, char const* last)
|
||||||
|
{
|
||||||
|
std::size_t const n = last - first;
|
||||||
|
if (n > N - size_)
|
||||||
|
return false;
|
||||||
|
std::memmove(&buf_[size_], first, n);
|
||||||
|
size_ += n;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void clear() noexcept
|
||||||
|
{
|
||||||
|
size_ = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
char* data() noexcept
|
||||||
|
{
|
||||||
|
return buf_;
|
||||||
|
}
|
||||||
|
|
||||||
|
char const* data() const noexcept
|
||||||
|
{
|
||||||
|
return buf_;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::size_t size() const noexcept
|
||||||
|
{
|
||||||
|
return size_;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool empty() const noexcept
|
||||||
|
{
|
||||||
|
return size_ == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::size_t size_= 0;
|
||||||
|
char buf_[N];
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
struct basic_parser_base
|
struct basic_parser_base
|
||||||
{
|
{
|
||||||
// limit on the size of the obs-fold buffer
|
// limit on the size of the obs-fold buffer
|
||||||
@@ -182,7 +234,7 @@ struct basic_parser_base
|
|||||||
char const* last,
|
char const* last,
|
||||||
string_view& name,
|
string_view& name,
|
||||||
string_view& value,
|
string_view& value,
|
||||||
static_string<max_obs_fold>& buf,
|
char_buffer<max_obs_fold>& buf,
|
||||||
error_code& ec);
|
error_code& ec);
|
||||||
|
|
||||||
BOOST_BEAST_DECL
|
BOOST_BEAST_DECL
|
||||||
|
@@ -491,7 +491,7 @@ parse_field(
|
|||||||
char const* last,
|
char const* last,
|
||||||
string_view& name,
|
string_view& name,
|
||||||
string_view& value,
|
string_view& value,
|
||||||
static_string<max_obs_fold>& buf,
|
char_buffer<max_obs_fold>& buf,
|
||||||
error_code& ec)
|
error_code& ec)
|
||||||
{
|
{
|
||||||
/* header-field = field-name ":" OWS field-value OWS
|
/* header-field = field-name ":" OWS field-value OWS
|
||||||
@@ -607,13 +607,14 @@ parse_field(
|
|||||||
if(token_last != first)
|
if(token_last != first)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
buf.resize(0);
|
buf.clear();
|
||||||
buf.append(first, token_last);
|
if (!buf.try_append(first, token_last))
|
||||||
BOOST_ASSERT(! buf.empty());
|
|
||||||
#ifndef BOOST_NO_EXCEPTIONS
|
|
||||||
try
|
|
||||||
#endif
|
|
||||||
{
|
{
|
||||||
|
ec = error::header_limit;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOST_ASSERT(! buf.empty());
|
||||||
for(;;)
|
for(;;)
|
||||||
{
|
{
|
||||||
// eat leading ' ' and '\t'
|
// eat leading ' ' and '\t'
|
||||||
@@ -646,8 +647,12 @@ parse_field(
|
|||||||
token_last = trim_back(token_last, first);
|
token_last = trim_back(token_last, first);
|
||||||
if(first != token_last)
|
if(first != token_last)
|
||||||
{
|
{
|
||||||
buf.push_back(' ');
|
if (!buf.try_push_back(' ') ||
|
||||||
buf.append(first, token_last);
|
!buf.try_append(first, token_last))
|
||||||
|
{
|
||||||
|
ec = error::header_limit;
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if(*p != ' ' && *p != '\t')
|
if(*p != ' ' && *p != '\t')
|
||||||
{
|
{
|
||||||
@@ -656,14 +661,6 @@ parse_field(
|
|||||||
}
|
}
|
||||||
++p;
|
++p;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
#ifndef BOOST_NO_EXCEPTIONS
|
|
||||||
catch(std::length_error const&)
|
|
||||||
{
|
|
||||||
ec = error::header_limit;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@@ -14,7 +14,6 @@
|
|||||||
#include <boost/beast/http/error.hpp>
|
#include <boost/beast/http/error.hpp>
|
||||||
#include <boost/beast/http/rfc7230.hpp>
|
#include <boost/beast/http/rfc7230.hpp>
|
||||||
#include <boost/beast/core/buffer_traits.hpp>
|
#include <boost/beast/core/buffer_traits.hpp>
|
||||||
#include <boost/beast/core/static_string.hpp>
|
|
||||||
#include <boost/beast/core/detail/clamp.hpp>
|
#include <boost/beast/core/detail/clamp.hpp>
|
||||||
#include <boost/beast/core/detail/config.hpp>
|
#include <boost/beast/core/detail/config.hpp>
|
||||||
#include <boost/asio/buffer.hpp>
|
#include <boost/asio/buffer.hpp>
|
||||||
@@ -405,7 +404,7 @@ parse_fields(char const*& in,
|
|||||||
string_view name;
|
string_view name;
|
||||||
string_view value;
|
string_view value;
|
||||||
// https://stackoverflow.com/questions/686217/maximum-on-http-header-values
|
// https://stackoverflow.com/questions/686217/maximum-on-http-header-values
|
||||||
static_string<max_obs_fold> buf;
|
detail::char_buffer<max_obs_fold> buf;
|
||||||
auto p = in;
|
auto p = in;
|
||||||
for(;;)
|
for(;;)
|
||||||
{
|
{
|
||||||
|
Reference in New Issue
Block a user