100db7afdSDavid E. O'Brien // Temporary buffer implementation -*- C++ -*-
200db7afdSDavid E. O'Brien
3*f8a1b7d9SAlexander Kabaev // Copyright (C) 2001, 2002, 2003, 2004, 2005, 2006
4*f8a1b7d9SAlexander Kabaev // Free Software Foundation, Inc.
500db7afdSDavid E. O'Brien //
600db7afdSDavid E. O'Brien // This file is part of the GNU ISO C++ Library. This library is free
700db7afdSDavid E. O'Brien // software; you can redistribute it and/or modify it under the
800db7afdSDavid E. O'Brien // terms of the GNU General Public License as published by the
900db7afdSDavid E. O'Brien // Free Software Foundation; either version 2, or (at your option)
1000db7afdSDavid E. O'Brien // any later version.
1100db7afdSDavid E. O'Brien
1200db7afdSDavid E. O'Brien // This library is distributed in the hope that it will be useful,
1300db7afdSDavid E. O'Brien // but WITHOUT ANY WARRANTY; without even the implied warranty of
1400db7afdSDavid E. O'Brien // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
1500db7afdSDavid E. O'Brien // GNU General Public License for more details.
1600db7afdSDavid E. O'Brien
1700db7afdSDavid E. O'Brien // You should have received a copy of the GNU General Public License along
1800db7afdSDavid E. O'Brien // with this library; see the file COPYING. If not, write to the Free
19*f8a1b7d9SAlexander Kabaev // Software Foundation, 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301,
2000db7afdSDavid E. O'Brien // USA.
2100db7afdSDavid E. O'Brien
2200db7afdSDavid E. O'Brien // As a special exception, you may use this file as part of a free software
2300db7afdSDavid E. O'Brien // library without restriction. Specifically, if other files instantiate
2400db7afdSDavid E. O'Brien // templates or use macros or inline functions from this file, or you compile
2500db7afdSDavid E. O'Brien // this file and link it with other files to produce an executable, this
2600db7afdSDavid E. O'Brien // file does not by itself cause the resulting executable to be covered by
2700db7afdSDavid E. O'Brien // the GNU General Public License. This exception does not however
2800db7afdSDavid E. O'Brien // invalidate any other reasons why the executable file might be covered by
2900db7afdSDavid E. O'Brien // the GNU General Public License.
3000db7afdSDavid E. O'Brien
3100db7afdSDavid E. O'Brien /*
3200db7afdSDavid E. O'Brien *
3300db7afdSDavid E. O'Brien * Copyright (c) 1994
3400db7afdSDavid E. O'Brien * Hewlett-Packard Company
3500db7afdSDavid E. O'Brien *
3600db7afdSDavid E. O'Brien * Permission to use, copy, modify, distribute and sell this software
3700db7afdSDavid E. O'Brien * and its documentation for any purpose is hereby granted without fee,
3800db7afdSDavid E. O'Brien * provided that the above copyright notice appear in all copies and
3900db7afdSDavid E. O'Brien * that both that copyright notice and this permission notice appear
4000db7afdSDavid E. O'Brien * in supporting documentation. Hewlett-Packard Company makes no
4100db7afdSDavid E. O'Brien * representations about the suitability of this software for any
4200db7afdSDavid E. O'Brien * purpose. It is provided "as is" without express or implied warranty.
4300db7afdSDavid E. O'Brien *
4400db7afdSDavid E. O'Brien *
4500db7afdSDavid E. O'Brien * Copyright (c) 1996,1997
4600db7afdSDavid E. O'Brien * Silicon Graphics Computer Systems, Inc.
4700db7afdSDavid E. O'Brien *
4800db7afdSDavid E. O'Brien * Permission to use, copy, modify, distribute and sell this software
4900db7afdSDavid E. O'Brien * and its documentation for any purpose is hereby granted without fee,
5000db7afdSDavid E. O'Brien * provided that the above copyright notice appear in all copies and
5100db7afdSDavid E. O'Brien * that both that copyright notice and this permission notice appear
5200db7afdSDavid E. O'Brien * in supporting documentation. Silicon Graphics makes no
5300db7afdSDavid E. O'Brien * representations about the suitability of this software for any
5400db7afdSDavid E. O'Brien * purpose. It is provided "as is" without express or implied warranty.
5500db7afdSDavid E. O'Brien */
5600db7afdSDavid E. O'Brien
5700db7afdSDavid E. O'Brien /** @file stl_tempbuf.h
5800db7afdSDavid E. O'Brien * This is an internal header file, included by other library headers.
5900db7afdSDavid E. O'Brien * You should not attempt to use it directly.
6000db7afdSDavid E. O'Brien */
6100db7afdSDavid E. O'Brien
62ffeaf689SAlexander Kabaev #ifndef _TEMPBUF_H
63ffeaf689SAlexander Kabaev #define _TEMPBUF_H 1
64ffeaf689SAlexander Kabaev
65ffeaf689SAlexander Kabaev #include <memory>
6600db7afdSDavid E. O'Brien
_GLIBCXX_BEGIN_NAMESPACE(std)67*f8a1b7d9SAlexander Kabaev _GLIBCXX_BEGIN_NAMESPACE(std)
68*f8a1b7d9SAlexander Kabaev
6900db7afdSDavid E. O'Brien /**
7000db7afdSDavid E. O'Brien * @if maint
71ffeaf689SAlexander Kabaev * This class is used in two places: stl_algo.h and ext/memory,
72ffeaf689SAlexander Kabaev * where it is wrapped as the temporary_buffer class. See
73ffeaf689SAlexander Kabaev * temporary_buffer docs for more notes.
7400db7afdSDavid E. O'Brien * @endif
7500db7afdSDavid E. O'Brien */
76ffeaf689SAlexander Kabaev template<typename _ForwardIterator, typename _Tp>
7700db7afdSDavid E. O'Brien class _Temporary_buffer
7800db7afdSDavid E. O'Brien {
7900db7afdSDavid E. O'Brien // concept requirements
80ffeaf689SAlexander Kabaev __glibcxx_class_requires(_ForwardIterator, _ForwardIteratorConcept)
8100db7afdSDavid E. O'Brien
82ffeaf689SAlexander Kabaev public:
83ffeaf689SAlexander Kabaev typedef _Tp value_type;
84ffeaf689SAlexander Kabaev typedef value_type* pointer;
85ffeaf689SAlexander Kabaev typedef pointer iterator;
86ffeaf689SAlexander Kabaev typedef ptrdiff_t size_type;
8700db7afdSDavid E. O'Brien
88ffeaf689SAlexander Kabaev protected:
89ffeaf689SAlexander Kabaev size_type _M_original_len;
90ffeaf689SAlexander Kabaev size_type _M_len;
91ffeaf689SAlexander Kabaev pointer _M_buffer;
9200db7afdSDavid E. O'Brien
93ffeaf689SAlexander Kabaev void
94ffeaf689SAlexander Kabaev _M_initialize_buffer(const _Tp&, __true_type) { }
9500db7afdSDavid E. O'Brien
96ffeaf689SAlexander Kabaev void
97*f8a1b7d9SAlexander Kabaev _M_initialize_buffer(const _Tp& __val, __false_type)
98*f8a1b7d9SAlexander Kabaev { std::uninitialized_fill_n(_M_buffer, _M_len, __val); }
9900db7afdSDavid E. O'Brien
10000db7afdSDavid E. O'Brien public:
10100db7afdSDavid E. O'Brien /// As per Table mumble.
102ffeaf689SAlexander Kabaev size_type
103ffeaf689SAlexander Kabaev size() const
104ffeaf689SAlexander Kabaev { return _M_len; }
10500db7afdSDavid E. O'Brien
106ffeaf689SAlexander Kabaev /// Returns the size requested by the constructor; may be >size().
107ffeaf689SAlexander Kabaev size_type
108ffeaf689SAlexander Kabaev requested_size() const
109ffeaf689SAlexander Kabaev { return _M_original_len; }
110ffeaf689SAlexander Kabaev
111ffeaf689SAlexander Kabaev /// As per Table mumble.
112ffeaf689SAlexander Kabaev iterator
113ffeaf689SAlexander Kabaev begin()
114ffeaf689SAlexander Kabaev { return _M_buffer; }
115ffeaf689SAlexander Kabaev
116ffeaf689SAlexander Kabaev /// As per Table mumble.
117ffeaf689SAlexander Kabaev iterator
118ffeaf689SAlexander Kabaev end()
119ffeaf689SAlexander Kabaev { return _M_buffer + _M_len; }
120ffeaf689SAlexander Kabaev
121ffeaf689SAlexander Kabaev /**
122ffeaf689SAlexander Kabaev * Constructs a temporary buffer of a size somewhere between
123ffeaf689SAlexander Kabaev * zero and the size of the given range.
124ffeaf689SAlexander Kabaev */
125ffeaf689SAlexander Kabaev _Temporary_buffer(_ForwardIterator __first, _ForwardIterator __last);
126ffeaf689SAlexander Kabaev
127ffeaf689SAlexander Kabaev ~_Temporary_buffer()
128ffeaf689SAlexander Kabaev {
129ffeaf689SAlexander Kabaev std::_Destroy(_M_buffer, _M_buffer + _M_len);
130ffeaf689SAlexander Kabaev std::return_temporary_buffer(_M_buffer);
131ffeaf689SAlexander Kabaev }
132ffeaf689SAlexander Kabaev
133ffeaf689SAlexander Kabaev private:
134ffeaf689SAlexander Kabaev // Disable copy constructor and assignment operator.
135ffeaf689SAlexander Kabaev _Temporary_buffer(const _Temporary_buffer&);
136ffeaf689SAlexander Kabaev
137ffeaf689SAlexander Kabaev void
138ffeaf689SAlexander Kabaev operator=(const _Temporary_buffer&);
139ffeaf689SAlexander Kabaev };
140ffeaf689SAlexander Kabaev
141ffeaf689SAlexander Kabaev
142ffeaf689SAlexander Kabaev template<typename _ForwardIterator, typename _Tp>
143ffeaf689SAlexander Kabaev _Temporary_buffer<_ForwardIterator, _Tp>::
_Temporary_buffer(_ForwardIterator __first,_ForwardIterator __last)144ffeaf689SAlexander Kabaev _Temporary_buffer(_ForwardIterator __first, _ForwardIterator __last)
145ffeaf689SAlexander Kabaev : _M_original_len(std::distance(__first, __last)),
146ffeaf689SAlexander Kabaev _M_len(0), _M_buffer(0)
147ffeaf689SAlexander Kabaev {
14800db7afdSDavid E. O'Brien // Workaround for a __type_traits bug in the pre-7.3 compiler.
149*f8a1b7d9SAlexander Kabaev typedef typename std::__is_scalar<_Tp>::__type _Trivial;
15000db7afdSDavid E. O'Brien
151ffeaf689SAlexander Kabaev try
152ffeaf689SAlexander Kabaev {
153ffeaf689SAlexander Kabaev pair<pointer, size_type> __p(get_temporary_buffer<
154ffeaf689SAlexander Kabaev value_type>(_M_original_len));
155ffeaf689SAlexander Kabaev _M_buffer = __p.first;
156ffeaf689SAlexander Kabaev _M_len = __p.second;
15700db7afdSDavid E. O'Brien if (_M_len > 0)
15800db7afdSDavid E. O'Brien _M_initialize_buffer(*__first, _Trivial());
15900db7afdSDavid E. O'Brien }
16000db7afdSDavid E. O'Brien catch(...)
16100db7afdSDavid E. O'Brien {
162ffeaf689SAlexander Kabaev std::return_temporary_buffer(_M_buffer);
16300db7afdSDavid E. O'Brien _M_buffer = 0;
16400db7afdSDavid E. O'Brien _M_len = 0;
16500db7afdSDavid E. O'Brien __throw_exception_again;
16600db7afdSDavid E. O'Brien }
16700db7afdSDavid E. O'Brien }
168*f8a1b7d9SAlexander Kabaev
169*f8a1b7d9SAlexander Kabaev _GLIBCXX_END_NAMESPACE
17000db7afdSDavid E. O'Brien
171ffeaf689SAlexander Kabaev #endif /* _TEMPBUF_H */
17200db7afdSDavid E. O'Brien
173