PBS::Debug - debugger support for PBS


PerlBuildSystem documentation Contained in the PerlBuildSystem distribution.

Index


Code Index:

NAME

Top

PBS::Debug - debugger support for PBS

SYNOPSIS

Top

	use PBS::Debug ;

	AddBreakpoint
		(
		  'hi'
		, DEPEND => 1
		, PRE => 1
		, ACTIONS =>
			[
			sub
				{
				PrintDebug "Hi there.\n" ;
				}
			]
		) ;

	ActivateBreakpoints('hi') ;

DESCRIPTION

Top

This module defines subs that manipulate PBS breakpoints (explained in PBS reference manual).

sub PrintBanner: display a friendly message when run under the perl debugger sub EnableDebugger sub p_h: display a help within the perl debugger sub p_b_wizard: not implemented sub p_list sub p_tree pretty print a tree sub p_dependencies pretty print the dependencies of a node sub p_node pretty prints a node

sub AddBreakpoint add a PBS breakpoint sub RemoveBreakpoints remove one or more PBS breakpoint according to the name regex passed as argument sub ListBreakpoints list all the breakpoints defined within PBS sub ActivateBreakpoints activates one or more PBS breakpoints sub DeactivateBreakpoints does the opposite of the above sub ActivatePerlDebugger activates wether a breakpoint (or breapoints) jumps to the perl debbugger sub DeactivatePerlDebugger does the opposite of the above

sub CheckBreakpoint

Aliases

p_list_breakpoints or p_L = ListBreakpoints p_activate_breakpoints or p_ab = ActivateBreakpoints p_deactivate_breakpoints or p_db = DeactivateBreakpoints p_jump_in_debugger or p_j = ActivatePerlDebugger p_B = RemoveBreakpoints

EXPORT

EnableDebugger AddBreakpoint RemoveBreakpoints ListBreakpoints ActivateBreakpoints DeactivateBreakpoints ActivatePerlDebugger DeactivatePerlDebugger CheckBreakpoint

and their aliases

BREAKPOINTs

Top

	AddBreakpoint
		(
		  'hi'
		, DEPEND => 1
		, PRE => 1
		, ACTIONS =>
			[
			sub
				{
				PrintDebug "Hi there.\n" ;
				}
			]
		) ;




Breakpoint position

Some breakpoints cab be triggered before or after (or both) something happends in the system.

if PRE is set, the breakpoints triggers before the action takes place. The breakpoint is triggered after the action takes place if after the action if POST is set. This allows you to take a snapshot before something happends and compare after it has happened.

Types

BUILD (PRE/POST): when building a node.

POST_BUILD (PRE/POST): when a post build action is to be performed

TREE: when a tree (or sub tree) is finished depending

INSERT When a node is inserted in the dependency tree.

VARIABLE: when a variable is set.

DEPEND (PRE/POST) when depending a node. DEPEND+TRIGGERED+POST can be used to trigger a breakpoint only when a rule has matched the node.

Breakpoint filtering

You can set any of the following to a qr// or string. Only actions matching all the regexes you set will trigger a breakpoint.

* RULE_REGEX
* NODE_REGEX
* PACKAGE_REGEX
* PBSFILE_REGEX

Actions

ACTIONS

ACTIONS is an array reference containing sub references. All the subs are run. All debugging functionality (ie activating or adding breakpoints) are available within the subs.

USE_DEBUGGER

USE_DEBUGGER if running under the perl debugger and USE_DEBUGGER is set, PBS will jump into the debugger after the breakpoint.

AUTHOR

Top

Khemir Nadim ibn Hamouda. nadim@khemir.net

SEE ALSO

Top

PBS reference manual / debug section.


PerlBuildSystem documentation Contained in the PerlBuildSystem distribution.

package PBS::Debug ;

use 5.006 ;

use strict ;
use warnings ;
use Data::Dumper ;
use Data::TreeDumper ;
use Carp ;

require Exporter ;
use AutoLoader qw(AUTOLOAD) ;

