< prev index next >

src/java.base/linux/classes/sun/nio/ch/EPollPoller.java

Print this page

  8  * particular file as subject to the "Classpath" exception as provided
  9  * by Oracle in the LICENSE file that accompanied this code.
 10  *
 11  * This code is distributed in the hope that it will be useful, but WITHOUT
 12  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 13  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 14  * version 2 for more details (a copy is included in the LICENSE file that
 15  * accompanied this code).
 16  *
 17  * You should have received a copy of the GNU General Public License version
 18  * 2 along with this work; if not, write to the Free Software Foundation,
 19  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 20  *
 21  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 22  * or visit www.oracle.com if you need additional information or have any
 23  * questions.
 24  */
 25 package sun.nio.ch;
 26 
 27 import java.io.IOException;


 28 import static sun.nio.ch.EPoll.*;
 29 
 30 /**
 31  * Poller implementation based on the epoll facility.
 32  */
 33 
 34 class EPollPoller extends Poller {
 35     private static final int ENOENT = 2;
 36 
 37     private final int epfd;
 38     private final int event;
 39     private final int maxEvents;
 40     private final long address;

 41 
 42     EPollPoller(boolean subPoller, boolean read) throws IOException {
 43         this.epfd = EPoll.create();

























 44         this.event = (read) ? EPOLLIN : EPOLLOUT;
 45         this.maxEvents = (subPoller) ? 64 : 512;
 46         this.address = EPoll.allocatePollArray(maxEvents);































 47     }
 48 
 49     @Override
 50     int fdVal() {
 51         return epfd;
 52     }
 53 
 54     @Override
 55     void implRegister(int fdVal) throws IOException {
 56         // re-arm
 57         int err = EPoll.ctl(epfd, EPOLL_CTL_MOD, fdVal, (event | EPOLLONESHOT));
 58         if (err == ENOENT)
 59             err = EPoll.ctl(epfd, EPOLL_CTL_ADD, fdVal, (event | EPOLLONESHOT));
 60         if (err != 0)
 61             throw new IOException("epoll_ctl failed: " + err);
 62     }
 63 
 64     @Override
 65     void implDeregister(int fdVal, boolean polled) {
 66         // event is disabled if already polled
 67         if (!polled) {
 68             EPoll.ctl(epfd, EPOLL_CTL_DEL, fdVal, 0);
 69         }
 70     }
 71 








 72     @Override
 73     int poll(int timeout) throws IOException {
 74         int n = EPoll.wait(epfd, address, maxEvents, timeout);

 75         int i = 0;
 76         while (i < n) {
 77             long eventAddress = EPoll.getEvent(address, i);
 78             int fdVal = EPoll.getDescriptor(eventAddress);
 79             polled(fdVal);



 80             i++;
 81         }
 82         return n;
 83     }
 84 }
 85 

  8  * particular file as subject to the "Classpath" exception as provided
  9  * by Oracle in the LICENSE file that accompanied this code.
 10  *
 11  * This code is distributed in the hope that it will be useful, but WITHOUT
 12  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 13  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 14  * version 2 for more details (a copy is included in the LICENSE file that
 15  * accompanied this code).
 16  *
 17  * You should have received a copy of the GNU General Public License version
 18  * 2 along with this work; if not, write to the Free Software Foundation,
 19  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 20  *
 21  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 22  * or visit www.oracle.com if you need additional information or have any
 23  * questions.
 24  */
 25 package sun.nio.ch;
 26 
 27 import java.io.IOException;
 28 import java.lang.ref.Cleaner.Cleanable;
 29 import jdk.internal.ref.CleanerFactory;
 30 import static sun.nio.ch.EPoll.*;
 31 
 32 /**
 33  * Poller implementation based on the epoll facility.
 34  */
 35 
 36 class EPollPoller extends Poller {
 37     private static final int ENOENT = 2;
 38 
 39     private final int epfd;
 40     private final int event;
 41     private final int maxEvents;
 42     private final long address;
 43     private final EventFD eventfd;  // wakeup event, used for shutdown
 44 
 45     // close action, and cleaner if this is subpoller
 46     private final Runnable closer;
 47     private final Cleanable cleaner;
 48 
 49     EPollPoller(Poller.Mode mode, boolean subPoller, boolean read) throws IOException {
 50         int maxEvents = (subPoller) ? 16 : 64;
 51 
 52         int epfd = EPoll.create();
 53         long address = 0L;
 54         EventFD eventfd = null;
 55         try {
 56             address = EPoll.allocatePollArray(maxEvents);
 57 
 58             // register event with epoll to allow for wakeup
 59             if (subPoller && (mode == Poller.Mode.POLLER_PER_CARRIER)) {
 60                 eventfd = new EventFD();
 61                 IOUtil.configureBlocking(eventfd.efd(), false);
 62                 EPoll.ctl(epfd, EPOLL_CTL_ADD, eventfd.efd(), EPOLLIN);
 63             }
 64         } catch (Throwable e) {
 65             FileDispatcherImpl.closeIntFD(epfd);
 66             if (address != 0L) EPoll.freePollArray(address);
 67             if (eventfd != null) eventfd.close();
 68             throw e;
 69         }
 70 
 71         this.epfd = epfd;
 72         this.event = (read) ? EPOLLIN : EPOLLOUT;
 73         this.maxEvents = maxEvents;
 74         this.address = address;
 75         this.eventfd = eventfd;
 76 
 77         // create action to close epoll instance, register cleaner if this is a subpoller
 78         this.closer = closer(epfd, address, eventfd);
 79         if (subPoller) {
 80             this.cleaner = CleanerFactory.cleaner().register(this, closer);
 81         } else {
 82             this.cleaner = null;
 83         }
 84     }
 85 
 86     /**
 87      * Returns an action to close the epoll instance and release other resources.
 88      */
 89     private static Runnable closer(int epfd, long address, EventFD eventfd) {
 90         return () -> {
 91             try {
 92                 FileDispatcherImpl.closeIntFD(epfd);
 93                 EPoll.freePollArray(address);
 94                 if (eventfd != null) eventfd.close();
 95             } catch (IOException _) { }
 96         };
 97     }
 98 
 99     @Override
100     void close() {
101         if (cleaner != null) {
102             cleaner.clean();
103         } else {
104             closer.run();
105         }
106     }
107 
108     @Override
109     int fdVal() {
110         return epfd;
111     }
112 
113     @Override
114     void implRegister(int fdVal) throws IOException {
115         // re-enable if already registered but disabled (previously polled)
116         int err = EPoll.ctl(epfd, EPOLL_CTL_MOD, fdVal, (event | EPOLLONESHOT));
117         if (err == ENOENT)
118             err = EPoll.ctl(epfd, EPOLL_CTL_ADD, fdVal, (event | EPOLLONESHOT));
119         if (err != 0)
120             throw new IOException("epoll_ctl failed: " + err);
121     }
122 
123     @Override
124     void implDeregister(int fdVal, boolean polled) {
125         // event is disabled if already polled
126         if (!polled) {
127             EPoll.ctl(epfd, EPOLL_CTL_DEL, fdVal, 0);
128         }
129     }
130 
131     @Override
132     void wakeupPoller() throws IOException {
133         if (eventfd == null) {
134             throw new UnsupportedOperationException();
135         }
136         eventfd.set();
137     }
138 
139     @Override
140     int poll(int timeout) throws IOException {
141         int n = EPoll.wait(epfd, address, maxEvents, timeout);
142         int polled = 0;
143         int i = 0;
144         while (i < n) {
145             long eventAddress = EPoll.getEvent(address, i);
146             int fd = EPoll.getDescriptor(eventAddress);
147             if (eventfd == null || fd != eventfd.efd()) {
148                 polled(fd);
149                 polled++;
150             }
151             i++;
152         }
153         return polled;
154     }
155 }
156 
< prev index next >