Forth-83 多任务解析

      FORTH 系统能够写多任务,是因为 FORTH 语言是集操做系统、编译程序、汇编程序等于一体的具备可扩充性的十分灵活的工具。
       用户变量及用户区:
一个做业或一个任务是在必定的程序环境中运行的。一个任务运行了一段时间而让给另外一个任务以前,首先必须把当前任务的运行环境保存起来,以便之后该任务再次得到 CPU 时得以继续运行。FORTH 系统使用一组变量跟踪,记录和保存当前程序运行环境。存放用户变量的值的 FORTH 存贮区域叫作用户区。每个任务须要的用户变量以下:
变量名 意义
TOS 指向参数堆栈顶
ENTRY 当任务被激活时转移的入口点
LINK 指向下一个任务
SP0 参数堆栈的起始地址
RP0 返回堆栈的起始地址
DP 词典的顶部
#OUT 已发送的字符数
#LINE 已打印的行数
OFFSET 当前文件的块位移
BASE 当前在输入/输出转换中采用的数制
HLD  (- addr) 指向在 PAD 区中刚转换完的字符
FILE 指向当前文件的 FCB
IN-FILE 指向输入文件的 FCB
PRINTING 打印机状态标志,其值为真时,打印机处于激活状态
  一、这些 变量的名字存在于主词典中(他们隶属于 FORTH 词汇),可是 它们的数值却保存在用户区中。每一个用户有本身单独的用户区及以上变量的付本。也正是由于这些重要的信息是由各个任务自身单独保存着的,因此任务之间的转换就变得很容易了,仅仅只有解释指针 IP 的值等三个别的信息须要在任务转换中显示保存。
    二、 用户变量并不使用参数域地址做为地址,而是使用在当前任务的用户区中该变量距用户区首地址的相对位移来编址。每一个用户有本身单独的用户区,在用户区内的用户变量的数值描述了该任务在任一时刻的运行环境。在终端上以交互方式运行的任务的用户区的起始地址能够方便地用 UP@ 获得,它所指着的用户区也就是系统生成时所划定的用户区域。在多任务环境中,每新建一个任务都要给它指定一块用户区。
      
       暂停及从新开始:
堆栈 功能
(PAUSE) ( - ) 中止当前任务的执行,把控制转交给下一个任务。

它把 IP 及 RP 存放到参数堆栈上,把参数堆栈指针存进用户变量 TOS 中,转移到由 LINK 所指着的代码,从而实现多任务转换。
RESTART ( - ) 与(PAUSE)相反。取回所存放的信息,执行在上一轮暂停的任务
SLEEP ( addr -  ) addr 是任一用户区的首地址,SLEEP 使该任务永远暂停
WAKE ( addr -  ) 唤醒一个在"睡觉"的任务,使它在下一次轮到时运行
STOP ( - ) 永远暂停当前任务
PAUSE ( - ) 在多任务环境中 PAUSE 执行 (PAUSE)
    F83 采用轮转法进行多任务调度,其中最关键的两个词是 (PAUSE)RESTART(PAUSE)显示存放当前任务的 IP返回堆栈指针以及 参数堆栈指针,而后执行转移指令转移到下一个任务的 ENTRY RESTART 把下一任务的用户变量 TOS 的地址送入 UP 恢复所保存的参数堆栈指针,返回堆栈指针及解释指针IPIP 指着该任务在暂停以前所要执行的下一个词,RESTART 最后执行 NEXT,因而所恢复的任务就继续执行。
:  STOP   ( - )    
        UP  @     返回当前任务的地址
        SLEEP      使当前任务“睡觉”
        PAUSE  ;  当即中止当前任务而开始执行下一个任务
CODE PAUSE
      NEXT
END-CODE 

在单任务运行方式下,PAUSE 当即返回什么也不作;但在多任务方式中,把(PAUSE)的代码指针域的内容填入到 PAUSE 的代码指针域中,因而一个任务执行到 PAUSE 时就暂停下来。

    多任务的创建:
    首先必须把一个任务定义为词典中的一个词,与此同时分配给它做为 用户区、参数堆栈、返回堆栈及词典区的主存区域。另外,还必须把它链入到轮转法调度循环中。上面这些工做均由定义 TASK: 承担。
堆栈 功能
#USER (- addr) 存放在用户区大小的变量
@LINK (- addr) 给出下一任务的ENTRY的地址
!LINK   (addr -) 把一相对距离赋给当前任务的 LINK。使之 LINK+(LINK) 等于下一任务用户区的首地址
LOCAL (base addr - addr1) base为下一任务用户区的首地址,addr 为本任务之某一用户变量的地址,addr1是下一任务的同一用户变量的地址。
SET-TASK (ip addr -) 使地址为addr的任务执行由ip指着的代码

:  TASK:    ( size   -  )   创建一个新任务和作有关的初始化工做(size   -)  // size 表示词典空间大小
          CREATE   创建新任务的首部
          TOS         当前任务用户区首址
          HERE       新任务的用户区今后处开始
          #USER @    取出用户区的大小           (size   当前用户区首址   新任务用户区首址   用户区大小   - )
          CMOVE   把现行任务的用户变量复制给新任务,初始化新任务的用户区     (size   -)
          @LINK     新任务的 ENTRY 的地址      (size   新任务入口地址   -)
          UP @ -ROT   把当前任务的用户区指针送到堆栈低暂存    (当前用户区起始地址   size   新任务入口地址    -)
          HERE UP !  使 UP 指向新任务的用户区    (当前用户区起始地址   size   新任务入口地址    -)
          !LINK       把现行任务用户区的地址存入到新任务的 LINK 中  (当前用户区起始地址   size   -)
          DUP HERE +    新用户区及size之和的下一地址   (当前用户区起始地址   size   新用户区和size和的下一个地址   -)
          DUP RP0 !        初始化新任务的返回堆栈指针      (当前用户区起始地址   size   新用户区和size和的下一个地址   -) 
          100 - SP0 !       初始化新任务的参数堆栈指针      (当前用户区起始地址   size  -)
          SWAP UP !        恢复 UP 指向现行任务           (size  -)
          HERE ENTRY LOCAL !LINK                把新任务的地址存进现行任务的 LINK 中   (size  -)
          HERE #USER @ +    新任务词典区首地址       (size  新任务词典区首地址  -)
          HERE DP LOCAL !    初始化新任务词典指针   (size  -)
          HERE SLEEP       首先使新任务处于"睡眠"状态       (size  -)
          ALLOT ;       分配 size 个字节给新任务      (-)
