2012-07-31 17:18:17 -04:00
/*
* Copyright 2012 Advanced Micro Devices , Inc .
*
* Permission is hereby granted , free of charge , to any person obtaining a
* copy of this software and associated documentation files ( the " Software " ) ,
* to deal in the Software without restriction , including without limitation
* the rights to use , copy , modify , merge , publish , distribute , sublicense ,
* and / or sell copies of the Software , and to permit persons to whom the
* Software is furnished to do so , subject to the following conditions :
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software .
*
* THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND , EXPRESS OR
* IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY ,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT . IN NO EVENT SHALL
* THE COPYRIGHT HOLDER ( S ) OR AUTHOR ( S ) BE LIABLE FOR ANY CLAIM , DAMAGES OR
* OTHER LIABILITY , WHETHER IN AN ACTION OF CONTRACT , TORT OR OTHERWISE ,
* ARISING FROM , OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
* OTHER DEALINGS IN THE SOFTWARE .
*
*/
2010-07-06 11:40:24 -04:00
# include <linux/pci.h>
# include <linux/acpi.h>
# include <linux/slab.h>
# include <acpi/acpi_drivers.h>
# include <acpi/acpi_bus.h>
# include "drmP.h"
# include "drm.h"
# include "drm_sarea.h"
# include "drm_crtc_helper.h"
# include "radeon.h"
2012-07-18 13:28:52 -04:00
# include "radeon_acpi.h"
2010-07-06 11:40:24 -04:00
# include <linux/vga_switcheroo.h>
/* Call the ATIF method
*/
2012-07-29 17:04:43 +02:00
static union acpi_object * radeon_atif_call ( acpi_handle handle , int function ,
struct acpi_buffer * params )
2010-07-06 11:40:24 -04:00
{
acpi_status status ;
union acpi_object atif_arg_elements [ 2 ] ;
struct acpi_object_list atif_arg ;
2012-07-29 17:04:43 +02:00
struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER , NULL } ;
2010-07-06 11:40:24 -04:00
atif_arg . count = 2 ;
atif_arg . pointer = & atif_arg_elements [ 0 ] ;
atif_arg_elements [ 0 ] . type = ACPI_TYPE_INTEGER ;
2012-07-29 17:04:43 +02:00
atif_arg_elements [ 0 ] . integer . value = function ;
if ( params ) {
atif_arg_elements [ 1 ] . type = ACPI_TYPE_BUFFER ;
atif_arg_elements [ 1 ] . buffer . length = params - > length ;
atif_arg_elements [ 1 ] . buffer . pointer = params - > pointer ;
} else {
/* We need a second fake parameter */
atif_arg_elements [ 1 ] . type = ACPI_TYPE_INTEGER ;
atif_arg_elements [ 1 ] . integer . value = 0 ;
}
2010-07-06 11:40:24 -04:00
status = acpi_evaluate_object ( handle , " ATIF " , & atif_arg , & buffer ) ;
/* Fail only if calling the method fails and ATIF is supported */
if ( ACPI_FAILURE ( status ) & & status ! = AE_NOT_FOUND ) {
2011-11-30 17:26:36 +01:00
DRM_DEBUG_DRIVER ( " failed to evaluate ATIF got %s \n " ,
acpi_format_exception ( status ) ) ;
2010-07-06 11:40:24 -04:00
kfree ( buffer . pointer ) ;
2012-07-29 17:04:43 +02:00
return NULL ;
2010-07-06 11:40:24 -04:00
}
2012-07-29 17:04:43 +02:00
return buffer . pointer ;
2010-07-06 11:40:24 -04:00
}
/* Call all ACPI methods here */
int radeon_acpi_init ( struct radeon_device * rdev )
{
acpi_handle handle ;
2012-07-29 17:04:43 +02:00
union acpi_object * info ;
int ret = 0 ;
2010-07-06 11:40:24 -04:00
/* Get the device handle */
handle = DEVICE_ACPI_HANDLE ( & rdev - > pdev - > dev ) ;
2011-11-30 17:36:39 +01:00
/* No need to proceed if we're sure that ATIF is not supported */
if ( ! ASIC_IS_AVIVO ( rdev ) | | ! rdev - > bios | | ! handle )
return 0 ;
2010-07-06 11:40:24 -04:00
/* Call the ATIF method */
2012-07-29 17:04:43 +02:00
info = radeon_atif_call ( handle , ATIF_FUNCTION_VERIFY_INTERFACE , NULL ) ;
if ( ! info )
ret = - EIO ;
2010-07-06 11:40:24 -04:00
2012-07-29 17:04:43 +02:00
kfree ( info ) ;
return ret ;
2010-07-06 11:40:24 -04:00
}