our @ISA = qw(Exporter) ;
our %EXPORT_TAGS = ('all' => [ qw() ]) ;
our @EXPORT_OK = ( @{ $EXPORT_TAGS{'all'} } ) ;
our @EXPORT = qw(
						EnableDebugger AddBreakpoint RemoveBreakpoints ListBreakpoints
						ActivateBreakpoints DeactivateBreakpoints 
						ActivatePerlDebugger DeactivatePerlDebugger 
						CheckBreakpoint
						) ;

our $VERSION = '0.03' ;

use PBS::PBS ;
use PBS::Output ;
use PBS::Information ; 

#-------------------------------------------------------------------------------

our $debug_enabled = 0 ;

# display the banner when running under the perl debugger.
PrintBanner() if(*DB::DB{CODE}) ;

#-------------------------------------------------------------------------------

sub EnableDebugger
{
my $switch = shift ;
my $file= shift ; # can contains breakpoint definitions

if($file ne '')
	{
	unless (my $return = do $file ) 
		{
		die ERROR "couldn't parse '$file ': $@" if $@;
		die ERROR "couldn't do '$file ': $!"    unless defined $return;
		die ERROR "couldn't run '$file '"       unless $return;
		}
	
	$debug_enabled = 1 ;
	}
}

#-------------------------------------------------------------------------------

sub PrintBanner
{
PrintDebug "Pbs support available, type 'p_h' for help.\n\n" ;
}

push @EXPORT, 'p_h' ;
sub p_h
{
PrintDebug <<EOH ;
commands: (no argument starts wizard if exists. RX means commands accepts a regex)

p_list_breakpoints or p_L           list breakpoints (RX)
p_activate_breakpoints or p_ab      activate breakpoints (RX)
p_deactivate_breakpoints or p_db    deactivate breakpoints (RX)
p_jump_in_debugger or p_j           breakpoints will jump into perl debugger (RX)
p_no_jump_in_debugger or p_nj       opposit of p_j
p_B                                 Remove breakpoints (RX)
p_node or p_n                       print node information
p_list or p_l                       list node (interactive)
p_tree or p_t                       pretty print tree
p_dependencies or p_d               lists the dependencies of a node
EOH
}

#----------------------------------------------------------------------

#~ p_b                                 insert a Pbs breakpoint 
#~ push @EXPORT, 'p_b' ;
#~ *p_b = \&AddBreakpoint ;

#~ sub p_b_wizard
#~ {
#~ PrintDebug "PBS - Not implemented!" ;

#~ PrintDebug "PBS - input breakpoint data:" ;

#~ my $data = <STDIN> ;
#~ chomp $data ;
#~ return if $data eq '' ;

#~ PrintDebug "breakpoint set at '$data'" ;
#~ }

push @EXPORT, 'p_list_breakpoints' ;
*p_list_breakpoints = \&ListBreakpoints ;

push @EXPORT, 'p_L' ;
*p_L = \&ListBreakpoints ;

#-------------------------------------------------------------------------------
push @EXPORT, 'p_activate_breakpoints' ;
*p_activate_breakpoints = \&ActivateBreakpoints ;

push @EXPORT, 'p_ab' ;
*p_ab= \&ActivateBreakpoints ;

push @EXPORT, 'p_deactivate_breakpoints' ;
*p_deactivate_breakpoints = \&DeactivateBreakpoints ;

push @EXPORT, 'p_db' ;
*p_db= \&DeactivateBreakpoints ;

#-------------------------------------------------------------------------------

push @EXPORT, 'p_jump_in_debugger' ;
*p_jump_in_debugger= \&ActivatePerlDebugger ;

push @EXPORT, 'p_j' ;
*p_j = \&ActivatePerlDebugger ;

push @EXPORT, 'p_no_jump_in_debugger' ;
*p_no_jump_in_debugger = \&DeactivatePerlDebugger ;

push @EXPORT, 'p_nj' ;
*p_nj = \&DeactivatePerlDebugger ;

#-------------------------------------------------------------------------------

push @EXPORT, 'p_B' ;
*p_B = \&RemoveBreakpoints ;

#-------------------------------------------------------------------------------