TASK:  完成一个新任务在词典中的创建和有关的初始化工做,但该新任务尚未被赋予具体的工做。赋予具体的新任务的工做由定义   BACKGROUND:  承担。在 F83 中能够定义不少个"后台做业"和一个"前台做业",后台做业的执行通常不占终端,而把前台留给使用者运行前台做业和检查后台做业的执行状况。这个前台做业和全部后台做业在一块儿按轮转法调度就造成了 F83 的多任务环境,使得多个任务能够共同执行。
:  BACKGROUND:    ( - )     创建一个词典空间为 400 个字节的新任务,初始化该任务去执行跟着的代码
         400  TASK:        创建一个以跟着的名字命令的新任
         HERE  IP            指向有待编译的代码,使得新任务能够执行它     新任务地址  当前解释指针位置  - )
         @LINK  2-         新任务的地址      (新任务地址  当前解释指针位置  新任务入口地址-2  -)
         SET-TASK          初始化新任务,让她执行 IP 指着的代码  (新任务地址  -  )
         !CSP            编译程序查错初始化       
         ]  ;                调用编译程序,编译要为新任务执行的跟着的代码    


多任务调度:
    分时操做系统按照轮转法调度来自多个终端上的多个做业:它让每一个做业每次运行必定的时间,时间一到就经过中断把 CPU 转让给下一个做业。F83 也是采用轮转法调度多任务的运行,不过不是经过中断,而是经过多个任务的主动合做和协调来实现;这样,在 F83 中就不须要有专职的调度程序。 在多任务的环境中词 PAUSE 的功能是中止当前任务的执行,把对 CPU 的控制转交给下一个任务。在 F83 的每一共行任务中,每隔适当的间隔就得安插上一个 PAUSE , 以便让其它的任务有机会投入运行。不然一个任务就会独占 CPU , 共行执行就将没法实现;另外一方面, 当一个任务须要 CPU 时,它将把 WAKE 代码放进他的用户变量 ENTRY 内,这样当下一轮对 CPU 的控制传给它时,该任务即可以继续执行。F83 中的多任务要具备相互合做的性质,这样作的优势是使每一个任务的起点、停点均为已知,并使得多任务的调度简单和迅速。F83 各共行任务的连接是依靠用户变量 LINK 。
堆栈 功能
MULTI (-) 创建多任务调度循环
SINGLE (-) 取消多任务调度循环

MULTI 在多任务调度中相当重要,它把(PAUSE)的代码指针域的内容放入 PAUSE 的代码指针域,这样就使得但凡含有 PAUSE 的任务执行到 PAUSE 时就暂停。MULTI 又把 RESTART 的代码指针域的内容放入到中断向量表中编号为 80H 的项目的第一个单元。此后,但凡 ENTRY 内含有 INT 80H 的任务轮转到时就可投入运行。

:  SINGLE    ( - )
       [']  PAUSE  >BODY
       [']  PAUSE  !  ;
定义中的第一行返回 PAUSE 的 pfa ,它指向 NEXT 。注意:PAUSE 是 CODE 定义,CODE 词的代码指针域内的内容是其参数域地址。定义第二行恢复 PAUSE 的代码指针域为常态。这样 PAUSE 就将当即返回而再也不执行(PASE)


     F83 中是用词 BACKGROUND: 定义一个新任务,它把一个任务的首部定义为一个变量,所以引用一个任务的名字就在参数堆栈上留下该任务的参数域地址,而它就是一个任务的用户区的首地址。所以, 键入一任务名 SLEEP 就把 JMP NOP 的代码存入该任务的 ENTRY 中,从而使任务处于“睡眠”状态。而键 入一任务名 WAKE 就把 INT 80H 的代码存入任务的 ENYTRY 中,因而在下一轮中轮到时就投入运行。在一个任务的创建过程当中,同时也把它加入到多任务链中, 一个任务的用户变量 LINK 自己的地址与 LINK 单元的内容之和老是指向链中的下一个任务。用这种方法把一个前台做业和多个后台做业彼此间连接起来 。
      多任务调度循环由定义 MULTI 创建。所以在执行多任务以前首先要执行 MULTI 执行 MULTI 激活 PAUSE,同时又把 RESTART 的地址送进中断向量表第 80H 项。而一个任务一旦被“唤醒”,它的用户变量 ENTRY 的内容就是 INT 80H ,而 ENTRY 的内容正是一个任务要执行的第一条指令。因而轮到该任务执行时就产生编号为 80H 的软中断,接着就转去执行 RESTART , 恢复该任务上次暂停时的环境,接着继续运行。因为每一个任务有本身单独的用户区,在那儿保存着它自身运行环境,因此任务间的转换就至关简单。前台任务使用的用户区就是词典中原有的用户区,而每一个后台做业的用户区及其余专用区域则在创建任务时加以指定。
相关文章
相关标签/搜索