4091 e1000g I217/I218 support
[illumos-gate.git] / usr / src / uts / common / io / e1000api / e1000_manage.c
blobc58d32d16b10809ae0ef0fa2dd0006f48cd09ca6
1 /******************************************************************************
3 Copyright (c) 2001-2013, Intel Corporation
4 All rights reserved.
6 Redistribution and use in source and binary forms, with or without
7 modification, are permitted provided that the following conditions are met:
9 1. Redistributions of source code must retain the above copyright notice,
10 this list of conditions and the following disclaimer.
12 2. Redistributions in binary form must reproduce the above copyright
13 notice, this list of conditions and the following disclaimer in the
14 documentation and/or other materials provided with the distribution.
16 3. Neither the name of the Intel Corporation nor the names of its
17 contributors may be used to endorse or promote products derived from
18 this software without specific prior written permission.
20 THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
21 AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22 IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23 ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
24 LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
25 CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
26 SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
27 INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
28 CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
29 ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
30 POSSIBILITY OF SUCH DAMAGE.
32 ******************************************************************************/
33 /*$FreeBSD$*/
35 #include "e1000_api.h"
37 /**
38 * e1000_calculate_checksum - Calculate checksum for buffer
39 * @buffer: pointer to EEPROM
40 * @length: size of EEPROM to calculate a checksum for
42 * Calculates the checksum for some buffer on a specified length. The
43 * checksum calculated is returned.
44 **/
45 u8 e1000_calculate_checksum(u8 *buffer, u32 length)
47 u32 i;
48 u8 sum = 0;
50 DEBUGFUNC("e1000_calculate_checksum");
52 if (!buffer)
53 return 0;
55 for (i = 0; i < length; i++)
56 sum += buffer[i];
58 return (u8) (0 - sum);
61 /**
62 * e1000_mng_enable_host_if_generic - Checks host interface is enabled
63 * @hw: pointer to the HW structure
65 * Returns E1000_success upon success, else E1000_ERR_HOST_INTERFACE_COMMAND
67 * This function checks whether the HOST IF is enabled for command operation
68 * and also checks whether the previous command is completed. It busy waits
69 * in case of previous command is not completed.
70 **/
71 s32 e1000_mng_enable_host_if_generic(struct e1000_hw *hw)
73 u32 hicr;
74 u8 i;
76 DEBUGFUNC("e1000_mng_enable_host_if_generic");
78 if (!hw->mac.arc_subsystem_valid) {
79 DEBUGOUT("ARC subsystem not valid.\n");
80 return -E1000_ERR_HOST_INTERFACE_COMMAND;
83 /* Check that the host interface is enabled. */
84 hicr = E1000_READ_REG(hw, E1000_HICR);
85 if (!(hicr & E1000_HICR_EN)) {
86 DEBUGOUT("E1000_HOST_EN bit disabled.\n");
87 return -E1000_ERR_HOST_INTERFACE_COMMAND;
89 /* check the previous command is completed */
90 for (i = 0; i < E1000_MNG_DHCP_COMMAND_TIMEOUT; i++) {
91 hicr = E1000_READ_REG(hw, E1000_HICR);
92 if (!(hicr & E1000_HICR_C))
93 break;
94 msec_delay_irq(1);
97 if (i == E1000_MNG_DHCP_COMMAND_TIMEOUT) {
98 DEBUGOUT("Previous command timeout failed .\n");
99 return -E1000_ERR_HOST_INTERFACE_COMMAND;
102 return E1000_SUCCESS;
106 * e1000_check_mng_mode_generic - Generic check management mode
107 * @hw: pointer to the HW structure
109 * Reads the firmware semaphore register and returns TRUE (>0) if
110 * manageability is enabled, else FALSE (0).
112 bool e1000_check_mng_mode_generic(struct e1000_hw *hw)
114 u32 fwsm = E1000_READ_REG(hw, E1000_FWSM);
116 DEBUGFUNC("e1000_check_mng_mode_generic");
119 return (fwsm & E1000_FWSM_MODE_MASK) ==
120 (E1000_MNG_IAMT_MODE << E1000_FWSM_MODE_SHIFT);
124 * e1000_enable_tx_pkt_filtering_generic - Enable packet filtering on Tx
125 * @hw: pointer to the HW structure
127 * Enables packet filtering on transmit packets if manageability is enabled
128 * and host interface is enabled.
130 bool e1000_enable_tx_pkt_filtering_generic(struct e1000_hw *hw)
132 struct e1000_host_mng_dhcp_cookie *hdr = &hw->mng_cookie;
133 u32 *buffer = (u32 *)&hw->mng_cookie;
134 u32 offset;
135 s32 ret_val, hdr_csum, csum;
136 u8 i, len;
138 DEBUGFUNC("e1000_enable_tx_pkt_filtering_generic");
140 hw->mac.tx_pkt_filtering = TRUE;
142 /* No manageability, no filtering */
143 if (!hw->mac.ops.check_mng_mode(hw)) {
144 hw->mac.tx_pkt_filtering = FALSE;
145 return hw->mac.tx_pkt_filtering;
148 /* If we can't read from the host interface for whatever
149 * reason, disable filtering.
151 ret_val = e1000_mng_enable_host_if_generic(hw);
152 if (ret_val != E1000_SUCCESS) {
153 hw->mac.tx_pkt_filtering = FALSE;
154 return hw->mac.tx_pkt_filtering;
157 /* Read in the header. Length and offset are in dwords. */
158 len = E1000_MNG_DHCP_COOKIE_LENGTH >> 2;
159 offset = E1000_MNG_DHCP_COOKIE_OFFSET >> 2;
160 for (i = 0; i < len; i++)
161 *(buffer + i) = E1000_READ_REG_ARRAY_DWORD(hw, E1000_HOST_IF,
162 offset + i);
163 hdr_csum = hdr->checksum;
164 hdr->checksum = 0;
165 csum = e1000_calculate_checksum((u8 *)hdr,
166 E1000_MNG_DHCP_COOKIE_LENGTH);
167 /* If either the checksums or signature don't match, then
168 * the cookie area isn't considered valid, in which case we
169 * take the safe route of assuming Tx filtering is enabled.
171 if ((hdr_csum != csum) || (hdr->signature != E1000_IAMT_SIGNATURE)) {
172 hw->mac.tx_pkt_filtering = TRUE;
173 return hw->mac.tx_pkt_filtering;
176 /* Cookie area is valid, make the final check for filtering. */
177 if (!(hdr->status & E1000_MNG_DHCP_COOKIE_STATUS_PARSING))
178 hw->mac.tx_pkt_filtering = FALSE;
180 return hw->mac.tx_pkt_filtering;
184 * e1000_mng_write_cmd_header_generic - Writes manageability command header
185 * @hw: pointer to the HW structure
186 * @hdr: pointer to the host interface command header
188 * Writes the command header after does the checksum calculation.
190 s32 e1000_mng_write_cmd_header_generic(struct e1000_hw *hw,
191 struct e1000_host_mng_command_header *hdr)
193 u16 i, length = sizeof(struct e1000_host_mng_command_header);
195 DEBUGFUNC("e1000_mng_write_cmd_header_generic");
197 /* Write the whole command header structure with new checksum. */
199 hdr->checksum = e1000_calculate_checksum((u8 *)hdr, length);
201 length >>= 2;
202 /* Write the relevant command block into the ram area. */
203 for (i = 0; i < length; i++) {
204 E1000_WRITE_REG_ARRAY_DWORD(hw, E1000_HOST_IF, i,
205 *((u32 *) hdr + i));
206 E1000_WRITE_FLUSH(hw);
209 return E1000_SUCCESS;
213 * e1000_mng_host_if_write_generic - Write to the manageability host interface
214 * @hw: pointer to the HW structure
215 * @buffer: pointer to the host interface buffer
216 * @length: size of the buffer
217 * @offset: location in the buffer to write to
218 * @sum: sum of the data (not checksum)
220 * This function writes the buffer content at the offset given on the host if.
221 * It also does alignment considerations to do the writes in most efficient
222 * way. Also fills up the sum of the buffer in *buffer parameter.
224 s32 e1000_mng_host_if_write_generic(struct e1000_hw *hw, u8 *buffer,
225 u16 length, u16 offset, u8 *sum)
227 u8 *tmp;
228 u8 *bufptr = buffer;
229 u32 data = 0;
230 u16 remaining, i, j, prev_bytes;
232 DEBUGFUNC("e1000_mng_host_if_write_generic");
234 /* sum = only sum of the data and it is not checksum */
236 if (length == 0 || offset + length > E1000_HI_MAX_MNG_DATA_LENGTH)
237 return -E1000_ERR_PARAM;
239 tmp = (u8 *)&data;
240 prev_bytes = offset & 0x3;
241 offset >>= 2;
243 if (prev_bytes) {
244 data = E1000_READ_REG_ARRAY_DWORD(hw, E1000_HOST_IF, offset);
245 for (j = prev_bytes; j < sizeof(u32); j++) {
246 *(tmp + j) = *bufptr++;
247 *sum += *(tmp + j);
249 E1000_WRITE_REG_ARRAY_DWORD(hw, E1000_HOST_IF, offset, data);
250 length -= j - prev_bytes;
251 offset++;
254 remaining = length & 0x3;
255 length -= remaining;
257 /* Calculate length in DWORDs */
258 length >>= 2;
260 /* The device driver writes the relevant command block into the
261 * ram area.
263 for (i = 0; i < length; i++) {
264 for (j = 0; j < sizeof(u32); j++) {
265 *(tmp + j) = *bufptr++;
266 *sum += *(tmp + j);
269 E1000_WRITE_REG_ARRAY_DWORD(hw, E1000_HOST_IF, offset + i,
270 data);
272 if (remaining) {
273 for (j = 0; j < sizeof(u32); j++) {
274 if (j < remaining)
275 *(tmp + j) = *bufptr++;
276 else
277 *(tmp + j) = 0;
279 *sum += *(tmp + j);
281 E1000_WRITE_REG_ARRAY_DWORD(hw, E1000_HOST_IF, offset + i,
282 data);
285 return E1000_SUCCESS;
289 * e1000_mng_write_dhcp_info_generic - Writes DHCP info to host interface
290 * @hw: pointer to the HW structure
291 * @buffer: pointer to the host interface
292 * @length: size of the buffer
294 * Writes the DHCP information to the host interface.
296 s32 e1000_mng_write_dhcp_info_generic(struct e1000_hw *hw, u8 *buffer,
297 u16 length)
299 struct e1000_host_mng_command_header hdr;
300 s32 ret_val;
301 u32 hicr;
303 DEBUGFUNC("e1000_mng_write_dhcp_info_generic");
305 hdr.command_id = E1000_MNG_DHCP_TX_PAYLOAD_CMD;
306 hdr.command_length = length;
307 hdr.reserved1 = 0;
308 hdr.reserved2 = 0;
309 hdr.checksum = 0;
311 /* Enable the host interface */
312 ret_val = e1000_mng_enable_host_if_generic(hw);
313 if (ret_val)
314 return ret_val;
316 /* Populate the host interface with the contents of "buffer". */
317 ret_val = e1000_mng_host_if_write_generic(hw, buffer, length,
318 sizeof(hdr), &(hdr.checksum));
319 if (ret_val)
320 return ret_val;
322 /* Write the manageability command header */
323 ret_val = e1000_mng_write_cmd_header_generic(hw, &hdr);
324 if (ret_val)
325 return ret_val;
327 /* Tell the ARC a new command is pending. */
328 hicr = E1000_READ_REG(hw, E1000_HICR);
329 E1000_WRITE_REG(hw, E1000_HICR, hicr | E1000_HICR_C);
331 return E1000_SUCCESS;
335 * e1000_enable_mng_pass_thru - Check if management passthrough is needed
336 * @hw: pointer to the HW structure
338 * Verifies the hardware needs to leave interface enabled so that frames can
339 * be directed to and from the management interface.
341 bool e1000_enable_mng_pass_thru(struct e1000_hw *hw)
343 u32 manc;
344 u32 fwsm, factps;
346 DEBUGFUNC("e1000_enable_mng_pass_thru");
348 if (!hw->mac.asf_firmware_present)
349 return FALSE;
351 manc = E1000_READ_REG(hw, E1000_MANC);
353 if (!(manc & E1000_MANC_RCV_TCO_EN))
354 return FALSE;
356 if (hw->mac.has_fwsm) {
357 fwsm = E1000_READ_REG(hw, E1000_FWSM);
358 factps = E1000_READ_REG(hw, E1000_FACTPS);
360 if (!(factps & E1000_FACTPS_MNGCG) &&
361 ((fwsm & E1000_FWSM_MODE_MASK) ==
362 (e1000_mng_mode_pt << E1000_FWSM_MODE_SHIFT)))
363 return TRUE;
364 } else if ((hw->mac.type == e1000_82574) ||
365 (hw->mac.type == e1000_82583)) {
366 u16 data;
368 factps = E1000_READ_REG(hw, E1000_FACTPS);
369 e1000_read_nvm(hw, NVM_INIT_CONTROL2_REG, 1, &data);
371 if (!(factps & E1000_FACTPS_MNGCG) &&
372 ((data & E1000_NVM_INIT_CTRL2_MNGM) ==
373 (e1000_mng_mode_pt << 13)))
374 return TRUE;
375 } else if ((manc & E1000_MANC_SMBUS_EN) &&
376 !(manc & E1000_MANC_ASF_EN)) {
377 return TRUE;
380 return FALSE;
384 * e1000_host_interface_command - Writes buffer to host interface
385 * @hw: pointer to the HW structure
386 * @buffer: contains a command to write
387 * @length: the byte length of the buffer, must be multiple of 4 bytes
389 * Writes a buffer to the Host Interface. Upon success, returns E1000_SUCCESS
390 * else returns E1000_ERR_HOST_INTERFACE_COMMAND.
392 s32 e1000_host_interface_command(struct e1000_hw *hw, u8 *buffer, u32 length)
394 u32 hicr, i;
396 DEBUGFUNC("e1000_host_interface_command");
398 if (!(hw->mac.arc_subsystem_valid)) {
399 DEBUGOUT("Hardware doesn't support host interface command.\n");
400 return E1000_SUCCESS;
403 if (!hw->mac.asf_firmware_present) {
404 DEBUGOUT("Firmware is not present.\n");
405 return E1000_SUCCESS;
408 if (length == 0 || length & 0x3 ||
409 length > E1000_HI_MAX_BLOCK_BYTE_LENGTH) {
410 DEBUGOUT("Buffer length failure.\n");
411 return -E1000_ERR_HOST_INTERFACE_COMMAND;
414 /* Check that the host interface is enabled. */
415 hicr = E1000_READ_REG(hw, E1000_HICR);
416 if (!(hicr & E1000_HICR_EN)) {
417 DEBUGOUT("E1000_HOST_EN bit disabled.\n");
418 return -E1000_ERR_HOST_INTERFACE_COMMAND;
421 /* Calculate length in DWORDs */
422 length >>= 2;
424 /* The device driver writes the relevant command block
425 * into the ram area.
427 for (i = 0; i < length; i++)
428 E1000_WRITE_REG_ARRAY_DWORD(hw, E1000_HOST_IF, i,
429 *((u32 *)buffer + i));
431 /* Setting this bit tells the ARC that a new command is pending. */
432 E1000_WRITE_REG(hw, E1000_HICR, hicr | E1000_HICR_C);
434 for (i = 0; i < E1000_HI_COMMAND_TIMEOUT; i++) {
435 hicr = E1000_READ_REG(hw, E1000_HICR);
436 if (!(hicr & E1000_HICR_C))
437 break;
438 msec_delay(1);
441 /* Check command successful completion. */
442 if (i == E1000_HI_COMMAND_TIMEOUT ||
443 (!(E1000_READ_REG(hw, E1000_HICR) & E1000_HICR_SV))) {
444 DEBUGOUT("Command has failed with no status valid.\n");
445 return -E1000_ERR_HOST_INTERFACE_COMMAND;
448 for (i = 0; i < length; i++)
449 *((u32 *)buffer + i) = E1000_READ_REG_ARRAY_DWORD(hw,
450 E1000_HOST_IF,
453 return E1000_SUCCESS;
456 * e1000_load_firmware - Writes proxy FW code buffer to host interface
457 * and execute.
458 * @hw: pointer to the HW structure
459 * @buffer: contains a firmware to write
460 * @length: the byte length of the buffer, must be multiple of 4 bytes
462 * Upon success returns E1000_SUCCESS, returns E1000_ERR_CONFIG if not enabled
463 * in HW else returns E1000_ERR_HOST_INTERFACE_COMMAND.
465 s32 e1000_load_firmware(struct e1000_hw *hw, u8 *buffer, u32 length)
467 u32 hicr, hibba, fwsm, icr, i;
469 DEBUGFUNC("e1000_load_firmware");
471 if (hw->mac.type < e1000_i210) {
472 DEBUGOUT("Hardware doesn't support loading FW by the driver\n");
473 return -E1000_ERR_CONFIG;
476 /* Check that the host interface is enabled. */
477 hicr = E1000_READ_REG(hw, E1000_HICR);
478 if (!(hicr & E1000_HICR_EN)) {
479 DEBUGOUT("E1000_HOST_EN bit disabled.\n");
480 return -E1000_ERR_CONFIG;
482 if (!(hicr & E1000_HICR_MEMORY_BASE_EN)) {
483 DEBUGOUT("E1000_HICR_MEMORY_BASE_EN bit disabled.\n");
484 return -E1000_ERR_CONFIG;
487 if (length == 0 || length & 0x3 || length > E1000_HI_FW_MAX_LENGTH) {
488 DEBUGOUT("Buffer length failure.\n");
489 return -E1000_ERR_INVALID_ARGUMENT;
492 /* Clear notification from ROM-FW by reading ICR register */
493 icr = E1000_READ_REG(hw, E1000_ICR_V2);
495 /* Reset ROM-FW */
496 hicr = E1000_READ_REG(hw, E1000_HICR);
497 hicr |= E1000_HICR_FW_RESET_ENABLE;
498 E1000_WRITE_REG(hw, E1000_HICR, hicr);
499 hicr |= E1000_HICR_FW_RESET;
500 E1000_WRITE_REG(hw, E1000_HICR, hicr);
501 E1000_WRITE_FLUSH(hw);
503 /* Wait till MAC notifies about its readiness after ROM-FW reset */
504 for (i = 0; i < (E1000_HI_COMMAND_TIMEOUT * 2); i++) {
505 icr = E1000_READ_REG(hw, E1000_ICR_V2);
506 if (icr & E1000_ICR_MNG)
507 break;
508 msec_delay(1);
511 /* Check for timeout */
512 if (i == E1000_HI_COMMAND_TIMEOUT) {
513 DEBUGOUT("FW reset failed.\n");
514 return -E1000_ERR_HOST_INTERFACE_COMMAND;
517 /* Wait till MAC is ready to accept new FW code */
518 for (i = 0; i < E1000_HI_COMMAND_TIMEOUT; i++) {
519 fwsm = E1000_READ_REG(hw, E1000_FWSM);
520 if ((fwsm & E1000_FWSM_FW_VALID) &&
521 ((fwsm & E1000_FWSM_MODE_MASK) >> E1000_FWSM_MODE_SHIFT ==
522 E1000_FWSM_HI_EN_ONLY_MODE))
523 break;
524 msec_delay(1);
527 /* Check for timeout */
528 if (i == E1000_HI_COMMAND_TIMEOUT) {
529 DEBUGOUT("FW reset failed.\n");
530 return -E1000_ERR_HOST_INTERFACE_COMMAND;
533 /* Calculate length in DWORDs */
534 length >>= 2;
536 /* The device driver writes the relevant FW code block
537 * into the ram area in DWORDs via 1kB ram addressing window.
539 for (i = 0; i < length; i++) {
540 if (!(i % E1000_HI_FW_BLOCK_DWORD_LENGTH)) {
541 /* Point to correct 1kB ram window */
542 hibba = E1000_HI_FW_BASE_ADDRESS +
543 ((E1000_HI_FW_BLOCK_DWORD_LENGTH << 2) *
544 (i / E1000_HI_FW_BLOCK_DWORD_LENGTH));
546 E1000_WRITE_REG(hw, E1000_HIBBA, hibba);
549 E1000_WRITE_REG_ARRAY_DWORD(hw, E1000_HOST_IF,
550 i % E1000_HI_FW_BLOCK_DWORD_LENGTH,
551 *((u32 *)buffer + i));
554 /* Setting this bit tells the ARC that a new FW is ready to execute. */
555 hicr = E1000_READ_REG(hw, E1000_HICR);
556 E1000_WRITE_REG(hw, E1000_HICR, hicr | E1000_HICR_C);
558 for (i = 0; i < E1000_HI_COMMAND_TIMEOUT; i++) {
559 hicr = E1000_READ_REG(hw, E1000_HICR);
560 if (!(hicr & E1000_HICR_C))
561 break;
562 msec_delay(1);
565 /* Check for successful FW start. */
566 if (i == E1000_HI_COMMAND_TIMEOUT) {
567 DEBUGOUT("New FW did not start within timeout period.\n");
568 return -E1000_ERR_HOST_INTERFACE_COMMAND;
571 return E1000_SUCCESS;