push @EXPORT, 'p_list' ;
sub p_list
{
my $tree = shift ;
my $node_regex = shift || '.*' ;
my $package_regex = shift || '.*' ;

if(!defined $tree || $tree eq 'HASH')
	{
	PrintDebug "p_list is expecting a reference to \$insereted_files\n" ;
	return() ;
	}
	
NodeChoice($tree, $node_regex) ;
}

push @EXPORT, 'p_l' ;
*p_l = \&p_list ;

#----------------------------------------------------------------------

push @EXPORT, 'p_tree' ;
sub p_tree
{
my $tree = shift ;
my $filter = shift || \&Data::TreeDumper::HashKeysSorter ;
my $indentation = shift || '' ;

unless(ref $tree eq 'HASH')
	{
	PrintDebug "p_tree is expecting a reference to a tree\n" ;
	return() ;
	}

PrintDebug
	(
	DumpTree
		(
		$tree
		, "Tree '$tree->{__NAME}': "
		, FILTER => $filter
		, INDENTATION => $indentation
		)
	) ;
}

push @EXPORT, 'p_t' ;
*p_t = \&p_tree ;

#----------------------------------------------------------------------

push @EXPORT, 'p_dependencies' ;
sub p_dependencies
{
my $tree = shift ;
my $GetDependenciesOnly = sub
			{
			my $tree = shift ;
			
			if('HASH' eq ref $tree)
				{
				return( 'HASH', undef, sort grep {! /^__/} keys %$tree) ;
				}
				
			#~ return (Data::TreeDumper::DefaultNodesToDisplay($tree)) ;
			} ;
			
p_tree($tree, $GetDependenciesOnly, '    ') ;
}

push @EXPORT, 'p_d' ;
*p_d = \&p_dependencies ;

#----------------------------------------------------------------------

push @EXPORT, 'p_node' ;
sub p_node
{
my $tree = shift ;
my $node_regex = shift || '.*' ;
my $package_regex = shift || '.*' ;

unless(ref $tree eq 'HASH')
	{
	PrintDebug "p_node is expecting a reference to a node or a tree\n" ;
	return() ;
	}

my $name = $tree->{__NAME} ;
if($name =~ /^__/)
	{
	PrintDebug "PBS - DisplayNodeInformation Can't display info about private node $name\n" ;
	PrintDebug "PBS - Which node would you like to see?\n" ;
	
	NodeChoice($tree, $node_regex) ;
	}
else
	{
	local $tree->{__PBS_CONFIG}{DISPLAY_NODE_ORIGIN} = 1 ;
	local $tree->{__PBS_CONFIG}{DISPLAY_NODE_DEPENDENCIES} = 1 ;
	local $tree->{__PBS_CONFIG}{DISPLAY_NODE_BUILD_CAUSE} = 1 ;
	local $tree->{__PBS_CONFIG}{DISPLAY_NODE_BUILD_RULES} = 1 ;
	local $tree->{__PBS_CONFIG}{DISPLAY_NODE_CONFIG} = 1 ;
	local $tree->{__PBS_CONFIG}{DISPLAY_NODE_BUILDER} = 1 ;
	local $tree->{__PBS_CONFIG}{DISPLAY_NODE_BUILD_POST_BUILD_COMMANDS} = 1 ;
	
	PBS::Information::DisplayNodeInformation($tree, $tree->{__PBS_CONFIG}) ;
	}
}

push @EXPORT, 'p_n' ;
*p_n = \&p_node ;

#----------------------------------------------------------------------

