14fc2cf1fSJustin Ernst // SPDX-License-Identifier: GPL-2.0-or-later
24fc2cf1fSJustin Ernst /*
34fc2cf1fSJustin Ernst * This file supports the /sys/firmware/sgi_uv topology tree on HPE UV.
44fc2cf1fSJustin Ernst *
54fc2cf1fSJustin Ernst * Copyright (c) 2020 Hewlett Packard Enterprise. All Rights Reserved.
64fc2cf1fSJustin Ernst * Copyright (c) Justin Ernst
74fc2cf1fSJustin Ernst */
84fc2cf1fSJustin Ernst
94fc2cf1fSJustin Ernst #include <linux/module.h>
104fc2cf1fSJustin Ernst #include <linux/kernel.h>
114fc2cf1fSJustin Ernst #include <linux/device.h>
124fc2cf1fSJustin Ernst #include <linux/slab.h>
134fc2cf1fSJustin Ernst #include <linux/kobject.h>
144fc2cf1fSJustin Ernst #include <asm/uv/bios.h>
154fc2cf1fSJustin Ernst #include <asm/uv/uv.h>
164fc2cf1fSJustin Ernst #include <asm/uv/uv_hub.h>
174fc2cf1fSJustin Ernst #include <asm/uv/uv_geo.h>
184fc2cf1fSJustin Ernst
194fc2cf1fSJustin Ernst #define INVALID_CNODE -1
204fc2cf1fSJustin Ernst
214fc2cf1fSJustin Ernst struct kobject *sgi_uv_kobj;
220c683e9dSZou Wei static struct kset *uv_pcibus_kset;
230c683e9dSZou Wei static struct kset *uv_hubs_kset;
244fc2cf1fSJustin Ernst static struct uv_bios_hub_info *hub_buf;
254fc2cf1fSJustin Ernst static struct uv_bios_port_info **port_buf;
264fc2cf1fSJustin Ernst static struct uv_hub **uv_hubs;
274fc2cf1fSJustin Ernst static struct uv_pci_top_obj **uv_pci_objs;
284fc2cf1fSJustin Ernst static int num_pci_lines;
294fc2cf1fSJustin Ernst static int num_cnodes;
304fc2cf1fSJustin Ernst static int *prev_obj_to_cnode;
314fc2cf1fSJustin Ernst static int uv_bios_obj_cnt;
324fc2cf1fSJustin Ernst static signed short uv_master_nasid = -1;
334fc2cf1fSJustin Ernst static void *uv_biosheap;
344fc2cf1fSJustin Ernst
uv_type_string(void)354fc2cf1fSJustin Ernst static const char *uv_type_string(void)
364fc2cf1fSJustin Ernst {
374fc2cf1fSJustin Ernst if (is_uv5_hub())
384fc2cf1fSJustin Ernst return "9.0";
394fc2cf1fSJustin Ernst else if (is_uv4a_hub())
404fc2cf1fSJustin Ernst return "7.1";
414fc2cf1fSJustin Ernst else if (is_uv4_hub())
424fc2cf1fSJustin Ernst return "7.0";
434fc2cf1fSJustin Ernst else if (is_uv3_hub())
444fc2cf1fSJustin Ernst return "5.0";
454fc2cf1fSJustin Ernst else if (is_uv2_hub())
464fc2cf1fSJustin Ernst return "3.0";
47433e817aSMike Travis else if (uv_get_hubless_system())
48433e817aSMike Travis return "0.1";
494fc2cf1fSJustin Ernst else
504fc2cf1fSJustin Ernst return "unknown";
514fc2cf1fSJustin Ernst }
524fc2cf1fSJustin Ernst
ordinal_to_nasid(int ordinal)534fc2cf1fSJustin Ernst static int ordinal_to_nasid(int ordinal)
544fc2cf1fSJustin Ernst {
554fc2cf1fSJustin Ernst if (ordinal < num_cnodes && ordinal >= 0)
564fc2cf1fSJustin Ernst return UV_PNODE_TO_NASID(uv_blade_to_pnode(ordinal));
574fc2cf1fSJustin Ernst else
584fc2cf1fSJustin Ernst return -1;
594fc2cf1fSJustin Ernst }
604fc2cf1fSJustin Ernst
cnode_to_geoid(int cnode)614fc2cf1fSJustin Ernst static union geoid_u cnode_to_geoid(int cnode)
624fc2cf1fSJustin Ernst {
634fc2cf1fSJustin Ernst union geoid_u geoid;
644fc2cf1fSJustin Ernst
654fc2cf1fSJustin Ernst uv_bios_get_geoinfo(ordinal_to_nasid(cnode), (u64)sizeof(union geoid_u), (u64 *)&geoid);
664fc2cf1fSJustin Ernst return geoid;
674fc2cf1fSJustin Ernst }
684fc2cf1fSJustin Ernst
location_to_bpos(char * location,int * rack,int * slot,int * blade)694fc2cf1fSJustin Ernst static int location_to_bpos(char *location, int *rack, int *slot, int *blade)
704fc2cf1fSJustin Ernst {
714fc2cf1fSJustin Ernst char type, r, b, h;
724fc2cf1fSJustin Ernst int idb, idh;
734fc2cf1fSJustin Ernst
744fc2cf1fSJustin Ernst if (sscanf(location, "%c%03d%c%02d%c%2d%c%d",
754fc2cf1fSJustin Ernst &r, rack, &type, slot, &b, &idb, &h, &idh) != 8)
764fc2cf1fSJustin Ernst return -1;
774fc2cf1fSJustin Ernst *blade = idb * 2 + idh;
784fc2cf1fSJustin Ernst
794fc2cf1fSJustin Ernst return 0;
804fc2cf1fSJustin Ernst }
814fc2cf1fSJustin Ernst
cache_obj_to_cnode(struct uv_bios_hub_info * obj)824fc2cf1fSJustin Ernst static int cache_obj_to_cnode(struct uv_bios_hub_info *obj)
834fc2cf1fSJustin Ernst {
844fc2cf1fSJustin Ernst int cnode;
854fc2cf1fSJustin Ernst union geoid_u geoid;
864fc2cf1fSJustin Ernst int obj_rack, obj_slot, obj_blade;
874fc2cf1fSJustin Ernst int rack, slot, blade;
884fc2cf1fSJustin Ernst
894fc2cf1fSJustin Ernst if (!obj->f.fields.this_part && !obj->f.fields.is_shared)
904fc2cf1fSJustin Ernst return 0;
914fc2cf1fSJustin Ernst
924fc2cf1fSJustin Ernst if (location_to_bpos(obj->location, &obj_rack, &obj_slot, &obj_blade))
934fc2cf1fSJustin Ernst return -1;
944fc2cf1fSJustin Ernst
954fc2cf1fSJustin Ernst for (cnode = 0; cnode < num_cnodes; cnode++) {
964fc2cf1fSJustin Ernst geoid = cnode_to_geoid(cnode);
974fc2cf1fSJustin Ernst rack = geo_rack(geoid);
984fc2cf1fSJustin Ernst slot = geo_slot(geoid);
994fc2cf1fSJustin Ernst blade = geo_blade(geoid);
1004fc2cf1fSJustin Ernst if (obj_rack == rack && obj_slot == slot && obj_blade == blade)
1014fc2cf1fSJustin Ernst prev_obj_to_cnode[obj->id] = cnode;
1024fc2cf1fSJustin Ernst }
1034fc2cf1fSJustin Ernst
1044fc2cf1fSJustin Ernst return 0;
1054fc2cf1fSJustin Ernst }
1064fc2cf1fSJustin Ernst
get_obj_to_cnode(int obj_id)1074fc2cf1fSJustin Ernst static int get_obj_to_cnode(int obj_id)
1084fc2cf1fSJustin Ernst {
1094fc2cf1fSJustin Ernst return prev_obj_to_cnode[obj_id];
1104fc2cf1fSJustin Ernst }
1114fc2cf1fSJustin Ernst
1124fc2cf1fSJustin Ernst struct uv_hub {
1134fc2cf1fSJustin Ernst struct kobject kobj;
1144fc2cf1fSJustin Ernst struct uv_bios_hub_info *hub_info;
1154fc2cf1fSJustin Ernst struct uv_port **ports;
1164fc2cf1fSJustin Ernst };
1174fc2cf1fSJustin Ernst
1184fc2cf1fSJustin Ernst #define to_uv_hub(kobj_ptr) container_of(kobj_ptr, struct uv_hub, kobj)
1194fc2cf1fSJustin Ernst
hub_name_show(struct uv_bios_hub_info * hub_info,char * buf)1204fc2cf1fSJustin Ernst static ssize_t hub_name_show(struct uv_bios_hub_info *hub_info, char *buf)
1214fc2cf1fSJustin Ernst {
122d240849cSye xingchen return sysfs_emit(buf, "%s\n", hub_info->name);
1234fc2cf1fSJustin Ernst }
1244fc2cf1fSJustin Ernst
hub_location_show(struct uv_bios_hub_info * hub_info,char * buf)1254fc2cf1fSJustin Ernst static ssize_t hub_location_show(struct uv_bios_hub_info *hub_info, char *buf)
1264fc2cf1fSJustin Ernst {
127d240849cSye xingchen return sysfs_emit(buf, "%s\n", hub_info->location);
1284fc2cf1fSJustin Ernst }
1294fc2cf1fSJustin Ernst
hub_partition_show(struct uv_bios_hub_info * hub_info,char * buf)1304fc2cf1fSJustin Ernst static ssize_t hub_partition_show(struct uv_bios_hub_info *hub_info, char *buf)
1314fc2cf1fSJustin Ernst {
1324fc2cf1fSJustin Ernst return sprintf(buf, "%d\n", hub_info->f.fields.this_part);
1334fc2cf1fSJustin Ernst }
1344fc2cf1fSJustin Ernst
hub_shared_show(struct uv_bios_hub_info * hub_info,char * buf)1354fc2cf1fSJustin Ernst static ssize_t hub_shared_show(struct uv_bios_hub_info *hub_info, char *buf)
1364fc2cf1fSJustin Ernst {
1374fc2cf1fSJustin Ernst return sprintf(buf, "%d\n", hub_info->f.fields.is_shared);
1384fc2cf1fSJustin Ernst }
hub_nasid_show(struct uv_bios_hub_info * hub_info,char * buf)1394fc2cf1fSJustin Ernst static ssize_t hub_nasid_show(struct uv_bios_hub_info *hub_info, char *buf)
1404fc2cf1fSJustin Ernst {
1414fc2cf1fSJustin Ernst int cnode = get_obj_to_cnode(hub_info->id);
1424fc2cf1fSJustin Ernst
1434fc2cf1fSJustin Ernst return sprintf(buf, "%d\n", ordinal_to_nasid(cnode));
1444fc2cf1fSJustin Ernst }
hub_cnode_show(struct uv_bios_hub_info * hub_info,char * buf)1454fc2cf1fSJustin Ernst static ssize_t hub_cnode_show(struct uv_bios_hub_info *hub_info, char *buf)
1464fc2cf1fSJustin Ernst {
1474fc2cf1fSJustin Ernst return sprintf(buf, "%d\n", get_obj_to_cnode(hub_info->id));
1484fc2cf1fSJustin Ernst }
1494fc2cf1fSJustin Ernst
1504fc2cf1fSJustin Ernst struct hub_sysfs_entry {
1514fc2cf1fSJustin Ernst struct attribute attr;
1524fc2cf1fSJustin Ernst ssize_t (*show)(struct uv_bios_hub_info *hub_info, char *buf);
1534fc2cf1fSJustin Ernst ssize_t (*store)(struct uv_bios_hub_info *hub_info, const char *buf, size_t sz);
1544fc2cf1fSJustin Ernst };
1554fc2cf1fSJustin Ernst
1564fc2cf1fSJustin Ernst static struct hub_sysfs_entry name_attribute =
1574fc2cf1fSJustin Ernst __ATTR(name, 0444, hub_name_show, NULL);
1584fc2cf1fSJustin Ernst static struct hub_sysfs_entry location_attribute =
1594fc2cf1fSJustin Ernst __ATTR(location, 0444, hub_location_show, NULL);
1604fc2cf1fSJustin Ernst static struct hub_sysfs_entry partition_attribute =
1614fc2cf1fSJustin Ernst __ATTR(this_partition, 0444, hub_partition_show, NULL);
1624fc2cf1fSJustin Ernst static struct hub_sysfs_entry shared_attribute =
1634fc2cf1fSJustin Ernst __ATTR(shared, 0444, hub_shared_show, NULL);
1644fc2cf1fSJustin Ernst static struct hub_sysfs_entry nasid_attribute =
1654fc2cf1fSJustin Ernst __ATTR(nasid, 0444, hub_nasid_show, NULL);
1664fc2cf1fSJustin Ernst static struct hub_sysfs_entry cnode_attribute =
1674fc2cf1fSJustin Ernst __ATTR(cnode, 0444, hub_cnode_show, NULL);
1684fc2cf1fSJustin Ernst
1694fc2cf1fSJustin Ernst static struct attribute *uv_hub_attrs[] = {
1704fc2cf1fSJustin Ernst &name_attribute.attr,
1714fc2cf1fSJustin Ernst &location_attribute.attr,
1724fc2cf1fSJustin Ernst &partition_attribute.attr,
1734fc2cf1fSJustin Ernst &shared_attribute.attr,
1744fc2cf1fSJustin Ernst &nasid_attribute.attr,
1754fc2cf1fSJustin Ernst &cnode_attribute.attr,
1764fc2cf1fSJustin Ernst NULL,
1774fc2cf1fSJustin Ernst };
178afca4cbeSGreg Kroah-Hartman ATTRIBUTE_GROUPS(uv_hub);
1794fc2cf1fSJustin Ernst
hub_release(struct kobject * kobj)1804fc2cf1fSJustin Ernst static void hub_release(struct kobject *kobj)
1814fc2cf1fSJustin Ernst {
1824fc2cf1fSJustin Ernst struct uv_hub *hub = to_uv_hub(kobj);
1834fc2cf1fSJustin Ernst
1844fc2cf1fSJustin Ernst kfree(hub);
1854fc2cf1fSJustin Ernst }
1864fc2cf1fSJustin Ernst
hub_type_show(struct kobject * kobj,struct attribute * attr,char * buf)1874fc2cf1fSJustin Ernst static ssize_t hub_type_show(struct kobject *kobj, struct attribute *attr,
1884fc2cf1fSJustin Ernst char *buf)
1894fc2cf1fSJustin Ernst {
1904fc2cf1fSJustin Ernst struct uv_hub *hub = to_uv_hub(kobj);
1914fc2cf1fSJustin Ernst struct uv_bios_hub_info *bios_hub_info = hub->hub_info;
1924fc2cf1fSJustin Ernst struct hub_sysfs_entry *entry;
1934fc2cf1fSJustin Ernst
1944fc2cf1fSJustin Ernst entry = container_of(attr, struct hub_sysfs_entry, attr);
1954fc2cf1fSJustin Ernst
1964fc2cf1fSJustin Ernst if (!entry->show)
1974fc2cf1fSJustin Ernst return -EIO;
1984fc2cf1fSJustin Ernst
1994fc2cf1fSJustin Ernst return entry->show(bios_hub_info, buf);
2004fc2cf1fSJustin Ernst }
2014fc2cf1fSJustin Ernst
2024fc2cf1fSJustin Ernst static const struct sysfs_ops hub_sysfs_ops = {
2034fc2cf1fSJustin Ernst .show = hub_type_show,
2044fc2cf1fSJustin Ernst };
2054fc2cf1fSJustin Ernst
206*ad76d9b8SThomas Weißschuh static const struct kobj_type hub_attr_type = {
2074fc2cf1fSJustin Ernst .release = hub_release,
2084fc2cf1fSJustin Ernst .sysfs_ops = &hub_sysfs_ops,
209afca4cbeSGreg Kroah-Hartman .default_groups = uv_hub_groups,
2104fc2cf1fSJustin Ernst };
2114fc2cf1fSJustin Ernst
uv_hubs_init(void)2124fc2cf1fSJustin Ernst static int uv_hubs_init(void)
2134fc2cf1fSJustin Ernst {
2144fc2cf1fSJustin Ernst s64 biosr;
2154fc2cf1fSJustin Ernst u64 sz;
2164fc2cf1fSJustin Ernst int i, ret;
2174fc2cf1fSJustin Ernst
2184fc2cf1fSJustin Ernst prev_obj_to_cnode = kmalloc_array(uv_bios_obj_cnt, sizeof(*prev_obj_to_cnode),
2194fc2cf1fSJustin Ernst GFP_KERNEL);
2204fc2cf1fSJustin Ernst if (!prev_obj_to_cnode)
2214fc2cf1fSJustin Ernst return -ENOMEM;
2224fc2cf1fSJustin Ernst
2234fc2cf1fSJustin Ernst for (i = 0; i < uv_bios_obj_cnt; i++)
2244fc2cf1fSJustin Ernst prev_obj_to_cnode[i] = INVALID_CNODE;
2254fc2cf1fSJustin Ernst
2264fc2cf1fSJustin Ernst uv_hubs_kset = kset_create_and_add("hubs", NULL, sgi_uv_kobj);
2274fc2cf1fSJustin Ernst if (!uv_hubs_kset) {
2284fc2cf1fSJustin Ernst ret = -ENOMEM;
2294fc2cf1fSJustin Ernst goto err_hubs_kset;
2304fc2cf1fSJustin Ernst }
2314fc2cf1fSJustin Ernst sz = uv_bios_obj_cnt * sizeof(*hub_buf);
2324fc2cf1fSJustin Ernst hub_buf = kzalloc(sz, GFP_KERNEL);
2334fc2cf1fSJustin Ernst if (!hub_buf) {
2344fc2cf1fSJustin Ernst ret = -ENOMEM;
2354fc2cf1fSJustin Ernst goto err_hub_buf;
2364fc2cf1fSJustin Ernst }
2374fc2cf1fSJustin Ernst
2384fc2cf1fSJustin Ernst biosr = uv_bios_enum_objs((u64)uv_master_nasid, sz, (u64 *)hub_buf);
2394fc2cf1fSJustin Ernst if (biosr) {
2404fc2cf1fSJustin Ernst ret = -EINVAL;
2414fc2cf1fSJustin Ernst goto err_enum_objs;
2424fc2cf1fSJustin Ernst }
2434fc2cf1fSJustin Ernst
2444fc2cf1fSJustin Ernst uv_hubs = kcalloc(uv_bios_obj_cnt, sizeof(*uv_hubs), GFP_KERNEL);
2454fc2cf1fSJustin Ernst if (!uv_hubs) {
2464fc2cf1fSJustin Ernst ret = -ENOMEM;
2474fc2cf1fSJustin Ernst goto err_enum_objs;
2484fc2cf1fSJustin Ernst }
2494fc2cf1fSJustin Ernst
2504fc2cf1fSJustin Ernst for (i = 0; i < uv_bios_obj_cnt; i++) {
2514fc2cf1fSJustin Ernst uv_hubs[i] = kzalloc(sizeof(*uv_hubs[i]), GFP_KERNEL);
2524fc2cf1fSJustin Ernst if (!uv_hubs[i]) {
2534fc2cf1fSJustin Ernst i--;
25418d047bdSDan Carpenter ret = -ENOMEM;
2554fc2cf1fSJustin Ernst goto err_hubs;
2564fc2cf1fSJustin Ernst }
2574fc2cf1fSJustin Ernst
2584fc2cf1fSJustin Ernst uv_hubs[i]->hub_info = &hub_buf[i];
2594fc2cf1fSJustin Ernst cache_obj_to_cnode(uv_hubs[i]->hub_info);
2604fc2cf1fSJustin Ernst
2614fc2cf1fSJustin Ernst uv_hubs[i]->kobj.kset = uv_hubs_kset;
2624fc2cf1fSJustin Ernst
2634fc2cf1fSJustin Ernst ret = kobject_init_and_add(&uv_hubs[i]->kobj, &hub_attr_type,
2644fc2cf1fSJustin Ernst NULL, "hub_%u", hub_buf[i].id);
2654fc2cf1fSJustin Ernst if (ret)
2664fc2cf1fSJustin Ernst goto err_hubs;
2674fc2cf1fSJustin Ernst kobject_uevent(&uv_hubs[i]->kobj, KOBJ_ADD);
2684fc2cf1fSJustin Ernst }
2694fc2cf1fSJustin Ernst return 0;
2704fc2cf1fSJustin Ernst
2714fc2cf1fSJustin Ernst err_hubs:
2724fc2cf1fSJustin Ernst for (; i >= 0; i--)
2734fc2cf1fSJustin Ernst kobject_put(&uv_hubs[i]->kobj);
2744fc2cf1fSJustin Ernst kfree(uv_hubs);
2754fc2cf1fSJustin Ernst err_enum_objs:
2764fc2cf1fSJustin Ernst kfree(hub_buf);
2774fc2cf1fSJustin Ernst err_hub_buf:
2784fc2cf1fSJustin Ernst kset_unregister(uv_hubs_kset);
2794fc2cf1fSJustin Ernst err_hubs_kset:
2804fc2cf1fSJustin Ernst kfree(prev_obj_to_cnode);
2814fc2cf1fSJustin Ernst return ret;
2824fc2cf1fSJustin Ernst
2834fc2cf1fSJustin Ernst }
2844fc2cf1fSJustin Ernst
uv_hubs_exit(void)2854fc2cf1fSJustin Ernst static void uv_hubs_exit(void)
2864fc2cf1fSJustin Ernst {
2874fc2cf1fSJustin Ernst int i;
2884fc2cf1fSJustin Ernst
2894fc2cf1fSJustin Ernst for (i = 0; i < uv_bios_obj_cnt; i++)
2904fc2cf1fSJustin Ernst kobject_put(&uv_hubs[i]->kobj);
2914fc2cf1fSJustin Ernst
2924fc2cf1fSJustin Ernst kfree(uv_hubs);
2934fc2cf1fSJustin Ernst kfree(hub_buf);
2944fc2cf1fSJustin Ernst kset_unregister(uv_hubs_kset);
2954fc2cf1fSJustin Ernst kfree(prev_obj_to_cnode);
2964fc2cf1fSJustin Ernst }
2974fc2cf1fSJustin Ernst
2984fc2cf1fSJustin Ernst struct uv_port {
2994fc2cf1fSJustin Ernst struct kobject kobj;
3004fc2cf1fSJustin Ernst struct uv_bios_port_info *port_info;
3014fc2cf1fSJustin Ernst };
3024fc2cf1fSJustin Ernst
3034fc2cf1fSJustin Ernst #define to_uv_port(kobj_ptr) container_of(kobj_ptr, struct uv_port, kobj)
3044fc2cf1fSJustin Ernst
uv_port_conn_hub_show(struct uv_bios_port_info * port,char * buf)3054fc2cf1fSJustin Ernst static ssize_t uv_port_conn_hub_show(struct uv_bios_port_info *port, char *buf)
3064fc2cf1fSJustin Ernst {
3074fc2cf1fSJustin Ernst return sprintf(buf, "%d\n", port->conn_id);
3084fc2cf1fSJustin Ernst }
3094fc2cf1fSJustin Ernst
uv_port_conn_port_show(struct uv_bios_port_info * port,char * buf)3104fc2cf1fSJustin Ernst static ssize_t uv_port_conn_port_show(struct uv_bios_port_info *port, char *buf)
3114fc2cf1fSJustin Ernst {
3124fc2cf1fSJustin Ernst return sprintf(buf, "%d\n", port->conn_port);
3134fc2cf1fSJustin Ernst }
3144fc2cf1fSJustin Ernst
3154fc2cf1fSJustin Ernst struct uv_port_sysfs_entry {
3164fc2cf1fSJustin Ernst struct attribute attr;
3174fc2cf1fSJustin Ernst ssize_t (*show)(struct uv_bios_port_info *port_info, char *buf);
3184fc2cf1fSJustin Ernst ssize_t (*store)(struct uv_bios_port_info *port_info, const char *buf, size_t size);
3194fc2cf1fSJustin Ernst };
3204fc2cf1fSJustin Ernst
3214fc2cf1fSJustin Ernst static struct uv_port_sysfs_entry uv_port_conn_hub_attribute =
3224fc2cf1fSJustin Ernst __ATTR(conn_hub, 0444, uv_port_conn_hub_show, NULL);
3234fc2cf1fSJustin Ernst static struct uv_port_sysfs_entry uv_port_conn_port_attribute =
3244fc2cf1fSJustin Ernst __ATTR(conn_port, 0444, uv_port_conn_port_show, NULL);
3254fc2cf1fSJustin Ernst
3264fc2cf1fSJustin Ernst static struct attribute *uv_port_attrs[] = {
3274fc2cf1fSJustin Ernst &uv_port_conn_hub_attribute.attr,
3284fc2cf1fSJustin Ernst &uv_port_conn_port_attribute.attr,
3294fc2cf1fSJustin Ernst NULL,
3304fc2cf1fSJustin Ernst };
331afca4cbeSGreg Kroah-Hartman ATTRIBUTE_GROUPS(uv_port);
3324fc2cf1fSJustin Ernst
uv_port_release(struct kobject * kobj)3334fc2cf1fSJustin Ernst static void uv_port_release(struct kobject *kobj)
3344fc2cf1fSJustin Ernst {
3354fc2cf1fSJustin Ernst struct uv_port *port = to_uv_port(kobj);
3364fc2cf1fSJustin Ernst
3374fc2cf1fSJustin Ernst kfree(port);
3384fc2cf1fSJustin Ernst }
3394fc2cf1fSJustin Ernst
uv_port_type_show(struct kobject * kobj,struct attribute * attr,char * buf)3404fc2cf1fSJustin Ernst static ssize_t uv_port_type_show(struct kobject *kobj, struct attribute *attr,
3414fc2cf1fSJustin Ernst char *buf)
3424fc2cf1fSJustin Ernst {
3434fc2cf1fSJustin Ernst struct uv_port *port = to_uv_port(kobj);
3444fc2cf1fSJustin Ernst struct uv_bios_port_info *port_info = port->port_info;
3454fc2cf1fSJustin Ernst struct uv_port_sysfs_entry *entry;
3464fc2cf1fSJustin Ernst
3474fc2cf1fSJustin Ernst entry = container_of(attr, struct uv_port_sysfs_entry, attr);
3484fc2cf1fSJustin Ernst
3494fc2cf1fSJustin Ernst if (!entry->show)
3504fc2cf1fSJustin Ernst return -EIO;
3514fc2cf1fSJustin Ernst
3524fc2cf1fSJustin Ernst return entry->show(port_info, buf);
3534fc2cf1fSJustin Ernst }
3544fc2cf1fSJustin Ernst
3554fc2cf1fSJustin Ernst static const struct sysfs_ops uv_port_sysfs_ops = {
3564fc2cf1fSJustin Ernst .show = uv_port_type_show,
3574fc2cf1fSJustin Ernst };
3584fc2cf1fSJustin Ernst
359*ad76d9b8SThomas Weißschuh static const struct kobj_type uv_port_attr_type = {
3604fc2cf1fSJustin Ernst .release = uv_port_release,
3614fc2cf1fSJustin Ernst .sysfs_ops = &uv_port_sysfs_ops,
362afca4cbeSGreg Kroah-Hartman .default_groups = uv_port_groups,
3634fc2cf1fSJustin Ernst };
3644fc2cf1fSJustin Ernst
uv_ports_init(void)3654fc2cf1fSJustin Ernst static int uv_ports_init(void)
3664fc2cf1fSJustin Ernst {
3674fc2cf1fSJustin Ernst s64 biosr;
3684fc2cf1fSJustin Ernst int j = 0, k = 0, ret, sz;
3694fc2cf1fSJustin Ernst
3704fc2cf1fSJustin Ernst port_buf = kcalloc(uv_bios_obj_cnt, sizeof(*port_buf), GFP_KERNEL);
3714fc2cf1fSJustin Ernst if (!port_buf)
3724fc2cf1fSJustin Ernst return -ENOMEM;
3734fc2cf1fSJustin Ernst
3744fc2cf1fSJustin Ernst for (j = 0; j < uv_bios_obj_cnt; j++) {
3754fc2cf1fSJustin Ernst sz = hub_buf[j].ports * sizeof(*port_buf[j]);
3764fc2cf1fSJustin Ernst port_buf[j] = kzalloc(sz, GFP_KERNEL);
3774fc2cf1fSJustin Ernst if (!port_buf[j]) {
3784fc2cf1fSJustin Ernst ret = -ENOMEM;
3794fc2cf1fSJustin Ernst j--;
3804fc2cf1fSJustin Ernst goto err_port_info;
3814fc2cf1fSJustin Ernst }
3824fc2cf1fSJustin Ernst biosr = uv_bios_enum_ports((u64)uv_master_nasid, (u64)hub_buf[j].id, sz,
3834fc2cf1fSJustin Ernst (u64 *)port_buf[j]);
3844fc2cf1fSJustin Ernst if (biosr) {
3854fc2cf1fSJustin Ernst ret = -EINVAL;
3864fc2cf1fSJustin Ernst goto err_port_info;
3874fc2cf1fSJustin Ernst }
3884fc2cf1fSJustin Ernst }
3894fc2cf1fSJustin Ernst for (j = 0; j < uv_bios_obj_cnt; j++) {
3904fc2cf1fSJustin Ernst uv_hubs[j]->ports = kcalloc(hub_buf[j].ports,
3914fc2cf1fSJustin Ernst sizeof(*uv_hubs[j]->ports), GFP_KERNEL);
3924fc2cf1fSJustin Ernst if (!uv_hubs[j]->ports) {
3934fc2cf1fSJustin Ernst ret = -ENOMEM;
3944fc2cf1fSJustin Ernst j--;
3954fc2cf1fSJustin Ernst goto err_ports;
3964fc2cf1fSJustin Ernst }
3974fc2cf1fSJustin Ernst }
3984fc2cf1fSJustin Ernst for (j = 0; j < uv_bios_obj_cnt; j++) {
3994fc2cf1fSJustin Ernst for (k = 0; k < hub_buf[j].ports; k++) {
4004fc2cf1fSJustin Ernst uv_hubs[j]->ports[k] = kzalloc(sizeof(*uv_hubs[j]->ports[k]), GFP_KERNEL);
4014fc2cf1fSJustin Ernst if (!uv_hubs[j]->ports[k]) {
4024fc2cf1fSJustin Ernst ret = -ENOMEM;
4034fc2cf1fSJustin Ernst k--;
4044fc2cf1fSJustin Ernst goto err_kobj_ports;
4054fc2cf1fSJustin Ernst }
4064fc2cf1fSJustin Ernst uv_hubs[j]->ports[k]->port_info = &port_buf[j][k];
4074fc2cf1fSJustin Ernst ret = kobject_init_and_add(&uv_hubs[j]->ports[k]->kobj, &uv_port_attr_type,
4084fc2cf1fSJustin Ernst &uv_hubs[j]->kobj, "port_%d", port_buf[j][k].port);
4094fc2cf1fSJustin Ernst if (ret)
4104fc2cf1fSJustin Ernst goto err_kobj_ports;
4114fc2cf1fSJustin Ernst kobject_uevent(&uv_hubs[j]->ports[k]->kobj, KOBJ_ADD);
4124fc2cf1fSJustin Ernst }
4134fc2cf1fSJustin Ernst }
4144fc2cf1fSJustin Ernst return 0;
4154fc2cf1fSJustin Ernst
4164fc2cf1fSJustin Ernst err_kobj_ports:
4174fc2cf1fSJustin Ernst for (; j >= 0; j--) {
4184fc2cf1fSJustin Ernst for (; k >= 0; k--)
4194fc2cf1fSJustin Ernst kobject_put(&uv_hubs[j]->ports[k]->kobj);
4204fc2cf1fSJustin Ernst if (j > 0)
4214fc2cf1fSJustin Ernst k = hub_buf[j-1].ports - 1;
4224fc2cf1fSJustin Ernst }
4234fc2cf1fSJustin Ernst j = uv_bios_obj_cnt - 1;
4244fc2cf1fSJustin Ernst err_ports:
4254fc2cf1fSJustin Ernst for (; j >= 0; j--)
4264fc2cf1fSJustin Ernst kfree(uv_hubs[j]->ports);
4274fc2cf1fSJustin Ernst j = uv_bios_obj_cnt - 1;
4284fc2cf1fSJustin Ernst err_port_info:
4294fc2cf1fSJustin Ernst for (; j >= 0; j--)
4304fc2cf1fSJustin Ernst kfree(port_buf[j]);
4314fc2cf1fSJustin Ernst kfree(port_buf);
4324fc2cf1fSJustin Ernst return ret;
4334fc2cf1fSJustin Ernst }
4344fc2cf1fSJustin Ernst
uv_ports_exit(void)4354fc2cf1fSJustin Ernst static void uv_ports_exit(void)
4364fc2cf1fSJustin Ernst {
4374fc2cf1fSJustin Ernst int j, k;
4384fc2cf1fSJustin Ernst
4394fc2cf1fSJustin Ernst for (j = 0; j < uv_bios_obj_cnt; j++) {
4404fc2cf1fSJustin Ernst for (k = hub_buf[j].ports - 1; k >= 0; k--)
4414fc2cf1fSJustin Ernst kobject_put(&uv_hubs[j]->ports[k]->kobj);
4424fc2cf1fSJustin Ernst }
4434fc2cf1fSJustin Ernst for (j = 0; j < uv_bios_obj_cnt; j++) {
4444fc2cf1fSJustin Ernst kfree(uv_hubs[j]->ports);
4454fc2cf1fSJustin Ernst kfree(port_buf[j]);
4464fc2cf1fSJustin Ernst }
4474fc2cf1fSJustin Ernst kfree(port_buf);
4484fc2cf1fSJustin Ernst }
4494fc2cf1fSJustin Ernst
4504fc2cf1fSJustin Ernst struct uv_pci_top_obj {
4514fc2cf1fSJustin Ernst struct kobject kobj;
4524fc2cf1fSJustin Ernst char *type;
4534fc2cf1fSJustin Ernst char *location;
4544fc2cf1fSJustin Ernst int iio_stack;
4554fc2cf1fSJustin Ernst char *ppb_addr;
4564fc2cf1fSJustin Ernst int slot;
4574fc2cf1fSJustin Ernst };
4584fc2cf1fSJustin Ernst
4594fc2cf1fSJustin Ernst #define to_uv_pci_top_obj(kobj_ptr) container_of(kobj_ptr, struct uv_pci_top_obj, kobj)
4604fc2cf1fSJustin Ernst
uv_pci_type_show(struct uv_pci_top_obj * top_obj,char * buf)4614fc2cf1fSJustin Ernst static ssize_t uv_pci_type_show(struct uv_pci_top_obj *top_obj, char *buf)
4624fc2cf1fSJustin Ernst {
463d240849cSye xingchen return sysfs_emit(buf, "%s\n", top_obj->type);
4644fc2cf1fSJustin Ernst }
4654fc2cf1fSJustin Ernst
uv_pci_location_show(struct uv_pci_top_obj * top_obj,char * buf)4664fc2cf1fSJustin Ernst static ssize_t uv_pci_location_show(struct uv_pci_top_obj *top_obj, char *buf)
4674fc2cf1fSJustin Ernst {
468d240849cSye xingchen return sysfs_emit(buf, "%s\n", top_obj->location);
4694fc2cf1fSJustin Ernst }
4704fc2cf1fSJustin Ernst
uv_pci_iio_stack_show(struct uv_pci_top_obj * top_obj,char * buf)4714fc2cf1fSJustin Ernst static ssize_t uv_pci_iio_stack_show(struct uv_pci_top_obj *top_obj, char *buf)
4724fc2cf1fSJustin Ernst {
4734fc2cf1fSJustin Ernst return sprintf(buf, "%d\n", top_obj->iio_stack);
4744fc2cf1fSJustin Ernst }
4754fc2cf1fSJustin Ernst
uv_pci_ppb_addr_show(struct uv_pci_top_obj * top_obj,char * buf)4764fc2cf1fSJustin Ernst static ssize_t uv_pci_ppb_addr_show(struct uv_pci_top_obj *top_obj, char *buf)
4774fc2cf1fSJustin Ernst {
478d240849cSye xingchen return sysfs_emit(buf, "%s\n", top_obj->ppb_addr);
4794fc2cf1fSJustin Ernst }
4804fc2cf1fSJustin Ernst
uv_pci_slot_show(struct uv_pci_top_obj * top_obj,char * buf)4814fc2cf1fSJustin Ernst static ssize_t uv_pci_slot_show(struct uv_pci_top_obj *top_obj, char *buf)
4824fc2cf1fSJustin Ernst {
4834fc2cf1fSJustin Ernst return sprintf(buf, "%d\n", top_obj->slot);
4844fc2cf1fSJustin Ernst }
4854fc2cf1fSJustin Ernst
4864fc2cf1fSJustin Ernst struct uv_pci_top_sysfs_entry {
4874fc2cf1fSJustin Ernst struct attribute attr;
4884fc2cf1fSJustin Ernst ssize_t (*show)(struct uv_pci_top_obj *top_obj, char *buf);
4894fc2cf1fSJustin Ernst ssize_t (*store)(struct uv_pci_top_obj *top_obj, const char *buf, size_t size);
4904fc2cf1fSJustin Ernst };
4914fc2cf1fSJustin Ernst
4924fc2cf1fSJustin Ernst static struct uv_pci_top_sysfs_entry uv_pci_type_attribute =
4934fc2cf1fSJustin Ernst __ATTR(type, 0444, uv_pci_type_show, NULL);
4944fc2cf1fSJustin Ernst static struct uv_pci_top_sysfs_entry uv_pci_location_attribute =
4954fc2cf1fSJustin Ernst __ATTR(location, 0444, uv_pci_location_show, NULL);
4964fc2cf1fSJustin Ernst static struct uv_pci_top_sysfs_entry uv_pci_iio_stack_attribute =
4974fc2cf1fSJustin Ernst __ATTR(iio_stack, 0444, uv_pci_iio_stack_show, NULL);
4984fc2cf1fSJustin Ernst static struct uv_pci_top_sysfs_entry uv_pci_ppb_addr_attribute =
4994fc2cf1fSJustin Ernst __ATTR(ppb_addr, 0444, uv_pci_ppb_addr_show, NULL);
5004fc2cf1fSJustin Ernst static struct uv_pci_top_sysfs_entry uv_pci_slot_attribute =
5014fc2cf1fSJustin Ernst __ATTR(slot, 0444, uv_pci_slot_show, NULL);
5024fc2cf1fSJustin Ernst
uv_pci_top_release(struct kobject * kobj)5034fc2cf1fSJustin Ernst static void uv_pci_top_release(struct kobject *kobj)
5044fc2cf1fSJustin Ernst {
5054fc2cf1fSJustin Ernst struct uv_pci_top_obj *top_obj = to_uv_pci_top_obj(kobj);
5064fc2cf1fSJustin Ernst
5074fc2cf1fSJustin Ernst kfree(top_obj->type);
5084fc2cf1fSJustin Ernst kfree(top_obj->location);
5094fc2cf1fSJustin Ernst kfree(top_obj->ppb_addr);
5104fc2cf1fSJustin Ernst kfree(top_obj);
5114fc2cf1fSJustin Ernst }
5124fc2cf1fSJustin Ernst
pci_top_type_show(struct kobject * kobj,struct attribute * attr,char * buf)5134fc2cf1fSJustin Ernst static ssize_t pci_top_type_show(struct kobject *kobj,
5144fc2cf1fSJustin Ernst struct attribute *attr, char *buf)
5154fc2cf1fSJustin Ernst {
5164fc2cf1fSJustin Ernst struct uv_pci_top_obj *top_obj = to_uv_pci_top_obj(kobj);
5174fc2cf1fSJustin Ernst struct uv_pci_top_sysfs_entry *entry;
5184fc2cf1fSJustin Ernst
5194fc2cf1fSJustin Ernst entry = container_of(attr, struct uv_pci_top_sysfs_entry, attr);
5204fc2cf1fSJustin Ernst
5214fc2cf1fSJustin Ernst if (!entry->show)
5224fc2cf1fSJustin Ernst return -EIO;
5234fc2cf1fSJustin Ernst
5244fc2cf1fSJustin Ernst return entry->show(top_obj, buf);
5254fc2cf1fSJustin Ernst }
5264fc2cf1fSJustin Ernst
5274fc2cf1fSJustin Ernst static const struct sysfs_ops uv_pci_top_sysfs_ops = {
5284fc2cf1fSJustin Ernst .show = pci_top_type_show,
5294fc2cf1fSJustin Ernst };
5304fc2cf1fSJustin Ernst
531*ad76d9b8SThomas Weißschuh static const struct kobj_type uv_pci_top_attr_type = {
5324fc2cf1fSJustin Ernst .release = uv_pci_top_release,
5334fc2cf1fSJustin Ernst .sysfs_ops = &uv_pci_top_sysfs_ops,
5344fc2cf1fSJustin Ernst };
5354fc2cf1fSJustin Ernst
init_pci_top_obj(struct uv_pci_top_obj * top_obj,char * line)5364fc2cf1fSJustin Ernst static int init_pci_top_obj(struct uv_pci_top_obj *top_obj, char *line)
5374fc2cf1fSJustin Ernst {
5384fc2cf1fSJustin Ernst char *start;
5394fc2cf1fSJustin Ernst char type[11], location[14], ppb_addr[15];
5404fc2cf1fSJustin Ernst int str_cnt, ret;
5414fc2cf1fSJustin Ernst unsigned int tmp_match[2];
5424fc2cf1fSJustin Ernst
5434fc2cf1fSJustin Ernst // Minimum line length
5444fc2cf1fSJustin Ernst if (strlen(line) < 36)
5454fc2cf1fSJustin Ernst return -EINVAL;
5464fc2cf1fSJustin Ernst
5474fc2cf1fSJustin Ernst //Line must match format "pcibus %4x:%2x" to be valid
5484fc2cf1fSJustin Ernst str_cnt = sscanf(line, "pcibus %4x:%2x", &tmp_match[0], &tmp_match[1]);
5494fc2cf1fSJustin Ernst if (str_cnt < 2)
5504fc2cf1fSJustin Ernst return -EINVAL;
5514fc2cf1fSJustin Ernst
5524fc2cf1fSJustin Ernst /* Connect pcibus to segment:bus number with '_'
5534fc2cf1fSJustin Ernst * to concatenate name tokens.
5544fc2cf1fSJustin Ernst * pcibus 0000:00 ... -> pcibus_0000:00 ...
5554fc2cf1fSJustin Ernst */
5564fc2cf1fSJustin Ernst line[6] = '_';
5574fc2cf1fSJustin Ernst
5584fc2cf1fSJustin Ernst /* Null terminate after the concatencated name tokens
5594fc2cf1fSJustin Ernst * to produce kobj name string.
5604fc2cf1fSJustin Ernst */
5614fc2cf1fSJustin Ernst line[14] = '\0';
5624fc2cf1fSJustin Ernst
5634fc2cf1fSJustin Ernst // Use start to index after name tokens string for remainder of line info.
5644fc2cf1fSJustin Ernst start = &line[15];
5654fc2cf1fSJustin Ernst
5664fc2cf1fSJustin Ernst top_obj->iio_stack = -1;
5674fc2cf1fSJustin Ernst top_obj->slot = -1;
5684fc2cf1fSJustin Ernst
5694fc2cf1fSJustin Ernst /* r001i01b00h0 BASE IO (IIO Stack 0)
5704fc2cf1fSJustin Ernst * r001i01b00h1 PCIe IO (IIO Stack 1)
5714fc2cf1fSJustin Ernst * r001i01b03h1 PCIe SLOT
5724fc2cf1fSJustin Ernst * r001i01b00h0 NODE IO
5734fc2cf1fSJustin Ernst * r001i01b00h0 Riser
5744fc2cf1fSJustin Ernst * (IIO Stack #) may not be present.
5754fc2cf1fSJustin Ernst */
5764fc2cf1fSJustin Ernst if (start[0] == 'r') {
5774fc2cf1fSJustin Ernst str_cnt = sscanf(start, "%13s %10[^(] %*s %*s %d)",
5784fc2cf1fSJustin Ernst location, type, &top_obj->iio_stack);
5794fc2cf1fSJustin Ernst if (str_cnt < 2)
5804fc2cf1fSJustin Ernst return -EINVAL;
5814fc2cf1fSJustin Ernst top_obj->type = kstrdup(type, GFP_KERNEL);
5824fc2cf1fSJustin Ernst if (!top_obj->type)
5834fc2cf1fSJustin Ernst return -ENOMEM;
5844fc2cf1fSJustin Ernst top_obj->location = kstrdup(location, GFP_KERNEL);
5854fc2cf1fSJustin Ernst if (!top_obj->location) {
5864fc2cf1fSJustin Ernst kfree(top_obj->type);
5874fc2cf1fSJustin Ernst return -ENOMEM;
5884fc2cf1fSJustin Ernst }
5894fc2cf1fSJustin Ernst }
5904fc2cf1fSJustin Ernst /* PPB at 0000:80:00.00 (slot 3)
5914fc2cf1fSJustin Ernst * (slot #) may not be present.
5924fc2cf1fSJustin Ernst */
5934fc2cf1fSJustin Ernst else if (start[0] == 'P') {
5944fc2cf1fSJustin Ernst str_cnt = sscanf(start, "%10s %*s %14s %*s %d)",
5954fc2cf1fSJustin Ernst type, ppb_addr, &top_obj->slot);
5964fc2cf1fSJustin Ernst if (str_cnt < 2)
5974fc2cf1fSJustin Ernst return -EINVAL;
5984fc2cf1fSJustin Ernst top_obj->type = kstrdup(type, GFP_KERNEL);
5994fc2cf1fSJustin Ernst if (!top_obj->type)
6004fc2cf1fSJustin Ernst return -ENOMEM;
6014fc2cf1fSJustin Ernst top_obj->ppb_addr = kstrdup(ppb_addr, GFP_KERNEL);
6024fc2cf1fSJustin Ernst if (!top_obj->ppb_addr) {
6034fc2cf1fSJustin Ernst kfree(top_obj->type);
6044fc2cf1fSJustin Ernst return -ENOMEM;
6054fc2cf1fSJustin Ernst }
6064fc2cf1fSJustin Ernst } else
6074fc2cf1fSJustin Ernst return -EINVAL;
6084fc2cf1fSJustin Ernst
6094fc2cf1fSJustin Ernst top_obj->kobj.kset = uv_pcibus_kset;
6104fc2cf1fSJustin Ernst
6114fc2cf1fSJustin Ernst ret = kobject_init_and_add(&top_obj->kobj, &uv_pci_top_attr_type, NULL, "%s", line);
6124fc2cf1fSJustin Ernst if (ret)
6134fc2cf1fSJustin Ernst goto err_add_sysfs;
6144fc2cf1fSJustin Ernst
6154fc2cf1fSJustin Ernst if (top_obj->type) {
6164fc2cf1fSJustin Ernst ret = sysfs_create_file(&top_obj->kobj, &uv_pci_type_attribute.attr);
6174fc2cf1fSJustin Ernst if (ret)
6184fc2cf1fSJustin Ernst goto err_add_sysfs;
6194fc2cf1fSJustin Ernst }
6204fc2cf1fSJustin Ernst if (top_obj->location) {
6214fc2cf1fSJustin Ernst ret = sysfs_create_file(&top_obj->kobj, &uv_pci_location_attribute.attr);
6224fc2cf1fSJustin Ernst if (ret)
6234fc2cf1fSJustin Ernst goto err_add_sysfs;
6244fc2cf1fSJustin Ernst }
6254fc2cf1fSJustin Ernst if (top_obj->iio_stack >= 0) {
6264fc2cf1fSJustin Ernst ret = sysfs_create_file(&top_obj->kobj, &uv_pci_iio_stack_attribute.attr);
6274fc2cf1fSJustin Ernst if (ret)
6284fc2cf1fSJustin Ernst goto err_add_sysfs;
6294fc2cf1fSJustin Ernst }
6304fc2cf1fSJustin Ernst if (top_obj->ppb_addr) {
6314fc2cf1fSJustin Ernst ret = sysfs_create_file(&top_obj->kobj, &uv_pci_ppb_addr_attribute.attr);
6324fc2cf1fSJustin Ernst if (ret)
6334fc2cf1fSJustin Ernst goto err_add_sysfs;
6344fc2cf1fSJustin Ernst }
6354fc2cf1fSJustin Ernst if (top_obj->slot >= 0) {
6364fc2cf1fSJustin Ernst ret = sysfs_create_file(&top_obj->kobj, &uv_pci_slot_attribute.attr);
6374fc2cf1fSJustin Ernst if (ret)
6384fc2cf1fSJustin Ernst goto err_add_sysfs;
6394fc2cf1fSJustin Ernst }
6404fc2cf1fSJustin Ernst
6414fc2cf1fSJustin Ernst kobject_uevent(&top_obj->kobj, KOBJ_ADD);
6424fc2cf1fSJustin Ernst return 0;
6434fc2cf1fSJustin Ernst
6444fc2cf1fSJustin Ernst err_add_sysfs:
6454fc2cf1fSJustin Ernst kobject_put(&top_obj->kobj);
6464fc2cf1fSJustin Ernst return ret;
6474fc2cf1fSJustin Ernst }
6484fc2cf1fSJustin Ernst
pci_topology_init(void)6494fc2cf1fSJustin Ernst static int pci_topology_init(void)
6504fc2cf1fSJustin Ernst {
6514fc2cf1fSJustin Ernst char *pci_top_str, *start, *found, *count;
6524fc2cf1fSJustin Ernst size_t sz;
6534fc2cf1fSJustin Ernst s64 biosr;
6544fc2cf1fSJustin Ernst int l = 0, k = 0;
6554fc2cf1fSJustin Ernst int len, ret;
6564fc2cf1fSJustin Ernst
6574fc2cf1fSJustin Ernst uv_pcibus_kset = kset_create_and_add("pcibuses", NULL, sgi_uv_kobj);
6584fc2cf1fSJustin Ernst if (!uv_pcibus_kset)
6594fc2cf1fSJustin Ernst return -ENOMEM;
6604fc2cf1fSJustin Ernst
6614fc2cf1fSJustin Ernst for (sz = PAGE_SIZE; sz < 16 * PAGE_SIZE; sz += PAGE_SIZE) {
6624fc2cf1fSJustin Ernst pci_top_str = kmalloc(sz, GFP_KERNEL);
6634fc2cf1fSJustin Ernst if (!pci_top_str) {
6644fc2cf1fSJustin Ernst ret = -ENOMEM;
6654fc2cf1fSJustin Ernst goto err_pci_top_str;
6664fc2cf1fSJustin Ernst }
6674fc2cf1fSJustin Ernst biosr = uv_bios_get_pci_topology((u64)sz, (u64 *)pci_top_str);
6684fc2cf1fSJustin Ernst if (biosr == BIOS_STATUS_SUCCESS) {
6694fc2cf1fSJustin Ernst len = strnlen(pci_top_str, sz);
6704fc2cf1fSJustin Ernst for (count = pci_top_str; count < pci_top_str + len; count++) {
6714fc2cf1fSJustin Ernst if (*count == '\n')
6724fc2cf1fSJustin Ernst l++;
6734fc2cf1fSJustin Ernst }
6744fc2cf1fSJustin Ernst num_pci_lines = l;
6754fc2cf1fSJustin Ernst
6764fc2cf1fSJustin Ernst uv_pci_objs = kcalloc(num_pci_lines,
6774fc2cf1fSJustin Ernst sizeof(*uv_pci_objs), GFP_KERNEL);
6784fc2cf1fSJustin Ernst if (!uv_pci_objs) {
6794fc2cf1fSJustin Ernst kfree(pci_top_str);
6804fc2cf1fSJustin Ernst ret = -ENOMEM;
6814fc2cf1fSJustin Ernst goto err_pci_top_str;
6824fc2cf1fSJustin Ernst }
6834fc2cf1fSJustin Ernst start = pci_top_str;
6844fc2cf1fSJustin Ernst while ((found = strsep(&start, "\n")) != NULL) {
6854fc2cf1fSJustin Ernst uv_pci_objs[k] = kzalloc(sizeof(*uv_pci_objs[k]), GFP_KERNEL);
6864fc2cf1fSJustin Ernst if (!uv_pci_objs[k]) {
6874fc2cf1fSJustin Ernst ret = -ENOMEM;
6884fc2cf1fSJustin Ernst goto err_pci_obj;
6894fc2cf1fSJustin Ernst }
6904fc2cf1fSJustin Ernst ret = init_pci_top_obj(uv_pci_objs[k], found);
6914fc2cf1fSJustin Ernst if (ret)
6924fc2cf1fSJustin Ernst goto err_pci_obj;
6934fc2cf1fSJustin Ernst k++;
6944fc2cf1fSJustin Ernst if (k == num_pci_lines)
6954fc2cf1fSJustin Ernst break;
6964fc2cf1fSJustin Ernst }
6974fc2cf1fSJustin Ernst }
6984fc2cf1fSJustin Ernst kfree(pci_top_str);
6994fc2cf1fSJustin Ernst if (biosr == BIOS_STATUS_SUCCESS || biosr == BIOS_STATUS_UNIMPLEMENTED)
7004fc2cf1fSJustin Ernst break;
7014fc2cf1fSJustin Ernst }
7024fc2cf1fSJustin Ernst
7034fc2cf1fSJustin Ernst return 0;
7044fc2cf1fSJustin Ernst err_pci_obj:
7054fc2cf1fSJustin Ernst k--;
7064fc2cf1fSJustin Ernst for (; k >= 0; k--)
7074fc2cf1fSJustin Ernst kobject_put(&uv_pci_objs[k]->kobj);
7084fc2cf1fSJustin Ernst kfree(uv_pci_objs);
7094fc2cf1fSJustin Ernst kfree(pci_top_str);
7104fc2cf1fSJustin Ernst err_pci_top_str:
7114fc2cf1fSJustin Ernst kset_unregister(uv_pcibus_kset);
7124fc2cf1fSJustin Ernst return ret;
7134fc2cf1fSJustin Ernst }
7144fc2cf1fSJustin Ernst
pci_topology_exit(void)7154fc2cf1fSJustin Ernst static void pci_topology_exit(void)
7164fc2cf1fSJustin Ernst {
7174fc2cf1fSJustin Ernst int k;
7184fc2cf1fSJustin Ernst
7194fc2cf1fSJustin Ernst for (k = 0; k < num_pci_lines; k++)
7204fc2cf1fSJustin Ernst kobject_put(&uv_pci_objs[k]->kobj);
7214fc2cf1fSJustin Ernst kset_unregister(uv_pcibus_kset);
7224fc2cf1fSJustin Ernst kfree(uv_pci_objs);
7234fc2cf1fSJustin Ernst }
7244fc2cf1fSJustin Ernst
partition_id_show(struct kobject * kobj,struct kobj_attribute * attr,char * buf)7254fc2cf1fSJustin Ernst static ssize_t partition_id_show(struct kobject *kobj,
7264fc2cf1fSJustin Ernst struct kobj_attribute *attr, char *buf)
7274fc2cf1fSJustin Ernst {
7284fc2cf1fSJustin Ernst return sprintf(buf, "%ld\n", sn_partition_id);
7294fc2cf1fSJustin Ernst }
7304fc2cf1fSJustin Ernst
coherence_id_show(struct kobject * kobj,struct kobj_attribute * attr,char * buf)7314fc2cf1fSJustin Ernst static ssize_t coherence_id_show(struct kobject *kobj,
7324fc2cf1fSJustin Ernst struct kobj_attribute *attr, char *buf)
7334fc2cf1fSJustin Ernst {
7344fc2cf1fSJustin Ernst return sprintf(buf, "%ld\n", sn_coherency_id);
7354fc2cf1fSJustin Ernst }
7364fc2cf1fSJustin Ernst
uv_type_show(struct kobject * kobj,struct kobj_attribute * attr,char * buf)7374fc2cf1fSJustin Ernst static ssize_t uv_type_show(struct kobject *kobj,
7384fc2cf1fSJustin Ernst struct kobj_attribute *attr, char *buf)
7394fc2cf1fSJustin Ernst {
740d240849cSye xingchen return sysfs_emit(buf, "%s\n", uv_type_string());
7414fc2cf1fSJustin Ernst }
7424fc2cf1fSJustin Ernst
uv_archtype_show(struct kobject * kobj,struct kobj_attribute * attr,char * buf)743612a0063SMike Travis static ssize_t uv_archtype_show(struct kobject *kobj,
744612a0063SMike Travis struct kobj_attribute *attr, char *buf)
745612a0063SMike Travis {
746612a0063SMike Travis return uv_get_archtype(buf, PAGE_SIZE);
747612a0063SMike Travis }
748612a0063SMike Travis
uv_hub_type_show(struct kobject * kobj,struct kobj_attribute * attr,char * buf)749612a0063SMike Travis static ssize_t uv_hub_type_show(struct kobject *kobj,
750612a0063SMike Travis struct kobj_attribute *attr, char *buf)
751612a0063SMike Travis {
752d240849cSye xingchen return sysfs_emit(buf, "0x%x\n", uv_hub_type());
753612a0063SMike Travis }
754612a0063SMike Travis
uv_hubless_show(struct kobject * kobj,struct kobj_attribute * attr,char * buf)755433e817aSMike Travis static ssize_t uv_hubless_show(struct kobject *kobj,
756433e817aSMike Travis struct kobj_attribute *attr, char *buf)
757433e817aSMike Travis {
758d240849cSye xingchen return sysfs_emit(buf, "0x%x\n", uv_get_hubless_system());
759433e817aSMike Travis }
760433e817aSMike Travis
7614fc2cf1fSJustin Ernst static struct kobj_attribute partition_id_attr =
7624fc2cf1fSJustin Ernst __ATTR(partition_id, 0444, partition_id_show, NULL);
7634fc2cf1fSJustin Ernst static struct kobj_attribute coherence_id_attr =
7644fc2cf1fSJustin Ernst __ATTR(coherence_id, 0444, coherence_id_show, NULL);
7654fc2cf1fSJustin Ernst static struct kobj_attribute uv_type_attr =
7664fc2cf1fSJustin Ernst __ATTR(uv_type, 0444, uv_type_show, NULL);
767612a0063SMike Travis static struct kobj_attribute uv_archtype_attr =
768612a0063SMike Travis __ATTR(archtype, 0444, uv_archtype_show, NULL);
769612a0063SMike Travis static struct kobj_attribute uv_hub_type_attr =
770612a0063SMike Travis __ATTR(hub_type, 0444, uv_hub_type_show, NULL);
771433e817aSMike Travis static struct kobj_attribute uv_hubless_attr =
772433e817aSMike Travis __ATTR(hubless, 0444, uv_hubless_show, NULL);
7734fc2cf1fSJustin Ernst
7744fc2cf1fSJustin Ernst static struct attribute *base_attrs[] = {
7754fc2cf1fSJustin Ernst &partition_id_attr.attr,
7764fc2cf1fSJustin Ernst &coherence_id_attr.attr,
7774fc2cf1fSJustin Ernst &uv_type_attr.attr,
778612a0063SMike Travis &uv_archtype_attr.attr,
779612a0063SMike Travis &uv_hub_type_attr.attr,
7804fc2cf1fSJustin Ernst NULL,
7814fc2cf1fSJustin Ernst };
7824fc2cf1fSJustin Ernst
78377d06ec6SRikard Falkeborn static const struct attribute_group base_attr_group = {
7844fc2cf1fSJustin Ernst .attrs = base_attrs
7854fc2cf1fSJustin Ernst };
7864fc2cf1fSJustin Ernst
initial_bios_setup(void)7874fc2cf1fSJustin Ernst static int initial_bios_setup(void)
7884fc2cf1fSJustin Ernst {
7894fc2cf1fSJustin Ernst u64 v;
7904fc2cf1fSJustin Ernst s64 biosr;
7914fc2cf1fSJustin Ernst
7924fc2cf1fSJustin Ernst biosr = uv_bios_get_master_nasid((u64)sizeof(uv_master_nasid), (u64 *)&uv_master_nasid);
7934fc2cf1fSJustin Ernst if (biosr)
7944fc2cf1fSJustin Ernst return -EINVAL;
7954fc2cf1fSJustin Ernst
7964fc2cf1fSJustin Ernst biosr = uv_bios_get_heapsize((u64)uv_master_nasid, (u64)sizeof(u64), &v);
7974fc2cf1fSJustin Ernst if (biosr)
7984fc2cf1fSJustin Ernst return -EINVAL;
7994fc2cf1fSJustin Ernst
8004fc2cf1fSJustin Ernst uv_biosheap = vmalloc(v);
8014fc2cf1fSJustin Ernst if (!uv_biosheap)
8024fc2cf1fSJustin Ernst return -ENOMEM;
8034fc2cf1fSJustin Ernst
8044fc2cf1fSJustin Ernst biosr = uv_bios_install_heap((u64)uv_master_nasid, v, (u64 *)uv_biosheap);
8054fc2cf1fSJustin Ernst if (biosr) {
8064fc2cf1fSJustin Ernst vfree(uv_biosheap);
8074fc2cf1fSJustin Ernst return -EINVAL;
8084fc2cf1fSJustin Ernst }
8094fc2cf1fSJustin Ernst
8104fc2cf1fSJustin Ernst biosr = uv_bios_obj_count((u64)uv_master_nasid, sizeof(u64), &v);
8114fc2cf1fSJustin Ernst if (biosr) {
8124fc2cf1fSJustin Ernst vfree(uv_biosheap);
8134fc2cf1fSJustin Ernst return -EINVAL;
8144fc2cf1fSJustin Ernst }
8154fc2cf1fSJustin Ernst uv_bios_obj_cnt = (int)v;
8164fc2cf1fSJustin Ernst
8174fc2cf1fSJustin Ernst return 0;
8184fc2cf1fSJustin Ernst }
8194fc2cf1fSJustin Ernst
820433e817aSMike Travis static struct attribute *hubless_base_attrs[] = {
821433e817aSMike Travis &partition_id_attr.attr,
822433e817aSMike Travis &uv_type_attr.attr,
823433e817aSMike Travis &uv_archtype_attr.attr,
824433e817aSMike Travis &uv_hubless_attr.attr,
825433e817aSMike Travis NULL,
826433e817aSMike Travis };
827433e817aSMike Travis
82877d06ec6SRikard Falkeborn static const struct attribute_group hubless_base_attr_group = {
829433e817aSMike Travis .attrs = hubless_base_attrs
830433e817aSMike Travis };
831433e817aSMike Travis
832433e817aSMike Travis
uv_sysfs_hubless_init(void)833433e817aSMike Travis static int __init uv_sysfs_hubless_init(void)
834433e817aSMike Travis {
835433e817aSMike Travis int ret;
836433e817aSMike Travis
837433e817aSMike Travis ret = sysfs_create_group(sgi_uv_kobj, &hubless_base_attr_group);
838433e817aSMike Travis if (ret) {
839433e817aSMike Travis pr_warn("sysfs_create_group hubless_base_attr_group failed\n");
840433e817aSMike Travis kobject_put(sgi_uv_kobj);
841433e817aSMike Travis }
842433e817aSMike Travis return ret;
843433e817aSMike Travis }
844433e817aSMike Travis
uv_sysfs_init(void)8454fc2cf1fSJustin Ernst static int __init uv_sysfs_init(void)
8464fc2cf1fSJustin Ernst {
8474fc2cf1fSJustin Ernst int ret = 0;
8484fc2cf1fSJustin Ernst
849433e817aSMike Travis if (!is_uv_system() && !uv_get_hubless_system())
8504fc2cf1fSJustin Ernst return -ENODEV;
8514fc2cf1fSJustin Ernst
8524fc2cf1fSJustin Ernst num_cnodes = uv_num_possible_blades();
8534fc2cf1fSJustin Ernst
8544fc2cf1fSJustin Ernst if (!sgi_uv_kobj)
8554fc2cf1fSJustin Ernst sgi_uv_kobj = kobject_create_and_add("sgi_uv", firmware_kobj);
8564fc2cf1fSJustin Ernst if (!sgi_uv_kobj) {
8574fc2cf1fSJustin Ernst pr_warn("kobject_create_and_add sgi_uv failed\n");
8584fc2cf1fSJustin Ernst return -EINVAL;
8594fc2cf1fSJustin Ernst }
860433e817aSMike Travis
861433e817aSMike Travis if (uv_get_hubless_system())
862433e817aSMike Travis return uv_sysfs_hubless_init();
863433e817aSMike Travis
8644fc2cf1fSJustin Ernst ret = sysfs_create_group(sgi_uv_kobj, &base_attr_group);
8654fc2cf1fSJustin Ernst if (ret) {
8664fc2cf1fSJustin Ernst pr_warn("sysfs_create_group base_attr_group failed\n");
8674fc2cf1fSJustin Ernst goto err_create_group;
8684fc2cf1fSJustin Ernst }
8694fc2cf1fSJustin Ernst
8704fc2cf1fSJustin Ernst ret = initial_bios_setup();
8714fc2cf1fSJustin Ernst if (ret)
8724fc2cf1fSJustin Ernst goto err_bios_setup;
8734fc2cf1fSJustin Ernst
8744fc2cf1fSJustin Ernst ret = uv_hubs_init();
8754fc2cf1fSJustin Ernst if (ret)
8764fc2cf1fSJustin Ernst goto err_hubs_init;
8774fc2cf1fSJustin Ernst
8784fc2cf1fSJustin Ernst ret = uv_ports_init();
8794fc2cf1fSJustin Ernst if (ret)
8804fc2cf1fSJustin Ernst goto err_ports_init;
8814fc2cf1fSJustin Ernst
8824fc2cf1fSJustin Ernst ret = pci_topology_init();
8834fc2cf1fSJustin Ernst if (ret)
8844fc2cf1fSJustin Ernst goto err_pci_init;
8854fc2cf1fSJustin Ernst
8864fc2cf1fSJustin Ernst return 0;
8874fc2cf1fSJustin Ernst
8884fc2cf1fSJustin Ernst err_pci_init:
8894fc2cf1fSJustin Ernst uv_ports_exit();
8904fc2cf1fSJustin Ernst err_ports_init:
8914fc2cf1fSJustin Ernst uv_hubs_exit();
8924fc2cf1fSJustin Ernst err_hubs_init:
8934fc2cf1fSJustin Ernst vfree(uv_biosheap);
8944fc2cf1fSJustin Ernst err_bios_setup:
8954fc2cf1fSJustin Ernst sysfs_remove_group(sgi_uv_kobj, &base_attr_group);
8964fc2cf1fSJustin Ernst err_create_group:
8974fc2cf1fSJustin Ernst kobject_put(sgi_uv_kobj);
8984fc2cf1fSJustin Ernst return ret;
8994fc2cf1fSJustin Ernst }
9004fc2cf1fSJustin Ernst
uv_sysfs_hubless_exit(void)901433e817aSMike Travis static void __exit uv_sysfs_hubless_exit(void)
902433e817aSMike Travis {
903433e817aSMike Travis sysfs_remove_group(sgi_uv_kobj, &hubless_base_attr_group);
904433e817aSMike Travis kobject_put(sgi_uv_kobj);
905433e817aSMike Travis }
906433e817aSMike Travis
uv_sysfs_exit(void)9074fc2cf1fSJustin Ernst static void __exit uv_sysfs_exit(void)
9084fc2cf1fSJustin Ernst {
909433e817aSMike Travis if (!is_uv_system()) {
910433e817aSMike Travis if (uv_get_hubless_system())
911433e817aSMike Travis uv_sysfs_hubless_exit();
9124fc2cf1fSJustin Ernst return;
913433e817aSMike Travis }
9144fc2cf1fSJustin Ernst
9154fc2cf1fSJustin Ernst pci_topology_exit();
9164fc2cf1fSJustin Ernst uv_ports_exit();
9174fc2cf1fSJustin Ernst uv_hubs_exit();
9184fc2cf1fSJustin Ernst vfree(uv_biosheap);
9194fc2cf1fSJustin Ernst sysfs_remove_group(sgi_uv_kobj, &base_attr_group);
9204fc2cf1fSJustin Ernst kobject_put(sgi_uv_kobj);
9214fc2cf1fSJustin Ernst }
9224fc2cf1fSJustin Ernst
9234fc2cf1fSJustin Ernst #ifndef MODULE
9244fc2cf1fSJustin Ernst device_initcall(uv_sysfs_init);
9254fc2cf1fSJustin Ernst #else
9264fc2cf1fSJustin Ernst module_init(uv_sysfs_init);
9274fc2cf1fSJustin Ernst #endif
9284fc2cf1fSJustin Ernst module_exit(uv_sysfs_exit);
9294fc2cf1fSJustin Ernst
9304fc2cf1fSJustin Ernst MODULE_AUTHOR("Hewlett Packard Enterprise");
9314fc2cf1fSJustin Ernst MODULE_LICENSE("GPL");
932