view usr/src/uts/common/io/usb/usba/genconsole.c @ 3943:302736821720

6415231 Galaxy 1 system panics w/BAD TRAP (page fault) in "ohci" due to NULL pointer
author sl147100
date Sun, 01 Apr 2007 22:45:05 -0700
parents 053dcaf72871
children e4dbffd35ebc
line wrap: on
line source

/*
 * CDDL HEADER START
 *
 * The contents of this file are subject to the terms of the
 * Common Development and Distribution License (the "License").
 * You may not use this file except in compliance with the License.
 *
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
 * or http://www.opensolaris.org/os/licensing.
 * See the License for the specific language governing permissions
 * and limitations under the License.
 *
 * When distributing Covered Code, include this CDDL HEADER in each
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
 * If applicable, add the following below this CDDL HEADER, with the
 * fields enclosed by brackets "[]" replaced with your own identifying
 * information: Portions Copyright [yyyy] [name of copyright owner]
 *
 * CDDL HEADER END
 */
/*
 * Copyright 2007 Sun Microsystems, Inc.  All rights reserved.
 * Use is subject to license terms.
 */
#pragma ident	"%Z%%M%	%I%	%E% SMI"

/*
 * USBA: Solaris USB Architecture support
 *
 * ISSUES:
 */
#define	USBA_FRAMEWORK
#include <sys/usb/usba.h>
#include <sys/usb/usba/hcdi.h>
#include <sys/usb/usba/genconsole.h>
#include <sys/usb/usba/usba_types.h>
#include <sys/usb/usba/usba_impl.h>

/*
 * Initialize USB polled support. This routine calls down to the lower
 * layers to initialize any state information.
 */
int
usb_console_input_init(dev_info_t		*dip,
			usb_pipe_handle_t	pipe_handle,
			uchar_t			**state_buf,
			usb_console_info_t	*console_input_info)
{
	int			ret;
	usba_device_t		*usba_device;
	usba_pipe_handle_data_t	*ph_data;
	usb_console_info_impl_t	*usb_console_input;

	if (dip == NULL) {

		return (USB_INVALID_ARGS);
	}

	if (DEVI_IS_DEVICE_REMOVED(dip)) {

		return (USB_FAILURE);
	}

	usb_console_input = kmem_zalloc(
	    sizeof (struct usb_console_info_impl), KM_SLEEP);

	/*
	 * Save the dip
	 */
	usb_console_input->uci_dip = dip;

	/*
	 * Translate the dip into a device.
	 */
	usba_device = usba_get_usba_device(dip);

	/*
	 * Get ph_data from pipe handle and hold the data
	 */
	if ((ph_data = usba_hold_ph_data(pipe_handle)) == NULL) {
		kmem_free(usb_console_input,
		    sizeof (struct usb_console_info_impl));

		return (USB_INVALID_PIPE);
	}

	/*
	 * Call the lower layer to initialize any state information
	 */
	ret = usba_device->usb_hcdi_ops->usba_hcdi_console_input_init(
	    ph_data, state_buf, usb_console_input);

	if (ret != USB_SUCCESS) {
		kmem_free(usb_console_input,
		    sizeof (struct usb_console_info_impl));
	} else {
		*console_input_info = (usb_console_info_t)usb_console_input;
	}

	usba_release_ph_data((usba_ph_impl_t *)pipe_handle);

	return (ret);
}


/*
 * Free up any resources that we allocated in the above initialization
 * routine.
 */
int
usb_console_input_fini(usb_console_info_t console_input_info)
{
	usb_console_info_impl_t		*usb_console_input;
	usba_device_t			*usba_device;
	int				ret;

	usb_console_input = (usb_console_info_impl_t *)console_input_info;

	/*
	 * Translate the dip into a device.
	 */
	usba_device = usba_get_usba_device(usb_console_input->uci_dip);

	/*
	 * Call the lower layer to free any state information.
	 */
	ret = usba_device->usb_hcdi_ops->usba_hcdi_console_input_fini(
		usb_console_input);

	if (ret == USB_FAILURE) {

		return (ret);
	}

	/*
	 * We won't be needing this information anymore.
	 */
	kmem_free(usb_console_input, sizeof (struct usb_console_info_impl));

	return (USB_SUCCESS);
}