sub NodeChoice
{
my $tree = shift ;
my $node_regex = shift || '.*' ;

my %nodes ;
my $index = 0 ;

for(sort keys %{$tree})
	{
	next if /^__/ ;
	next unless /$node_regex/ ;
	
	PrintDebug "$index/ '$_'\n" ;
	$nodes{$index} = $tree->{$_} ;
	$index++ ;
	}

if(0 == $index)
	{
	PrintDebug "No node to display.\n" ;
	}
	
if(1 == $index)
	{
	local $nodes{0}->{__PBS_CONFIG}{DISPLAY_NODE_ORIGIN} = 1 ;
	local $nodes{0}->{__PBS_CONFIG}{DISPLAY_NODE_DEPENDENCIES} = 1 ;
	local $nodes{0}->{__PBS_CONFIG}{DISPLAY_NODE_BUILD_CAUSE} = 1 ;
	local $nodes{0}->{__PBS_CONFIG}{DISPLAY_NODE_BUILD_RULES} = 1 ;
	local $nodes{0}->{__PBS_CONFIG}{DISPLAY_NODE_CONFIG} = 1 ;
	local $nodes{0}->{__PBS_CONFIG}{DISPLAY_NODE_BUILDER} = 1 ;
	local $nodes{0}->{__PBS_CONFIG}{DISPLAY_NODE_BUILD_POST_BUILD_COMMANDS} = 1 ;
	
	PBS::Information::DisplayNodeInformation($nodes{0}, $nodes{0}->{__PBS_CONFIG}) ;
	return() ;
	}

while(1)
	{
	PrintDebug "PBS> " ;
	my $user_choice = <STDIN> ;
	chomp $user_choice;
	return if $user_choice eq '' ;
	
	if(exists $nodes{$user_choice})
		{
		local $nodes{$user_choice}->{__PBS_CONFIG}{DISPLAY_NODE_ORIGIN} = 1 ;
		local $nodes{$user_choice}->{__PBS_CONFIG}{DISPLAY_NODE_DEPENDENCIES} = 1 ;
		local $nodes{$user_choice}->{__PBS_CONFIG}{DISPLAY_NODE_BUILD_CAUSE} = 1 ;
		local $nodes{$user_choice}->{__PBS_CONFIG}{DISPLAY_NODE_BUILD_RULES} = 1 ;
		local $nodes{$user_choice}->{__PBS_CONFIG}{DISPLAY_NODE_CONFIG} = 1 ;
		local $nodes{$user_choice}->{__PBS_CONFIG}{DISPLAY_NODE_BUILDER} = 1 ;
		local $nodes{$user_choice}->{__PBS_CONFIG}{DISPLAY_NODE_BUILD_POST_BUILD_COMMANDS} = 1 ;
		PBS::Information::DisplayNodeInformation($nodes{$user_choice}, $nodes{$user_choice}->{__PBS_CONFIG}) ;
		}
	}
}

#----------------------------------------------------------------------

my %breakpoints ;

sub AddBreakpoint
{
my $breakpoint_name = shift ;

PrintWarning("Redefining breakpoint '$breakpoint_name'.\n") if(exists $breakpoints{$breakpoint_name}) ;

$breakpoints{$breakpoint_name} = {@_} ;
}

#----------------------------------------------------------------------

sub RemoveBreakpoints
{
my $breakpoint_regex = shift || '.*';

for my $breakpoint_name (sort keys %breakpoints)
	{
	next unless $breakpoint_name =~ $breakpoint_regex ;
	
	PrintDebug("Breakpoint '$breakpoint_name' removed. \n") ;
	delete $breakpoints{$breakpoint_name} ;
	}
}

#----------------------------------------------------------------------

sub ListBreakpoints
{
my $breakpoint_regex = shift || '.*';

for my $breakpoint_name (sort keys %breakpoints)
	{
	next unless $breakpoint_name =~ $breakpoint_regex ;
	
	PrintDebug(DumpTree($breakpoints{$breakpoint_name}, "$breakpoint_name:")) ;
	}
}

#----------------------------------------------------------------------

sub ActivateBreakpoints
{
my @breakpoint_regexes = @_ ;

for my $breakpoint_name (sort keys %breakpoints)
	{
	for my $breakpoint_regex (@breakpoint_regexes)
		{
		next unless $breakpoint_name =~ $breakpoint_regex ;
		
		$breakpoints{$breakpoint_name}{ACTIVE} = 1 ;
		PrintDebug("Breakpoint '$breakpoint_name' activated. \n") ;
		}
	}
}

#----------------------------------------------------------------------

sub DeactivateBreakpoints
{
my @breakpoint_regexes = @_ ;

for my $breakpoint_name (sort keys %breakpoints)
	{
	for my $breakpoint_regex (@breakpoint_regexes)
		{
		next unless $breakpoint_name =~ $breakpoint_regex ;
		
		$breakpoints{$breakpoint_name}{ACTIVE} = 0 ;
		PrintDebug("Breakpoint '$breakpoint_name' deactivated. \n") ;
		}
	}
}

