From b57634863a38700b3b26e3d4a9d47a4a6c61eef1 Mon Sep 17 00:00:00 2001 From: Dan Carpenter Date: Fri, 17 Mar 2023 08:00:27 +0000 Subject: [PATCH] mm/mempolicy.c: fix out of bounds write in mpol_parse_str() stable inclusion from stable-v4.19.102 commit 732ecd4aad51d336b49b9be431219d173ac826c8 category: bugfix bugzilla: https://gitee.com/openeuler/kernel/issues/I6L0EC CVE: NA Reference: https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git/commit/?id=732ecd4aad51d336b49b9be431219d173ac826c8 -------------------------------- commit c7a91bc7c2e17e0a9c8b9745a2cb118891218fd1 upstream. What we are trying to do is change the '=' character to a NUL terminator and then at the end of the function we restore it back to an '='. The problem is there are two error paths where we jump to the end of the function before we have replaced the '=' with NUL. We end up putting the '=' in the wrong place (possibly one element before the start of the buffer). Link: http://lkml.kernel.org/r/20200115055426.vdjwvry44nfug7yy@kili.mountain Reported-by: syzbot+e64a13c5369a194d67df@syzkaller.appspotmail.com Fixes: 095f1fc4ebf3 ("mempolicy: rework shmem mpol parsing and display") Signed-off-by: Dan Carpenter Acked-by: Vlastimil Babka Dmitry Vyukov Cc: Michal Hocko Cc: Dan Carpenter Cc: Lee Schermerhorn Cc: Andrea Arcangeli Cc: Hugh Dickins Cc: Signed-off-by: Andrew Morton Signed-off-by: Linus Torvalds Signed-off-by: Greg Kroah-Hartman Signed-off-by: ZhangPeng Reviewed-by: Kefeng Wang Reviewed-by: tong tiangen Signed-off-by: Yongqiang Liu --- mm/mempolicy.c | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/mm/mempolicy.c b/mm/mempolicy.c index 4cac46d56f38..4769ed2ed7f3 100644 --- a/mm/mempolicy.c +++ b/mm/mempolicy.c @@ -2886,6 +2886,9 @@ int mpol_parse_str(char *str, struct mempolicy **mpol) char *flags = strchr(str, '='); int err = 1; + if (flags) + *flags++ = '\0'; /* terminate mode string */ + if (nodelist) { /* NUL-terminate mode or flags string */ *nodelist++ = '\0'; @@ -2896,9 +2899,6 @@ int mpol_parse_str(char *str, struct mempolicy **mpol) } else nodes_clear(nodes); - if (flags) - *flags++ = '\0'; /* terminate mode string */ - for (mode = 0; mode < MPOL_MAX; mode++) { if (!strcmp(str, policy_modes[mode])) { break; -- GitLab