/*
 * This is the routine that OBP calls to save the USB state information
 * before using the USB keyboard as an input device.  This routine,
 * and all of the routines that it calls, are responsible for saving
 * any state information so that it can be restored when OBP mode is
 * over.  At this layer, this code is mainly just a pass through.
 *
 * Warning:  this code runs in polled mode.
 */
int
usb_console_input_enter(usb_console_info_t console_input_info)
{
	usba_device_t				*usba_device;
	usb_console_info_impl_t			*usb_console_input;

	usb_console_input = (usb_console_info_impl_t *)console_input_info;

	/*
	 * Translate the dip into a device.
	 * Do this by directly looking at the dip, do not call
	 * usba_get_usba_device() because this function calls into the DDI.
	 * The ddi then tries to acquire a mutex and the machine hard hangs.
	 */
	usba_device = usba_polled_get_usba_device(usb_console_input->uci_dip);

	/*
	 * Call the lower layer to save state information.
	 */
	usba_device->usb_hcdi_ops->usba_hcdi_console_input_enter(
		usb_console_input);

	return (USB_SUCCESS);
}


/*
 * This is the routine that OBP calls when it wants to read a character.
 * We will call to the lower layers to see if there is any input data
 * available.  At this layer, this code is mainly just a pass through.
 *
 * Warning: This code runs in polled mode.
 */
int
usb_console_read(usb_console_info_t console_input_info, uint_t *num_characters)
{
	usba_device_t				*usba_device;
	usb_console_info_impl_t			*usb_console_input;

	usb_console_input = (usb_console_info_impl_t *)console_input_info;

	/*
	 * Translate the dip into a device.
	 * Do this by directly looking at the dip, do not call
	 * usba_get_usba_device() because this function calls into the DDI.
	 * The ddi then tries to acquire a mutex and the machine hard hangs.
	 */
	usba_device = usba_polled_get_usba_device(usb_console_input->uci_dip);

	/*
	 * Call the lower layer to get a a character.  Return the number
	 * of characters read into the buffer.
	 */
	return (usba_device->usb_hcdi_ops->usba_hcdi_console_read(
		usb_console_input, num_characters));
}


/*
 * This is the routine that OBP calls when it is giving up control of the
 * USB keyboard.  This routine, and the lower layer routines that it calls,
 * are responsible for restoring the controller state to the state it was
 * in before OBP took control. At this layer, this code is mainly just a
 * pass through.
 *
 * Warning: This code runs in polled mode.
 */
int
usb_console_input_exit(usb_console_info_t console_input_info)
{
	usba_device_t				*usba_device;
	usb_console_info_impl_t			*usb_console_input;

	usb_console_input = (usb_console_info_impl_t *)console_input_info;

	/*
	 * Translate the dip into a device.
	 * Do this by directly looking at the dip, do not call
	 * usba_get_usba_device() because this function calls into the DDI.
	 * The ddi then tries to acquire a mutex and the machine hard hangs.
	 */
	usba_device = usba_polled_get_usba_device(usb_console_input->uci_dip);

	/*
	 * Restore the state information.
	 */
	usba_device->usb_hcdi_ops->usba_hcdi_console_input_exit(
		usb_console_input);

	return (USB_SUCCESS);
}

/*
 * Initialize USB OBP support.  This routine calls down to the lower
 * layers to initialize any state information.
 */
int
usb_console_output_init(
	dev_info_t		*dip,
	usb_pipe_handle_t	pipe_handle,
	usb_console_info_t	*console_output_info)
{
	usba_device_t		*usb_device;
	usb_console_info_impl_t	*usb_console_output;
	int			ret;

	/* Translate the dip into a device and check hcdi ops  */
	usb_device = usba_get_usba_device(dip);
	if (usb_device->usb_hcdi_ops->usba_hcdi_ops_version <
	    HCDI_OPS_VERSION_1 ||
	    usb_device->usb_hcdi_ops->usba_hcdi_console_output_init == NULL)

		return (USB_FAILURE);

	usb_console_output = kmem_zalloc(sizeof (struct usb_console_info_impl),
		KM_SLEEP);
	usb_console_output->uci_dip = dip;

	/*
	 * Call the lower layer to initialize any state information
	 */
	ret = usb_device->usb_hcdi_ops->usba_hcdi_console_output_init(
		usba_get_ph_data(pipe_handle), usb_console_output);

	if (ret == USB_FAILURE) {
		kmem_free(usb_console_output,
			sizeof (struct usb_console_info_impl));

		return (ret);
	}

	*console_output_info = (usb_console_info_t)usb_console_output;

	return (USB_SUCCESS);
}

