cmm.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481
  1. /*
  2. * arch/s390/mm/cmm.c
  3. *
  4. * S390 version
  5. * Copyright (C) 2003 IBM Deutschland Entwicklung GmbH, IBM Corporation
  6. * Author(s): Martin Schwidefsky (schwidefsky@de.ibm.com)
  7. *
  8. * Collaborative memory management interface.
  9. */
  10. #include <linux/errno.h>
  11. #include <linux/fs.h>
  12. #include <linux/init.h>
  13. #include <linux/module.h>
  14. #include <linux/sched.h>
  15. #include <linux/sysctl.h>
  16. #include <linux/ctype.h>
  17. #include <linux/swap.h>
  18. #include <linux/kthread.h>
  19. #include <asm/pgalloc.h>
  20. #include <asm/uaccess.h>
  21. #include <asm/diag.h>
  22. static char *sender = "VMRMSVM";
  23. module_param(sender, charp, 0400);
  24. MODULE_PARM_DESC(sender,
  25. "Guest name that may send SMSG messages (default VMRMSVM)");
  26. #include "../../../drivers/s390/net/smsgiucv.h"
  27. #define CMM_NR_PAGES ((PAGE_SIZE / sizeof(unsigned long)) - 2)
  28. struct cmm_page_array {
  29. struct cmm_page_array *next;
  30. unsigned long index;
  31. unsigned long pages[CMM_NR_PAGES];
  32. };
  33. static long cmm_pages;
  34. static long cmm_timed_pages;
  35. static volatile long cmm_pages_target;
  36. static volatile long cmm_timed_pages_target;
  37. static long cmm_timeout_pages;
  38. static long cmm_timeout_seconds;
  39. static struct cmm_page_array *cmm_page_list;
  40. static struct cmm_page_array *cmm_timed_page_list;
  41. static DEFINE_SPINLOCK(cmm_lock);
  42. static struct task_struct *cmm_thread_ptr;
  43. static wait_queue_head_t cmm_thread_wait;
  44. static struct timer_list cmm_timer;
  45. static void cmm_timer_fn(unsigned long);
  46. static void cmm_set_timer(void);
  47. static long
  48. cmm_alloc_pages(long nr, long *counter, struct cmm_page_array **list)
  49. {
  50. struct cmm_page_array *pa, *npa;
  51. unsigned long addr;
  52. while (nr) {
  53. addr = __get_free_page(GFP_NOIO);
  54. if (!addr)
  55. break;
  56. spin_lock(&cmm_lock);
  57. pa = *list;
  58. if (!pa || pa->index >= CMM_NR_PAGES) {
  59. /* Need a new page for the page list. */
  60. spin_unlock(&cmm_lock);
  61. npa = (struct cmm_page_array *)
  62. __get_free_page(GFP_NOIO);
  63. if (!npa) {
  64. free_page(addr);
  65. break;
  66. }
  67. spin_lock(&cmm_lock);
  68. pa = *list;
  69. if (!pa || pa->index >= CMM_NR_PAGES) {
  70. npa->next = pa;
  71. npa->index = 0;
  72. pa = npa;
  73. *list = pa;
  74. } else
  75. free_page((unsigned long) npa);
  76. }
  77. diag10(addr);
  78. pa->pages[pa->index++] = addr;
  79. (*counter)++;
  80. spin_unlock(&cmm_lock);
  81. nr--;
  82. }
  83. return nr;
  84. }
  85. static long
  86. cmm_free_pages(long nr, long *counter, struct cmm_page_array **list)
  87. {
  88. struct cmm_page_array *pa;
  89. unsigned long addr;
  90. spin_lock(&cmm_lock);
  91. pa = *list;
  92. while (nr) {
  93. if (!pa || pa->index <= 0)
  94. break;
  95. addr = pa->pages[--pa->index];
  96. if (pa->index == 0) {
  97. pa = pa->next;
  98. free_page((unsigned long) *list);
  99. *list = pa;
  100. }
  101. free_page(addr);
  102. (*counter)--;
  103. nr--;
  104. }
  105. spin_unlock(&cmm_lock);
  106. return nr;
  107. }
  108. static int cmm_oom_notify(struct notifier_block *self,
  109. unsigned long dummy, void *parm)
  110. {
  111. unsigned long *freed = parm;
  112. long nr = 256;
  113. nr = cmm_free_pages(nr, &cmm_timed_pages, &cmm_timed_page_list);
  114. if (nr > 0)
  115. nr = cmm_free_pages(nr, &cmm_pages, &cmm_page_list);
  116. cmm_pages_target = cmm_pages;
  117. cmm_timed_pages_target = cmm_timed_pages;
  118. *freed += 256 - nr;
  119. return NOTIFY_OK;
  120. }
  121. static struct notifier_block cmm_oom_nb = {
  122. .notifier_call = cmm_oom_notify
  123. };
  124. static int
  125. cmm_thread(void *dummy)
  126. {
  127. int rc;
  128. while (1) {
  129. rc = wait_event_interruptible(cmm_thread_wait,
  130. (cmm_pages != cmm_pages_target ||
  131. cmm_timed_pages != cmm_timed_pages_target ||
  132. kthread_should_stop()));
  133. if (kthread_should_stop() || rc == -ERESTARTSYS) {
  134. cmm_pages_target = cmm_pages;
  135. cmm_timed_pages_target = cmm_timed_pages;
  136. break;
  137. }
  138. if (cmm_pages_target > cmm_pages) {
  139. if (cmm_alloc_pages(1, &cmm_pages, &cmm_page_list))
  140. cmm_pages_target = cmm_pages;
  141. } else if (cmm_pages_target < cmm_pages) {
  142. cmm_free_pages(1, &cmm_pages, &cmm_page_list);
  143. }
  144. if (cmm_timed_pages_target > cmm_timed_pages) {
  145. if (cmm_alloc_pages(1, &cmm_timed_pages,
  146. &cmm_timed_page_list))
  147. cmm_timed_pages_target = cmm_timed_pages;
  148. } else if (cmm_timed_pages_target < cmm_timed_pages) {
  149. cmm_free_pages(1, &cmm_timed_pages,
  150. &cmm_timed_page_list);
  151. }
  152. if (cmm_timed_pages > 0 && !timer_pending(&cmm_timer))
  153. cmm_set_timer();
  154. }
  155. return 0;
  156. }
  157. static void
  158. cmm_kick_thread(void)
  159. {
  160. wake_up(&cmm_thread_wait);
  161. }
  162. static void
  163. cmm_set_timer(void)
  164. {
  165. if (cmm_timed_pages_target <= 0 || cmm_timeout_seconds <= 0) {
  166. if (timer_pending(&cmm_timer))
  167. del_timer(&cmm_timer);
  168. return;
  169. }
  170. if (timer_pending(&cmm_timer)) {
  171. if (mod_timer(&cmm_timer, jiffies + cmm_timeout_seconds*HZ))
  172. return;
  173. }
  174. cmm_timer.function = cmm_timer_fn;
  175. cmm_timer.data = 0;
  176. cmm_timer.expires = jiffies + cmm_timeout_seconds*HZ;
  177. add_timer(&cmm_timer);
  178. }
  179. static void
  180. cmm_timer_fn(unsigned long ignored)
  181. {
  182. long nr;
  183. nr = cmm_timed_pages_target - cmm_timeout_pages;
  184. if (nr < 0)
  185. cmm_timed_pages_target = 0;
  186. else
  187. cmm_timed_pages_target = nr;
  188. cmm_kick_thread();
  189. cmm_set_timer();
  190. }
  191. void
  192. cmm_set_pages(long nr)
  193. {
  194. cmm_pages_target = nr;
  195. cmm_kick_thread();
  196. }
  197. long
  198. cmm_get_pages(void)
  199. {
  200. return cmm_pages;
  201. }
  202. void
  203. cmm_add_timed_pages(long nr)
  204. {
  205. cmm_timed_pages_target += nr;
  206. cmm_kick_thread();
  207. }
  208. long
  209. cmm_get_timed_pages(void)
  210. {
  211. return cmm_timed_pages;
  212. }
  213. void
  214. cmm_set_timeout(long nr, long seconds)
  215. {
  216. cmm_timeout_pages = nr;
  217. cmm_timeout_seconds = seconds;
  218. cmm_set_timer();
  219. }
  220. static int
  221. cmm_skip_blanks(char *cp, char **endp)
  222. {
  223. char *str;
  224. for (str = cp; *str == ' ' || *str == '\t'; str++);
  225. *endp = str;
  226. return str != cp;
  227. }
  228. #ifdef CONFIG_CMM_PROC
  229. static struct ctl_table cmm_table[];
  230. static int
  231. cmm_pages_handler(ctl_table *ctl, int write, struct file *filp,
  232. void __user *buffer, size_t *lenp, loff_t *ppos)
  233. {
  234. char buf[16], *p;
  235. long nr;
  236. int len;
  237. if (!*lenp || (*ppos && !write)) {
  238. *lenp = 0;
  239. return 0;
  240. }
  241. if (write) {
  242. len = *lenp;
  243. if (copy_from_user(buf, buffer,
  244. len > sizeof(buf) ? sizeof(buf) : len))
  245. return -EFAULT;
  246. buf[sizeof(buf) - 1] = '\0';
  247. cmm_skip_blanks(buf, &p);
  248. nr = simple_strtoul(p, &p, 0);
  249. if (ctl == &cmm_table[0])
  250. cmm_set_pages(nr);
  251. else
  252. cmm_add_timed_pages(nr);
  253. } else {
  254. if (ctl == &cmm_table[0])
  255. nr = cmm_get_pages();
  256. else
  257. nr = cmm_get_timed_pages();
  258. len = sprintf(buf, "%ld\n", nr);
  259. if (len > *lenp)
  260. len = *lenp;
  261. if (copy_to_user(buffer, buf, len))
  262. return -EFAULT;
  263. }
  264. *lenp = len;
  265. *ppos += len;
  266. return 0;
  267. }
  268. static int
  269. cmm_timeout_handler(ctl_table *ctl, int write, struct file *filp,
  270. void __user *buffer, size_t *lenp, loff_t *ppos)
  271. {
  272. char buf[64], *p;
  273. long nr, seconds;
  274. int len;
  275. if (!*lenp || (*ppos && !write)) {
  276. *lenp = 0;
  277. return 0;
  278. }
  279. if (write) {
  280. len = *lenp;
  281. if (copy_from_user(buf, buffer,
  282. len > sizeof(buf) ? sizeof(buf) : len))
  283. return -EFAULT;
  284. buf[sizeof(buf) - 1] = '\0';
  285. cmm_skip_blanks(buf, &p);
  286. nr = simple_strtoul(p, &p, 0);
  287. cmm_skip_blanks(p, &p);
  288. seconds = simple_strtoul(p, &p, 0);
  289. cmm_set_timeout(nr, seconds);
  290. } else {
  291. len = sprintf(buf, "%ld %ld\n",
  292. cmm_timeout_pages, cmm_timeout_seconds);
  293. if (len > *lenp)
  294. len = *lenp;
  295. if (copy_to_user(buffer, buf, len))
  296. return -EFAULT;
  297. }
  298. *lenp = len;
  299. *ppos += len;
  300. return 0;
  301. }
  302. static struct ctl_table cmm_table[] = {
  303. {
  304. .ctl_name = VM_CMM_PAGES,
  305. .procname = "cmm_pages",
  306. .mode = 0644,
  307. .proc_handler = &cmm_pages_handler,
  308. },
  309. {
  310. .ctl_name = VM_CMM_TIMED_PAGES,
  311. .procname = "cmm_timed_pages",
  312. .mode = 0644,
  313. .proc_handler = &cmm_pages_handler,
  314. },
  315. {
  316. .ctl_name = VM_CMM_TIMEOUT,
  317. .procname = "cmm_timeout",
  318. .mode = 0644,
  319. .proc_handler = &cmm_timeout_handler,
  320. },
  321. { .ctl_name = 0 }
  322. };
  323. static struct ctl_table cmm_dir_table[] = {
  324. {
  325. .ctl_name = CTL_VM,
  326. .procname = "vm",
  327. .maxlen = 0,
  328. .mode = 0555,
  329. .child = cmm_table,
  330. },
  331. { .ctl_name = 0 }
  332. };
  333. #endif
  334. #ifdef CONFIG_CMM_IUCV
  335. #define SMSG_PREFIX "CMM"
  336. static void
  337. cmm_smsg_target(char *from, char *msg)
  338. {
  339. long nr, seconds;
  340. if (strlen(sender) > 0 && strcmp(from, sender) != 0)
  341. return;
  342. if (!cmm_skip_blanks(msg + strlen(SMSG_PREFIX), &msg))
  343. return;
  344. if (strncmp(msg, "SHRINK", 6) == 0) {
  345. if (!cmm_skip_blanks(msg + 6, &msg))
  346. return;
  347. nr = simple_strtoul(msg, &msg, 0);
  348. cmm_skip_blanks(msg, &msg);
  349. if (*msg == '\0')
  350. cmm_set_pages(nr);
  351. } else if (strncmp(msg, "RELEASE", 7) == 0) {
  352. if (!cmm_skip_blanks(msg + 7, &msg))
  353. return;
  354. nr = simple_strtoul(msg, &msg, 0);
  355. cmm_skip_blanks(msg, &msg);
  356. if (*msg == '\0')
  357. cmm_add_timed_pages(nr);
  358. } else if (strncmp(msg, "REUSE", 5) == 0) {
  359. if (!cmm_skip_blanks(msg + 5, &msg))
  360. return;
  361. nr = simple_strtoul(msg, &msg, 0);
  362. if (!cmm_skip_blanks(msg, &msg))
  363. return;
  364. seconds = simple_strtoul(msg, &msg, 0);
  365. cmm_skip_blanks(msg, &msg);
  366. if (*msg == '\0')
  367. cmm_set_timeout(nr, seconds);
  368. }
  369. }
  370. #endif
  371. static struct ctl_table_header *cmm_sysctl_header;
  372. static int
  373. cmm_init (void)
  374. {
  375. int rc = -ENOMEM;
  376. #ifdef CONFIG_CMM_PROC
  377. cmm_sysctl_header = register_sysctl_table(cmm_dir_table);
  378. if (!cmm_sysctl_header)
  379. goto out;
  380. #endif
  381. #ifdef CONFIG_CMM_IUCV
  382. rc = smsg_register_callback(SMSG_PREFIX, cmm_smsg_target);
  383. if (rc < 0)
  384. goto out_smsg;
  385. #endif
  386. rc = register_oom_notifier(&cmm_oom_nb);
  387. if (rc < 0)
  388. goto out_oom_notify;
  389. init_waitqueue_head(&cmm_thread_wait);
  390. init_timer(&cmm_timer);
  391. cmm_thread_ptr = kthread_run(cmm_thread, NULL, "cmmthread");
  392. rc = IS_ERR(cmm_thread_ptr) ? PTR_ERR(cmm_thread_ptr) : 0;
  393. if (!rc)
  394. goto out;
  395. /*
  396. * kthread_create failed. undo all the stuff from above again.
  397. */
  398. unregister_oom_notifier(&cmm_oom_nb);
  399. out_oom_notify:
  400. #ifdef CONFIG_CMM_IUCV
  401. smsg_unregister_callback(SMSG_PREFIX, cmm_smsg_target);
  402. out_smsg:
  403. #endif
  404. #ifdef CONFIG_CMM_PROC
  405. unregister_sysctl_table(cmm_sysctl_header);
  406. #endif
  407. out:
  408. return rc;
  409. }
  410. static void
  411. cmm_exit(void)
  412. {
  413. kthread_stop(cmm_thread_ptr);
  414. unregister_oom_notifier(&cmm_oom_nb);
  415. cmm_free_pages(cmm_pages, &cmm_pages, &cmm_page_list);
  416. cmm_free_pages(cmm_timed_pages, &cmm_timed_pages, &cmm_timed_page_list);
  417. #ifdef CONFIG_CMM_PROC
  418. unregister_sysctl_table(cmm_sysctl_header);
  419. #endif
  420. #ifdef CONFIG_CMM_IUCV
  421. smsg_unregister_callback(SMSG_PREFIX, cmm_smsg_target);
  422. #endif
  423. }
  424. module_init(cmm_init);
  425. module_exit(cmm_exit);
  426. EXPORT_SYMBOL(cmm_set_pages);
  427. EXPORT_SYMBOL(cmm_get_pages);
  428. EXPORT_SYMBOL(cmm_add_timed_pages);
  429. EXPORT_SYMBOL(cmm_get_timed_pages);
  430. EXPORT_SYMBOL(cmm_set_timeout);
  431. MODULE_LICENSE("GPL");