qeth_core_sys.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666
  1. /*
  2. * drivers/s390/net/qeth_core_sys.c
  3. *
  4. * Copyright IBM Corp. 2007
  5. * Author(s): Utz Bacher <utz.bacher@de.ibm.com>,
  6. * Frank Pavlic <fpavlic@de.ibm.com>,
  7. * Thomas Spatzier <tspat@de.ibm.com>,
  8. * Frank Blaschka <frank.blaschka@de.ibm.com>
  9. */
  10. #include <linux/list.h>
  11. #include <linux/rwsem.h>
  12. #include <asm/ebcdic.h>
  13. #include "qeth_core.h"
  14. static ssize_t qeth_dev_state_show(struct device *dev,
  15. struct device_attribute *attr, char *buf)
  16. {
  17. struct qeth_card *card = dev_get_drvdata(dev);
  18. if (!card)
  19. return -EINVAL;
  20. switch (card->state) {
  21. case CARD_STATE_DOWN:
  22. return sprintf(buf, "DOWN\n");
  23. case CARD_STATE_HARDSETUP:
  24. return sprintf(buf, "HARDSETUP\n");
  25. case CARD_STATE_SOFTSETUP:
  26. return sprintf(buf, "SOFTSETUP\n");
  27. case CARD_STATE_UP:
  28. if (card->lan_online)
  29. return sprintf(buf, "UP (LAN ONLINE)\n");
  30. else
  31. return sprintf(buf, "UP (LAN OFFLINE)\n");
  32. case CARD_STATE_RECOVER:
  33. return sprintf(buf, "RECOVER\n");
  34. default:
  35. return sprintf(buf, "UNKNOWN\n");
  36. }
  37. }
  38. static DEVICE_ATTR(state, 0444, qeth_dev_state_show, NULL);
  39. static ssize_t qeth_dev_chpid_show(struct device *dev,
  40. struct device_attribute *attr, char *buf)
  41. {
  42. struct qeth_card *card = dev_get_drvdata(dev);
  43. if (!card)
  44. return -EINVAL;
  45. return sprintf(buf, "%02X\n", card->info.chpid);
  46. }
  47. static DEVICE_ATTR(chpid, 0444, qeth_dev_chpid_show, NULL);
  48. static ssize_t qeth_dev_if_name_show(struct device *dev,
  49. struct device_attribute *attr, char *buf)
  50. {
  51. struct qeth_card *card = dev_get_drvdata(dev);
  52. if (!card)
  53. return -EINVAL;
  54. return sprintf(buf, "%s\n", QETH_CARD_IFNAME(card));
  55. }
  56. static DEVICE_ATTR(if_name, 0444, qeth_dev_if_name_show, NULL);
  57. static ssize_t qeth_dev_card_type_show(struct device *dev,
  58. struct device_attribute *attr, char *buf)
  59. {
  60. struct qeth_card *card = dev_get_drvdata(dev);
  61. if (!card)
  62. return -EINVAL;
  63. return sprintf(buf, "%s\n", qeth_get_cardname_short(card));
  64. }
  65. static DEVICE_ATTR(card_type, 0444, qeth_dev_card_type_show, NULL);
  66. static inline const char *qeth_get_bufsize_str(struct qeth_card *card)
  67. {
  68. if (card->qdio.in_buf_size == 16384)
  69. return "16k";
  70. else if (card->qdio.in_buf_size == 24576)
  71. return "24k";
  72. else if (card->qdio.in_buf_size == 32768)
  73. return "32k";
  74. else if (card->qdio.in_buf_size == 40960)
  75. return "40k";
  76. else
  77. return "64k";
  78. }
  79. static ssize_t qeth_dev_inbuf_size_show(struct device *dev,
  80. struct device_attribute *attr, char *buf)
  81. {
  82. struct qeth_card *card = dev_get_drvdata(dev);
  83. if (!card)
  84. return -EINVAL;
  85. return sprintf(buf, "%s\n", qeth_get_bufsize_str(card));
  86. }
  87. static DEVICE_ATTR(inbuf_size, 0444, qeth_dev_inbuf_size_show, NULL);
  88. static ssize_t qeth_dev_portno_show(struct device *dev,
  89. struct device_attribute *attr, char *buf)
  90. {
  91. struct qeth_card *card = dev_get_drvdata(dev);
  92. if (!card)
  93. return -EINVAL;
  94. return sprintf(buf, "%i\n", card->info.portno);
  95. }
  96. static ssize_t qeth_dev_portno_store(struct device *dev,
  97. struct device_attribute *attr, const char *buf, size_t count)
  98. {
  99. struct qeth_card *card = dev_get_drvdata(dev);
  100. char *tmp;
  101. unsigned int portno, limit;
  102. if (!card)
  103. return -EINVAL;
  104. if ((card->state != CARD_STATE_DOWN) &&
  105. (card->state != CARD_STATE_RECOVER))
  106. return -EPERM;
  107. portno = simple_strtoul(buf, &tmp, 16);
  108. if (portno > QETH_MAX_PORTNO)
  109. return -EINVAL;
  110. limit = (card->ssqd.pcnt ? card->ssqd.pcnt - 1 : card->ssqd.pcnt);
  111. if (portno > limit)
  112. return -EINVAL;
  113. card->info.portno = portno;
  114. return count;
  115. }
  116. static DEVICE_ATTR(portno, 0644, qeth_dev_portno_show, qeth_dev_portno_store);
  117. static ssize_t qeth_dev_portname_show(struct device *dev,
  118. struct device_attribute *attr, char *buf)
  119. {
  120. struct qeth_card *card = dev_get_drvdata(dev);
  121. char portname[9] = {0, };
  122. if (!card)
  123. return -EINVAL;
  124. if (card->info.portname_required) {
  125. memcpy(portname, card->info.portname + 1, 8);
  126. EBCASC(portname, 8);
  127. return sprintf(buf, "%s\n", portname);
  128. } else
  129. return sprintf(buf, "no portname required\n");
  130. }
  131. static ssize_t qeth_dev_portname_store(struct device *dev,
  132. struct device_attribute *attr, const char *buf, size_t count)
  133. {
  134. struct qeth_card *card = dev_get_drvdata(dev);
  135. char *tmp;
  136. int i;
  137. if (!card)
  138. return -EINVAL;
  139. if ((card->state != CARD_STATE_DOWN) &&
  140. (card->state != CARD_STATE_RECOVER))
  141. return -EPERM;
  142. tmp = strsep((char **) &buf, "\n");
  143. if ((strlen(tmp) > 8) || (strlen(tmp) == 0))
  144. return -EINVAL;
  145. card->info.portname[0] = strlen(tmp);
  146. /* for beauty reasons */
  147. for (i = 1; i < 9; i++)
  148. card->info.portname[i] = ' ';
  149. strcpy(card->info.portname + 1, tmp);
  150. ASCEBC(card->info.portname + 1, 8);
  151. return count;
  152. }
  153. static DEVICE_ATTR(portname, 0644, qeth_dev_portname_show,
  154. qeth_dev_portname_store);
  155. static ssize_t qeth_dev_prioqing_show(struct device *dev,
  156. struct device_attribute *attr, char *buf)
  157. {
  158. struct qeth_card *card = dev_get_drvdata(dev);
  159. if (!card)
  160. return -EINVAL;
  161. switch (card->qdio.do_prio_queueing) {
  162. case QETH_PRIO_Q_ING_PREC:
  163. return sprintf(buf, "%s\n", "by precedence");
  164. case QETH_PRIO_Q_ING_TOS:
  165. return sprintf(buf, "%s\n", "by type of service");
  166. default:
  167. return sprintf(buf, "always queue %i\n",
  168. card->qdio.default_out_queue);
  169. }
  170. }
  171. static ssize_t qeth_dev_prioqing_store(struct device *dev,
  172. struct device_attribute *attr, const char *buf, size_t count)
  173. {
  174. struct qeth_card *card = dev_get_drvdata(dev);
  175. char *tmp;
  176. if (!card)
  177. return -EINVAL;
  178. if ((card->state != CARD_STATE_DOWN) &&
  179. (card->state != CARD_STATE_RECOVER))
  180. return -EPERM;
  181. /* check if 1920 devices are supported ,
  182. * if though we have to permit priority queueing
  183. */
  184. if (card->qdio.no_out_queues == 1) {
  185. card->qdio.do_prio_queueing = QETH_PRIOQ_DEFAULT;
  186. return -EPERM;
  187. }
  188. tmp = strsep((char **) &buf, "\n");
  189. if (!strcmp(tmp, "prio_queueing_prec"))
  190. card->qdio.do_prio_queueing = QETH_PRIO_Q_ING_PREC;
  191. else if (!strcmp(tmp, "prio_queueing_tos"))
  192. card->qdio.do_prio_queueing = QETH_PRIO_Q_ING_TOS;
  193. else if (!strcmp(tmp, "no_prio_queueing:0")) {
  194. card->qdio.do_prio_queueing = QETH_NO_PRIO_QUEUEING;
  195. card->qdio.default_out_queue = 0;
  196. } else if (!strcmp(tmp, "no_prio_queueing:1")) {
  197. card->qdio.do_prio_queueing = QETH_NO_PRIO_QUEUEING;
  198. card->qdio.default_out_queue = 1;
  199. } else if (!strcmp(tmp, "no_prio_queueing:2")) {
  200. card->qdio.do_prio_queueing = QETH_NO_PRIO_QUEUEING;
  201. card->qdio.default_out_queue = 2;
  202. } else if (!strcmp(tmp, "no_prio_queueing:3")) {
  203. card->qdio.do_prio_queueing = QETH_NO_PRIO_QUEUEING;
  204. card->qdio.default_out_queue = 3;
  205. } else if (!strcmp(tmp, "no_prio_queueing")) {
  206. card->qdio.do_prio_queueing = QETH_NO_PRIO_QUEUEING;
  207. card->qdio.default_out_queue = QETH_DEFAULT_QUEUE;
  208. } else {
  209. return -EINVAL;
  210. }
  211. return count;
  212. }
  213. static DEVICE_ATTR(priority_queueing, 0644, qeth_dev_prioqing_show,
  214. qeth_dev_prioqing_store);
  215. static ssize_t qeth_dev_bufcnt_show(struct device *dev,
  216. struct device_attribute *attr, char *buf)
  217. {
  218. struct qeth_card *card = dev_get_drvdata(dev);
  219. if (!card)
  220. return -EINVAL;
  221. return sprintf(buf, "%i\n", card->qdio.in_buf_pool.buf_count);
  222. }
  223. static ssize_t qeth_dev_bufcnt_store(struct device *dev,
  224. struct device_attribute *attr, const char *buf, size_t count)
  225. {
  226. struct qeth_card *card = dev_get_drvdata(dev);
  227. char *tmp;
  228. int cnt, old_cnt;
  229. int rc;
  230. if (!card)
  231. return -EINVAL;
  232. if ((card->state != CARD_STATE_DOWN) &&
  233. (card->state != CARD_STATE_RECOVER))
  234. return -EPERM;
  235. old_cnt = card->qdio.in_buf_pool.buf_count;
  236. cnt = simple_strtoul(buf, &tmp, 10);
  237. cnt = (cnt < QETH_IN_BUF_COUNT_MIN) ? QETH_IN_BUF_COUNT_MIN :
  238. ((cnt > QETH_IN_BUF_COUNT_MAX) ? QETH_IN_BUF_COUNT_MAX : cnt);
  239. if (old_cnt != cnt) {
  240. rc = qeth_realloc_buffer_pool(card, cnt);
  241. }
  242. return count;
  243. }
  244. static DEVICE_ATTR(buffer_count, 0644, qeth_dev_bufcnt_show,
  245. qeth_dev_bufcnt_store);
  246. static ssize_t qeth_dev_recover_store(struct device *dev,
  247. struct device_attribute *attr, const char *buf, size_t count)
  248. {
  249. struct qeth_card *card = dev_get_drvdata(dev);
  250. char *tmp;
  251. int i;
  252. if (!card)
  253. return -EINVAL;
  254. if (card->state != CARD_STATE_UP)
  255. return -EPERM;
  256. i = simple_strtoul(buf, &tmp, 16);
  257. if (i == 1)
  258. qeth_schedule_recovery(card);
  259. return count;
  260. }
  261. static DEVICE_ATTR(recover, 0200, NULL, qeth_dev_recover_store);
  262. static ssize_t qeth_dev_performance_stats_show(struct device *dev,
  263. struct device_attribute *attr, char *buf)
  264. {
  265. struct qeth_card *card = dev_get_drvdata(dev);
  266. if (!card)
  267. return -EINVAL;
  268. return sprintf(buf, "%i\n", card->options.performance_stats ? 1:0);
  269. }
  270. static ssize_t qeth_dev_performance_stats_store(struct device *dev,
  271. struct device_attribute *attr, const char *buf, size_t count)
  272. {
  273. struct qeth_card *card = dev_get_drvdata(dev);
  274. char *tmp;
  275. int i;
  276. if (!card)
  277. return -EINVAL;
  278. i = simple_strtoul(buf, &tmp, 16);
  279. if ((i == 0) || (i == 1)) {
  280. if (i == card->options.performance_stats)
  281. return count;
  282. card->options.performance_stats = i;
  283. if (i == 0)
  284. memset(&card->perf_stats, 0,
  285. sizeof(struct qeth_perf_stats));
  286. card->perf_stats.initial_rx_packets = card->stats.rx_packets;
  287. card->perf_stats.initial_tx_packets = card->stats.tx_packets;
  288. } else {
  289. return -EINVAL;
  290. }
  291. return count;
  292. }
  293. static DEVICE_ATTR(performance_stats, 0644, qeth_dev_performance_stats_show,
  294. qeth_dev_performance_stats_store);
  295. static ssize_t qeth_dev_layer2_show(struct device *dev,
  296. struct device_attribute *attr, char *buf)
  297. {
  298. struct qeth_card *card = dev_get_drvdata(dev);
  299. if (!card)
  300. return -EINVAL;
  301. return sprintf(buf, "%i\n", card->options.layer2);
  302. }
  303. static ssize_t qeth_dev_layer2_store(struct device *dev,
  304. struct device_attribute *attr, const char *buf, size_t count)
  305. {
  306. struct qeth_card *card = dev_get_drvdata(dev);
  307. char *tmp;
  308. int i, rc;
  309. enum qeth_discipline_id newdis;
  310. if (!card)
  311. return -EINVAL;
  312. if (((card->state != CARD_STATE_DOWN) &&
  313. (card->state != CARD_STATE_RECOVER)))
  314. return -EPERM;
  315. i = simple_strtoul(buf, &tmp, 16);
  316. switch (i) {
  317. case 0:
  318. newdis = QETH_DISCIPLINE_LAYER3;
  319. break;
  320. case 1:
  321. newdis = QETH_DISCIPLINE_LAYER2;
  322. break;
  323. default:
  324. return -EINVAL;
  325. }
  326. if (card->options.layer2 == newdis) {
  327. return count;
  328. } else {
  329. if (card->discipline.ccwgdriver) {
  330. card->discipline.ccwgdriver->remove(card->gdev);
  331. qeth_core_free_discipline(card);
  332. }
  333. }
  334. rc = qeth_core_load_discipline(card, newdis);
  335. if (rc)
  336. return rc;
  337. rc = card->discipline.ccwgdriver->probe(card->gdev);
  338. if (rc)
  339. return rc;
  340. return count;
  341. }
  342. static DEVICE_ATTR(layer2, 0644, qeth_dev_layer2_show,
  343. qeth_dev_layer2_store);
  344. #define ATTR_QETH_ISOLATION_NONE ("none")
  345. #define ATTR_QETH_ISOLATION_FWD ("forward")
  346. #define ATTR_QETH_ISOLATION_DROP ("drop")
  347. static ssize_t qeth_dev_isolation_show(struct device *dev,
  348. struct device_attribute *attr, char *buf)
  349. {
  350. struct qeth_card *card = dev_get_drvdata(dev);
  351. if (!card)
  352. return -EINVAL;
  353. switch (card->options.isolation) {
  354. case ISOLATION_MODE_NONE:
  355. return snprintf(buf, 6, "%s\n", ATTR_QETH_ISOLATION_NONE);
  356. case ISOLATION_MODE_FWD:
  357. return snprintf(buf, 9, "%s\n", ATTR_QETH_ISOLATION_FWD);
  358. case ISOLATION_MODE_DROP:
  359. return snprintf(buf, 6, "%s\n", ATTR_QETH_ISOLATION_DROP);
  360. default:
  361. return snprintf(buf, 5, "%s\n", "N/A");
  362. }
  363. }
  364. static ssize_t qeth_dev_isolation_store(struct device *dev,
  365. struct device_attribute *attr, const char *buf, size_t count)
  366. {
  367. struct qeth_card *card = dev_get_drvdata(dev);
  368. enum qeth_ipa_isolation_modes isolation;
  369. int rc = 0;
  370. char *tmp, *curtoken;
  371. curtoken = (char *) buf;
  372. if (!card) {
  373. rc = -EINVAL;
  374. goto out;
  375. }
  376. /* check for unknown, too, in case we do not yet know who we are */
  377. if (card->info.type != QETH_CARD_TYPE_OSAE &&
  378. card->info.type != QETH_CARD_TYPE_UNKNOWN) {
  379. rc = -EOPNOTSUPP;
  380. dev_err(&card->gdev->dev, "Adapter does not "
  381. "support QDIO data connection isolation\n");
  382. goto out;
  383. }
  384. /* parse input into isolation mode */
  385. tmp = strsep(&curtoken, "\n");
  386. if (!strcmp(tmp, ATTR_QETH_ISOLATION_NONE)) {
  387. isolation = ISOLATION_MODE_NONE;
  388. } else if (!strcmp(tmp, ATTR_QETH_ISOLATION_FWD)) {
  389. isolation = ISOLATION_MODE_FWD;
  390. } else if (!strcmp(tmp, ATTR_QETH_ISOLATION_DROP)) {
  391. isolation = ISOLATION_MODE_DROP;
  392. } else {
  393. rc = -EINVAL;
  394. goto out;
  395. }
  396. rc = count;
  397. /* defer IP assist if device is offline (until discipline->set_online)*/
  398. card->options.isolation = isolation;
  399. if (card->state == CARD_STATE_SOFTSETUP ||
  400. card->state == CARD_STATE_UP) {
  401. int ipa_rc = qeth_set_access_ctrl_online(card);
  402. if (ipa_rc != 0)
  403. rc = ipa_rc;
  404. }
  405. out:
  406. return rc;
  407. }
  408. static DEVICE_ATTR(isolation, 0644, qeth_dev_isolation_show,
  409. qeth_dev_isolation_store);
  410. static ssize_t qeth_dev_blkt_show(char *buf, struct qeth_card *card, int value)
  411. {
  412. if (!card)
  413. return -EINVAL;
  414. return sprintf(buf, "%i\n", value);
  415. }
  416. static ssize_t qeth_dev_blkt_store(struct qeth_card *card,
  417. const char *buf, size_t count, int *value, int max_value)
  418. {
  419. char *tmp;
  420. int i;
  421. if (!card)
  422. return -EINVAL;
  423. if ((card->state != CARD_STATE_DOWN) &&
  424. (card->state != CARD_STATE_RECOVER))
  425. return -EPERM;
  426. i = simple_strtoul(buf, &tmp, 10);
  427. if (i <= max_value) {
  428. *value = i;
  429. } else {
  430. return -EINVAL;
  431. }
  432. return count;
  433. }
  434. static ssize_t qeth_dev_blkt_total_show(struct device *dev,
  435. struct device_attribute *attr, char *buf)
  436. {
  437. struct qeth_card *card = dev_get_drvdata(dev);
  438. return qeth_dev_blkt_show(buf, card, card->info.blkt.time_total);
  439. }
  440. static ssize_t qeth_dev_blkt_total_store(struct device *dev,
  441. struct device_attribute *attr, const char *buf, size_t count)
  442. {
  443. struct qeth_card *card = dev_get_drvdata(dev);
  444. return qeth_dev_blkt_store(card, buf, count,
  445. &card->info.blkt.time_total, 5000);
  446. }
  447. static DEVICE_ATTR(total, 0644, qeth_dev_blkt_total_show,
  448. qeth_dev_blkt_total_store);
  449. static ssize_t qeth_dev_blkt_inter_show(struct device *dev,
  450. struct device_attribute *attr, char *buf)
  451. {
  452. struct qeth_card *card = dev_get_drvdata(dev);
  453. return qeth_dev_blkt_show(buf, card, card->info.blkt.inter_packet);
  454. }
  455. static ssize_t qeth_dev_blkt_inter_store(struct device *dev,
  456. struct device_attribute *attr, const char *buf, size_t count)
  457. {
  458. struct qeth_card *card = dev_get_drvdata(dev);
  459. return qeth_dev_blkt_store(card, buf, count,
  460. &card->info.blkt.inter_packet, 1000);
  461. }
  462. static DEVICE_ATTR(inter, 0644, qeth_dev_blkt_inter_show,
  463. qeth_dev_blkt_inter_store);
  464. static ssize_t qeth_dev_blkt_inter_jumbo_show(struct device *dev,
  465. struct device_attribute *attr, char *buf)
  466. {
  467. struct qeth_card *card = dev_get_drvdata(dev);
  468. return qeth_dev_blkt_show(buf, card,
  469. card->info.blkt.inter_packet_jumbo);
  470. }
  471. static ssize_t qeth_dev_blkt_inter_jumbo_store(struct device *dev,
  472. struct device_attribute *attr, const char *buf, size_t count)
  473. {
  474. struct qeth_card *card = dev_get_drvdata(dev);
  475. return qeth_dev_blkt_store(card, buf, count,
  476. &card->info.blkt.inter_packet_jumbo, 1000);
  477. }
  478. static DEVICE_ATTR(inter_jumbo, 0644, qeth_dev_blkt_inter_jumbo_show,
  479. qeth_dev_blkt_inter_jumbo_store);
  480. static struct attribute *qeth_blkt_device_attrs[] = {
  481. &dev_attr_total.attr,
  482. &dev_attr_inter.attr,
  483. &dev_attr_inter_jumbo.attr,
  484. NULL,
  485. };
  486. static struct attribute_group qeth_device_blkt_group = {
  487. .name = "blkt",
  488. .attrs = qeth_blkt_device_attrs,
  489. };
  490. static struct attribute *qeth_device_attrs[] = {
  491. &dev_attr_state.attr,
  492. &dev_attr_chpid.attr,
  493. &dev_attr_if_name.attr,
  494. &dev_attr_card_type.attr,
  495. &dev_attr_inbuf_size.attr,
  496. &dev_attr_portno.attr,
  497. &dev_attr_portname.attr,
  498. &dev_attr_priority_queueing.attr,
  499. &dev_attr_buffer_count.attr,
  500. &dev_attr_recover.attr,
  501. &dev_attr_performance_stats.attr,
  502. &dev_attr_layer2.attr,
  503. &dev_attr_isolation.attr,
  504. NULL,
  505. };
  506. static struct attribute_group qeth_device_attr_group = {
  507. .attrs = qeth_device_attrs,
  508. };
  509. static struct attribute *qeth_osn_device_attrs[] = {
  510. &dev_attr_state.attr,
  511. &dev_attr_chpid.attr,
  512. &dev_attr_if_name.attr,
  513. &dev_attr_card_type.attr,
  514. &dev_attr_buffer_count.attr,
  515. &dev_attr_recover.attr,
  516. NULL,
  517. };
  518. static struct attribute_group qeth_osn_device_attr_group = {
  519. .attrs = qeth_osn_device_attrs,
  520. };
  521. int qeth_core_create_device_attributes(struct device *dev)
  522. {
  523. int ret;
  524. ret = sysfs_create_group(&dev->kobj, &qeth_device_attr_group);
  525. if (ret)
  526. return ret;
  527. ret = sysfs_create_group(&dev->kobj, &qeth_device_blkt_group);
  528. if (ret)
  529. sysfs_remove_group(&dev->kobj, &qeth_device_attr_group);
  530. return 0;
  531. }
  532. void qeth_core_remove_device_attributes(struct device *dev)
  533. {
  534. sysfs_remove_group(&dev->kobj, &qeth_device_attr_group);
  535. sysfs_remove_group(&dev->kobj, &qeth_device_blkt_group);
  536. }
  537. int qeth_core_create_osn_attributes(struct device *dev)
  538. {
  539. return sysfs_create_group(&dev->kobj, &qeth_osn_device_attr_group);
  540. }
  541. void qeth_core_remove_osn_attributes(struct device *dev)
  542. {
  543. sysfs_remove_group(&dev->kobj, &qeth_osn_device_attr_group);
  544. return;
  545. }