/*
 * Free up any resources that we allocated in the above initialization
 * routine.
 */
int
usb_console_output_fini(usb_console_info_t console_output_info)
{
	usb_console_info_impl_t	*usb_console_output;
	usba_device_t		*usb_device;
	int			ret;

	usb_console_output = (usb_console_info_impl_t *)console_output_info;

	/*
	 * Translate the dip into a device.
	 */
	usb_device = usba_polled_get_usba_device(usb_console_output->uci_dip);

	/*
	 * Call the lower layer to free any state information.
	 */
	ret = usb_device->usb_hcdi_ops->usba_hcdi_console_output_fini(
		usb_console_output);

	if (ret == USB_FAILURE) {

		return (ret);
	}

	/*
	 * We won't be needing this information anymore.
	 */
	kmem_free(usb_console_output, sizeof (struct usb_console_info_impl));

	return (USB_SUCCESS);
}

/*
 * This is the routine that OBP calls to save the USB state information
 * before using the USB device as an output device.  This routine,
 * and all of the routines that it calls, are responsible for saving
 * any state information so that it can be restored when OBP mode is
 * over.  At this layer, this code is mainly just a pass through.
 */
int
usb_console_output_enter(usb_console_info_t console_output_info)
{
	usba_device_t			    *usb_device;
	usb_console_info_impl_t		 *usb_console_output;

	usb_console_output = (usb_console_info_impl_t *)console_output_info;

	/*
	 * Translate the dip into a device.
	 */
	usb_device = usba_polled_get_usba_device(usb_console_output->uci_dip);

	/*
	 * Call the lower layer to save state information.
	 */
	usb_device->usb_hcdi_ops->usba_hcdi_console_output_enter(
		usb_console_output);

	return (USB_SUCCESS);
}

/*
 * This is the routine that OBP calls when it wants to write a character.
 * We will call to the lower layers to write any data
 * At this layer, this code is mainly just a pass through.
 */
int
usb_console_write(usb_console_info_t console_output_info,
	uchar_t *buf, uint_t num_characters, uint_t *num_characters_written)
{
	usba_device_t		*usb_device;
	usb_console_info_impl_t	*usb_console_output;

	usb_console_output = (usb_console_info_impl_t *)console_output_info;

	/*
	 * Translate the dip into a device.
	 */
	usb_device = usba_polled_get_usba_device(usb_console_output->uci_dip);

	/*
	 * Call the lower layer to get a a character.  Return the number
	 * of characters read into the buffer.
	 */
	return (usb_device->usb_hcdi_ops->usba_hcdi_console_write(
		usb_console_output, buf, num_characters,
		num_characters_written));
}

/*
 * This is the routine that OBP calls when it is giving up control of the
 * USB output device.  This routine, and the lower layer routines that it
 * calls, are responsible for restoring the controller state to the state
 * it was in before OBP took control. At this layer, this code is mainly
 * just a pass through.
 */
int
usb_console_output_exit(usb_console_info_t console_output_info)
{
	usba_device_t			 *usb_device;
	usb_console_info_impl_t		 *usb_console_output;

	usb_console_output = (usb_console_info_impl_t *)console_output_info;

	/*
	 * Translate the dip into a device.
	 */
	usb_device = usba_polled_get_usba_device(usb_console_output->uci_dip);

	/*
	 * Restore the state information.
	 */
	usb_device->usb_hcdi_ops->usba_hcdi_console_output_exit(
		usb_console_output);

	return (USB_SUCCESS);
}