blob: 2a81b2cfd760f169172e081f2bfe27ec8bdff7fa [file] [log] [blame]
// Copyright 2019 Google LLC
//
// Licensed 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
//
// https://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.
#ifndef IREE_BINDINGS_PYTHON_PYIREE_BINDING_H_
#define IREE_BINDINGS_PYTHON_PYIREE_BINDING_H_
#include <vector>
#include "absl/types/optional.h"
#include "iree/base/api.h"
#include "pybind11/pybind11.h"
#include "pybind11/stl.h"
namespace pybind11 {
namespace detail {
#if !defined(ABSL_HAVE_STD_OPTIONAL)
// Make absl::optional act like the future C++17 optional for pybind11.
// If ABSL_HAVE_STD_OPTIONAL is defined then absl::optional == std::optional
// and the default type caster is sufficient.
template <typename T>
struct type_caster<absl::optional<T>> : optional_caster<absl::optional<T>> {};
#endif
} // namespace detail
} // namespace pybind11
namespace iree {
namespace python {
namespace py = pybind11;
template <typename T>
struct ApiPtrAdapter {};
template <typename Self, typename T>
class ApiRefCounted {
public:
ApiRefCounted() : instance_(nullptr) {}
ApiRefCounted(ApiRefCounted&& other) : instance_(other.instance_) {
other.instance_ = nullptr;
}
ApiRefCounted& operator=(ApiRefCounted&& other) {
instance_ = other.instance_;
other.instance_ = nullptr;
return *this;
}
void operator=(const ApiRefCounted&) = delete;
~ApiRefCounted() { Release(); }
// Creates an instance of the ref counted wrapper based on an instance
// that has already been retained. Ownership is transferred to the
// wrapper.
static Self CreateRetained(T* retained_inst) {
auto self = Self();
self.instance_ = retained_inst;
return self;
}
// Creates a new instance, retaining the underlying object.
static Self RetainAndCreate(T* non_retained_inst) {
auto self = Self();
self.instance_ = non_retained_inst;
if (non_retained_inst) {
ApiPtrAdapter<T>::Retain(non_retained_inst);
}
return self;
}
// Whether it is nullptr.
operator bool() const { return instance_; }
T* steal_raw_ptr() {
T* ret = instance_;
instance_ = nullptr;
return ret;
}
T* raw_ptr() {
if (!instance_) {
throw std::invalid_argument("API object is null");
}
return instance_;
}
const T* raw_ptr() const {
return const_cast<ApiRefCounted*>(this)->raw_ptr();
}
void Retain() {
if (instance_) {
ApiPtrAdapter<T>::Retain(instance_);
}
}
void Release() {
if (instance_) {
ApiPtrAdapter<T>::Release(instance_);
}
}
private:
T* instance_;
};
} // namespace python
} // namespace iree
#endif // IREE_BINDINGS_PYTHON_PYIREE_BINDING_H_