qeth_core_sys.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651
  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;
  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. PRINT_WARN("portno 0x%X is out of range\n", portno);
  110. return -EINVAL;
  111. }
  112. card->info.portno = portno;
  113. return count;
  114. }
  115. static DEVICE_ATTR(portno, 0644, qeth_dev_portno_show, qeth_dev_portno_store);
  116. static ssize_t qeth_dev_portname_show(struct device *dev,
  117. struct device_attribute *attr, char *buf)
  118. {
  119. struct qeth_card *card = dev_get_drvdata(dev);
  120. char portname[9] = {0, };
  121. if (!card)
  122. return -EINVAL;
  123. if (card->info.portname_required) {
  124. memcpy(portname, card->info.portname + 1, 8);
  125. EBCASC(portname, 8);
  126. return sprintf(buf, "%s\n", portname);
  127. } else
  128. return sprintf(buf, "no portname required\n");
  129. }
  130. static ssize_t qeth_dev_portname_store(struct device *dev,
  131. struct device_attribute *attr, const char *buf, size_t count)
  132. {
  133. struct qeth_card *card = dev_get_drvdata(dev);
  134. char *tmp;
  135. int i;
  136. if (!card)
  137. return -EINVAL;
  138. if ((card->state != CARD_STATE_DOWN) &&
  139. (card->state != CARD_STATE_RECOVER))
  140. return -EPERM;
  141. tmp = strsep((char **) &buf, "\n");
  142. if ((strlen(tmp) > 8) || (strlen(tmp) == 0))
  143. return -EINVAL;
  144. card->info.portname[0] = strlen(tmp);
  145. /* for beauty reasons */
  146. for (i = 1; i < 9; i++)
  147. card->info.portname[i] = ' ';
  148. strcpy(card->info.portname + 1, tmp);
  149. ASCEBC(card->info.portname + 1, 8);
  150. return count;
  151. }
  152. static DEVICE_ATTR(portname, 0644, qeth_dev_portname_show,
  153. qeth_dev_portname_store);
  154. static ssize_t qeth_dev_prioqing_show(struct device *dev,
  155. struct device_attribute *attr, char *buf)
  156. {
  157. struct qeth_card *card = dev_get_drvdata(dev);
  158. if (!card)
  159. return -EINVAL;
  160. switch (card->qdio.do_prio_queueing) {
  161. case QETH_PRIO_Q_ING_PREC:
  162. return sprintf(buf, "%s\n", "by precedence");
  163. case QETH_PRIO_Q_ING_TOS:
  164. return sprintf(buf, "%s\n", "by type of service");
  165. default:
  166. return sprintf(buf, "always queue %i\n",
  167. card->qdio.default_out_queue);
  168. }
  169. }
  170. static ssize_t qeth_dev_prioqing_store(struct device *dev,
  171. struct device_attribute *attr, const char *buf, size_t count)
  172. {
  173. struct qeth_card *card = dev_get_drvdata(dev);
  174. char *tmp;
  175. if (!card)
  176. return -EINVAL;
  177. if ((card->state != CARD_STATE_DOWN) &&
  178. (card->state != CARD_STATE_RECOVER))
  179. return -EPERM;
  180. /* check if 1920 devices are supported ,
  181. * if though we have to permit priority queueing
  182. */
  183. if (card->qdio.no_out_queues == 1) {
  184. PRINT_WARN("Priority queueing disabled due "
  185. "to hardware limitations!\n");
  186. card->qdio.do_prio_queueing = QETH_PRIOQ_DEFAULT;
  187. return -EPERM;
  188. }
  189. tmp = strsep((char **) &buf, "\n");
  190. if (!strcmp(tmp, "prio_queueing_prec"))
  191. card->qdio.do_prio_queueing = QETH_PRIO_Q_ING_PREC;
  192. else if (!strcmp(tmp, "prio_queueing_tos"))
  193. card->qdio.do_prio_queueing = QETH_PRIO_Q_ING_TOS;
  194. else if (!strcmp(tmp, "no_prio_queueing:0")) {
  195. card->qdio.do_prio_queueing = QETH_NO_PRIO_QUEUEING;
  196. card->qdio.default_out_queue = 0;
  197. } else if (!strcmp(tmp, "no_prio_queueing:1")) {
  198. card->qdio.do_prio_queueing = QETH_NO_PRIO_QUEUEING;
  199. card->qdio.default_out_queue = 1;
  200. } else if (!strcmp(tmp, "no_prio_queueing:2")) {
  201. card->qdio.do_prio_queueing = QETH_NO_PRIO_QUEUEING;
  202. card->qdio.default_out_queue = 2;
  203. } else if (!strcmp(tmp, "no_prio_queueing:3")) {
  204. card->qdio.do_prio_queueing = QETH_NO_PRIO_QUEUEING;
  205. card->qdio.default_out_queue = 3;
  206. } else if (!strcmp(tmp, "no_prio_queueing")) {
  207. card->qdio.do_prio_queueing = QETH_NO_PRIO_QUEUEING;
  208. card->qdio.default_out_queue = QETH_DEFAULT_QUEUE;
  209. } else {
  210. PRINT_WARN("Unknown queueing type '%s'\n", tmp);
  211. return -EINVAL;
  212. }
  213. return count;
  214. }
  215. static DEVICE_ATTR(priority_queueing, 0644, qeth_dev_prioqing_show,
  216. qeth_dev_prioqing_store);
  217. static ssize_t qeth_dev_bufcnt_show(struct device *dev,
  218. struct device_attribute *attr, char *buf)
  219. {
  220. struct qeth_card *card = dev_get_drvdata(dev);
  221. if (!card)
  222. return -EINVAL;
  223. return sprintf(buf, "%i\n", card->qdio.in_buf_pool.buf_count);
  224. }
  225. static ssize_t qeth_dev_bufcnt_store(struct device *dev,
  226. struct device_attribute *attr, const char *buf, size_t count)
  227. {
  228. struct qeth_card *card = dev_get_drvdata(dev);
  229. char *tmp;
  230. int cnt, old_cnt;
  231. int rc;
  232. if (!card)
  233. return -EINVAL;
  234. if ((card->state != CARD_STATE_DOWN) &&
  235. (card->state != CARD_STATE_RECOVER))
  236. return -EPERM;
  237. old_cnt = card->qdio.in_buf_pool.buf_count;
  238. cnt = simple_strtoul(buf, &tmp, 10);
  239. cnt = (cnt < QETH_IN_BUF_COUNT_MIN) ? QETH_IN_BUF_COUNT_MIN :
  240. ((cnt > QETH_IN_BUF_COUNT_MAX) ? QETH_IN_BUF_COUNT_MAX : cnt);
  241. if (old_cnt != cnt) {
  242. rc = qeth_realloc_buffer_pool(card, cnt);
  243. if (rc)
  244. PRINT_WARN("Error (%d) while setting "
  245. "buffer count.\n", rc);
  246. }
  247. return count;
  248. }
  249. static DEVICE_ATTR(buffer_count, 0644, qeth_dev_bufcnt_show,
  250. qeth_dev_bufcnt_store);
  251. static ssize_t qeth_dev_recover_store(struct device *dev,
  252. struct device_attribute *attr, const char *buf, size_t count)
  253. {
  254. struct qeth_card *card = dev_get_drvdata(dev);
  255. char *tmp;
  256. int i;
  257. if (!card)
  258. return -EINVAL;
  259. if (card->state != CARD_STATE_UP)
  260. return -EPERM;
  261. i = simple_strtoul(buf, &tmp, 16);
  262. if (i == 1)
  263. qeth_schedule_recovery(card);
  264. return count;
  265. }
  266. static DEVICE_ATTR(recover, 0200, NULL, qeth_dev_recover_store);
  267. static ssize_t qeth_dev_performance_stats_show(struct device *dev,
  268. struct device_attribute *attr, char *buf)
  269. {
  270. struct qeth_card *card = dev_get_drvdata(dev);
  271. if (!card)
  272. return -EINVAL;
  273. return sprintf(buf, "%i\n", card->options.performance_stats ? 1:0);
  274. }
  275. static ssize_t qeth_dev_performance_stats_store(struct device *dev,
  276. struct device_attribute *attr, const char *buf, size_t count)
  277. {
  278. struct qeth_card *card = dev_get_drvdata(dev);
  279. char *tmp;
  280. int i;
  281. if (!card)
  282. return -EINVAL;
  283. i = simple_strtoul(buf, &tmp, 16);
  284. if ((i == 0) || (i == 1)) {
  285. if (i == card->options.performance_stats)
  286. return count;
  287. card->options.performance_stats = i;
  288. if (i == 0)
  289. memset(&card->perf_stats, 0,
  290. sizeof(struct qeth_perf_stats));
  291. card->perf_stats.initial_rx_packets = card->stats.rx_packets;
  292. card->perf_stats.initial_tx_packets = card->stats.tx_packets;
  293. } else {
  294. PRINT_WARN("performance_stats: write 0 or 1 to this file!\n");
  295. return -EINVAL;
  296. }
  297. return count;
  298. }
  299. static DEVICE_ATTR(performance_stats, 0644, qeth_dev_performance_stats_show,
  300. qeth_dev_performance_stats_store);
  301. static ssize_t qeth_dev_layer2_show(struct device *dev,
  302. struct device_attribute *attr, char *buf)
  303. {
  304. struct qeth_card *card = dev_get_drvdata(dev);
  305. if (!card)
  306. return -EINVAL;
  307. return sprintf(buf, "%i\n", card->options.layer2 ? 1:0);
  308. }
  309. static ssize_t qeth_dev_layer2_store(struct device *dev,
  310. struct device_attribute *attr, const char *buf, size_t count)
  311. {
  312. struct qeth_card *card = dev_get_drvdata(dev);
  313. char *tmp;
  314. int i, rc;
  315. enum qeth_discipline_id newdis;
  316. if (!card)
  317. return -EINVAL;
  318. if (((card->state != CARD_STATE_DOWN) &&
  319. (card->state != CARD_STATE_RECOVER)))
  320. return -EPERM;
  321. i = simple_strtoul(buf, &tmp, 16);
  322. switch (i) {
  323. case 0:
  324. newdis = QETH_DISCIPLINE_LAYER3;
  325. break;
  326. case 1:
  327. newdis = QETH_DISCIPLINE_LAYER2;
  328. break;
  329. default:
  330. PRINT_WARN("layer2: write 0 or 1 to this file!\n");
  331. return -EINVAL;
  332. }
  333. if (card->options.layer2 == newdis) {
  334. return count;
  335. } else {
  336. if (card->discipline.ccwgdriver) {
  337. card->discipline.ccwgdriver->remove(card->gdev);
  338. qeth_core_free_discipline(card);
  339. }
  340. }
  341. rc = qeth_core_load_discipline(card, newdis);
  342. if (rc)
  343. return rc;
  344. rc = card->discipline.ccwgdriver->probe(card->gdev);
  345. if (rc)
  346. return rc;
  347. return count;
  348. }
  349. static DEVICE_ATTR(layer2, 0644, qeth_dev_layer2_show,
  350. qeth_dev_layer2_store);
  351. static ssize_t qeth_dev_large_send_show(struct device *dev,
  352. struct device_attribute *attr, char *buf)
  353. {
  354. struct qeth_card *card = dev_get_drvdata(dev);
  355. if (!card)
  356. return -EINVAL;
  357. switch (card->options.large_send) {
  358. case QETH_LARGE_SEND_NO:
  359. return sprintf(buf, "%s\n", "no");
  360. case QETH_LARGE_SEND_EDDP:
  361. return sprintf(buf, "%s\n", "EDDP");
  362. case QETH_LARGE_SEND_TSO:
  363. return sprintf(buf, "%s\n", "TSO");
  364. default:
  365. return sprintf(buf, "%s\n", "N/A");
  366. }
  367. }
  368. static ssize_t qeth_dev_large_send_store(struct device *dev,
  369. struct device_attribute *attr, const char *buf, size_t count)
  370. {
  371. struct qeth_card *card = dev_get_drvdata(dev);
  372. enum qeth_large_send_types type;
  373. int rc = 0;
  374. char *tmp;
  375. if (!card)
  376. return -EINVAL;
  377. tmp = strsep((char **) &buf, "\n");
  378. if (!strcmp(tmp, "no")) {
  379. type = QETH_LARGE_SEND_NO;
  380. } else if (!strcmp(tmp, "EDDP")) {
  381. type = QETH_LARGE_SEND_EDDP;
  382. } else if (!strcmp(tmp, "TSO")) {
  383. type = QETH_LARGE_SEND_TSO;
  384. } else {
  385. PRINT_WARN("large_send: invalid mode %s!\n", tmp);
  386. return -EINVAL;
  387. }
  388. if (card->options.large_send == type)
  389. return count;
  390. rc = qeth_set_large_send(card, type);
  391. if (rc)
  392. return rc;
  393. return count;
  394. }
  395. static DEVICE_ATTR(large_send, 0644, qeth_dev_large_send_show,
  396. qeth_dev_large_send_store);
  397. static ssize_t qeth_dev_blkt_show(char *buf, struct qeth_card *card, int value)
  398. {
  399. if (!card)
  400. return -EINVAL;
  401. return sprintf(buf, "%i\n", value);
  402. }
  403. static ssize_t qeth_dev_blkt_store(struct qeth_card *card,
  404. const char *buf, size_t count, int *value, int max_value)
  405. {
  406. char *tmp;
  407. int i;
  408. if (!card)
  409. return -EINVAL;
  410. if ((card->state != CARD_STATE_DOWN) &&
  411. (card->state != CARD_STATE_RECOVER))
  412. return -EPERM;
  413. i = simple_strtoul(buf, &tmp, 10);
  414. if (i <= max_value) {
  415. *value = i;
  416. } else {
  417. PRINT_WARN("blkt total time: write values between"
  418. " 0 and %d to this file!\n", max_value);
  419. return -EINVAL;
  420. }
  421. return count;
  422. }
  423. static ssize_t qeth_dev_blkt_total_show(struct device *dev,
  424. struct device_attribute *attr, char *buf)
  425. {
  426. struct qeth_card *card = dev_get_drvdata(dev);
  427. return qeth_dev_blkt_show(buf, card, card->info.blkt.time_total);
  428. }
  429. static ssize_t qeth_dev_blkt_total_store(struct device *dev,
  430. struct device_attribute *attr, const char *buf, size_t count)
  431. {
  432. struct qeth_card *card = dev_get_drvdata(dev);
  433. return qeth_dev_blkt_store(card, buf, count,
  434. &card->info.blkt.time_total, 1000);
  435. }
  436. static DEVICE_ATTR(total, 0644, qeth_dev_blkt_total_show,
  437. qeth_dev_blkt_total_store);
  438. static ssize_t qeth_dev_blkt_inter_show(struct device *dev,
  439. struct device_attribute *attr, char *buf)
  440. {
  441. struct qeth_card *card = dev_get_drvdata(dev);
  442. return qeth_dev_blkt_show(buf, card, card->info.blkt.inter_packet);
  443. }
  444. static ssize_t qeth_dev_blkt_inter_store(struct device *dev,
  445. struct device_attribute *attr, const char *buf, size_t count)
  446. {
  447. struct qeth_card *card = dev_get_drvdata(dev);
  448. return qeth_dev_blkt_store(card, buf, count,
  449. &card->info.blkt.inter_packet, 100);
  450. }
  451. static DEVICE_ATTR(inter, 0644, qeth_dev_blkt_inter_show,
  452. qeth_dev_blkt_inter_store);
  453. static ssize_t qeth_dev_blkt_inter_jumbo_show(struct device *dev,
  454. struct device_attribute *attr, char *buf)
  455. {
  456. struct qeth_card *card = dev_get_drvdata(dev);
  457. return qeth_dev_blkt_show(buf, card,
  458. card->info.blkt.inter_packet_jumbo);
  459. }
  460. static ssize_t qeth_dev_blkt_inter_jumbo_store(struct device *dev,
  461. struct device_attribute *attr, const char *buf, size_t count)
  462. {
  463. struct qeth_card *card = dev_get_drvdata(dev);
  464. return qeth_dev_blkt_store(card, buf, count,
  465. &card->info.blkt.inter_packet_jumbo, 100);
  466. }
  467. static DEVICE_ATTR(inter_jumbo, 0644, qeth_dev_blkt_inter_jumbo_show,
  468. qeth_dev_blkt_inter_jumbo_store);
  469. static struct attribute *qeth_blkt_device_attrs[] = {
  470. &dev_attr_total.attr,
  471. &dev_attr_inter.attr,
  472. &dev_attr_inter_jumbo.attr,
  473. NULL,
  474. };
  475. static struct attribute_group qeth_device_blkt_group = {
  476. .name = "blkt",
  477. .attrs = qeth_blkt_device_attrs,
  478. };
  479. static struct attribute *qeth_device_attrs[] = {
  480. &dev_attr_state.attr,
  481. &dev_attr_chpid.attr,
  482. &dev_attr_if_name.attr,
  483. &dev_attr_card_type.attr,
  484. &dev_attr_inbuf_size.attr,
  485. &dev_attr_portno.attr,
  486. &dev_attr_portname.attr,
  487. &dev_attr_priority_queueing.attr,
  488. &dev_attr_buffer_count.attr,
  489. &dev_attr_recover.attr,
  490. &dev_attr_performance_stats.attr,
  491. &dev_attr_layer2.attr,
  492. &dev_attr_large_send.attr,
  493. NULL,
  494. };
  495. static struct attribute_group qeth_device_attr_group = {
  496. .attrs = qeth_device_attrs,
  497. };
  498. static struct attribute *qeth_osn_device_attrs[] = {
  499. &dev_attr_state.attr,
  500. &dev_attr_chpid.attr,
  501. &dev_attr_if_name.attr,
  502. &dev_attr_card_type.attr,
  503. &dev_attr_buffer_count.attr,
  504. &dev_attr_recover.attr,
  505. NULL,
  506. };
  507. static struct attribute_group qeth_osn_device_attr_group = {
  508. .attrs = qeth_osn_device_attrs,
  509. };
  510. int qeth_core_create_device_attributes(struct device *dev)
  511. {
  512. int ret;
  513. ret = sysfs_create_group(&dev->kobj, &qeth_device_attr_group);
  514. if (ret)
  515. return ret;
  516. ret = sysfs_create_group(&dev->kobj, &qeth_device_blkt_group);
  517. if (ret)
  518. sysfs_remove_group(&dev->kobj, &qeth_device_attr_group);
  519. return 0;
  520. }
  521. void qeth_core_remove_device_attributes(struct device *dev)
  522. {
  523. sysfs_remove_group(&dev->kobj, &qeth_device_attr_group);
  524. sysfs_remove_group(&dev->kobj, &qeth_device_blkt_group);
  525. }
  526. int qeth_core_create_osn_attributes(struct device *dev)
  527. {
  528. return sysfs_create_group(&dev->kobj, &qeth_osn_device_attr_group);
  529. }
  530. void qeth_core_remove_osn_attributes(struct device *dev)
  531. {
  532. sysfs_remove_group(&dev->kobj, &qeth_osn_device_attr_group);
  533. return;
  534. }