180814287SRaphael Isemann //===-- AppleGetThreadItemInfoHandler.cpp ---------------------------------===//
22fd83355SJason Molenda //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
62fd83355SJason Molenda //
72fd83355SJason Molenda //===----------------------------------------------------------------------===//
82fd83355SJason Molenda 
92fd83355SJason Molenda #include "AppleGetThreadItemInfoHandler.h"
102fd83355SJason Molenda 
118be30215SAlex Langford #include "Plugins/TypeSystem/Clang/TypeSystemClang.h"
122fd83355SJason Molenda #include "lldb/Core/Module.h"
132fd83355SJason Molenda #include "lldb/Core/Value.h"
14579e70c9SSean Callanan #include "lldb/Expression/DiagnosticManager.h"
15151c032cSJim Ingham #include "lldb/Expression/Expression.h"
16151c032cSJim Ingham #include "lldb/Expression/FunctionCaller.h"
17151c032cSJim Ingham #include "lldb/Expression/UtilityFunction.h"
182fd83355SJason Molenda #include "lldb/Symbol/Symbol.h"
192fd83355SJason Molenda #include "lldb/Target/ExecutionContext.h"
202fd83355SJason Molenda #include "lldb/Target/Process.h"
21151c032cSJim Ingham #include "lldb/Target/StackFrame.h"
222fd83355SJason Molenda #include "lldb/Target/Target.h"
232fd83355SJason Molenda #include "lldb/Target/Thread.h"
24bf9a7730SZachary Turner #include "lldb/Utility/ConstString.h"
25*c34698a8SPavel Labath #include "lldb/Utility/LLDBLog.h"
266f9e6901SZachary Turner #include "lldb/Utility/Log.h"
27bf9a7730SZachary Turner #include "lldb/Utility/StreamString.h"
28579e70c9SSean Callanan #include "lldb/lldb-private.h"
292fd83355SJason Molenda 
302fd83355SJason Molenda using namespace lldb;
312fd83355SJason Molenda using namespace lldb_private;
322fd83355SJason Molenda 
33b9c1b51eSKate Stone const char
34b9c1b51eSKate Stone     *AppleGetThreadItemInfoHandler::g_get_thread_item_info_function_name =
35b9c1b51eSKate Stone         "__lldb_backtrace_recording_get_thread_item_info";
36b9c1b51eSKate Stone const char
37b9c1b51eSKate Stone     *AppleGetThreadItemInfoHandler::g_get_thread_item_info_function_code =
38b9c1b51eSKate Stone         "                                  \n\
392fd83355SJason Molenda extern \"C\"                                                                                                    \n\
402fd83355SJason Molenda {                                                                                                               \n\
412fd83355SJason Molenda     /*                                                                                                          \n\
422fd83355SJason Molenda      * mach defines                                                                                             \n\
432fd83355SJason Molenda      */                                                                                                         \n\
442fd83355SJason Molenda                                                                                                                 \n\
452fd83355SJason Molenda     typedef unsigned int uint32_t;                                                                              \n\
462fd83355SJason Molenda     typedef unsigned long long uint64_t;                                                                        \n\
472fd83355SJason Molenda     typedef uint32_t mach_port_t;                                                                               \n\
482fd83355SJason Molenda     typedef mach_port_t vm_map_t;                                                                               \n\
492fd83355SJason Molenda     typedef int kern_return_t;                                                                                  \n\
502fd83355SJason Molenda     typedef uint64_t mach_vm_address_t;                                                                         \n\
512fd83355SJason Molenda     typedef uint64_t mach_vm_size_t;                                                                            \n\
522fd83355SJason Molenda                                                                                                                 \n\
532fd83355SJason Molenda     mach_port_t mach_task_self ();                                                                              \n\
542fd83355SJason Molenda     kern_return_t mach_vm_deallocate (vm_map_t target, mach_vm_address_t address, mach_vm_size_t size);         \n\
552fd83355SJason Molenda                                                                                                                 \n\
562fd83355SJason Molenda     typedef void *pthread_t;                                                                                    \n\
572fd83355SJason Molenda     extern int printf(const char *format, ...);                                                                 \n\
582fd83355SJason Molenda     extern pthread_t pthread_self(void);                                                                        \n\
592fd83355SJason Molenda                                                                                                                 \n\
602fd83355SJason Molenda     /*                                                                                                          \n\
612fd83355SJason Molenda      * libBacktraceRecording defines                                                                            \n\
622fd83355SJason Molenda      */                                                                                                         \n\
632fd83355SJason Molenda                                                                                                                 \n\
642fd83355SJason Molenda     typedef uint32_t queue_list_scope_t;                                                                        \n\
652fd83355SJason Molenda     typedef void *dispatch_queue_t;                                                                             \n\
662fd83355SJason Molenda     typedef void *introspection_dispatch_queue_info_t;                                                          \n\
672fd83355SJason Molenda     typedef void *introspection_dispatch_item_info_ref;                                                         \n\
682fd83355SJason Molenda                                                                                                                 \n\
69da276f90SJason Molenda     extern void __introspection_dispatch_thread_get_item_info (uint64_t  thread_id,                             \n\
702fd83355SJason Molenda                                                  introspection_dispatch_item_info_ref *returned_queues_buffer,  \n\
712fd83355SJason Molenda                                                  uint64_t *returned_queues_buffer_size);                        \n\
722fd83355SJason Molenda                                                                                                                 \n\
732fd83355SJason Molenda     /*                                                                                                          \n\
742fd83355SJason Molenda      * return type define                                                                                       \n\
752fd83355SJason Molenda      */                                                                                                         \n\
762fd83355SJason Molenda                                                                                                                 \n\
772fd83355SJason Molenda     struct get_thread_item_info_return_values                                                                      \n\
782fd83355SJason Molenda     {                                                                                                           \n\
792fd83355SJason Molenda         uint64_t item_info_buffer_ptr;    /* the address of the items buffer from libBacktraceRecording */  \n\
802fd83355SJason Molenda         uint64_t item_info_buffer_size;   /* the size of the items buffer from libBacktraceRecording */     \n\
812fd83355SJason Molenda     };                                                                                                          \n\
822fd83355SJason Molenda                                                                                                                 \n\
832fd83355SJason Molenda     void  __lldb_backtrace_recording_get_thread_item_info                                                          \n\
842fd83355SJason Molenda                                                (struct get_thread_item_info_return_values *return_buffer,          \n\
852fd83355SJason Molenda                                                 int debug,                                                      \n\
86da276f90SJason Molenda                                                 uint64_t thread_id,                                             \n\
872fd83355SJason Molenda                                                 void *page_to_free,                                             \n\
882fd83355SJason Molenda                                                 uint64_t page_to_free_size)                                     \n\
892fd83355SJason Molenda {                                                                                                               \n\
902fd83355SJason Molenda     void *pthread_id = pthread_self ();                                                                         \n\
912fd83355SJason Molenda     if (debug)                                                                                                  \n\
92da276f90SJason Molenda       printf (\"entering get_thread_item_info with args return_buffer == %p, debug == %d, thread id == 0x%llx, page_to_free == %p, page_to_free_size == 0x%llx\\n\", return_buffer, debug, (uint64_t) thread_id, page_to_free, page_to_free_size); \n\
932fd83355SJason Molenda     if (page_to_free != 0)                                                                                      \n\
942fd83355SJason Molenda     {                                                                                                           \n\
952fd83355SJason Molenda         mach_vm_deallocate (mach_task_self(), (mach_vm_address_t) page_to_free, (mach_vm_size_t) page_to_free_size); \n\
962fd83355SJason Molenda     }                                                                                                           \n\
972fd83355SJason Molenda                                                                                                                 \n\
98da276f90SJason Molenda     __introspection_dispatch_thread_get_item_info (thread_id,                                                  \n\
992fd83355SJason Molenda                                                   (void**)&return_buffer->item_info_buffer_ptr,                 \n\
1002fd83355SJason Molenda                                                   &return_buffer->item_info_buffer_size);                       \n\
1012fd83355SJason Molenda }                                                                                                               \n\
1022fd83355SJason Molenda }                                                                                                               \n\
1032fd83355SJason Molenda ";
1042fd83355SJason Molenda 
AppleGetThreadItemInfoHandler(Process * process)10516ff8604SSaleem Abdulrasool AppleGetThreadItemInfoHandler::AppleGetThreadItemInfoHandler(Process *process)
106b9c1b51eSKate Stone     : m_process(process), m_get_thread_item_info_impl_code(),
1072fd83355SJason Molenda       m_get_thread_item_info_function_mutex(),
1082fd83355SJason Molenda       m_get_thread_item_info_return_buffer_addr(LLDB_INVALID_ADDRESS),
109b9c1b51eSKate Stone       m_get_thread_item_info_retbuffer_mutex() {}
1102fd83355SJason Molenda 
111fd2433e1SJonas Devlieghere AppleGetThreadItemInfoHandler::~AppleGetThreadItemInfoHandler() = default;
1122fd83355SJason Molenda 
Detach()113b9c1b51eSKate Stone void AppleGetThreadItemInfoHandler::Detach() {
1142fd83355SJason Molenda 
115b9c1b51eSKate Stone   if (m_process && m_process->IsAlive() &&
116b9c1b51eSKate Stone       m_get_thread_item_info_return_buffer_addr != LLDB_INVALID_ADDRESS) {
117b9c1b51eSKate Stone     std::unique_lock<std::mutex> lock(m_get_thread_item_info_retbuffer_mutex,
118b9c1b51eSKate Stone                                       std::defer_lock);
1190c86dfb8SReid Kleckner     (void)lock.try_lock(); // Even if we don't get the lock, deallocate the buffer
1202fd83355SJason Molenda     m_process->DeallocateMemory(m_get_thread_item_info_return_buffer_addr);
1212fd83355SJason Molenda   }
1222fd83355SJason Molenda }
1232fd83355SJason Molenda 
124b9c1b51eSKate Stone // Compile our __lldb_backtrace_recording_get_thread_item_info() function (from
12505097246SAdrian Prantl // the source above in g_get_thread_item_info_function_code) if we don't find
12605097246SAdrian Prantl // that function in the inferior already with USE_BUILTIN_FUNCTION defined.
12705097246SAdrian Prantl // (e.g. this would be the case for testing.)
1282fd83355SJason Molenda //
129b9c1b51eSKate Stone // Insert the __lldb_backtrace_recording_get_thread_item_info into the inferior
130b9c1b51eSKate Stone // process if needed.
1312fd83355SJason Molenda //
132b9c1b51eSKate Stone // Write the get_thread_item_info_arglist into the inferior's memory space to
133b9c1b51eSKate Stone // prepare for the call.
1342fd83355SJason Molenda //
135b9c1b51eSKate Stone // Returns the address of the arguments written down in the inferior process,
13605097246SAdrian Prantl // which can be used to make the function call.
1372fd83355SJason Molenda 
SetupGetThreadItemInfoFunction(Thread & thread,ValueList & get_thread_item_info_arglist)138b9c1b51eSKate Stone lldb::addr_t AppleGetThreadItemInfoHandler::SetupGetThreadItemInfoFunction(
139b9c1b51eSKate Stone     Thread &thread, ValueList &get_thread_item_info_arglist) {
1406896b355SJim Ingham   ThreadSP thread_sp(thread.shared_from_this());
1416896b355SJim Ingham   ExecutionContext exe_ctx(thread_sp);
1422fd83355SJason Molenda   Address impl_code_address;
143579e70c9SSean Callanan   DiagnosticManager diagnostics;
144a007a6d8SPavel Labath   Log *log = GetLog(LLDBLog::SystemRuntime);
1452fd83355SJason Molenda   lldb::addr_t args_addr = LLDB_INVALID_ADDRESS;
146151c032cSJim Ingham   FunctionCaller *get_thread_item_info_caller = nullptr;
1472fd83355SJason Molenda 
1482fd83355SJason Molenda   // Scope for mutex locker:
1492fd83355SJason Molenda   {
15016ff8604SSaleem Abdulrasool     std::lock_guard<std::mutex> guard(m_get_thread_item_info_function_mutex);
1512fd83355SJason Molenda 
1522fd83355SJason Molenda     // First stage is to make the ClangUtility to hold our injected function:
1532fd83355SJason Molenda 
15470355aceSJonas Devlieghere     if (!m_get_thread_item_info_impl_code) {
15597206d57SZachary Turner       Status error;
156248a1305SKonrad Kleine       if (g_get_thread_item_info_function_code != nullptr) {
157de346cf2SJonas Devlieghere         auto utility_fn_or_error = exe_ctx.GetTargetRef().CreateUtilityFunction(
158de346cf2SJonas Devlieghere             g_get_thread_item_info_function_code,
159de346cf2SJonas Devlieghere             g_get_thread_item_info_function_name, eLanguageTypeC, exe_ctx);
160de346cf2SJonas Devlieghere         if (!utility_fn_or_error) {
161de346cf2SJonas Devlieghere           LLDB_LOG_ERROR(log, utility_fn_or_error.takeError(),
16263e5fb76SJonas Devlieghere                          "Failed to get UtilityFunction for "
163de346cf2SJonas Devlieghere                          "get-thread-item-info introspection: {0}.");
164151c032cSJim Ingham           return args_addr;
165151c032cSJim Ingham         }
166de346cf2SJonas Devlieghere         m_get_thread_item_info_impl_code = std::move(*utility_fn_or_error);
167b9c1b51eSKate Stone       } else {
16863e5fb76SJonas Devlieghere         LLDB_LOGF(log, "No get-thread-item-info introspection code found.");
1692fd83355SJason Molenda         return LLDB_INVALID_ADDRESS;
1702fd83355SJason Molenda       }
1712fd83355SJason Molenda 
172151c032cSJim Ingham       // Also make the FunctionCaller for this UtilityFunction:
173151c032cSJim Ingham 
174594308c7SRaphael Isemann       TypeSystemClang *clang_ast_context = ScratchTypeSystemClang::GetForTarget(
175594308c7SRaphael Isemann           thread.GetProcess()->GetTarget());
176b9c1b51eSKate Stone       CompilerType get_thread_item_info_return_type =
177b9c1b51eSKate Stone           clang_ast_context->GetBasicType(eBasicTypeVoid).GetPointerType();
178151c032cSJim Ingham 
179b9c1b51eSKate Stone       get_thread_item_info_caller =
180b9c1b51eSKate Stone           m_get_thread_item_info_impl_code->MakeFunctionCaller(
181b9c1b51eSKate Stone               get_thread_item_info_return_type, get_thread_item_info_arglist,
182b9c1b51eSKate Stone               thread_sp, error);
183ed2977faSJason Molenda       if (error.Fail() || get_thread_item_info_caller == nullptr) {
18463e5fb76SJonas Devlieghere         LLDB_LOGF(log,
18563e5fb76SJonas Devlieghere                   "Failed to install get-thread-item-info introspection "
186b9c1b51eSKate Stone                   "caller: %s.",
187b9c1b51eSKate Stone                   error.AsCString());
188151c032cSJim Ingham         m_get_thread_item_info_impl_code.reset();
189151c032cSJim Ingham         return args_addr;
190151c032cSJim Ingham       }
191151c032cSJim Ingham 
192b9c1b51eSKate Stone     } else {
193b9c1b51eSKate Stone       get_thread_item_info_caller =
194b9c1b51eSKate Stone           m_get_thread_item_info_impl_code->GetFunctionCaller();
1952fd83355SJason Molenda     }
1962fd83355SJason Molenda   }
1972fd83355SJason Molenda 
198579e70c9SSean Callanan   diagnostics.Clear();
1992fd83355SJason Molenda 
200b9c1b51eSKate Stone   // Now write down the argument values for this particular call.  This looks
20105097246SAdrian Prantl   // like it might be a race condition if other threads were calling into here,
20205097246SAdrian Prantl   // but actually it isn't because we allocate a new args structure for this
20305097246SAdrian Prantl   // call by passing args_addr = LLDB_INVALID_ADDRESS...
2042fd83355SJason Molenda 
205b9c1b51eSKate Stone   if (!get_thread_item_info_caller->WriteFunctionArguments(
206b9c1b51eSKate Stone           exe_ctx, args_addr, get_thread_item_info_arglist, diagnostics)) {
207b9c1b51eSKate Stone     if (log) {
20863e5fb76SJonas Devlieghere       LLDB_LOGF(log, "Error writing get-thread-item-info function arguments");
209579e70c9SSean Callanan       diagnostics.Dump(log);
210579e70c9SSean Callanan     }
2112fd83355SJason Molenda     return args_addr;
2122fd83355SJason Molenda   }
2132fd83355SJason Molenda 
2142fd83355SJason Molenda   return args_addr;
2152fd83355SJason Molenda }
2162fd83355SJason Molenda 
2172fd83355SJason Molenda AppleGetThreadItemInfoHandler::GetThreadItemInfoReturnInfo
GetThreadItemInfo(Thread & thread,tid_t thread_id,addr_t page_to_free,uint64_t page_to_free_size,Status & error)218b9c1b51eSKate Stone AppleGetThreadItemInfoHandler::GetThreadItemInfo(Thread &thread,
219b9c1b51eSKate Stone                                                  tid_t thread_id,
220b9c1b51eSKate Stone                                                  addr_t page_to_free,
221b9c1b51eSKate Stone                                                  uint64_t page_to_free_size,
22297206d57SZachary Turner                                                  Status &error) {
2232fd83355SJason Molenda   lldb::StackFrameSP thread_cur_frame = thread.GetStackFrameAtIndex(0);
2242fd83355SJason Molenda   ProcessSP process_sp(thread.CalculateProcess());
2252fd83355SJason Molenda   TargetSP target_sp(thread.CalculateTarget());
226594308c7SRaphael Isemann   TypeSystemClang *clang_ast_context =
227594308c7SRaphael Isemann       ScratchTypeSystemClang::GetForTarget(*target_sp);
228a007a6d8SPavel Labath   Log *log = GetLog(LLDBLog::SystemRuntime);
2292fd83355SJason Molenda 
2302fd83355SJason Molenda   GetThreadItemInfoReturnInfo return_value;
2312fd83355SJason Molenda   return_value.item_buffer_ptr = LLDB_INVALID_ADDRESS;
2322fd83355SJason Molenda   return_value.item_buffer_size = 0;
2332fd83355SJason Molenda 
2342fd83355SJason Molenda   error.Clear();
2352fd83355SJason Molenda 
236a6682a41SJonas Devlieghere   if (!thread.SafeToCallFunctions()) {
23763e5fb76SJonas Devlieghere     LLDB_LOGF(log, "Not safe to call functions on thread 0x%" PRIx64,
238b9c1b51eSKate Stone               thread.GetID());
239b4892cd2SJason Molenda     error.SetErrorString("Not safe to call functions on this thread.");
240b4892cd2SJason Molenda     return return_value;
241b4892cd2SJason Molenda   }
242b4892cd2SJason Molenda 
2432fd83355SJason Molenda   // Set up the arguments for a call to
2442fd83355SJason Molenda 
24505097246SAdrian Prantl   // struct get_thread_item_info_return_values {
246b9c1b51eSKate Stone   //     uint64_t item_info_buffer_ptr;    /* the address of the items buffer
247b9c1b51eSKate Stone   //     from libBacktraceRecording */
248b9c1b51eSKate Stone   //     uint64_t item_info_buffer_size;   /* the size of the items buffer from
249b9c1b51eSKate Stone   //     libBacktraceRecording */
2502fd83355SJason Molenda   // };
2512fd83355SJason Molenda   //
2522fd83355SJason Molenda   // void  __lldb_backtrace_recording_get_thread_item_info
253b9c1b51eSKate Stone   //                                            (struct
254b9c1b51eSKate Stone   //                                            get_thread_item_info_return_values
255b9c1b51eSKate Stone   //                                            *return_buffer,
2562fd83355SJason Molenda   //                                             int debug,
2572fd83355SJason Molenda   //                                             void *page_to_free,
2582fd83355SJason Molenda   //                                             uint64_t page_to_free_size)
2592fd83355SJason Molenda 
260b9c1b51eSKate Stone   // Where the return_buffer argument points to a 24 byte region of memory
26105097246SAdrian Prantl   // already allocated by lldb in the inferior process.
2622fd83355SJason Molenda 
263b9c1b51eSKate Stone   CompilerType clang_void_ptr_type =
264b9c1b51eSKate Stone       clang_ast_context->GetBasicType(eBasicTypeVoid).GetPointerType();
2652fd83355SJason Molenda   Value return_buffer_ptr_value;
266057efa99SAdrian Prantl   return_buffer_ptr_value.SetValueType(Value::ValueType::Scalar);
26799558cc4SGreg Clayton   return_buffer_ptr_value.SetCompilerType(clang_void_ptr_type);
2682fd83355SJason Molenda 
269a1e5dc86SGreg Clayton   CompilerType clang_int_type = clang_ast_context->GetBasicType(eBasicTypeInt);
2702fd83355SJason Molenda   Value debug_value;
271057efa99SAdrian Prantl   debug_value.SetValueType(Value::ValueType::Scalar);
27299558cc4SGreg Clayton   debug_value.SetCompilerType(clang_int_type);
2732fd83355SJason Molenda 
274b9c1b51eSKate Stone   CompilerType clang_uint64_type =
275b9c1b51eSKate Stone       clang_ast_context->GetBasicType(eBasicTypeUnsignedLongLong);
276da276f90SJason Molenda   Value thread_id_value;
277057efa99SAdrian Prantl   thread_id_value.SetValueType(Value::ValueType::Scalar);
27899558cc4SGreg Clayton   thread_id_value.SetCompilerType(clang_uint64_type);
279da276f90SJason Molenda 
2802fd83355SJason Molenda   Value page_to_free_value;
281057efa99SAdrian Prantl   page_to_free_value.SetValueType(Value::ValueType::Scalar);
28299558cc4SGreg Clayton   page_to_free_value.SetCompilerType(clang_void_ptr_type);
2832fd83355SJason Molenda 
2842fd83355SJason Molenda   Value page_to_free_size_value;
285057efa99SAdrian Prantl   page_to_free_size_value.SetValueType(Value::ValueType::Scalar);
28699558cc4SGreg Clayton   page_to_free_size_value.SetCompilerType(clang_uint64_type);
2872fd83355SJason Molenda 
28816ff8604SSaleem Abdulrasool   std::lock_guard<std::mutex> guard(m_get_thread_item_info_retbuffer_mutex);
289b9c1b51eSKate Stone   if (m_get_thread_item_info_return_buffer_addr == LLDB_INVALID_ADDRESS) {
290b9c1b51eSKate Stone     addr_t bufaddr = process_sp->AllocateMemory(
291b9c1b51eSKate Stone         32, ePermissionsReadable | ePermissionsWritable, error);
292b9c1b51eSKate Stone     if (!error.Success() || bufaddr == LLDB_INVALID_ADDRESS) {
29363e5fb76SJonas Devlieghere       LLDB_LOGF(log, "Failed to allocate memory for return buffer for get "
294b9c1b51eSKate Stone                      "current queues func call");
2952fd83355SJason Molenda       return return_value;
2962fd83355SJason Molenda     }
2972fd83355SJason Molenda     m_get_thread_item_info_return_buffer_addr = bufaddr;
2982fd83355SJason Molenda   }
2992fd83355SJason Molenda 
3002fd83355SJason Molenda   ValueList argument_values;
3012fd83355SJason Molenda 
302b9c1b51eSKate Stone   return_buffer_ptr_value.GetScalar() =
303b9c1b51eSKate Stone       m_get_thread_item_info_return_buffer_addr;
3042fd83355SJason Molenda   argument_values.PushValue(return_buffer_ptr_value);
3052fd83355SJason Molenda 
3062fd83355SJason Molenda   debug_value.GetScalar() = 0;
3072fd83355SJason Molenda   argument_values.PushValue(debug_value);
3082fd83355SJason Molenda 
309da276f90SJason Molenda   thread_id_value.GetScalar() = thread_id;
310da276f90SJason Molenda   argument_values.PushValue(thread_id_value);
311da276f90SJason Molenda 
3122fd83355SJason Molenda   if (page_to_free != LLDB_INVALID_ADDRESS)
3132fd83355SJason Molenda     page_to_free_value.GetScalar() = page_to_free;
3142fd83355SJason Molenda   else
3152fd83355SJason Molenda     page_to_free_value.GetScalar() = 0;
3162fd83355SJason Molenda   argument_values.PushValue(page_to_free_value);
3172fd83355SJason Molenda 
3182fd83355SJason Molenda   page_to_free_size_value.GetScalar() = page_to_free_size;
3192fd83355SJason Molenda   argument_values.PushValue(page_to_free_size_value);
3202fd83355SJason Molenda 
3212fd83355SJason Molenda   addr_t args_addr = SetupGetThreadItemInfoFunction(thread, argument_values);
3222fd83355SJason Molenda 
323579e70c9SSean Callanan   DiagnosticManager diagnostics;
3242fd83355SJason Molenda   ExecutionContext exe_ctx;
3252fd83355SJason Molenda   EvaluateExpressionOptions options;
326151c032cSJim Ingham   FunctionCaller *get_thread_item_info_caller = nullptr;
327151c032cSJim Ingham 
3282fd83355SJason Molenda   options.SetUnwindOnError(true);
3292fd83355SJason Molenda   options.SetIgnoreBreakpoints(true);
3302fd83355SJason Molenda   options.SetStopOthers(true);
3315b0a687dSJason Molenda #if __has_feature(address_sanitizer)
3324c03ea14SAdrian Prantl   options.SetTimeout(process_sp->GetUtilityExpressionTimeout());
3335b0a687dSJason Molenda #else
3345b0a687dSJason Molenda   options.SetTimeout(std::chrono::milliseconds(500));
3355b0a687dSJason Molenda #endif
336a0560152SJason Molenda   options.SetTryAllThreads(false);
337c01783a8SRaphael Isemann   options.SetIsForUtilityExpr(true);
3382fd83355SJason Molenda   thread.CalculateExecutionContext(exe_ctx);
3392fd83355SJason Molenda 
340b9c1b51eSKate Stone   if (!m_get_thread_item_info_impl_code) {
341b9c1b51eSKate Stone     error.SetErrorString("Unable to compile function to call "
342b9c1b51eSKate Stone                          "__introspection_dispatch_thread_get_item_info");
3432fd83355SJason Molenda     return return_value;
3442fd83355SJason Molenda   }
3452fd83355SJason Molenda 
346b9c1b51eSKate Stone   get_thread_item_info_caller =
347b9c1b51eSKate Stone       m_get_thread_item_info_impl_code->GetFunctionCaller();
348151c032cSJim Ingham 
349b9c1b51eSKate Stone   if (!get_thread_item_info_caller) {
350b9c1b51eSKate Stone     error.SetErrorString("Unable to compile function caller for "
351b9c1b51eSKate Stone                          "__introspection_dispatch_thread_get_item_info");
352151c032cSJim Ingham     return return_value;
353151c032cSJim Ingham   }
3542fd83355SJason Molenda 
3551624a2d3SJim Ingham   ExpressionResults func_call_ret;
3562fd83355SJason Molenda   Value results;
357b9c1b51eSKate Stone   func_call_ret = get_thread_item_info_caller->ExecuteFunction(
358b9c1b51eSKate Stone       exe_ctx, &args_addr, options, diagnostics, results);
359b9c1b51eSKate Stone   if (func_call_ret != eExpressionCompleted || !error.Success()) {
36063e5fb76SJonas Devlieghere     LLDB_LOGF(log,
36163e5fb76SJonas Devlieghere               "Unable to call "
362b9c1b51eSKate Stone               "__introspection_dispatch_thread_get_item_info(), got "
363b9c1b51eSKate Stone               "ExpressionResults %d, error contains %s",
364b9c1b51eSKate Stone               func_call_ret, error.AsCString(""));
365b9c1b51eSKate Stone     error.SetErrorString("Unable to call "
366b9c1b51eSKate Stone                          "__introspection_dispatch_thread_get_item_info() for "
367b9c1b51eSKate Stone                          "list of queues");
3682fd83355SJason Molenda     return return_value;
3692fd83355SJason Molenda   }
3702fd83355SJason Molenda 
371b9c1b51eSKate Stone   return_value.item_buffer_ptr = m_process->ReadUnsignedIntegerFromMemory(
372b9c1b51eSKate Stone       m_get_thread_item_info_return_buffer_addr, 8, LLDB_INVALID_ADDRESS,
373b9c1b51eSKate Stone       error);
374b9c1b51eSKate Stone   if (!error.Success() ||
375b9c1b51eSKate Stone       return_value.item_buffer_ptr == LLDB_INVALID_ADDRESS) {
3762fd83355SJason Molenda     return_value.item_buffer_ptr = LLDB_INVALID_ADDRESS;
3772fd83355SJason Molenda     return return_value;
3782fd83355SJason Molenda   }
3792fd83355SJason Molenda 
380b9c1b51eSKate Stone   return_value.item_buffer_size = m_process->ReadUnsignedIntegerFromMemory(
381b9c1b51eSKate Stone       m_get_thread_item_info_return_buffer_addr + 8, 8, 0, error);
3822fd83355SJason Molenda 
383b9c1b51eSKate Stone   if (!error.Success()) {
3842fd83355SJason Molenda     return_value.item_buffer_ptr = LLDB_INVALID_ADDRESS;
3852fd83355SJason Molenda     return return_value;
3862fd83355SJason Molenda   }
3872fd83355SJason Molenda 
38863e5fb76SJonas Devlieghere   LLDB_LOGF(log,
38963e5fb76SJonas Devlieghere             "AppleGetThreadItemInfoHandler called "
390b9c1b51eSKate Stone             "__introspection_dispatch_thread_get_item_info (page_to_free "
391b9c1b51eSKate Stone             "== 0x%" PRIx64 ", size = %" PRId64
392b9c1b51eSKate Stone             "), returned page is at 0x%" PRIx64 ", size %" PRId64,
393b9c1b51eSKate Stone             page_to_free, page_to_free_size, return_value.item_buffer_ptr,
394b9c1b51eSKate Stone             return_value.item_buffer_size);
3952a6c252dSJason Molenda 
3962fd83355SJason Molenda   return return_value;
3972fd83355SJason Molenda }
398