1.\" Copyright (c) 1998, 1999, Nicolas Souchu 2.\" All rights reserved. 3.\" 4.\" Redistribution and use in source and binary forms, with or without 5.\" modification, are permitted provided that the following conditions 6.\" are met: 7.\" 1. Redistributions of source code must retain the above copyright 8.\" notice, this list of conditions and the following disclaimer. 9.\" 2. Redistributions in binary form must reproduce the above copyright 10.\" notice, this list of conditions and the following disclaimer in the 11.\" documentation and/or other materials provided with the distribution. 12.\" 13.\" THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 14.\" ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 15.\" IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 16.\" ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 17.\" FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 18.\" DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 19.\" OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 20.\" HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 21.\" LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 22.\" OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 23.\" SUCH DAMAGE. 24.\" 25.Dd June 6, 1998 26.Dt MICROSEQ 9 27.Os 28.Sh NAME 29.Nm microseq 30.Nd ppbus microsequencer developer's guide 31.Sh SYNOPSIS 32.In sys/types.h 33.In dev/ppbus/ppbconf.h 34.In dev/ppbus/ppb_msq.h 35.Sh DESCRIPTION 36See 37.Xr ppbus 4 38for ppbus description and general info about the microsequencer. 39.Pp 40The purpose of this document is to encourage developers to use the 41microsequencer mechanism in order to have: 42.Bl -enum -offset indent 43.It 44a uniform programming model 45.It 46efficient code 47.El 48.Pp 49Before using microsequences, you are encouraged to look at 50.Xr ppc 4 51microsequencer implementation and an example of how using it in 52.Xr ppi 4 . 53.Sh PPBUS register model 54.Ss Background 55The parallel port model chosen for ppbus is the PC parallel port model. 56Thus, any register described later has the same semantic than its counterpart 57in a PC parallel port. 58For more info about ISA/ECP programming, get the 59Microsoft standard referenced as "Extended Capabilities Port Protocol and 60ISA interface Standard". 61Registers described later are standard parallel port 62registers. 63.Pp 64Mask macros are defined in the standard ppbus include files for each valid 65bit of parallel port registers. 66.Ss Data register 67In compatible or nibble mode, writing to this register will drive data to the 68parallel port data lines. 69In any other mode, drivers may be tri-stated by 70setting the direction bit (PCD) in the control register. 71Reads to this register 72return the value on the data lines. 73.Ss Device status register 74This read-only register reflects the inputs on the parallel port interface. 75.Pp 76.Bl -column "Bit" "Name" "Description" -compact 77.It Em Bit Ta Em Name Ta Em Description 78.It 7 Ta nBUSY Ta "inverted version of parallel port Busy signal" 79.It 6 Ta nACK Ta "version of parallel port nAck signal" 80.It 5 Ta PERROR Ta "version of parallel port PERROR signal" 81.It 4 Ta SELECT Ta "version of parallel port Select signal" 82.It 3 Ta nFAULT Ta "version of parallel port nFault signal" 83.El 84.Pp 85Others are reserved and return undefined result when read. 86.Ss Device control register 87This register directly controls several output signals as well as enabling 88some functions. 89.Pp 90.Bl -column "Bit" "Name " "Description" -compact 91.It Em Bit Ta Em Name Ta Em Description 92.It 5 Ta PCD Ta "direction bit in extended modes" 93.It 4 Ta IRQENABLE Ta "1 enables an interrupt on the rising edge of nAck" 94.It 3 Ta SELECTIN Ta "inverted and driven as parallel port nSelectin signal" 95.It 2 Ta nINIT Ta "driven as parallel port nInit signal" 96.It 1 Ta AUTOFEED Ta "inverted and driven as parallel port nAutoFd signal" 97.It 0 Ta STROBE Ta "inverted and driven as parallel port nStrobe signal" 98.El 99.Sh MICROINSTRUCTIONS 100.Ss Description 101.Em Microinstructions 102are either parallel port accesses, program iterations, submicrosequence or 103C calls. 104The parallel port must be considered as the logical model described in 105.Xr ppbus 4 . 106.Pp 107Available microinstructions are: 108.Bd -literal 109#define MS_OP_GET 0 /* get <ptr>, <len> */ 110#define MS_OP_PUT 1 /* put <ptr>, <len> */ 111#define MS_OP_RFETCH 2 /* rfetch <reg>, <mask>, <ptr> */ 112#define MS_OP_RSET 3 /* rset <reg>, <mask>, <mask> */ 113#define MS_OP_RASSERT 4 /* rassert <reg>, <mask> */ 114#define MS_OP_DELAY 5 /* delay <val> */ 115#define MS_OP_SET 6 /* set <val> */ 116#define MS_OP_DBRA 7 /* dbra <offset> */ 117#define MS_OP_BRSET 8 /* brset <mask>, <offset> */ 118#define MS_OP_BRCLEAR 9 /* brclear <mask>, <offset> */ 119#define MS_OP_RET 10 /* ret <retcode> */ 120#define MS_OP_C_CALL 11 /* c_call <function>, <parameter> */ 121#define MS_OP_PTR 12 /* ptr <pointer> */ 122#define MS_OP_ADELAY 13 /* adelay <val> */ 123#define MS_OP_BRSTAT 14 /* brstat <mask>, <mask>, <offset> */ 124#define MS_OP_SUBRET 15 /* subret <code> */ 125#define MS_OP_CALL 16 /* call <microsequence> */ 126#define MS_OP_RASSERT_P 17 /* rassert_p <iter>, <reg> */ 127#define MS_OP_RFETCH_P 18 /* rfetch_p <iter>, <reg>, <mask> */ 128#define MS_OP_TRIG 19 /* trigger <reg>, <len>, <array> */ 129.Ed 130.Ss Execution context 131The 132.Em execution context 133of microinstructions is: 134.Bl -bullet -offset indent 135.It 136the 137.Em program counter 138which points to the next microinstruction to execute either in the main 139microsequence or in a subcall 140.It 141the current value of 142.Em ptr 143which points to the next char to send/receive 144.It 145the current value of the internal 146.Em branch register 147.El 148.Pp 149This data is modified by some of the microinstructions, not all. 150.Ss MS_OP_GET and MS_OP_PUT 151are microinstructions used to do either predefined standard IEEE1284-1994 152transfers or programmed non-standard io. 153.Ss MS_OP_RFETCH - Register FETCH 154is used to retrieve the current value of a parallel port register, apply a 155mask and save it in a buffer. 156.Pp 157Parameters: 158.Bl -enum -offset indent 159.It 160register 161.It 162character mask 163.It 164pointer to the buffer 165.El 166.Pp 167Predefined macro: MS_RFETCH(reg,mask,ptr) 168.Ss MS_OP_RSET - Register SET 169is used to assert/clear some bits of a particular parallel port register, 170two masks are applied. 171.Pp 172Parameters: 173.Bl -enum -offset indent 174.It 175register 176.It 177mask of bits to assert 178.It 179mask of bits to clear 180.El 181.Pp 182Predefined macro: MS_RSET(reg,assert,clear) 183.Ss MS_OP_RASSERT - Register ASSERT 184is used to assert all bits of a particular parallel port register. 185.Pp 186Parameters: 187.Bl -enum -offset indent 188.It 189register 190.It 191byte to assert 192.El 193.Pp 194Predefined macro: MS_RASSERT(reg,byte) 195.Ss MS_OP_DELAY - microsecond DELAY 196is used to delay the execution of the microsequence. 197.Pp 198Parameter: 199.Bl -enum -offset indent 200.It 201delay in microseconds 202.El 203.Pp 204Predefined macro: MS_DELAY(delay) 205.Ss MS_OP_SET - SET internal branch register 206is used to set the value of the internal branch register. 207.Pp 208Parameter: 209.Bl -enum -offset indent 210.It 211integer value 212.El 213.Pp 214Predefined macro: MS_SET(accum) 215.Ss MS_OP_DBRA - \&Do BRAnch 216is used to branch if internal branch register decremented by one result value 217is positive. 218.Pp 219Parameter: 220.Bl -enum -offset indent 221.It 222integer offset in the current executed (sub)microsequence. 223Offset is added to 224the index of the next microinstruction to execute. 225.El 226.Pp 227Predefined macro: MS_DBRA(offset) 228.Ss MS_OP_BRSET - BRanch on SET 229is used to branch if some of the status register bits of the parallel port 230are set. 231.Pp 232Parameter: 233.Bl -enum -offset indent 234.It 235bits of the status register 236.It 237integer offset in the current executed (sub)microsequence. 238Offset is added to 239the index of the next microinstruction to execute. 240.El 241.Pp 242Predefined macro: MS_BRSET(mask,offset) 243.Ss MS_OP_BRCLEAR - BRanch on CLEAR 244is used to branch if some of the status register bits of the parallel port 245are cleared. 246.Pp 247Parameter: 248.Bl -enum -offset indent 249.It 250bits of the status register 251.It 252integer offset in the current executed (sub)microsequence. 253Offset is added to 254the index of the next microinstruction to execute. 255.El 256.Pp 257Predefined macro: MS_BRCLEAR(mask,offset) 258.Ss MS_OP_RET - RETurn 259is used to return from a microsequence. 260This instruction is mandatory. 261This 262is the only way for the microsequencer to detect the end of the microsequence. 263The return code is returned in the integer pointed by the (int *) parameter 264of the ppb_MS_microseq(). 265.Pp 266Parameter: 267.Bl -enum -offset indent 268.It 269integer return code 270.El 271.Pp 272Predefined macro: MS_RET(code) 273.Ss MS_OP_C_CALL - C function CALL 274is used to call C functions from microsequence execution. 275This may be useful 276when a non-standard i/o is performed to retrieve a data character from the 277parallel port. 278.Pp 279Parameter: 280.Bl -enum -offset indent 281.It 282the C function to call 283.It 284the parameter to pass to the function call 285.El 286.Pp 287The C function shall be declared as a 288.Ft int(*)(void *p, char *ptr) . 289The ptr parameter is the current position in the buffer currently scanned. 290.Pp 291Predefined macro: MS_C_CALL(func,param) 292.Ss MS_OP_PTR - initialize internal PTR 293is used to initialize the internal pointer to the currently scanned buffer. 294This pointer is passed to any C call (see above). 295.Pp 296Parameter: 297.Bl -enum -offset indent 298.It 299pointer to the buffer that shall be accessed by xxx_P() microsequence calls. 300Note that this pointer is automatically incremented during xxx_P() calls 301.El 302.Pp 303Predefined macro: MS_PTR(ptr) 304.Ss MS_OP_ADELAY - do an Asynchronous DELAY 305is used to make a tsleep() during microsequence execution. 306The tsleep is 307executed at PPBPRI level. 308.Pp 309Parameter: 310.Bl -enum -offset indent 311.It 312delay in ms 313.El 314.Pp 315Predefined macro: MS_ADELAY(delay) 316.Ss MS_OP_BRSTAT - BRanch on STATe 317is used to branch on status register state condition. 318.Pp 319Parameter: 320.Bl -enum -offset indent 321.It 322mask of asserted bits. 323Bits that shall be asserted in the status register 324are set in the mask 325.It 326mask of cleared bits. 327Bits that shall be cleared in the status register 328are set in the mask 329.It 330integer offset in the current executed (sub)microsequence. 331Offset is added 332to the index of the next microinstruction to execute. 333.El 334.Pp 335Predefined macro: MS_BRSTAT(asserted_bits,clear_bits,offset) 336.Ss MS_OP_SUBRET - SUBmicrosequence RETurn 337is used to return from the submicrosequence call. 338This action is mandatory 339before a RET call. 340Some microinstructions (PUT, GET) may not be callable 341within a submicrosequence. 342.Pp 343No parameter. 344.Pp 345Predefined macro: MS_SUBRET() 346.Ss MS_OP_CALL - submicrosequence CALL 347is used to call a submicrosequence. 348A submicrosequence is a microsequence with 349a SUBRET call. 350Parameter: 351.Bl -enum -offset indent 352.It 353the submicrosequence to execute 354.El 355.Pp 356Predefined macro: MS_CALL(microseq) 357.Ss MS_OP_RASSERT_P - Register ASSERT from internal PTR 358is used to assert a register with data currently pointed by the internal PTR 359pointer. 360Parameter: 361.Bl -enum -offset indent 362.It 363amount of data to write to the register 364.It 365register 366.El 367.Pp 368Predefined macro: MS_RASSERT_P(iter,reg) 369.Ss MS_OP_RFETCH_P - Register FETCH to internal PTR 370is used to fetch data from a register. 371Data is stored in the buffer currently 372pointed by the internal PTR pointer. 373Parameter: 374.Bl -enum -offset indent 375.It 376amount of data to read from the register 377.It 378register 379.It 380mask applied to fetched data 381.El 382.Pp 383Predefined macro: MS_RFETCH_P(iter,reg,mask) 384.Ss MS_OP_TRIG - TRIG register 385is used to trigger the parallel port. 386This microinstruction is intended to 387provide a very efficient control of the parallel port. 388Triggering a register 389is writing data, wait a while, write data, wait a while... 390This allows to 391write magic sequences to the port. 392Parameter: 393.Bl -enum -offset indent 394.It 395amount of data to read from the register 396.It 397register 398.It 399size of the array 400.It 401array of unsigned chars. 402Each couple of u_chars define the data to write to 403the register and the delay in us to wait. 404The delay is limited to 255 us to 405simplify and reduce the size of the array. 406.El 407.Pp 408Predefined macro: MS_TRIG(reg,len,array) 409.Sh MICROSEQUENCES 410.Ss C structures 411.Bd -literal 412union ppb_insarg { 413 int i; 414 char c; 415 void *p; 416 int (* f)(void *, char *); 417}; 418 419struct ppb_microseq { 420 int opcode; /* microins. opcode */ 421 union ppb_insarg arg[PPB_MS_MAXARGS]; /* arguments */ 422}; 423.Ed 424.Ss Using microsequences 425To instantiate a microsequence, just declare an array of ppb_microseq 426structures and initialize it as needed. 427You may either use predefined macros 428or code directly your microinstructions according to the ppb_microseq 429definition. 430For example, 431.Bd -literal 432 struct ppb_microseq select_microseq[] = { 433 434 /* parameter list 435 */ 436 #define SELECT_TARGET MS_PARAM(0, 1, MS_TYP_INT) 437 #define SELECT_INITIATOR MS_PARAM(3, 1, MS_TYP_INT) 438 439 /* send the select command to the drive */ 440 MS_DASS(MS_UNKNOWN), 441 MS_CASS(H_nAUTO | H_nSELIN | H_INIT | H_STROBE), 442 MS_CASS( H_AUTO | H_nSELIN | H_INIT | H_STROBE), 443 MS_DASS(MS_UNKNOWN), 444 MS_CASS( H_AUTO | H_nSELIN | H_nINIT | H_STROBE), 445 446 /* now, wait until the drive is ready */ 447 MS_SET(VP0_SELTMO), 448/* loop: */ MS_BRSET(H_ACK, 2 /* ready */), 449 MS_DBRA(-2 /* loop */), 450/* error: */ MS_RET(1), 451/* ready: */ MS_RET(0) 452 }; 453.Ed 454.Pp 455Here, some parameters are undefined and must be filled before executing 456the microsequence. 457In order to initialize each microsequence, one 458should use the ppb_MS_init_msq() function like this: 459.Bd -literal 460 ppb_MS_init_msq(select_microseq, 2, 461 SELECT_TARGET, 1 << target, 462 SELECT_INITIATOR, 1 << initiator); 463.Ed 464.Pp 465and then execute the microsequence. 466.Ss The microsequencer 467The microsequencer is executed either at ppbus or adapter level (see 468.Xr ppbus 4 469for info about ppbus system layers). 470Most of the microsequencer is executed 471at ppc level to avoid ppbus to adapter function call overhead. 472But some 473actions like deciding whereas the transfer is IEEE1284-1994 compliant are 474executed at ppbus layer. 475.Sh SEE ALSO 476.Xr ppbus 4 , 477.Xr ppc 4 , 478.Xr ppi 4 479.Sh HISTORY 480The 481.Nm 482manual page first appeared in 483.Fx 3.0 . 484.Sh AUTHORS 485This 486manual page was written by 487.An Nicolas Souchu . 488.Sh BUGS 489Only one level of submicrosequences is allowed. 490.Pp 491When triggering the port, maximum delay allowed is 255 us. 492