blob: 1793e277937534f627062e7806f20bd606787f20 [file] [log] [blame]
/***************************************************************************
*
* 25.for_each.cpp - test exercising 25.1.1 [lib.alg.foreach]
*
* $Id$
*
***************************************************************************
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed
* with this work for additional information regarding copyright
* ownership. The ASF licenses this file to you under the Apache
* License, Version 2.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of
* the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
* implied. See the License for the specific language governing
* permissions and limitations under the License.
*
* Copyright 1994-2006 Rogue Wave Software.
*
**************************************************************************/
#include <algorithm> // for for_each
#include <cstddef> // for ptrdiff_t
#include <alg_test.h>
#include <rw_value.h> // for UserClass
#include <driver.h> // for rw_test(), ...
/**************************************************************************/
struct FunctionBase
{
static std::size_t funcalls_;
static int sum_;
bool is_const_;
FunctionBase (bool is_const)
: is_const_ (is_const) {
funcalls_ = 0;
sum_ = 0;
}
private:
void operator= (FunctionBase&); // not assignable
};
std::size_t FunctionBase::funcalls_;
int FunctionBase::sum_;
struct ConstFunction: FunctionBase
{
// dummy arguments provided to prevent the class
// from being default constructible
ConstFunction (int, int): FunctionBase (true) { }
void operator() (UserClass val) /* not const */ {
++funcalls_;
sum_ += val.data_.val_;
}
static const char* name () { return "ConstFunction"; }
};
struct MutableFunction: FunctionBase
{
// dummy arguments provided to prevent the class
// from being default constructible
MutableFunction (int, int): FunctionBase (false) { }
void operator() (UserClass &val) /* not const */ {
++funcalls_;
val.data_.val_ = -val.data_.val_;
}
static const char* name () { return "MutableFunction"; }
};
// exercises std::for_each()
template <class InputIterator, class T, class Function>
void test_for_each (std::size_t N, InputIterator dummy, T*, Function*)
{
static const char* const itname = type_name (dummy, (T*)0);
static const char* const fnname = Function::name ();
rw_info (0, 0, 0, "std::for_each (%s, %1$s, %s)", itname, fnname);
// generate sequential values for each default constructed UserClass
UserClass::gen_ = gen_seq;
UserClass *buf = new UserClass [N];
const int first_val = buf [0].data_.val_;
for (std::size_t i = 0; i != N; ++i) {
UserClass* const buf_end = buf + i;
const InputIterator first (buf, buf, buf_end);
const InputIterator last (buf_end, buf_end, buf_end);
// create a const function object and zero out all its counters
const Function fun (0, 0);
std::for_each (first, last, fun);
// compute the sum of elements in the sequence and check each
// element's value against the expected one
int sum = 0;
for (std::size_t j = 0; j != i; ++j) {
int expect;
if (fun.is_const_) {
// const function object doesn't modify the subject
// sequence; the expected value of the element is
// the same as the original value
expect = first_val + int (j);
}
else {
// non-const function object negates each argument
expect = -(first_val + int (j));
}
// compute the sum (computed by the const function object)
sum += buf [j].data_.val_;
// assert the element value as the same as the expected value
rw_assert (expect == buf [j].data_.val_, 0, __LINE__,
"for_each (%s, %1$s, %s); element [%zu] == %d, got %d",
itname, fnname, j, expect, buf [j].data_.val_);
if (expect != buf [j].data_.val_) {
// break out of both loops on failure
i = N;
break;
}
// restore the original value of the element
buf [j].data_.val_ = first_val + int (j);
}
// assert that for_each invoked the function object's operator()
// exactly as many times as necessary and required
rw_assert (i == fun.funcalls_, 0, __LINE__,
"for_each (%s, %1$s, %s); expected %zu invocations of "
"Function::operator(), got %zu", itname, fnname,
i, fun.funcalls_);
if (fun.is_const_) {
rw_assert (sum == fun.sum_, 0, __LINE__,
"for_each (%s, %1$s, %s); sum of %zu elements == %d, "
"got %d", itname, fnname, sum, fun.sum_);
}
}
delete[] (buf);
}
/**************************************************************************/
static int rw_opt_nloops = 32; // --nloops=#
static int rw_opt_no_input_iter; // --no-InputIterator
static int rw_opt_no_fwd_iter; // --no-ForwardIterator
static int rw_opt_no_bidir_iter; // --no-BidirectionalIterator
static int rw_opt_no_rnd_iter; // --no-RandomAccessIterator
static int
run_test (int, char*[])
{
const std::size_t N = std::size_t (rw_opt_nloops);
rw_info (0, 0, 0,
"template <class %s, class %s> "
"%2$s std::for_each (%1$s, %1$s, %2$s)",
"InputIterator", "Function");
if (rw_opt_no_input_iter) {
rw_note (0, __FILE__, __LINE__, "InputIterator test disabled");
}
else {
test_for_each (N, InputIter<UserClass>(0, 0, 0), (UserClass*)0,
(ConstFunction*)0);
}
if (rw_opt_no_fwd_iter) {
rw_note (0, __FILE__, __LINE__, "ForwardIterator test disabled");
}
else {
test_for_each (N, ConstFwdIter<UserClass>(), (UserClass*)0,
(ConstFunction*)0);
test_for_each (N, FwdIter<UserClass>(), (UserClass*)0,
(ConstFunction*)0);
test_for_each (N, FwdIter<UserClass>(), (UserClass*)0,
(MutableFunction*)0);
}
if (rw_opt_no_bidir_iter) {
rw_note (0, __FILE__, __LINE__, "BidirectionalIterator test disabled");
}
else {
test_for_each (N, ConstBidirIter<UserClass>(), (UserClass*)0,
(ConstFunction*)0);
test_for_each (N, BidirIter<UserClass>(), (UserClass*)0,
(ConstFunction*)0);
test_for_each (N, BidirIter<UserClass>(), (UserClass*)0,
(MutableFunction*)0);
}
if (rw_opt_no_rnd_iter) {
rw_note (0, __FILE__, __LINE__, "RandomAccessIterator test disabled");
}
else {
test_for_each (N, ConstRandomAccessIter<UserClass>(), (UserClass*)0,
(ConstFunction*)0);
test_for_each (N, RandomAccessIter<UserClass>(), (UserClass*)0,
(ConstFunction*)0);
test_for_each (N, RandomAccessIter<UserClass>(), (UserClass*)0,
(MutableFunction*)0);
}
return 0;
}
/**************************************************************************/
int main (int argc, char *argv[])
{
return rw_test (argc, argv, __FILE__,
"lib.alg.foreach",
0 /* no comment */, run_test,
"|-nloops#0 " // must be non-negative
"|-no-InputIterator# "
"|-no-ForwardIterator# "
"|-no-BidirectionalIterator# "
"|-no-RandomAccessIterator#",
&rw_opt_nloops,
&rw_opt_no_input_iter,
&rw_opt_no_fwd_iter,
&rw_opt_no_bidir_iter,
&rw_opt_no_rnd_iter);
}