1ab519a01SNathan Fontenot /* 2ab519a01SNathan Fontenot * Support for dynamic reconfiguration for PCI, Memory, and CPU 3ab519a01SNathan Fontenot * Hotplug and Dynamic Logical Partitioning on RPA platforms. 4ab519a01SNathan Fontenot * 5ab519a01SNathan Fontenot * Copyright (C) 2009 Nathan Fontenot 6ab519a01SNathan Fontenot * Copyright (C) 2009 IBM Corporation 7ab519a01SNathan Fontenot * 8ab519a01SNathan Fontenot * This program is free software; you can redistribute it and/or 9ab519a01SNathan Fontenot * modify it under the terms of the GNU General Public License version 10ab519a01SNathan Fontenot * 2 as published by the Free Software Foundation. 11ab519a01SNathan Fontenot */ 12ab519a01SNathan Fontenot 13999e2dadSNathan Fontenot #define pr_fmt(fmt) "dlpar: " fmt 14999e2dadSNathan Fontenot 15ab519a01SNathan Fontenot #include <linux/kernel.h> 16ab519a01SNathan Fontenot #include <linux/notifier.h> 17ab519a01SNathan Fontenot #include <linux/spinlock.h> 18ab519a01SNathan Fontenot #include <linux/cpu.h> 195a0e3ad6STejun Heo #include <linux/slab.h> 201cf3d8b3SNathan Fontenot #include <linux/of.h> 2106bacefcSAndy Shevchenko 2206bacefcSAndy Shevchenko #include "of_helpers.h" 231217d34bSAnton Blanchard #include "pseries.h" 24ab519a01SNathan Fontenot 25ab519a01SNathan Fontenot #include <asm/prom.h> 26ab519a01SNathan Fontenot #include <asm/machdep.h> 277c0f6ba6SLinus Torvalds #include <linux/uaccess.h> 28ab519a01SNathan Fontenot #include <asm/rtas.h> 29ab519a01SNathan Fontenot 307c98bd72SDaniel Axtens static struct workqueue_struct *pseries_hp_wq; 319054619eSJohn Allen 329054619eSJohn Allen struct pseries_hp_work { 339054619eSJohn Allen struct work_struct work; 349054619eSJohn Allen struct pseries_hp_errorlog *errlog; 359054619eSJohn Allen }; 369054619eSJohn Allen 37ab519a01SNathan Fontenot struct cc_workarea { 38d6f1e7abSBharata B Rao __be32 drc_index; 39d6f1e7abSBharata B Rao __be32 zero; 40d6f1e7abSBharata B Rao __be32 name_offset; 41d6f1e7abSBharata B Rao __be32 prop_length; 42d6f1e7abSBharata B Rao __be32 prop_offset; 43ab519a01SNathan Fontenot }; 44ab519a01SNathan Fontenot 4520648974SNathan Fontenot void dlpar_free_cc_property(struct property *prop) 46ab519a01SNathan Fontenot { 47ab519a01SNathan Fontenot kfree(prop->name); 48ab519a01SNathan Fontenot kfree(prop->value); 49ab519a01SNathan Fontenot kfree(prop); 50ab519a01SNathan Fontenot } 51ab519a01SNathan Fontenot 52ab519a01SNathan Fontenot static struct property *dlpar_parse_cc_property(struct cc_workarea *ccwa) 53ab519a01SNathan Fontenot { 54ab519a01SNathan Fontenot struct property *prop; 55ab519a01SNathan Fontenot char *name; 56ab519a01SNathan Fontenot char *value; 57ab519a01SNathan Fontenot 58ab519a01SNathan Fontenot prop = kzalloc(sizeof(*prop), GFP_KERNEL); 59ab519a01SNathan Fontenot if (!prop) 60ab519a01SNathan Fontenot return NULL; 61ab519a01SNathan Fontenot 62d6f1e7abSBharata B Rao name = (char *)ccwa + be32_to_cpu(ccwa->name_offset); 63ab519a01SNathan Fontenot prop->name = kstrdup(name, GFP_KERNEL); 64*efa9ace6SGen Zhang if (!prop->name) { 65*efa9ace6SGen Zhang dlpar_free_cc_property(prop); 66*efa9ace6SGen Zhang return NULL; 67*efa9ace6SGen Zhang } 68ab519a01SNathan Fontenot 69d6f1e7abSBharata B Rao prop->length = be32_to_cpu(ccwa->prop_length); 70d6f1e7abSBharata B Rao value = (char *)ccwa + be32_to_cpu(ccwa->prop_offset); 71e72ed6b5SNishanth Aravamudan prop->value = kmemdup(value, prop->length, GFP_KERNEL); 72ab519a01SNathan Fontenot if (!prop->value) { 73ab519a01SNathan Fontenot dlpar_free_cc_property(prop); 74ab519a01SNathan Fontenot return NULL; 75ab519a01SNathan Fontenot } 76ab519a01SNathan Fontenot 77ab519a01SNathan Fontenot return prop; 78ab519a01SNathan Fontenot } 79ab519a01SNathan Fontenot 8006665989SRob Herring static struct device_node *dlpar_parse_cc_node(struct cc_workarea *ccwa) 81ab519a01SNathan Fontenot { 82ab519a01SNathan Fontenot struct device_node *dn; 8306665989SRob Herring const char *name; 848d5ff320STyrel Datwyler 85ab519a01SNathan Fontenot dn = kzalloc(sizeof(*dn), GFP_KERNEL); 86ab519a01SNathan Fontenot if (!dn) 87ab519a01SNathan Fontenot return NULL; 88ab519a01SNathan Fontenot 8906665989SRob Herring name = (const char *)ccwa + be32_to_cpu(ccwa->name_offset); 9006665989SRob Herring dn->full_name = kstrdup(name, GFP_KERNEL); 91ab519a01SNathan Fontenot if (!dn->full_name) { 92ab519a01SNathan Fontenot kfree(dn); 93ab519a01SNathan Fontenot return NULL; 94ab519a01SNathan Fontenot } 95ab519a01SNathan Fontenot 961578cb76STyrel Datwyler of_node_set_flag(dn, OF_DYNAMIC); 9797a9a717STyrel Datwyler of_node_init(dn); 981578cb76STyrel Datwyler 99ab519a01SNathan Fontenot return dn; 100ab519a01SNathan Fontenot } 101ab519a01SNathan Fontenot 102ab519a01SNathan Fontenot static void dlpar_free_one_cc_node(struct device_node *dn) 103ab519a01SNathan Fontenot { 104ab519a01SNathan Fontenot struct property *prop; 105ab519a01SNathan Fontenot 106ab519a01SNathan Fontenot while (dn->properties) { 107ab519a01SNathan Fontenot prop = dn->properties; 108ab519a01SNathan Fontenot dn->properties = prop->next; 109ab519a01SNathan Fontenot dlpar_free_cc_property(prop); 110ab519a01SNathan Fontenot } 111ab519a01SNathan Fontenot 112ab519a01SNathan Fontenot kfree(dn->full_name); 113ab519a01SNathan Fontenot kfree(dn); 114ab519a01SNathan Fontenot } 115ab519a01SNathan Fontenot 11620648974SNathan Fontenot void dlpar_free_cc_nodes(struct device_node *dn) 117ab519a01SNathan Fontenot { 118ab519a01SNathan Fontenot if (dn->child) 119ab519a01SNathan Fontenot dlpar_free_cc_nodes(dn->child); 120ab519a01SNathan Fontenot 121ab519a01SNathan Fontenot if (dn->sibling) 122ab519a01SNathan Fontenot dlpar_free_cc_nodes(dn->sibling); 123ab519a01SNathan Fontenot 124ab519a01SNathan Fontenot dlpar_free_one_cc_node(dn); 125ab519a01SNathan Fontenot } 126ab519a01SNathan Fontenot 1279c740025SAnton Blanchard #define COMPLETE 0 128ab519a01SNathan Fontenot #define NEXT_SIBLING 1 129ab519a01SNathan Fontenot #define NEXT_CHILD 2 130ab519a01SNathan Fontenot #define NEXT_PROPERTY 3 131ab519a01SNathan Fontenot #define PREV_PARENT 4 132ab519a01SNathan Fontenot #define MORE_MEMORY 5 133ab519a01SNathan Fontenot #define CALL_AGAIN -2 134ab519a01SNathan Fontenot #define ERR_CFG_USE -9003 135ab519a01SNathan Fontenot 136d6f1e7abSBharata B Rao struct device_node *dlpar_configure_connector(__be32 drc_index, 1378d5ff320STyrel Datwyler struct device_node *parent) 138ab519a01SNathan Fontenot { 139ab519a01SNathan Fontenot struct device_node *dn; 140ab519a01SNathan Fontenot struct device_node *first_dn = NULL; 141ab519a01SNathan Fontenot struct device_node *last_dn = NULL; 142ab519a01SNathan Fontenot struct property *property; 143ab519a01SNathan Fontenot struct property *last_property = NULL; 144ab519a01SNathan Fontenot struct cc_workarea *ccwa; 14593f68f1eSNathan Fontenot char *data_buf; 146ab519a01SNathan Fontenot int cc_token; 14793f68f1eSNathan Fontenot int rc = -1; 148ab519a01SNathan Fontenot 149ab519a01SNathan Fontenot cc_token = rtas_token("ibm,configure-connector"); 150ab519a01SNathan Fontenot if (cc_token == RTAS_UNKNOWN_SERVICE) 151ab519a01SNathan Fontenot return NULL; 152ab519a01SNathan Fontenot 15393f68f1eSNathan Fontenot data_buf = kzalloc(RTAS_DATA_BUF_SIZE, GFP_KERNEL); 15493f68f1eSNathan Fontenot if (!data_buf) 15593f68f1eSNathan Fontenot return NULL; 15693f68f1eSNathan Fontenot 15793f68f1eSNathan Fontenot ccwa = (struct cc_workarea *)&data_buf[0]; 158ab519a01SNathan Fontenot ccwa->drc_index = drc_index; 159ab519a01SNathan Fontenot ccwa->zero = 0; 160ab519a01SNathan Fontenot 16193f68f1eSNathan Fontenot do { 16293f68f1eSNathan Fontenot /* Since we release the rtas_data_buf lock between configure 16393f68f1eSNathan Fontenot * connector calls we want to re-populate the rtas_data_buffer 16493f68f1eSNathan Fontenot * with the contents of the previous call. 16593f68f1eSNathan Fontenot */ 16693f68f1eSNathan Fontenot spin_lock(&rtas_data_buf_lock); 16793f68f1eSNathan Fontenot 16893f68f1eSNathan Fontenot memcpy(rtas_data_buf, data_buf, RTAS_DATA_BUF_SIZE); 169ab519a01SNathan Fontenot rc = rtas_call(cc_token, 2, 1, NULL, rtas_data_buf, NULL); 17093f68f1eSNathan Fontenot memcpy(data_buf, rtas_data_buf, RTAS_DATA_BUF_SIZE); 17193f68f1eSNathan Fontenot 17293f68f1eSNathan Fontenot spin_unlock(&rtas_data_buf_lock); 17393f68f1eSNathan Fontenot 174ab519a01SNathan Fontenot switch (rc) { 1759c740025SAnton Blanchard case COMPLETE: 1769c740025SAnton Blanchard break; 1779c740025SAnton Blanchard 178ab519a01SNathan Fontenot case NEXT_SIBLING: 17906665989SRob Herring dn = dlpar_parse_cc_node(ccwa); 180ab519a01SNathan Fontenot if (!dn) 181ab519a01SNathan Fontenot goto cc_error; 182ab519a01SNathan Fontenot 183ab519a01SNathan Fontenot dn->parent = last_dn->parent; 184ab519a01SNathan Fontenot last_dn->sibling = dn; 185ab519a01SNathan Fontenot last_dn = dn; 186ab519a01SNathan Fontenot break; 187ab519a01SNathan Fontenot 188ab519a01SNathan Fontenot case NEXT_CHILD: 18906665989SRob Herring dn = dlpar_parse_cc_node(ccwa); 190ab519a01SNathan Fontenot if (!dn) 191ab519a01SNathan Fontenot goto cc_error; 192ab519a01SNathan Fontenot 1938d5ff320STyrel Datwyler if (!first_dn) { 1948d5ff320STyrel Datwyler dn->parent = parent; 195ab519a01SNathan Fontenot first_dn = dn; 1968d5ff320STyrel Datwyler } else { 197ab519a01SNathan Fontenot dn->parent = last_dn; 198ab519a01SNathan Fontenot if (last_dn) 199ab519a01SNathan Fontenot last_dn->child = dn; 200ab519a01SNathan Fontenot } 201ab519a01SNathan Fontenot 202ab519a01SNathan Fontenot last_dn = dn; 203ab519a01SNathan Fontenot break; 204ab519a01SNathan Fontenot 205ab519a01SNathan Fontenot case NEXT_PROPERTY: 206ab519a01SNathan Fontenot property = dlpar_parse_cc_property(ccwa); 207ab519a01SNathan Fontenot if (!property) 208ab519a01SNathan Fontenot goto cc_error; 209ab519a01SNathan Fontenot 210ab519a01SNathan Fontenot if (!last_dn->properties) 211ab519a01SNathan Fontenot last_dn->properties = property; 212ab519a01SNathan Fontenot else 213ab519a01SNathan Fontenot last_property->next = property; 214ab519a01SNathan Fontenot 215ab519a01SNathan Fontenot last_property = property; 216ab519a01SNathan Fontenot break; 217ab519a01SNathan Fontenot 218ab519a01SNathan Fontenot case PREV_PARENT: 219ab519a01SNathan Fontenot last_dn = last_dn->parent; 220ab519a01SNathan Fontenot break; 221ab519a01SNathan Fontenot 222ab519a01SNathan Fontenot case CALL_AGAIN: 223ab519a01SNathan Fontenot break; 224ab519a01SNathan Fontenot 225ab519a01SNathan Fontenot case MORE_MEMORY: 226ab519a01SNathan Fontenot case ERR_CFG_USE: 227ab519a01SNathan Fontenot default: 228ab519a01SNathan Fontenot printk(KERN_ERR "Unexpected Error (%d) " 229ab519a01SNathan Fontenot "returned from configure-connector\n", rc); 230ab519a01SNathan Fontenot goto cc_error; 231ab519a01SNathan Fontenot } 23293f68f1eSNathan Fontenot } while (rc); 233ab519a01SNathan Fontenot 234ab519a01SNathan Fontenot cc_error: 23593f68f1eSNathan Fontenot kfree(data_buf); 23693f68f1eSNathan Fontenot 23793f68f1eSNathan Fontenot if (rc) { 238ab519a01SNathan Fontenot if (first_dn) 239ab519a01SNathan Fontenot dlpar_free_cc_nodes(first_dn); 24093f68f1eSNathan Fontenot 241ab519a01SNathan Fontenot return NULL; 242ab519a01SNathan Fontenot } 243ab519a01SNathan Fontenot 24493f68f1eSNathan Fontenot return first_dn; 24593f68f1eSNathan Fontenot } 24693f68f1eSNathan Fontenot 247215ee763SRob Herring int dlpar_attach_node(struct device_node *dn, struct device_node *parent) 248ab519a01SNathan Fontenot { 249ab519a01SNathan Fontenot int rc; 250ab519a01SNathan Fontenot 251215ee763SRob Herring dn->parent = parent; 252ab519a01SNathan Fontenot 2531cf3d8b3SNathan Fontenot rc = of_attach_node(dn); 2543aef19f0SAkinobu Mita if (rc) { 255b7c670d6SRob Herring printk(KERN_ERR "Failed to add device node %pOF\n", dn); 2563aef19f0SAkinobu Mita return rc; 257ab519a01SNathan Fontenot } 258ab519a01SNathan Fontenot 259ab519a01SNathan Fontenot return 0; 260ab519a01SNathan Fontenot } 261ab519a01SNathan Fontenot 262ab519a01SNathan Fontenot int dlpar_detach_node(struct device_node *dn) 263ab519a01SNathan Fontenot { 2645935ff43STyrel Datwyler struct device_node *child; 2651cf3d8b3SNathan Fontenot int rc; 266ab519a01SNathan Fontenot 2675935ff43STyrel Datwyler child = of_get_next_child(dn, NULL); 2685935ff43STyrel Datwyler while (child) { 2695935ff43STyrel Datwyler dlpar_detach_node(child); 2705935ff43STyrel Datwyler child = of_get_next_child(dn, child); 2715935ff43STyrel Datwyler } 2725935ff43STyrel Datwyler 2731cf3d8b3SNathan Fontenot rc = of_detach_node(dn); 2741cf3d8b3SNathan Fontenot if (rc) 2751cf3d8b3SNathan Fontenot return rc; 2761cf3d8b3SNathan Fontenot 2775b3f5c40SFrank Rowand of_node_put(dn); 2785b3f5c40SFrank Rowand 279ab519a01SNathan Fontenot return 0; 280ab519a01SNathan Fontenot } 281ab519a01SNathan Fontenot 282ab519a01SNathan Fontenot #define DR_ENTITY_SENSE 9003 283ab519a01SNathan Fontenot #define DR_ENTITY_PRESENT 1 284ab519a01SNathan Fontenot #define DR_ENTITY_UNUSABLE 2 285ab519a01SNathan Fontenot #define ALLOCATION_STATE 9003 286ab519a01SNathan Fontenot #define ALLOC_UNUSABLE 0 287ab519a01SNathan Fontenot #define ALLOC_USABLE 1 288ab519a01SNathan Fontenot #define ISOLATION_STATE 9001 289ab519a01SNathan Fontenot #define ISOLATE 0 290ab519a01SNathan Fontenot #define UNISOLATE 1 291ab519a01SNathan Fontenot 292ab519a01SNathan Fontenot int dlpar_acquire_drc(u32 drc_index) 293ab519a01SNathan Fontenot { 294ab519a01SNathan Fontenot int dr_status, rc; 295ab519a01SNathan Fontenot 296ab519a01SNathan Fontenot rc = rtas_call(rtas_token("get-sensor-state"), 2, 2, &dr_status, 297ab519a01SNathan Fontenot DR_ENTITY_SENSE, drc_index); 298ab519a01SNathan Fontenot if (rc || dr_status != DR_ENTITY_UNUSABLE) 299ab519a01SNathan Fontenot return -1; 300ab519a01SNathan Fontenot 301ab519a01SNathan Fontenot rc = rtas_set_indicator(ALLOCATION_STATE, drc_index, ALLOC_USABLE); 302ab519a01SNathan Fontenot if (rc) 303ab519a01SNathan Fontenot return rc; 304ab519a01SNathan Fontenot 305ab519a01SNathan Fontenot rc = rtas_set_indicator(ISOLATION_STATE, drc_index, UNISOLATE); 306ab519a01SNathan Fontenot if (rc) { 307ab519a01SNathan Fontenot rtas_set_indicator(ALLOCATION_STATE, drc_index, ALLOC_UNUSABLE); 308ab519a01SNathan Fontenot return rc; 309ab519a01SNathan Fontenot } 310ab519a01SNathan Fontenot 311ab519a01SNathan Fontenot return 0; 312ab519a01SNathan Fontenot } 313ab519a01SNathan Fontenot 314ab519a01SNathan Fontenot int dlpar_release_drc(u32 drc_index) 315ab519a01SNathan Fontenot { 316ab519a01SNathan Fontenot int dr_status, rc; 317ab519a01SNathan Fontenot 318ab519a01SNathan Fontenot rc = rtas_call(rtas_token("get-sensor-state"), 2, 2, &dr_status, 319ab519a01SNathan Fontenot DR_ENTITY_SENSE, drc_index); 320ab519a01SNathan Fontenot if (rc || dr_status != DR_ENTITY_PRESENT) 321ab519a01SNathan Fontenot return -1; 322ab519a01SNathan Fontenot 323ab519a01SNathan Fontenot rc = rtas_set_indicator(ISOLATION_STATE, drc_index, ISOLATE); 324ab519a01SNathan Fontenot if (rc) 325ab519a01SNathan Fontenot return rc; 326ab519a01SNathan Fontenot 327ab519a01SNathan Fontenot rc = rtas_set_indicator(ALLOCATION_STATE, drc_index, ALLOC_UNUSABLE); 328ab519a01SNathan Fontenot if (rc) { 329ab519a01SNathan Fontenot rtas_set_indicator(ISOLATION_STATE, drc_index, UNISOLATE); 330ab519a01SNathan Fontenot return rc; 331ab519a01SNathan Fontenot } 332ab519a01SNathan Fontenot 333ab519a01SNathan Fontenot return 0; 334ab519a01SNathan Fontenot } 335ab519a01SNathan Fontenot 336fd12527aSNathan Fontenot int handle_dlpar_errorlog(struct pseries_hp_errorlog *hp_elog) 337999e2dadSNathan Fontenot { 338999e2dadSNathan Fontenot int rc; 339999e2dadSNathan Fontenot 340999e2dadSNathan Fontenot /* pseries error logs are in BE format, convert to cpu type */ 341999e2dadSNathan Fontenot switch (hp_elog->id_type) { 342999e2dadSNathan Fontenot case PSERIES_HP_ELOG_ID_DRC_COUNT: 343999e2dadSNathan Fontenot hp_elog->_drc_u.drc_count = 344999e2dadSNathan Fontenot be32_to_cpu(hp_elog->_drc_u.drc_count); 345999e2dadSNathan Fontenot break; 346999e2dadSNathan Fontenot case PSERIES_HP_ELOG_ID_DRC_INDEX: 347999e2dadSNathan Fontenot hp_elog->_drc_u.drc_index = 348999e2dadSNathan Fontenot be32_to_cpu(hp_elog->_drc_u.drc_index); 349333f7b76SSahil Mehta break; 350333f7b76SSahil Mehta case PSERIES_HP_ELOG_ID_DRC_IC: 351333f7b76SSahil Mehta hp_elog->_drc_u.ic.count = 352333f7b76SSahil Mehta be32_to_cpu(hp_elog->_drc_u.ic.count); 353333f7b76SSahil Mehta hp_elog->_drc_u.ic.index = 354333f7b76SSahil Mehta be32_to_cpu(hp_elog->_drc_u.ic.index); 355999e2dadSNathan Fontenot } 356999e2dadSNathan Fontenot 357999e2dadSNathan Fontenot switch (hp_elog->resource) { 358999e2dadSNathan Fontenot case PSERIES_HP_ELOG_RESOURCE_MEM: 359999e2dadSNathan Fontenot rc = dlpar_memory(hp_elog); 360999e2dadSNathan Fontenot break; 361e9d764f8SNathan Fontenot case PSERIES_HP_ELOG_RESOURCE_CPU: 362e9d764f8SNathan Fontenot rc = dlpar_cpu(hp_elog); 363e9d764f8SNathan Fontenot break; 3644c5d87dbSOliver O'Halloran case PSERIES_HP_ELOG_RESOURCE_PMEM: 3654c5d87dbSOliver O'Halloran rc = dlpar_hp_pmem(hp_elog); 3664c5d87dbSOliver O'Halloran break; 3674c5d87dbSOliver O'Halloran 368999e2dadSNathan Fontenot default: 369999e2dadSNathan Fontenot pr_warn_ratelimited("Invalid resource (%d) specified\n", 370999e2dadSNathan Fontenot hp_elog->resource); 371999e2dadSNathan Fontenot rc = -EINVAL; 372999e2dadSNathan Fontenot } 373999e2dadSNathan Fontenot 374999e2dadSNathan Fontenot return rc; 375999e2dadSNathan Fontenot } 376999e2dadSNathan Fontenot 3777c98bd72SDaniel Axtens static void pseries_hp_work_fn(struct work_struct *work) 3789054619eSJohn Allen { 3799054619eSJohn Allen struct pseries_hp_work *hp_work = 3809054619eSJohn Allen container_of(work, struct pseries_hp_work, work); 3819054619eSJohn Allen 3829054619eSJohn Allen handle_dlpar_errorlog(hp_work->errlog); 3839054619eSJohn Allen 3849054619eSJohn Allen kfree(hp_work->errlog); 3859054619eSJohn Allen kfree((void *)work); 3869054619eSJohn Allen } 3879054619eSJohn Allen 388fd12527aSNathan Fontenot void queue_hotplug_event(struct pseries_hp_errorlog *hp_errlog) 3899054619eSJohn Allen { 3909054619eSJohn Allen struct pseries_hp_work *work; 3919054619eSJohn Allen struct pseries_hp_errorlog *hp_errlog_copy; 3929054619eSJohn Allen 3939054619eSJohn Allen hp_errlog_copy = kmalloc(sizeof(struct pseries_hp_errorlog), 3949054619eSJohn Allen GFP_KERNEL); 3959054619eSJohn Allen memcpy(hp_errlog_copy, hp_errlog, sizeof(struct pseries_hp_errorlog)); 3969054619eSJohn Allen 3979054619eSJohn Allen work = kmalloc(sizeof(struct pseries_hp_work), GFP_KERNEL); 3989054619eSJohn Allen if (work) { 3999054619eSJohn Allen INIT_WORK((struct work_struct *)work, pseries_hp_work_fn); 4009054619eSJohn Allen work->errlog = hp_errlog_copy; 4019054619eSJohn Allen queue_work(pseries_hp_wq, (struct work_struct *)work); 4029054619eSJohn Allen } else { 40390ce3514SAndrew Donnellan kfree(hp_errlog_copy); 4049054619eSJohn Allen } 4059054619eSJohn Allen } 4069054619eSJohn Allen 40725b587fbSNathan Fontenot static int dlpar_parse_resource(char **cmd, struct pseries_hp_errorlog *hp_elog) 40825b587fbSNathan Fontenot { 40925b587fbSNathan Fontenot char *arg; 41025b587fbSNathan Fontenot 41125b587fbSNathan Fontenot arg = strsep(cmd, " "); 41225b587fbSNathan Fontenot if (!arg) 41325b587fbSNathan Fontenot return -EINVAL; 41425b587fbSNathan Fontenot 41525b587fbSNathan Fontenot if (sysfs_streq(arg, "memory")) { 41625b587fbSNathan Fontenot hp_elog->resource = PSERIES_HP_ELOG_RESOURCE_MEM; 41725b587fbSNathan Fontenot } else if (sysfs_streq(arg, "cpu")) { 41825b587fbSNathan Fontenot hp_elog->resource = PSERIES_HP_ELOG_RESOURCE_CPU; 41925b587fbSNathan Fontenot } else { 42025b587fbSNathan Fontenot pr_err("Invalid resource specified.\n"); 42125b587fbSNathan Fontenot return -EINVAL; 42225b587fbSNathan Fontenot } 42325b587fbSNathan Fontenot 42425b587fbSNathan Fontenot return 0; 42525b587fbSNathan Fontenot } 42625b587fbSNathan Fontenot 42725b587fbSNathan Fontenot static int dlpar_parse_action(char **cmd, struct pseries_hp_errorlog *hp_elog) 42825b587fbSNathan Fontenot { 42925b587fbSNathan Fontenot char *arg; 43025b587fbSNathan Fontenot 43125b587fbSNathan Fontenot arg = strsep(cmd, " "); 43225b587fbSNathan Fontenot if (!arg) 43325b587fbSNathan Fontenot return -EINVAL; 43425b587fbSNathan Fontenot 43525b587fbSNathan Fontenot if (sysfs_streq(arg, "add")) { 43625b587fbSNathan Fontenot hp_elog->action = PSERIES_HP_ELOG_ACTION_ADD; 43725b587fbSNathan Fontenot } else if (sysfs_streq(arg, "remove")) { 43825b587fbSNathan Fontenot hp_elog->action = PSERIES_HP_ELOG_ACTION_REMOVE; 43925b587fbSNathan Fontenot } else { 44025b587fbSNathan Fontenot pr_err("Invalid action specified.\n"); 44125b587fbSNathan Fontenot return -EINVAL; 44225b587fbSNathan Fontenot } 44325b587fbSNathan Fontenot 44425b587fbSNathan Fontenot return 0; 44525b587fbSNathan Fontenot } 44625b587fbSNathan Fontenot 44725b587fbSNathan Fontenot static int dlpar_parse_id_type(char **cmd, struct pseries_hp_errorlog *hp_elog) 44825b587fbSNathan Fontenot { 44925b587fbSNathan Fontenot char *arg; 45025b587fbSNathan Fontenot u32 count, index; 45125b587fbSNathan Fontenot 45225b587fbSNathan Fontenot arg = strsep(cmd, " "); 45325b587fbSNathan Fontenot if (!arg) 45425b587fbSNathan Fontenot return -EINVAL; 45525b587fbSNathan Fontenot 456333f7b76SSahil Mehta if (sysfs_streq(arg, "indexed-count")) { 457333f7b76SSahil Mehta hp_elog->id_type = PSERIES_HP_ELOG_ID_DRC_IC; 458333f7b76SSahil Mehta arg = strsep(cmd, " "); 459333f7b76SSahil Mehta if (!arg) { 460333f7b76SSahil Mehta pr_err("No DRC count specified.\n"); 461333f7b76SSahil Mehta return -EINVAL; 462333f7b76SSahil Mehta } 463333f7b76SSahil Mehta 464333f7b76SSahil Mehta if (kstrtou32(arg, 0, &count)) { 465333f7b76SSahil Mehta pr_err("Invalid DRC count specified.\n"); 466333f7b76SSahil Mehta return -EINVAL; 467333f7b76SSahil Mehta } 468333f7b76SSahil Mehta 469333f7b76SSahil Mehta arg = strsep(cmd, " "); 470333f7b76SSahil Mehta if (!arg) { 471333f7b76SSahil Mehta pr_err("No DRC Index specified.\n"); 472333f7b76SSahil Mehta return -EINVAL; 473333f7b76SSahil Mehta } 474333f7b76SSahil Mehta 475333f7b76SSahil Mehta if (kstrtou32(arg, 0, &index)) { 476333f7b76SSahil Mehta pr_err("Invalid DRC Index specified.\n"); 477333f7b76SSahil Mehta return -EINVAL; 478333f7b76SSahil Mehta } 479333f7b76SSahil Mehta 480333f7b76SSahil Mehta hp_elog->_drc_u.ic.count = cpu_to_be32(count); 481333f7b76SSahil Mehta hp_elog->_drc_u.ic.index = cpu_to_be32(index); 482333f7b76SSahil Mehta } else if (sysfs_streq(arg, "index")) { 48325b587fbSNathan Fontenot hp_elog->id_type = PSERIES_HP_ELOG_ID_DRC_INDEX; 48425b587fbSNathan Fontenot arg = strsep(cmd, " "); 48525b587fbSNathan Fontenot if (!arg) { 48625b587fbSNathan Fontenot pr_err("No DRC Index specified.\n"); 48725b587fbSNathan Fontenot return -EINVAL; 48825b587fbSNathan Fontenot } 48925b587fbSNathan Fontenot 49025b587fbSNathan Fontenot if (kstrtou32(arg, 0, &index)) { 49125b587fbSNathan Fontenot pr_err("Invalid DRC Index specified.\n"); 49225b587fbSNathan Fontenot return -EINVAL; 49325b587fbSNathan Fontenot } 49425b587fbSNathan Fontenot 49525b587fbSNathan Fontenot hp_elog->_drc_u.drc_index = cpu_to_be32(index); 49625b587fbSNathan Fontenot } else if (sysfs_streq(arg, "count")) { 49725b587fbSNathan Fontenot hp_elog->id_type = PSERIES_HP_ELOG_ID_DRC_COUNT; 49825b587fbSNathan Fontenot arg = strsep(cmd, " "); 49925b587fbSNathan Fontenot if (!arg) { 50025b587fbSNathan Fontenot pr_err("No DRC count specified.\n"); 50125b587fbSNathan Fontenot return -EINVAL; 50225b587fbSNathan Fontenot } 50325b587fbSNathan Fontenot 50425b587fbSNathan Fontenot if (kstrtou32(arg, 0, &count)) { 50525b587fbSNathan Fontenot pr_err("Invalid DRC count specified.\n"); 50625b587fbSNathan Fontenot return -EINVAL; 50725b587fbSNathan Fontenot } 50825b587fbSNathan Fontenot 50925b587fbSNathan Fontenot hp_elog->_drc_u.drc_count = cpu_to_be32(count); 51025b587fbSNathan Fontenot } else { 51125b587fbSNathan Fontenot pr_err("Invalid id_type specified.\n"); 51225b587fbSNathan Fontenot return -EINVAL; 51325b587fbSNathan Fontenot } 51425b587fbSNathan Fontenot 51525b587fbSNathan Fontenot return 0; 51625b587fbSNathan Fontenot } 51725b587fbSNathan Fontenot 518999e2dadSNathan Fontenot static ssize_t dlpar_store(struct class *class, struct class_attribute *attr, 519999e2dadSNathan Fontenot const char *buf, size_t count) 520999e2dadSNathan Fontenot { 521fd12527aSNathan Fontenot struct pseries_hp_errorlog hp_elog; 52225b587fbSNathan Fontenot char *argbuf; 52325b587fbSNathan Fontenot char *args; 524999e2dadSNathan Fontenot int rc; 525999e2dadSNathan Fontenot 52625b587fbSNathan Fontenot args = argbuf = kstrdup(buf, GFP_KERNEL); 527fd12527aSNathan Fontenot if (!argbuf) { 52825b587fbSNathan Fontenot pr_info("Could not allocate resources for DLPAR operation\n"); 52925b587fbSNathan Fontenot kfree(argbuf); 53025b587fbSNathan Fontenot return -ENOMEM; 531999e2dadSNathan Fontenot } 532999e2dadSNathan Fontenot 53325b587fbSNathan Fontenot /* 53425b587fbSNathan Fontenot * Parse out the request from the user, this will be in the form: 535999e2dadSNathan Fontenot * <resource> <action> <id_type> <id> 536999e2dadSNathan Fontenot */ 537fd12527aSNathan Fontenot rc = dlpar_parse_resource(&args, &hp_elog); 53825b587fbSNathan Fontenot if (rc) 539999e2dadSNathan Fontenot goto dlpar_store_out; 540999e2dadSNathan Fontenot 541fd12527aSNathan Fontenot rc = dlpar_parse_action(&args, &hp_elog); 54225b587fbSNathan Fontenot if (rc) 543999e2dadSNathan Fontenot goto dlpar_store_out; 544999e2dadSNathan Fontenot 545fd12527aSNathan Fontenot rc = dlpar_parse_id_type(&args, &hp_elog); 54625b587fbSNathan Fontenot if (rc) 547999e2dadSNathan Fontenot goto dlpar_store_out; 548999e2dadSNathan Fontenot 549fd12527aSNathan Fontenot rc = handle_dlpar_errorlog(&hp_elog); 550999e2dadSNathan Fontenot 551999e2dadSNathan Fontenot dlpar_store_out: 55225b587fbSNathan Fontenot kfree(argbuf); 55325b587fbSNathan Fontenot 55425b587fbSNathan Fontenot if (rc) 55525b587fbSNathan Fontenot pr_err("Could not handle DLPAR request \"%s\"\n", buf); 55625b587fbSNathan Fontenot 557999e2dadSNathan Fontenot return rc ? rc : count; 558999e2dadSNathan Fontenot } 559999e2dadSNathan Fontenot 560673bc435SNathan Fontenot static ssize_t dlpar_show(struct class *class, struct class_attribute *attr, 561673bc435SNathan Fontenot char *buf) 562673bc435SNathan Fontenot { 563673bc435SNathan Fontenot return sprintf(buf, "%s\n", "memory,cpu"); 564673bc435SNathan Fontenot } 565673bc435SNathan Fontenot 5666f428096SGreg Kroah-Hartman static CLASS_ATTR_RW(dlpar); 567999e2dadSNathan Fontenot 568e2d59152SMichael Ellerman int __init dlpar_workqueue_init(void) 5691a8061c4SNathan Fontenot { 570e2d59152SMichael Ellerman if (pseries_hp_wq) 571e2d59152SMichael Ellerman return 0; 572e2d59152SMichael Ellerman 5739054619eSJohn Allen pseries_hp_wq = alloc_workqueue("pseries hotplug workqueue", 5749054619eSJohn Allen WQ_UNBOUND, 1); 575e2d59152SMichael Ellerman 576e2d59152SMichael Ellerman return pseries_hp_wq ? 0 : -ENOMEM; 577e2d59152SMichael Ellerman } 578e2d59152SMichael Ellerman 579e2d59152SMichael Ellerman static int __init dlpar_sysfs_init(void) 580e2d59152SMichael Ellerman { 581e2d59152SMichael Ellerman int rc; 582e2d59152SMichael Ellerman 583e2d59152SMichael Ellerman rc = dlpar_workqueue_init(); 584e2d59152SMichael Ellerman if (rc) 585e2d59152SMichael Ellerman return rc; 586e2d59152SMichael Ellerman 587183deeeaSNathan Fontenot return sysfs_create_file(kernel_kobj, &class_attr_dlpar.attr); 5881a8061c4SNathan Fontenot } 589e2d59152SMichael Ellerman machine_device_initcall(pseries, dlpar_sysfs_init); 5901a8061c4SNathan Fontenot 591