source: xtideuniversalbios/trunk/XTIDE_Universal_BIOS/Src/Initialization/AdvancedAta/Vision.asm@ 569

Last change on this file since 569 was 566, checked in by krille_n_@…, 11 years ago

Changes to XTIDE Universal BIOS:

  • Fixed a very old bug (from r150) in IdeError.asm where GetBiosErrorCodeToAHfromStatusAndErrorRegistersInAX would return RET_HD_SEEK_FAIL instead of RET_HD_STATUSERR.
  • Optimizations and other fixes.
File size: 8.9 KB
Line 
1; Project name : XTIDE Universal BIOS
2; Description : Functions for initializing QDI Vision
3; QD6500 and QD6580 VLB IDE Controllers.
4
5;
6; XTIDE Universal BIOS and Associated Tools
7; Copyright (C) 2009-2010 by Tomi Tilli, 2011-2013 by XTIDE Universal BIOS Team.
8;
9; This program is free software; you can redistribute it and/or modify
10; it under the terms of the GNU General Public License as published by
11; the Free Software Foundation; either version 2 of the License, or
12; (at your option) any later version.
13;
14; This program is distributed in the hope that it will be useful,
15; but WITHOUT ANY WARRANTY; without even the implied warranty of
16; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17; GNU General Public License for more details.
18; Visit http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
19;
20
21; Section containing code
22SECTION .text
23
24;--------------------------------------------------------------------
25; Vision_DetectAndReturnIDinAXandPortInDXifControllerPresent
26; Parameters:
27; Nothing
28; Returns:
29; AX: ID WORD specific for QDI Vision Controllers
30; (AL = QD65xx Config Register contents)
31; (AH = QDI Vision Controller ID (bits 4...7))
32; DX: Controller port (not IDE port)
33; ZF: Set if controller found
34; Cleared if supported controller not found (AX,DX = undefined)
35; Corrupts registers:
36; Nothing
37;--------------------------------------------------------------------
38Vision_DetectAndReturnIDinAXandPortInDXifControllerPresent:
39 ; Check QD65xx base port
40 mov dx, QD65XX_BASE_PORT
41 in al, QD65XX_BASE_PORT + QD65XX_CONFIG_REGISTER_in
42
43%ifdef DANGEROUS_DETECTION
44 ; Checking alternative base port is currently commented away
45 ; since Intel PIIX4 south bridge mirrors Interrupt Controller registers
46 ; from Axh to Bxh.
47 call IsConfigRegisterWithIDinAL
48 je SHORT VisionControllerDetected
49
50 ; Check QD65xx alternative base port
51 or dl, QD65XX_ALTERNATIVE_BASE_PORT
52 in al, QD65XX_ALTERNATIVE_BASE_PORT + QD65XX_CONFIG_REGISTER_in
53%endif ; DANGEROUS_DETECTION
54 ; Fall to IsConfigRegisterWithIDinAL
55
56;--------------------------------------------------------------------
57; IsConfigRegisterWithIDinAL
58; Parameters:
59; AL: Possible QD65xx Config Register contents
60; Returns:
61; AH QDI Vision Controller ID or undefined
62; ZF: Set if controller found
63; Cleared if supported controller not found (AH = undefined)
64; Corrupts registers:
65; Nothing
66;--------------------------------------------------------------------
67IsConfigRegisterWithIDinAL:
68 mov ah, al
69 and al, MASK_QDCONFIG_CONTROLLER_ID
70 cmp al, ID_QD6500 << 4
71 je SHORT VisionControllerDetected
72 cmp al, ID_QD6580 << 4
73 je SHORT VisionControllerDetected
74 cmp al, ID_QD6580_ALTERNATE << 4
75VisionControllerDetected:
76 xchg ah, al
77 ret
78
79
80;--------------------------------------------------------------------
81; Vision_DoesIdePortInBXbelongToControllerWithIDinAX
82; Parameters:
83; AL: QD65xx Config Register contents
84; AH: QDI Vision Controller ID (bits 4...7)
85; BX: IDE Base port to check
86; DX: Vision Controller port
87; Returns:
88; ZF: Set if port belongs to controller
89; Cleared if port belongs to another controller
90; Corrupts registers:
91; Nothing
92;--------------------------------------------------------------------
93Vision_DoesIdePortInBXbelongToControllerWithIDinAX:
94 cmp ah, ID_QD6500 << 4
95 je SHORT .DoesIdePortInDXbelongToQD6500
96
97 ; QD6580 always have Primary IDE at 1F0h
98 ; Secondary IDE at 170h can be enabled or disabled
99 cmp bx, DEVICE_ATA_PRIMARY_PORT
100 je SHORT .ReturnResultInZF
101
102 ; Check if Secondary IDE channel is enabled
103 push ax
104 add dx, BYTE QD6580_CONTROL_REGISTER
105 in al, dx
106 sub dx, BYTE QD6580_CONTROL_REGISTER
107 test al, FLG_QDCONTROL_SECONDARY_DISABLED_in
108 pop ax
109 jz SHORT .CompareBXtoSecondaryIDE
110 ret
111
112 ; QD6500 has only one IDE channel that can be at 1F0h or 170h
113.DoesIdePortInDXbelongToQD6500:
114 test al, FLG_QDCONFIG_PRIMARY_IDE
115 jz SHORT .CompareBXtoSecondaryIDE
116 cmp bx, DEVICE_ATA_PRIMARY_PORT
117 ret
118
119.CompareBXtoSecondaryIDE:
120 cmp bx, DEVICE_ATA_SECONDARY_PORT
121.ReturnResultInZF:
122 ret
123
124
125;--------------------------------------------------------------------
126; Vision_GetMaxPioModeToALandMinCycleTimeToBX
127; Parameters:
128; AL: QD65xx Config Register contents
129; AH: QDI Vision Controller ID (bits 4...7)
130; Returns:
131; AL: Max supported PIO mode
132; AH: FLGH_DPT_IORDY if IORDY supported, zero otherwise
133; BX: Min PIO Cycle Time (only if CF set)
134; CF: Set if PIO limit necessary
135; Cleared if no need to limit timings
136; Corrupts registers:
137; (AX if CF cleared)
138; Corrupts registers:
139; Nothing
140;--------------------------------------------------------------------
141Vision_GetMaxPioModeToALandMinCycleTimeToBX:
142 cmp ah, ID_QD6500 << 4
143 clc
144 jne SHORT .NoNeedToLimitForQD6580
145
146 mov ax, 2 ; Limit to PIO 2 because QD6500 does not support IORDY
147 mov bx, PIO_2_MIN_CYCLE_TIME_NS
148 stc
149.NoNeedToLimitForQD6580:
150 ret
151
152
153;--------------------------------------------------------------------
154; Vision_InitializeWithIDinAHandConfigInAL
155; Parameters:
156; AL: QD65xx Config Register contents
157; AH: QDI Vision Controller ID (bits 4...7)
158; DS:DI: Ptr to DPT for Single or Slave Drive
159; SI: Offset to Master DPT if Slave Drive present
160; Zero if Slave Drive not present
161; Returns:
162; CF: Cleared if success
163; Set if error
164; Corrupts registers:
165; AX, BX, CX, DX, BP
166;--------------------------------------------------------------------
167Vision_InitializeWithIDinAHandConfigInAL:
168 ; QD6580 has a Control Register that needs to be programmed
169 mov dx, [di+DPT_ADVANCED_ATA.wControllerBasePort]
170 cmp ah, ID_QD6500 << 4
171 je SHORT .CalculateTimingForQD6500
172
173 ; Program QD6580 Control Register (not available on QD6500) to
174 ; Enable or Disable Read-Ahead and Post-Write Buffer to match
175 ; jumper setting on the multi I/O card.
176 xor ax, ax
177 add dx, BYTE QD6580_CONTROL_REGISTER
178 in al, dx ; Read to get ATAPI jumper status
179 test al, FLG_QDCONTROL_HDONLY_in
180 eCMOVNZ ah, FLG_QDCONTROL_NONATAPI ; Enable Read-Ahead and Post-Write Buffers
181 mov al, ah
182 or al, MASK_QDCONTROL_FLAGS_TO_SET
183 out dx, al
184 sub dx, BYTE QD6580_CONTROL_REGISTER
185
186 ; Now we need to determine is the drive connected to the Primary or Secondary channel.
187 ; QD6500 has only one channel that can be Primary at 1F0h or Secondary at 170h.
188 ; QD6580 always has Primary channel at 1F0h. Secondary channel at 170h can be Enabled or Disabled.
189 cmp WORD [di+DPT.wBasePort], DEVICE_ATA_PRIMARY_PORT
190 je SHORT .CalculateTimingTicksForQD6580 ; Primary Channel so no need to modify DX
191 times 2 inc dx ; Secondary Channel IDE Timing Register
192
193 ; QD6500 and QD6580 require slightly different calculations.
194.CalculateTimingTicksForQD6580:
195 mov bp, QD6580_MAX_ACTIVE_TIME_CLOCKS | (QD6580_MIN_ACTIVE_TIME_CLOCKS << 8)
196 jmp SHORT .CalculateTimingsForQD65xx
197
198.CalculateTimingForQD6500:
199 mov bp, QD6500_MAX_ACTIVE_TIME_CLOCKS | (QD6500_MIN_ACTIVE_TIME_CLOCKS << 8)
200
201 ; We need the PIO Cycle Time in CX to calculate Active and Recovery Times.
202.CalculateTimingsForQD65xx:
203 call AdvAtaInit_SelectSlowestCommonPioTimingsToBXandCXfromDSSIandDSDI
204
205 ; Calculate Active Time value for QD65xx IDE Timing Register
206 call AtaID_GetActiveTimeToAXfromPioModeInBX
207 call ConvertNanosecsFromAXwithLimitsInBPtoRegisterValue
208 xchg bp, ax
209
210 ; Calculate Recovery Time value for QD65xx IDE Timing Register
211 call AtaID_GetRecoveryTimeToAXfromPioModeInBXandCycleTimeInCX
212 mov bx, bp ; Active Time value now in BL
213 mov bp, QD65xx_MAX_RECOVERY_TIME_CLOCKS | (QD65xx_MIN_RECOVERY_TIME_CLOCKS << 8)
214 call ConvertNanosecsFromAXwithLimitsInBPtoRegisterValue
215
216 ; Merge the values to a single byte to output
217 eSHL_IM al, POSITION_QD65XXIDE_RECOVERY_TIME
218 or al, bl
219 out dx, al
220 ret ; Return with CF cleared
221
222
223;--------------------------------------------------------------------
224; ConvertNanosecsFromAXwithLimitsInBPtoRegisterValue
225; Parameters:
226; AX: Nanosecs to convert
227; BP: Low Byte: Maximum allowed ticks
228; High Byte: Minimum allowed ticks
229; DS:DI: Ptr to DPT for Single or Slave Drive
230; Returns:
231; AL: Timing value for QD65xx register
232; Corrupts registers:
233; Nothing
234;--------------------------------------------------------------------
235ConvertNanosecsFromAXwithLimitsInBPtoRegisterValue:
236 push cx
237
238 ; Get VLB Cycle Time in nanosecs
239 mov cl, VLB_33MHZ_CYCLE_TIME ; Assume 33 MHz or slower VLB bus
240 test BYTE [di+DPT_ADVANCED_ATA.wControllerID], FLG_QDCONFIG_ID3
241 eCMOVZ cl, VLB_40MHZ_CYCLE_TIME
242
243 ; Convert value in AX to VLB ticks
244 div cl ; AL = VLB ticks
245 inc ax ; Round up
246
247 ; Limit value to QD65xx limits
248 mov cx, bp
249 MAX_U al, ch ; Make sure not below minimum
250 MIN_U al, cl ; Make sure not above maximum
251
252 ; Not done yet, we need to invert the ticks since 0 is the slowest
253 ; value on the timing register
254 sub cl, al
255 xchg ax, cx ; Return in AL
256
257 pop cx
258 ret
Note: See TracBrowser for help on using the repository browser.