#----------------------------------------------------------------------

sub ActivatePerlDebugger
{
my @breakpoint_regexes = @_ ;

for my $breakpoint_name (sort keys %breakpoints)
	{
	for my $breakpoint_regex (@breakpoint_regexes)
		{
		next unless $breakpoint_name =~ $breakpoint_regex ;
		
		$breakpoints{$breakpoint_name}{USE_DEBUGGER} = 1 ;
		PrintDebug("Breakpoint '$breakpoint_name' will activate the perl debugger. \n") ;
		}
	}
}

#----------------------------------------------------------------------

sub DeactivatePerlDebugger
{
my @breakpoint_regexes = @_ ;

for my $breakpoint_name (sort keys %breakpoints)
	{
	for my $breakpoint_regex (@breakpoint_regexes)
		{
		next unless $breakpoint_name =~ $breakpoint_regex ;
		
		$breakpoints{$breakpoint_name}{USE_DEBUGGER} = 0 ;
		PrintDebug("Breakpoint '$breakpoint_name' will NOT activate the perl debugger. \n") ;
		}
	}
}

#----------------------------------------------------------------------

sub CheckBreakpoint
{
my %point = @_ ;

return(0) unless $debug_enabled ;

my $use_debugger = 0 ;

for my $breakpoint_name (keys %breakpoints)
	{
	my $breakpoint = $breakpoints{$breakpoint_name} ;
	
	next unless $breakpoint->{ACTIVE} ;
	
	if(uc $breakpoint->{TYPE} eq 'BUILD')
		{
		next unless $point{TYPE} eq 'BUILD' ;
		}
		
	if(uc $breakpoint->{TYPE} eq 'POST_BUILD')
		{
		next unless $point{TYPE} eq 'POST_BUILD' ;
		}
		
	if(uc $breakpoint->{TYPE} eq 'TREE')
		{
		next unless $point{TYPE} eq 'TREE' ;
		}
	
	if(uc $breakpoint->{TYPE} eq 'INSERT')
		{
		next unless $point{TYPE} eq 'INSERT' ;
		}
	
	if(uc $breakpoint->{TYPE} eq 'VARIABLE')
		{
		next unless $point{TYPE} eq 'VARIABLE' ;
		}
	
	if(uc $breakpoint->{TYPE} eq 'DEPEND')
		{
		next unless $point{TYPE} eq 'DEPEND' ;
		}
		
	if(defined $breakpoint->{RULE_REGEX})
		{
		next unless $point{RULE_NAME} =~ /$breakpoint->{RULE_REGEX}/ ;
		}
	
	if(defined $breakpoint->{NODE_REGEX})
		{
		next unless $point{NODE_NAME} =~ /$breakpoint->{NODE_REGEX}/ ;
		}
		
	if(defined $breakpoint->{PACKAGE_REGEX})
		{
		next unless $point{PACKAGE_NAME} =~ /$breakpoint->{PACKAGE_REGEX}/ ;
		}
	
	if(defined $breakpoint->{PBSFILE_REGEX})
		{
		next unless $point{PBSFILE} =~ /$breakpoint->{PBSFILE_REGEX}/ ;
		}
		
	if($point{PRE})
		{
		next unless $breakpoint->{PRE} ;
		}
		
	if($point{POST})
		{
		next unless $breakpoint->{POST} ;
		}
		
	if($breakpoint->{TRIGGERED})
		{
		next unless $point{TRIGGERED} ;
		}

	$use_debugger++ if $breakpoint->{USE_DEBUGGER} ;
	
	for my $action (@{$breakpoint->{ACTIONS}})
		{
		$action->(%point, BREAKPOINT_NAME => $breakpoint_name) ;
		}
		
	if($breakpoint->{USE_DEBUGGER})
		{
		$use_debugger++  ;
		}
	}
	

return($use_debugger) ;
}

#----------------------------------------------------------------------

1 ;

__END__