|
Message-ID: <79bac827-4092-8a4d-9dc6-6019419b2486@ssi.gouv.fr> Date: Thu, 27 Jun 2019 18:18:12 +0200 From: Mickaël Salaün <mickael.salaun@....gouv.fr> To: Al Viro <viro@...iv.linux.org.uk>, Mickaël Salaün <mic@...ikod.net> CC: <linux-kernel@...r.kernel.org>, Aleksa Sarai <cyphar@...har.com>, Alexei Starovoitov <ast@...nel.org>, Andrew Morton <akpm@...ux-foundation.org>, Andy Lutomirski <luto@...capital.net>, Arnaldo Carvalho de Melo <acme@...nel.org>, Casey Schaufler <casey@...aufler-ca.com>, Daniel Borkmann <daniel@...earbox.net>, David Drysdale <drysdale@...gle.com>, "David S . Miller" <davem@...emloft.net>, "Eric W . Biederman" <ebiederm@...ssion.com>, James Morris <jmorris@...ei.org>, Jann Horn <jann@...jh.net>, John Johansen <john.johansen@...onical.com>, Jonathan Corbet <corbet@....net>, Kees Cook <keescook@...omium.org>, Michael Kerrisk <mtk.manpages@...il.com>, Paul Moore <paul@...l-moore.com>, Sargun Dhillon <sargun@...gun.me>, "Serge E . Hallyn" <serge@...lyn.com>, Shuah Khan <shuah@...nel.org>, Stephen Smalley <sds@...ho.nsa.gov>, Tejun Heo <tj@...nel.org>, Tetsuo Handa <penguin-kernel@...ove.sakura.ne.jp>, Thomas Graf <tgraf@...g.ch>, Tycho Andersen <tycho@...ho.ws>, Will Drewry <wad@...omium.org>, <kernel-hardening@...ts.openwall.com>, <linux-api@...r.kernel.org>, <linux-fsdevel@...r.kernel.org>, <linux-security-module@...r.kernel.org>, <netdev@...r.kernel.org> Subject: Re: [PATCH bpf-next v9 05/10] bpf,landlock: Add a new map type: inode On 26/06/2019 00:52, Al Viro wrote: > On Tue, Jun 25, 2019 at 11:52:34PM +0200, Mickaël Salaün wrote: >> +/* must call iput(inode) after this call */ >> +static struct inode *inode_from_fd(int ufd, bool check_access) >> +{ >> + struct inode *ret; >> + struct fd f; >> + int deny; >> + >> + f = fdget(ufd); >> + if (unlikely(!f.file || !file_inode(f.file))) { >> + ret = ERR_PTR(-EBADF); >> + goto put_fd; >> + } > > Just when does one get a NULL file_inode()? The reason I'm asking is > that arseloads of code would break if one managed to create such > a beast... I didn't find any API documentation about this guarantee, so I followed a defensive programming approach. I'll remove the file_inode() check. > > Incidentally, that should be return ERR_PTR(-EBADF); fdput() is wrong there. Right, I'll fix that. > >> + } >> + /* check if the FD is tied to a mount point */ >> + /* TODO: add this check when called from an eBPF program too */ >> + if (unlikely(!f.file->f_path.mnt > > Again, the same question - when the hell can that happen? Defensive programming again, I'll remove it. > If you are > sitting on an exploitable roothole, do share it... > > || f.file->f_path.mnt->mnt_flags & >> + MNT_INTERNAL)) { >> + ret = ERR_PTR(-EINVAL); >> + goto put_fd; > > What does it have to do with mountpoints, anyway? I want to only manage inodes tied to a userspace-visible file system (this check may not be enough though). It doesn't make sense to be able to add inodes which are not mounted, to this kind of map. > >> +/* called from syscall */ >> +static int sys_inode_map_delete_elem(struct bpf_map *map, struct inode *key) >> +{ >> + struct inode_array *array = container_of(map, struct inode_array, map); >> + struct inode *inode; >> + int i; >> + >> + WARN_ON_ONCE(!rcu_read_lock_held()); >> + for (i = 0; i < array->map.max_entries; i++) { >> + if (array->elems[i].inode == key) { >> + inode = xchg(&array->elems[i].inode, NULL); >> + array->nb_entries--; > > Umm... Is that intended to be atomic in any sense? nb_entries is not used as a bound check but to avoid walking uselessly through the (pre-allocated) array when adding a new element, but I'll use an atomic to avoid inconsistencies anyway. > >> + iput(inode); >> + return 0; >> + } >> + } >> + return -ENOENT; >> +} >> + >> +/* called from syscall */ >> +int bpf_inode_map_delete_elem(struct bpf_map *map, int *key) >> +{ >> + struct inode *inode; >> + int err; >> + >> + inode = inode_from_fd(*key, false); >> + if (IS_ERR(inode)) >> + return PTR_ERR(inode); >> + err = sys_inode_map_delete_elem(map, inode); >> + iput(inode); >> + return err; >> +} > > Wait a sec... So we have those beasties that can have long-term > references to arbitrary inodes stuck in them? What will happen > if you get umount(2) called while such a thing exists? I though an umount would be denied but no, we get a self-destructed busy inode and a bug! What about wrapping the inode's superblock->s_op->destroy_inode() to first remove the element from the map and then call the real destroy_inode(), if any? Or I could update fs/inode.c:destroy_inode() to call inode->free_inode() if it is set, and set it when such inode is referenced by a map? Or maybe I could hold the referencing file in the map and then wrap its f_op? -- Mickaël Salaün ANSSI/SDE/ST/LAM Les données à caractère personnel recueillies et traitées dans le cadre de cet échange, le sont à seule fin d’exécution d’une relation professionnelle et s’opèrent dans cette seule finalité et pour la durée nécessaire à cette relation. Si vous souhaitez faire usage de vos droits de consultation, de rectification et de suppression de vos données, veuillez contacter contact.rgpd@...sn.gouv.fr. Si vous avez reçu ce message par erreur, nous vous remercions d’en informer l’expéditeur et de détruire le message. The personal data collected and processed during this exchange aims solely at completing a business relationship and is limited to the necessary duration of that relationship. If you wish to use your rights of consultation, rectification and deletion of your data, please contact: contact.rgpd@...sn.gouv.fr. If you have received this message in error, we thank you for informing the sender and destroying the message.
Powered by blists - more mailing lists
Confused about mailing lists and their use? Read about mailing lists on Wikipedia and check out these guidelines on proper formatting of your messages.