Blame usr/scsi.c

Packit eace71
/*
Packit eace71
 * The following is from the linux kernel scsi_error.c
Packit eace71
 *
Packit eace71
 *  scsi_error.c Copyright (C) 1997 Eric Youngdale
Packit eace71
 *
Packit eace71
 *  SCSI error/timeout handling
Packit eace71
 *      Initial versions: Eric Youngdale.  Based upon conversations with
Packit eace71
 *                        Leonard Zubkoff and David Miller at Linux Expo,
Packit eace71
 *                        ideas originating from all over the place.
Packit eace71
 *
Packit eace71
 *	Restructured scsi_unjam_host and associated functions.
Packit eace71
 *	September 04, 2002 Mike Anderson (andmike@us.ibm.com)
Packit eace71
 *
Packit eace71
 *	Forward port of Russell King's (rmk@arm.linux.org.uk) changes and
Packit eace71
 *	minor  cleanups.
Packit eace71
 *	September 30, 2002 Mike Anderson (andmike@us.ibm.com)
Packit eace71
 */
Packit eace71
Packit eace71
#include <string.h>
Packit eace71
#include "scsi.h"
Packit eace71
Packit eace71
/**
Packit eace71
 * scsi_normalize_sense - normalize main elements from either fixed or
Packit eace71
 *			descriptor sense data format into a common format.
Packit eace71
 *
Packit eace71
 * @sense_buffer:	byte array containing sense data returned by device
Packit eace71
 * @sb_len:		number of valid bytes in sense_buffer
Packit eace71
 * @sshdr:		pointer to instance of structure that common
Packit eace71
 *			elements are written to.
Packit eace71
 *
Packit eace71
 * Notes:
Packit eace71
 *	The "main elements" from sense data are: response_code, sense_key,
Packit eace71
 *	asc, ascq and additional_length (only for descriptor format).
Packit eace71
 *
Packit eace71
 *	Typically this function can be called after a device has
Packit eace71
 *	responded to a SCSI command with the CHECK_CONDITION status.
Packit eace71
 *
Packit eace71
 * Return value:
Packit eace71
 *	1 if valid sense data information found, else 0;
Packit eace71
 **/
Packit eace71
int scsi_normalize_sense(const uint8_t *sense_buffer, int sb_len,
Packit eace71
                         struct scsi_sense_hdr *sshdr)
Packit eace71
{
Packit eace71
	if (!sense_buffer || !sb_len)
Packit eace71
		return 0;
Packit eace71
Packit eace71
	memset(sshdr, 0, sizeof(struct scsi_sense_hdr));
Packit eace71
Packit eace71
	sshdr->response_code = (sense_buffer[0] & 0x7f);
Packit eace71
Packit eace71
	if (!scsi_sense_valid(sshdr))
Packit eace71
		return 0;
Packit eace71
Packit eace71
	if (sshdr->response_code >= 0x72) {
Packit eace71
		/*
Packit eace71
		 * descriptor format
Packit eace71
		 */
Packit eace71
		if (sb_len > 1)
Packit eace71
			sshdr->sense_key = (sense_buffer[1] & 0xf);
Packit eace71
		if (sb_len > 2)
Packit eace71
			sshdr->asc = sense_buffer[2];
Packit eace71
		if (sb_len > 3)
Packit eace71
			sshdr->ascq = sense_buffer[3];
Packit eace71
		if (sb_len > 7)
Packit eace71
			sshdr->additional_length = sense_buffer[7];
Packit eace71
	} else {
Packit eace71
		/* 
Packit eace71
		 * fixed format
Packit eace71
		 */
Packit eace71
		if (sb_len > 2)
Packit eace71
			sshdr->sense_key = (sense_buffer[2] & 0xf);
Packit eace71
		if (sb_len > 7) {
Packit eace71
			sb_len = (sb_len < (sense_buffer[7] + 8)) ?
Packit eace71
					 sb_len : (sense_buffer[7] + 8);
Packit eace71
			if (sb_len > 12)
Packit eace71
				sshdr->asc = sense_buffer[12];
Packit eace71
			if (sb_len > 13)
Packit eace71
				sshdr->ascq = sense_buffer[13];
Packit eace71
		}
Packit eace71
	}
Packit eace71
Packit eace71
	return 1;
Packit eace71
}