[Boost-bugs] [Boost C++ Libraries] #4400: BOOST_PP_SEQ_REPLACE fails in corner cases

Subject: [Boost-bugs] [Boost C++ Libraries] #4400: BOOST_PP_SEQ_REPLACE fails in corner cases
From: Boost C++ Libraries (noreply_at_[hidden])
Date: 2010-07-02 22:08:44


#4400: BOOST_PP_SEQ_REPLACE fails in corner cases
------------------------------------------+---------------------------------
 Reporter: Wolf Lammen <ookami1@…> | Owner: no-maintainer
     Type: Bugs | Status: new
Milestone: Boost 1.44.0 | Component: preprocessor
  Version: Boost 1.44.0 | Severity: Problem
 Keywords: |
------------------------------------------+---------------------------------
 The following code replaces individual elements (first the 255th, then the
 256th) in a given preprocessor sequence of length 256. I noticed that
 sometimes the element is not replaced, but inserted instead, yielding a
 longer sequence.

 Using Boost version 1.4.3, the following code will not compile

 --------------------------------------------------

 {{{
 /*
  * booster.cpp
  *
  * Created on: 25.06.2010
  * Author: Wolf Lammen
  */

 #include "boost/preprocessor.hpp"

 #define SEQ_256 \
         (1)(2)(3)(4)(5)(6)(7)(8)(9) \
         (10)(11)(12)(13)(14)(15)(16)(17)(18)(19) \
         (20)(21)(22)(23)(24)(25)(26)(27)(28)(29) \
         (30)(31)(32)(33)(34)(35)(36)(37)(38)(39) \
         (40)(41)(42)(43)(44)(45)(46)(47)(48)(49) \
         (50)(51)(52)(53)(54)(55)(56)(57)(58)(59) \
         (60)(61)(62)(63)(64)(65)(66)(67)(68)(69) \
         (70)(71)(72)(73)(74)(75)(76)(77)(78)(79) \
         (80)(81)(82)(83)(84)(85)(86)(87)(88)(89) \
         (90)(91)(92)(93)(94)(95)(96)(97)(98)(99) \
         (100)(101)(102)(103)(104)(105)(106)(107)(108)(109) \
         (110)(111)(112)(113)(114)(115)(116)(117)(118)(119) \
         (120)(121)(122)(123)(124)(125)(126)(127)(128)(129) \
         (130)(131)(132)(133)(134)(135)(136)(137)(138)(139) \
         (140)(141)(142)(143)(144)(145)(146)(147)(148)(149) \
         (150)(151)(152)(153)(154)(155)(156)(157)(158)(159) \
         (160)(161)(162)(163)(164)(165)(166)(167)(168)(169) \
         (170)(171)(172)(173)(174)(175)(176)(177)(178)(179) \
         (180)(181)(182)(183)(184)(185)(186)(187)(188)(189) \
         (190)(191)(192)(193)(194)(195)(196)(197)(198)(199) \
         (200)(201)(202)(203)(204)(205)(206)(207)(208)(209) \
         (210)(211)(212)(213)(214)(215)(216)(217)(218)(219) \
         (220)(221)(222)(223)(224)(225)(226)(227)(228)(229) \
         (230)(231)(232)(233)(234)(235)(236)(237)(238)(239) \
         (240)(241)(242)(243)(244)(245)(246)(247)(248)(249) \
         (250)(251)(252)(253)(254)(255)(256)

 #if BOOST_PP_SEQ_SIZE(BOOST_PP_SEQ_REPLACE(SEQ_256, 254, a)) != 256
 # error "replacement at 254: size of sequence has changed"
 #endif
 #if BOOST_PP_SEQ_SIZE(BOOST_PP_SEQ_REPLACE(SEQ_256, 255, a)) != 256
 # error "replacement at 255: size of sequence has changed"
 #endif

 int main()
 {
         return 0;
 }
 }}}

 --------------------------------------------------

 Here is the error message:

 Invoking: GCC C++ Compiler
 g++ -I"/home/wolf/workspace/booster" -O0 -g3 -Wall -c -fmessage-length=0
 -MMD -MP -MF"booster.d" -MT"booster.d" -o"booster.o" "../booster.cpp"
 ../booster.cpp:42:3: error: #error "replacement at 255: size of sequence
 has changed"

 Note that the first call to BOOST_PP_SEQ_REPLACE succeeded.

 I use the GNU Compiler collection version (Ubuntu 4.4.1-4ubuntu9) 4.4.1 on
 x86_64 GNU/Linux

 =================== ANALYSIS OF THE BUG ======================

 According to the documentation, valid ranges of lengths of boost
 preprocessor sequences are from 1 to BOOST_PP_LIMIT_SEQ (=256). Especially
 empty sequences are not allowed. This overall restriction imposes limits
 to parameters of macros related to sequences.
 In particular, you must not call

 {{{
 BOOST_PP_SEQ_REST_N(i, seq)
 }}}

 with i == BOOST_PP_SEQ_SIZE(seq), as this would force BOOST_PP_REST_N to
 return an empty sequence.
 BOOST_PP_SEQ_REPLACE does use BOOST_PP_REST_N in this way, thus relying on
 undefined behavior. Here is how BOOST_PP_SEQ_REPLACE is defined:

 {{{
 # define BOOST_PP_SEQ_REPLACE(seq, i, elem) BOOST_PP_SEQ_FIRST_N(i,
 seq) (elem) BOOST_PP_SEQ_REST_N(BOOST_PP_INC(i), seq)

 }}}
 If i points to the last element of the sequence, it equals (size(seq) - 1)
 and BOOST_PP_INC(i) yields size(seq), an invalid argument to
 BOOST_PP_REST_N.
 I called BOOST_PP_REST_N with an offset equal to size(seq) for various
 sequences, and even though this was outside of the specifications, it
 expanded to an empty string - most of the time. An empty string would have
 been fine for BOOST_PP_SEQ_REPLACE, but unfortunately, BOOST_PP_REST_N
 failed to do so for sequences of maximum size.
 It internally increments its first parameter, and since incrementation
 saturates at 256, the algorithm delivers a different result then.

 ========================= FIX ======================

 Here is a suggestion on how to fix this:


 Extend the domain of BOOST_PP_REST_N and allow the first parameter to be
 equal to the size of the sequence. Make the macro reliably expand to an
 empty string then:

 {{{
 #define BOOST_PP_SEQ_REST_N(n, seq) BOOST_PP_IF(n, BOOST_PP_TUPLE_ELEM(2,
 1, BOOST_PP_SEQ_SPLIT(BOOST_PP_INC(BOOST_PP_DEC(n)), seq
 BOOST_PP_EMPTY))(), seq)

 }}}
 This is quick and dirty, and not optimal, because BOOST_PP_TUPLE_ELEM is
 always evaluated, but according to my tests it works. As I am not an
 experienced macro programmer, I would like to leave the details to your
 experts.

 With kind regards
 Wolf Lammen

-- 
Ticket URL: <https://svn.boost.org/trac/boost/ticket/4400>
Boost C++ Libraries <http://www.boost.org/>
Boost provides free peer-reviewed portable C++ source libraries.

This archive was generated by hypermail 2.1.7 : 2017-02-16 18:50:03 UTC