⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 segmacs.asi

📁 PIC16CXX C语言工具
💻 ASI
字号:
******************************************************************************
* Permission to use all or part of these macros in your code as long as you
* include the following comment:
******************************************************************************
* Segment handling routines by Don Lekei
******************************************************************************
* First use:
*	DEFSEG	<name>,<start>,<end+1>	;to define a memory segment
* then use:
* 	SEG	NAME	;to change segments
*
* Other useful macros:
*	PADSEG		;Pads the current segment with $FF
*	OLDSEG		;Change back to the PREVIOUS SEGMENT
*	FILLSEG	<value>	;Fill the rest of the segment a byte value
*	SEG_REPORT	;Generate a final segment report
*	PC(NAME)	;*Function* to return next address in segment
******************************************************************************

***************************************************************
* Global variables used by various SEG macros                 *
***************************************************************

..SEG$CNT	=	0	;Number of segments defined
..CUR$SEG	=	0	;Current segment number
..OLD$SEG	=	0	;Previous segment number

******************************************************************************
* DEFSEG --- Segment definition macro
* Usage:
*	DEFSEG	<name>,<start>,<end+1>
* where:
*	<name>	is the SEG name (eg. ROM, RAM, CODE, etc.) max. 7 chrs
*	<start>	is the starting address
*	<end+1>	is the first address NOT in the segment
*
* DEFSEG creates the following labels (where name represents <name> and
*	n is the current segment # (SEG$CNT+1) as a HEX digit:
*
* S$name:	The segment # of the segment
* ..SEG$n:	The next address in the segment
* END$n:	The value of <end+1>
*
* It also defines a macro "SEGREP$n" to print the segment report line info
*
******************************************************************************

DEFSEG		.MACRO	*[1],[2],[3]	;Name,start,end+1
..SEG$CNT	=	..SEG$CNT+1	;get next segment #
S$[1]		=	..SEG$CNT	;define the segment tag label
		make$seg (S$[1]),([2]),([3]),[1]  ;Define segment labels
		.ENDM

******************************************************************************
* SEG --- CHANGE SEGMENTS
* Usage:
*	SEG	<name>
* where:
*	<name> is the same name used in the DEFSEG statement
******************************************************************************

SEG		.MACRO	*[1]	;change segment to [1] (NAME)
			;(S$[1]) expands to the value of label S$NAME
		set$seg	(S$[1]),(..CUR$SEG)	;pass #s to SET$SEG
		.ENDM

******************************************************************************
* OLDSEG --- CHANGE TO PREVIOUS SEGMENT
******************************************************************************

OLDSEG		.MACRO	*	;change to previous segment
		set$seg	(..OLD$SEG),(..CUR$SEG) ;pass #s to SET$SEG
		.ENDM

******************************************************************************
* PADSEG --- PAD WITH $FF TO END OF CURRENT SEGMENT
******************************************************************************

PADSEG		.MACRO	*	;fill rest of current segment with $FF
		pad$seg	(..CUR$SEG),$FF
		.ENDM

******************************************************************************
* FILLSEG --- FILL WITH [1] TO END OF CURRENT SEGMENT
******************************************************************************

FILLSEG		.MACRO	*[1]	;fill rest of current segment with [1]
		pad$seg	(..CUR$SEG),[1]
		.ENDM

******************************************************************************
* SEG_REPORT --- PRINT THE FINAL STATUS OF ALL SEGMENTS
*	The report will print to the current ECHO device or file (at the
*	end of each pass), AND will expand to the listing.
******************************************************************************

SEG_REPORT	.MACRO	*	;print the final status of all segments

	.if	..SEG$CNT=0	;make sure there are segments defined
	.EXITM			;nothing to report
	.endif

***************************************************************
* Update current segment's PC (forgets previous segment)      *
***************************************************************
	set$seg	(..CUR$SEG),(..CUR$SEG) ;pass #s to SET$SEG
***************************************************************
* Print a nice heading.                                       *
***************************************************************
	.mxp	on
 echo		SEGMENT	START	END	PC	USED	FREE
 echo		-------	-----	-----	-----	-----	-----
	.mxp	off

	.do	..SEG$CNT	;print all segment reports
	seg$rep (.._index)	;report each segment
	.loop

	.ENDM

******************************************************************************
* PC(name) (function macro) --- Returns the current PC for named segment
*	eg.	PC(RAM)
******************************************************************************

PC(	.MACRO [1])	;return next PC for SEG [1]
  ( P$C( (S$[1]) ) )	;pass segment # from S$NAME as a string to P$C()
	.ENDM

******************************************************************************
* P$C((segnum),(CUR$SEG)) find segment PC
*	Invoked with parms in parentheses to pass the values as $n
******************************************************************************

P$C(	.MACRO	[1])	;find PC for use by PC() function
  ( ([1] == ..CUR$SEG) ? * : ..SEG[1] ) ;return PC if current SEG is requested
	.ENDM

******************************************************************************
* MAKE$SEG: invoked by DEFSEG to create segment labels
*	Invoked with parms 1, 2 & 3 in parentheses to pass the values as $n
******************************************************************************

MAKE$SEG	.MACRO	[1],[2],[3],[4]	;(segnum), (start), (end), name
..SEG[1]	=	[2]	;Set initial segment PC
END[1]		=	[3]	;Set the segment end address
		setenv '_SEG[1]=','[4]'

***************************************************************
* Define a macro to display the segment                       *
* Note: [1], [2], [3], and [4] are replaced during the        *
*       the expansion of *** MAKE$SEG ***, while "end" is     *
*       replaced by the expansion of *** SEGREP ***           *
***************************************************************
SEGREP[1]	.macro *pc		;report on SEG [4] (# [1])
;			(name,start,end,pc,used,free)
		seg$echo [4],[2],[3],(pc),(pc-[2]),([3]-pc)

		.endm
		.ENDM

******************************************************************************
* SEG$ECHO --- Invoked by SEGREP$n TO CREATE SEGMENT REPORT LINE
*	Invoked with parms in parentheses to pass the values as $n
******************************************************************************
SEG$ECHO	.MACRO	*name,start,end,pc,used,free	;seg report line
		.mxp	on
 echo		name	start	end	pc	used	free
		.mxp	off
		.ENDM

******************************************************************************
* SEG$REP --- Invoked by SEG_REPORT to convert seg # to SEGREP$n invovation
*	Invoked with parm in parentheses to pass the value as $n
******************************************************************************
SEG$REP		.MACRO *[1]	;(segnum) invoke SEGREP$n
		segrep[1] ..SEG[1]	;print final report
		.ENDM

******************************************************************************
* SET$SEG Invoked by SEG to set segment variables for segment change
*	Invoked with parms in parentheses to pass the values as $n
******************************************************************************

SET$SEG		.MACRO	[1],[2]		;(new segnum),(current segnum)
		.if	..CUR$SEG	;skip if first segment
..OLD$SEG	=	..CUR$SEG	 ;SAVE PREVIOUS SEGMENT
..SEG[2]	=	*		 ;UPDATE SEGMENT POINTER
		  .IF	(* > END[2]) && (END[2] != 0)
		     seg$fail [2],(*)	  ;PC is past end of segment
		  .ENDIF
		.endif
		*=	..SEG[1]	;SET PC TO NEW SEGMENT
..CUR$SEG	=	[1]		;SET CURRENT SEGMENT NUMBER
		.ENDM

******************************************************************************
* SEG$FAIL --- Invoked by SET$SEG when SEGMENT OVERFLOW occurs
*	Invoked with parms in parentheses to pass the values as $n
******************************************************************************
SEG$FAIL	.MACRO	*[1],[2]
	.mxp	on
 echo		SEGMENT	START	END	PC	USED	FREE
 echo		-------	-----	-----	-----	-----	-----
	.mxp	off
		.segrep[1] ([2]-1)	;print report info for this segment
		.ERROR 'Segment ',#_SEG[1], ' overflow at PC = [2]'
		.ENDM

******************************************************************************
* PAD$SEG --- Invoked by PADSEG fill rest of segment
*	Invoked with parms in parentheses to pass the values as $n
******************************************************************************

PAD$SEG		.MACRO	[1],[2]
		FILL	[2],END[1]-*
		.ENDM

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -