|  | /* hvapi.c: Hypervisor API management. | 
|  | * | 
|  | * Copyright (C) 2007 David S. Miller <davem@davemloft.net> | 
|  | */ | 
|  | #include <linux/kernel.h> | 
|  | #include <linux/export.h> | 
|  | #include <linux/init.h> | 
|  |  | 
|  | #include <asm/hypervisor.h> | 
|  | #include <asm/oplib.h> | 
|  |  | 
|  | /* If the hypervisor indicates that the API setting | 
|  | * calls are unsupported, by returning HV_EBADTRAP or | 
|  | * HV_ENOTSUPPORTED, we assume that API groups with the | 
|  | * PRE_API flag set are major 1 minor 0. | 
|  | */ | 
|  | struct api_info { | 
|  | unsigned long group; | 
|  | unsigned long major; | 
|  | unsigned long minor; | 
|  | unsigned int refcnt; | 
|  | unsigned int flags; | 
|  | #define FLAG_PRE_API		0x00000001 | 
|  | }; | 
|  |  | 
|  | static struct api_info api_table[] = { | 
|  | { .group = HV_GRP_SUN4V,	.flags = FLAG_PRE_API	}, | 
|  | { .group = HV_GRP_CORE,		.flags = FLAG_PRE_API	}, | 
|  | { .group = HV_GRP_INTR,					}, | 
|  | { .group = HV_GRP_SOFT_STATE,				}, | 
|  | { .group = HV_GRP_TM,					}, | 
|  | { .group = HV_GRP_PCI,		.flags = FLAG_PRE_API	}, | 
|  | { .group = HV_GRP_LDOM,					}, | 
|  | { .group = HV_GRP_SVC_CHAN,	.flags = FLAG_PRE_API	}, | 
|  | { .group = HV_GRP_NCS,		.flags = FLAG_PRE_API	}, | 
|  | { .group = HV_GRP_RNG,					}, | 
|  | { .group = HV_GRP_PBOOT,				}, | 
|  | { .group = HV_GRP_TPM,					}, | 
|  | { .group = HV_GRP_SDIO,					}, | 
|  | { .group = HV_GRP_SDIO_ERR,				}, | 
|  | { .group = HV_GRP_REBOOT_DATA,				}, | 
|  | { .group = HV_GRP_ATU,		.flags = FLAG_PRE_API	}, | 
|  | { .group = HV_GRP_NIAG_PERF,	.flags = FLAG_PRE_API	}, | 
|  | { .group = HV_GRP_FIRE_PERF,				}, | 
|  | { .group = HV_GRP_N2_CPU,				}, | 
|  | { .group = HV_GRP_NIU,					}, | 
|  | { .group = HV_GRP_VF_CPU,				}, | 
|  | { .group = HV_GRP_KT_CPU,				}, | 
|  | { .group = HV_GRP_VT_CPU,				}, | 
|  | { .group = HV_GRP_T5_CPU,				}, | 
|  | { .group = HV_GRP_DIAG,		.flags = FLAG_PRE_API	}, | 
|  | { .group = HV_GRP_M7_PERF,				}, | 
|  | }; | 
|  |  | 
|  | static DEFINE_SPINLOCK(hvapi_lock); | 
|  |  | 
|  | static struct api_info *__get_info(unsigned long group) | 
|  | { | 
|  | int i; | 
|  |  | 
|  | for (i = 0; i < ARRAY_SIZE(api_table); i++) { | 
|  | if (api_table[i].group == group) | 
|  | return &api_table[i]; | 
|  | } | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | static void __get_ref(struct api_info *p) | 
|  | { | 
|  | p->refcnt++; | 
|  | } | 
|  |  | 
|  | static void __put_ref(struct api_info *p) | 
|  | { | 
|  | if (--p->refcnt == 0) { | 
|  | unsigned long ignore; | 
|  |  | 
|  | sun4v_set_version(p->group, 0, 0, &ignore); | 
|  | p->major = p->minor = 0; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* Register a hypervisor API specification.  It indicates the | 
|  | * API group and desired major+minor. | 
|  | * | 
|  | * If an existing API registration exists '0' (success) will | 
|  | * be returned if it is compatible with the one being registered. | 
|  | * Otherwise a negative error code will be returned. | 
|  | * | 
|  | * Otherwise an attempt will be made to negotiate the requested | 
|  | * API group/major/minor with the hypervisor, and errors returned | 
|  | * if that does not succeed. | 
|  | */ | 
|  | int sun4v_hvapi_register(unsigned long group, unsigned long major, | 
|  | unsigned long *minor) | 
|  | { | 
|  | struct api_info *p; | 
|  | unsigned long flags; | 
|  | int ret; | 
|  |  | 
|  | spin_lock_irqsave(&hvapi_lock, flags); | 
|  | p = __get_info(group); | 
|  | ret = -EINVAL; | 
|  | if (p) { | 
|  | if (p->refcnt) { | 
|  | ret = -EINVAL; | 
|  | if (p->major == major) { | 
|  | *minor = p->minor; | 
|  | ret = 0; | 
|  | } | 
|  | } else { | 
|  | unsigned long actual_minor; | 
|  | unsigned long hv_ret; | 
|  |  | 
|  | hv_ret = sun4v_set_version(group, major, *minor, | 
|  | &actual_minor); | 
|  | ret = -EINVAL; | 
|  | if (hv_ret == HV_EOK) { | 
|  | *minor = actual_minor; | 
|  | p->major = major; | 
|  | p->minor = actual_minor; | 
|  | ret = 0; | 
|  | } else if (hv_ret == HV_EBADTRAP || | 
|  | hv_ret == HV_ENOTSUPPORTED) { | 
|  | if (p->flags & FLAG_PRE_API) { | 
|  | if (major == 1) { | 
|  | p->major = 1; | 
|  | p->minor = 0; | 
|  | *minor = 0; | 
|  | ret = 0; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | if (ret == 0) | 
|  | __get_ref(p); | 
|  | } | 
|  | spin_unlock_irqrestore(&hvapi_lock, flags); | 
|  |  | 
|  | return ret; | 
|  | } | 
|  | EXPORT_SYMBOL(sun4v_hvapi_register); | 
|  |  | 
|  | void sun4v_hvapi_unregister(unsigned long group) | 
|  | { | 
|  | struct api_info *p; | 
|  | unsigned long flags; | 
|  |  | 
|  | spin_lock_irqsave(&hvapi_lock, flags); | 
|  | p = __get_info(group); | 
|  | if (p) | 
|  | __put_ref(p); | 
|  | spin_unlock_irqrestore(&hvapi_lock, flags); | 
|  | } | 
|  | EXPORT_SYMBOL(sun4v_hvapi_unregister); | 
|  |  | 
|  | int sun4v_hvapi_get(unsigned long group, | 
|  | unsigned long *major, | 
|  | unsigned long *minor) | 
|  | { | 
|  | struct api_info *p; | 
|  | unsigned long flags; | 
|  | int ret; | 
|  |  | 
|  | spin_lock_irqsave(&hvapi_lock, flags); | 
|  | ret = -EINVAL; | 
|  | p = __get_info(group); | 
|  | if (p && p->refcnt) { | 
|  | *major = p->major; | 
|  | *minor = p->minor; | 
|  | ret = 0; | 
|  | } | 
|  | spin_unlock_irqrestore(&hvapi_lock, flags); | 
|  |  | 
|  | return ret; | 
|  | } | 
|  | EXPORT_SYMBOL(sun4v_hvapi_get); | 
|  |  | 
|  | void __init sun4v_hvapi_init(void) | 
|  | { | 
|  | unsigned long group, major, minor; | 
|  |  | 
|  | group = HV_GRP_SUN4V; | 
|  | major = 1; | 
|  | minor = 0; | 
|  | if (sun4v_hvapi_register(group, major, &minor)) | 
|  | goto bad; | 
|  |  | 
|  | group = HV_GRP_CORE; | 
|  | major = 1; | 
|  | minor = 1; | 
|  | if (sun4v_hvapi_register(group, major, &minor)) | 
|  | goto bad; | 
|  |  | 
|  | return; | 
|  |  | 
|  | bad: | 
|  | prom_printf("HVAPI: Cannot register API group " | 
|  | "%lx with major(%lu) minor(%lu)\n", | 
|  | group, major, minor); | 
|  | prom_halt(); | 
|  | } |