/* * Copyright (C) 2011-2017 Intel Corporation. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * * Neither the name of Intel Corporation nor the names of its * contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * */ #ifndef _TCS_H_ #define _TCS_H_ #include "arch.h" #include "se_wrapper.h" #include "util.h" #include "sgx_error.h" #include "se_debugger_lib.h" #include "se_lock.hpp" #include #include "node.h" using namespace std; typedef int (*bridge_fn_t)(const void*); class CEnclave; class CTrustThread: private Uncopyable { public: CTrustThread(tcs_t *tcs, CEnclave* enclave); ~CTrustThread(); int get_reference() { return m_reference; } void increase_ref() { m_reference++; } void decrease_ref() { m_reference--; } tcs_t *get_tcs() { return m_tcs; } CEnclave *get_enclave() { return m_enclave; } se_handle_t get_event(); void reset_ref() { m_reference = 0; } debug_tcs_info_t* get_debug_info(){return &m_tcs_info;} void push_ocall_frame(ocall_frame_t* frame_point); void pop_ocall_frame(); private: tcs_t *m_tcs; CEnclave *m_enclave; int m_reference; //it will increase by 1 before ecall, and decrease after ecall. se_handle_t m_event; debug_tcs_info_t m_tcs_info; }; class CTrustThreadPool: private Uncopyable { public: CTrustThreadPool(uint32_t tcs_min_pool); virtual ~CTrustThreadPool(); CTrustThread * acquire_thread(bool is_initialize_ecall = false); void release_thread(CTrustThread * const trust_thread); CTrustThread *add_thread(tcs_t * const tcs, CEnclave * const enclave, bool is_unallocated); CTrustThread *get_bound_thread(const tcs_t *tcs); std::vector get_thread_list(); void reset(); void wake_threads(); sgx_status_t new_thread(); sgx_status_t fill_tcs_mini_pool(); bool need_to_new_thread(); bool is_dynamic_thread_exist(); protected: virtual int garbage_collect() = 0; inline int find_thread(vector &thread_vector, se_thread_id_t thread_id); inline CTrustThread * get_free_thread(); int bind_thread(const se_thread_id_t thread_id, CTrustThread * const trust_thread); CTrustThread * get_bound_thread(const se_thread_id_t thread_id); void add_to_free_thread_vector(CTrustThread* it); vector m_free_thread_vector; vector m_unallocated_threads; Node *m_thread_list; Mutex m_thread_mutex; //protect thread_cache list. The mutex is recursive. //Thread can operate the list when it get the mutex Mutex m_free_thread_mutex; //protect free threads. Cond m_need_to_wait_for_new_thread_cond; private: CTrustThread * _acquire_thread(); CTrustThread *m_utility_thread; uint64_t m_tcs_min_pool; bool m_need_to_wait_for_new_thread; }; class CThreadPoolBindMode : public CTrustThreadPool { public: CThreadPoolBindMode(uint32_t tcs_min_pool):CTrustThreadPool(tcs_min_pool){} private: virtual int garbage_collect(); }; class CThreadPoolUnBindMode : public CTrustThreadPool { public: CThreadPoolUnBindMode(uint32_t tcs_min_pool):CTrustThreadPool(tcs_min_pool){} private: virtual int garbage_collect(); }; #endif