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