1 /* Copyright (C) 2016 The Android Open Source Project
2  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
3  *
4  * This file implements interfaces from the file jvmti.h. This implementation
5  * is licensed under the same terms as the file jvmti.h.  The
6  * copyright and license information for the file jvmti.h follows.
7  *
8  * Copyright (c) 2003, 2011, Oracle and/or its affiliates. All rights reserved.
9  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
10  *
11  * This code is free software; you can redistribute it and/or modify it
12  * under the terms of the GNU General Public License version 2 only, as
13  * published by the Free Software Foundation.  Oracle designates this
14  * particular file as subject to the "Classpath" exception as provided
15  * by Oracle in the LICENSE file that accompanied this code.
16  *
17  * This code is distributed in the hope that it will be useful, but WITHOUT
18  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
19  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
20  * version 2 for more details (a copy is included in the LICENSE file that
21  * accompanied this code).
22  *
23  * You should have received a copy of the GNU General Public License version
24  * 2 along with this work; if not, write to the Free Software Foundation,
25  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
26  *
27  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
28  * or visit www.oracle.com if you need additional information or have any
29  * questions.
30  */
31 
32 #ifndef ART_OPENJDKJVMTI_JVMTI_ALLOCATOR_H_
33 #define ART_OPENJDKJVMTI_JVMTI_ALLOCATOR_H_
34 
35 #include <android-base/logging.h>
36 #include <android-base/macros.h>
37 
38 #include "jvmti.h"
39 
40 #include "ti_allocator.h"
41 
42 namespace openjdkjvmti {
43 
44 template <typename T> class JvmtiAllocator;
45 
46 template <>
47 class JvmtiAllocator<void> {
48  public:
49   using value_type = void;
50   using pointer = void*;
51   using const_pointer = const void*;
52 
53   template <typename U>
54   struct rebind {
55     using other = JvmtiAllocator<U>;
56   };
57 
JvmtiAllocator(jvmtiEnv * env)58   explicit JvmtiAllocator(jvmtiEnv* env) : env_(env) {}
JvmtiAllocator()59   JvmtiAllocator() : env_(nullptr) {}
60 
61   template <typename U>
JvmtiAllocator(const JvmtiAllocator<U> & other)62   JvmtiAllocator(const JvmtiAllocator<U>& other)
63       : env_(other.env_) {}
64 
65   JvmtiAllocator(const JvmtiAllocator& other) = default;
66   JvmtiAllocator& operator=(const JvmtiAllocator& other) = default;
67   ~JvmtiAllocator() = default;
68 
69  private:
70   jvmtiEnv* env_;
71 
72   template <typename U>
73   friend class JvmtiAllocator;
74 
75   template <typename U>
76   friend bool operator==(const JvmtiAllocator<U>& lhs, const JvmtiAllocator<U>& rhs);
77 };
78 
79 template <typename T>
80 class JvmtiAllocator {
81  public:
82   using value_type = T;
83   using pointer = T*;
84   using reference = T&;
85   using const_pointer = const T*;
86   using const_reference = const T&;
87   using size_type = size_t;
88   using difference_type = ptrdiff_t;
89 
90   template <typename U>
91   struct rebind {
92     using other = JvmtiAllocator<U>;
93   };
94 
JvmtiAllocator(jvmtiEnv * env)95   explicit JvmtiAllocator(jvmtiEnv* env) : env_(env) {}
JvmtiAllocator()96   JvmtiAllocator() : env_(nullptr) {}
97 
98   template <typename U>
JvmtiAllocator(const JvmtiAllocator<U> & other)99   JvmtiAllocator(const JvmtiAllocator<U>& other)
100       : env_(other.env_) {}
101 
102   JvmtiAllocator(const JvmtiAllocator& other) = default;
103   JvmtiAllocator& operator=(const JvmtiAllocator& other) = default;
104   ~JvmtiAllocator() = default;
105 
max_size()106   size_type max_size() const {
107     return static_cast<size_type>(-1) / sizeof(T);
108   }
109 
address(reference x)110   pointer address(reference x) const { return &x; }
address(const_reference x)111   const_pointer address(const_reference x) const { return &x; }
112 
113   pointer allocate(size_type n, [[maybe_unused]] JvmtiAllocator<void>::pointer hint = nullptr) {
114     DCHECK_LE(n, max_size());
115     if (env_ == nullptr) {
116       T* result = reinterpret_cast<T*>(AllocUtil::AllocateImpl(n * sizeof(T)));
117       CHECK(result != nullptr || n == 0u);  // Abort if AllocateImpl() fails.
118       return result;
119     } else {
120       unsigned char* result;
121       jvmtiError alloc_error = env_->Allocate(n * sizeof(T), &result);
122       CHECK(alloc_error == JVMTI_ERROR_NONE);
123       return reinterpret_cast<T*>(result);
124     }
125   }
deallocate(pointer p,size_type n)126   void deallocate(pointer p, [[maybe_unused]] size_type n) {
127     if (env_ == nullptr) {
128       AllocUtil::DeallocateImpl(reinterpret_cast<unsigned char*>(p));
129     } else {
130       jvmtiError dealloc_error = env_->Deallocate(reinterpret_cast<unsigned char*>(p));
131       CHECK(dealloc_error == JVMTI_ERROR_NONE);
132     }
133   }
134 
construct(pointer p,const_reference val)135   void construct(pointer p, const_reference val) {
136     new (static_cast<void*>(p)) value_type(val);
137   }
138   template <class U, class... Args>
construct(U * p,Args &&...args)139   void construct(U* p, Args&&... args) {
140     ::new (static_cast<void*>(p)) U(std::forward<Args>(args)...);
141   }
destroy(pointer p)142   void destroy(pointer p) {
143     p->~value_type();
144   }
145 
146   inline bool operator==(JvmtiAllocator const& other) {
147     return env_ == other.env_;
148   }
149   inline bool operator!=(JvmtiAllocator const& other) {
150     return !operator==(other);
151   }
152 
153  private:
154   jvmtiEnv* env_;
155 
156   template <typename U>
157   friend class JvmtiAllocator;
158 
159   template <typename U>
160   friend bool operator==(const JvmtiAllocator<U>& lhs, const JvmtiAllocator<U>& rhs);
161 };
162 
163 template <typename T>
164 inline bool operator==(const JvmtiAllocator<T>& lhs, const JvmtiAllocator<T>& rhs) {
165   return lhs.env_ == rhs.env_;
166 }
167 
168 template <typename T>
169 inline bool operator!=(const JvmtiAllocator<T>& lhs, const JvmtiAllocator<T>& rhs) {
170   return !(lhs == rhs);
171 }
172 
173 }  // namespace openjdkjvmti
174 
175 #endif  // ART_OPENJDKJVMTI_JVMTI_ALLOCATOR_H_
176