File: array.h

package info (click to toggle)
sfs 1%3A0.5k-8
  • links: PTS
  • area: main
  • in suites: woody
  • size: 5,388 kB
  • ctags: 8,556
  • sloc: cpp: 43,410; ansic: 17,574; sh: 8,412; makefile: 771; yacc: 277; lex: 96; sed: 47
file content (112 lines) | stat: -rw-r--r-- 3,263 bytes parent folder | download
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
// -*-c++-*-
/* $Id: array.h,v 1.5 1999/11/19 08:35:05 dm Exp $ */

/*
 *
 * Copyright (C) 1998 David Mazieres (dm@uun.org)
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License as
 * published by the Free Software Foundation; either version 2, or (at
 * your option) any later version.
 *
 * This program is distributed in the hope that it will be useful, but
 * WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
 * USA
 *
 */

/* ***** WHY:
 *
 * C arrays (e.g. "int c[64];") vastly complicate some aspects of C++
 * template programming.  Suppose you have a template tmpl with a
 * class parameter T, and T is instantiated with an array:
 *
 *    class elm {
 *      //...
 *    };
 *
 *    template<class T> tmpl {
 *      //...
 *    };
 *
 *    typedef tmpl<elm[64]> problem_t;
 *
 * If, for instance, tmpl generally needs to allocate an object of
 * type T, a function in tmpl might have code like this:
 *
 *    T *objp = new T;
 *
 * However, this won't work when T is elm[64], because the code
 * "new elm[64]" returns an "elm *", not a "(*) elm[64]".
 *
 * Worse yet, any code that uses placement new or calls destructors
 * will not work.  If T is an array, then allocating a "new T" invokes
 * operator new[] rather than operator new, and that generally
 * requires more than sizeof(T) bytes.
 *
 * Finally, a lot of template classes require things like copy
 * constructors or assignment to work, and neither of those does with
 * C arrays.
 *
 *
 * ***** WHAT:
 *
 * The simple solution to all these problems is simply not to use C
 * arrays.  The dirt-simple type "array<type, size>" is simply an
 * array wrapped in a structure.  These arrays can be allocated with
 * the ordinary scalar new, and things like assignment and copy
 * construction will work fine.
 *
 * The macro "toarray" converts a C array type to a template array.
 */

#ifndef _ARRAY_H_WITH_TOARRAY_
#define _ARRAY_H_WITH_TOARRAY_ 1

#include <stddef.h>

template<class T, size_t n> struct array;

template<class T> struct __toarray {
  typedef T type;
};
template<class T, size_t n> struct __toarray<T[n]> {
  typedef array<__toarray<T>::type, n> type;
};
#define toarray(T) __toarray<T>::type

template<class T, size_t n> class array {
public:
  typedef toarray(T) elm_t;
  enum { nelm = n };

private:
  elm_t a[nelm];

#ifdef CHECK_BOUNDS
  void bcheck (size_t i) const { assert (i < nelm); }
#else /* !CHECK_BOUNDS */
  void bcheck (size_t) const {}
#endif /* !CHECK_BOUNDS */

public:
  static size_t size () { return nelm; }

  elm_t *base () { return a; }
  const elm_t *base () const { return a; }

  elm_t *lim () { return a + nelm; }
  const elm_t *lim () const { return a + nelm; }

  elm_t &operator[] (ptrdiff_t i) { bcheck (i); return a[i]; }
  const elm_t &operator[] (ptrdiff_t i) const { bcheck (i); return a[i]; }
};

#endif /* _ARRAY_H_WITH_TOARRAY_ */