Memory handling with struct epoll_event
Asked Answered
M

4

20

I'm developing a server in C with the epoll library and I have a question as to how memory is handled for struct epoll_event. I've noticed in some online examples that, when making epoll_ctl calls, the events argument is allocated on the stack and then the pointer is passed, like so:

struct epoll_event ev;
ev.events = EPOLLIN;
epoll_ctl(epfd, EPOLL_CTL_ADD, sockfd, &ev);

Now we all know what happens to ev when the function returns. My question is: does the epoll library make copies of these values internally or does it rely on the struct you passed to be heap allocated? Will the above example totally break my reactor implementation? If so, what is the best way of keeping track of my heap allocated epoll_event structs?

Thanks for your time.

Motorbike answered 19/10, 2012 at 20:58 Comment(2)
If "we all know what happens", then why do you ask?Phaedra
When I say "we all know what happens" I mean that the stack allocated memory is released when the function returns.Motorbike
P
20

Everything is fine. The epoll_ctl function is a simple wrapper around a system call which will be entirely complete when the function returns. No further data from userspace is required. The struct is simply a way to package the arguments.

Phaedra answered 19/10, 2012 at 21:3 Comment(1)
I wish this information was in the manpage and didn't require a roundtrip to SO.Arratoon
B
13

It's absolutely fine to immediately throw away or reuse your epoll_event struct.

The kernel will copy the parameters out of the epoll_event struct.

This is exactly the same as if you used an ioctl which takes a struct as a parameter, or a socket operation (e.g. bind) which takes a struct sockaddr_in.

The kernel takes what it needs, and it's immediately ok for you to free it.

The only thing you need to worry about is the "user data", which is only relevant to you. The kernel will store it, but you need to know what it means when you get an event.

Bodycheck answered 19/10, 2012 at 21:5 Comment(1)
Thank you. Upvoted. I will mark Kerrek SB's answer correct however, because he answered about a minute before you.Motorbike
A
6

epoll is a set of syscalls, not a library. When you call the epoll syscalls you enter the kernel, and the kernel generally doesn't trust these user mode buffers to necessarily be valid or stick around, but rather copies into kernel memory via copy_from_user etc. So yes, you can set up structs on the stack, pass their addresses to the syscall, then discard them after it returns.

Arthur answered 19/10, 2012 at 21:6 Comment(0)
G
1

As all say, the memory the struct epoll_event * parameter points to can be freed or reused after epoll_ctl().

In linux/v5.8/source/fs/eventpoll.c#L2288, we see the kernel makes a copy of the struct epoll_event, that confirms what we believe.

SYSCALL_DEFINE4(epoll_ctl, int, epfd, int, op, int, fd,
        struct epoll_event __user *, event)
{
    struct epoll_event epds;

    if (ep_op_has_event(op) &&
        copy_from_user(&epds, event, sizeof(struct epoll_event)))
        return -EFAULT;

    return do_epoll_ctl(epfd, op, fd, &epds, false);
}
Gunar answered 18/8, 2020 at 4:6 Comment(2)
However, I still wonder why the parameter is not struct epoll_event const *, but without const.Haul
There is a nice explanation about that here: https://mcmap.net/q/662709/-may-epoll_ctl-modify-the-epoll_event-structure-passed-to-itMccallion

© 2022 - 2025 — McMap. All rights reserved.