mirror of
https://github.com/genodelabs/genode.git
synced 2025-06-19 15:43:56 +00:00
committed by
Norman Feske
parent
ed15a46ca4
commit
b7ffeb51aa
@ -1,79 +0,0 @@
|
|||||||
/*
|
|
||||||
* \brief Client-side CPU thread interface
|
|
||||||
* \author Norman Feske
|
|
||||||
* \date 2016-05-10
|
|
||||||
*/
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Copyright (C) 2016-2017 Genode Labs GmbH
|
|
||||||
*
|
|
||||||
* This file is part of the Genode OS framework, which is distributed
|
|
||||||
* under the terms of the GNU Affero General Public License version 3.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef _INCLUDE__CPU_THREAD__CLIENT_H_
|
|
||||||
#define _INCLUDE__CPU_THREAD__CLIENT_H_
|
|
||||||
|
|
||||||
#include <cpu_thread/cpu_thread.h>
|
|
||||||
#include <base/rpc_client.h>
|
|
||||||
|
|
||||||
namespace Genode { struct Cpu_thread_client; }
|
|
||||||
|
|
||||||
|
|
||||||
struct Genode::Cpu_thread_client : Rpc_client<Cpu_thread>
|
|
||||||
{
|
|
||||||
explicit Cpu_thread_client(Thread_capability cap)
|
|
||||||
: Rpc_client<Cpu_thread>(cap) { }
|
|
||||||
|
|
||||||
Dataspace_capability utcb() override {
|
|
||||||
return call<Rpc_utcb>(); }
|
|
||||||
|
|
||||||
void start(addr_t ip, addr_t sp) override {
|
|
||||||
call<Rpc_start>(ip, sp); }
|
|
||||||
|
|
||||||
void pause() override {
|
|
||||||
|
|
||||||
for (;;) {
|
|
||||||
|
|
||||||
call<Rpc_pause>();
|
|
||||||
|
|
||||||
try {
|
|
||||||
/* check if the thread state is valid */
|
|
||||||
state();
|
|
||||||
/* the thread is paused */
|
|
||||||
return;
|
|
||||||
} catch (State_access_failed) {
|
|
||||||
/* the thread is (most likely) running on a different CPU */
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void resume() override {
|
|
||||||
call<Rpc_resume>(); }
|
|
||||||
|
|
||||||
Thread_state state() override {
|
|
||||||
return call<Rpc_get_state>(); }
|
|
||||||
|
|
||||||
void state(Thread_state const &state) override {
|
|
||||||
call<Rpc_set_state>(state); }
|
|
||||||
|
|
||||||
void exception_sigh(Signal_context_capability handler) override {
|
|
||||||
call<Rpc_exception_sigh>(handler); }
|
|
||||||
|
|
||||||
void single_step(bool enabled) override {
|
|
||||||
call<Rpc_single_step>(enabled); }
|
|
||||||
|
|
||||||
void affinity(Affinity::Location location) override {
|
|
||||||
call<Rpc_affinity>(location); }
|
|
||||||
|
|
||||||
unsigned trace_control_index() override {
|
|
||||||
return call<Rpc_trace_control_index>(); }
|
|
||||||
|
|
||||||
Dataspace_capability trace_buffer() override {
|
|
||||||
return call<Rpc_trace_buffer>(); }
|
|
||||||
|
|
||||||
Dataspace_capability trace_policy() override {
|
|
||||||
return call<Rpc_trace_policy>(); }
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif /* _INCLUDE__CPU_THREAD__CLIENT_H_ */
|
|
@ -77,24 +77,32 @@ void Cpu_sampler::Cpu_thread_component::take_sample()
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
enum { MAX_LOOP_CNT = 100 };
|
||||||
|
unsigned loop_cnt = 0;
|
||||||
|
for (; loop_cnt < MAX_LOOP_CNT; loop_cnt++) {
|
||||||
|
|
||||||
_parent_cpu_thread.pause();
|
_parent_cpu_thread.pause();
|
||||||
|
|
||||||
Thread_state thread_state = _parent_cpu_thread.state();
|
try {
|
||||||
|
|
||||||
|
Thread_state thread_state = _parent_cpu_thread.state();
|
||||||
|
|
||||||
|
_sample_buf[_sample_buf_index++] = thread_state.ip;
|
||||||
|
|
||||||
|
} catch (State_access_failed) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
_parent_cpu_thread.resume();
|
_parent_cpu_thread.resume();
|
||||||
|
|
||||||
_sample_buf[_sample_buf_index++] = thread_state.ip;
|
|
||||||
|
|
||||||
if (_sample_buf_index == SAMPLE_BUF_SIZE)
|
if (_sample_buf_index == SAMPLE_BUF_SIZE)
|
||||||
flush();
|
flush();
|
||||||
|
|
||||||
} catch (Cpu_thread::State_access_failed) {
|
break;
|
||||||
|
|
||||||
Genode::log("thread state access failed");
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (loop_cnt >= MAX_LOOP_CNT)
|
||||||
|
log("thread state access failed, ", _label.string());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -237,7 +237,26 @@ void Cpu_thread_component::start(addr_t ip, addr_t sp)
|
|||||||
|
|
||||||
void Cpu_thread_component::pause()
|
void Cpu_thread_component::pause()
|
||||||
{
|
{
|
||||||
_parent_cpu_thread.pause();
|
unsigned loop_cnt = 0;
|
||||||
|
|
||||||
|
/* required semantic for gdb is that thread is paused with valid state */
|
||||||
|
for (;;) {
|
||||||
|
|
||||||
|
_parent_cpu_thread.pause();
|
||||||
|
|
||||||
|
try {
|
||||||
|
/* check if the thread state is valid */
|
||||||
|
_parent_cpu_thread.state();
|
||||||
|
/* the thread is paused */
|
||||||
|
return;
|
||||||
|
} catch (State_access_failed) {
|
||||||
|
loop_cnt ++;
|
||||||
|
|
||||||
|
if (loop_cnt % 100 == 0)
|
||||||
|
Genode::warning("pausing thread failed ", loop_cnt,
|
||||||
|
". times, continue looping");
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user