shithub: purgatorio

ref: cb5057c6b3f549328a09fb08b300a0f74f671520
dir: /os/port/portbreak.c/

View raw version
#include "u.h"
#include "../port/lib.h"
#include "mem.h"
#include "dat.h"
#include "portfns.h"
#include "ureg.h"
#include "../port/error.h"

//
// These bits used to be in port/devdbg but were removed in
// order to allow for using hardware debug features on certain
// architectures
//

extern void breakset(Bkpt *b);
extern void breakrestore(Bkpt *b);
extern Bkpt* breakclear(int id);
extern void breaknotify(Bkpt *b, Proc *p);
extern int breakmatch(BkptCond *cond, Ureg *ur, Proc *p);

void	skipfree(Bkpt *b);
Bkpt*newskip(ulong addr, Bkpt *skipb, Proc *skipp);
Bkpt *skipalloc;
extern Bkpt	*breakpoints;
typedef struct SkipArg SkipArg;
struct SkipArg
{
	Bkpt *b;
	Proc *p;
};

void
skiphandler(Bkpt *b)
{
	SkipArg *a = b->aux;
	Bkpt *l;

	if(breakclear(b->id) == nil)
		panic("skiphandler: breakclear() failed");
	breakrestore(a->b);
	l = a->b->link;
	while(l != nil) {
		breakrestore(l);
		l = l->link;
	}
	skipfree(b);
	a->p->dbgstop = 0;		// Whoo!
	if(a->p->state == Stopped)
		ready(a->p);
}

Bkpt*
newskip(ulong addr, Bkpt *skipb, Proc *skipp)
{
	Bkpt *b;
	SkipArg *a;

	b = skipalloc;
	if(b == nil)
		panic("newskip(): no free skips\n");
	skipalloc = b->next;

	b->addr = addr;
	b->conditions->val = addr;
	b->link = nil;
	a = b->aux;
	a->b = skipb;
	a->p = skipp;

	return b;
}

void
skipfree(Bkpt *b)
{
	b->next = skipalloc;
	skipalloc = b;
}

//
// Called from the exception handler when a breakpoint instruction has been
// hit.  This cannot not be called unless at least one breakpoint with this
// address is in the list of breakpoints.  (All breakpoint notifications must
// previously have been set via setbreak())
//
//	foreach breakpoint in list
//		if breakpoint matches conditions
//			notify the break handler
//	if no breakpoints matched the conditions
//		pick a random breakpoint set to this address
//
//		set a breakpoint at the next instruction to be executed,
//		and pass the current breakpoint to the "skiphandler"
//
//		clear the current breakpoint
//
//		Tell the scheduler to stop scheduling, so the caller is
//		guaranteed to execute the instruction, followed by the
//		added breakpoint.
//
//
int
breakhit(Ureg *ur, Proc *p)
{
	Bkpt *b;
	int nmatched;
	Bkpt *skip;

	nmatched = 0;
	for(b = breakpoints; b != nil; b = b->next) {
		if(breakmatch(b->conditions, ur, p)) {
			breaknotify(b, p);
			++nmatched;
		}
	}

	if(nmatched)
		return BrkSched;

	skip = nil;
	for(b = breakpoints; b != nil;  b = b->next) {
		if(b->addr == ur->pc) {
			if(breakclear(b->id) == nil)
				panic("breakhit: breakclear() failed");

			if(skip == nil)
				skip = newskip(machnextaddr(ur), b, p);
			else {
				b->link = skip->link;
				skip->link = b;
			}
		}
	}
	if(skip == nil)
		return BrkSched;
	breakset(skip);
	return BrkNoSched;
}

void
portbreakinit(void)
{
	Bkpt *b;
	int i;

	skipalloc = mallocz(conf.nproc*(sizeof(Bkpt)+sizeof(BkptCond)+sizeof(SkipArg)), 1);
	if(skipalloc == nil)
		error(Enomem);

	b = skipalloc;
	for(i=0; i < conf.nproc-1; i++) {
		b->id = -(i+1);
		b->conditions = (BkptCond*)((uchar*)b + sizeof(Bkpt));
		b->conditions->op = 'b';
		b->handler = skiphandler;
		b->aux = (SkipArg*)((uchar*)b+sizeof(Bkpt)+sizeof(BkptCond));
		b->next = (Bkpt*)((uchar*)b+sizeof(Bkpt)+sizeof(BkptCond)+sizeof(SkipArg));
		b = b->next;
	}
	b->next = nil;
}