The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.
package IO::Async::Loop::KQueue;
BEGIN {
  $IO::Async::Loop::KQueue::VERSION = '0.02';
}

use strict;
use warnings;
use Carp;

use IO::KQueue;

use base qw( IO::Async::Loop );

use constant API_VERSION => '0.33';

=head1 NAME

IO::Async::Loop::KQueue - use C<IO::Async> with C<kqueue>

=head1 VERSION

Version 0.02

=head1 SYNOPSIS

Like L<IO::Async::Loop::Epoll> for Linux, This module provides native loop management for
BSD like operating systems that have KQueue present, using C<IO::KQueue>.

    use IO::Async::Loop::KQueue;
    
    my $loop = IO::Async::Loop::KQueue->new();
    
    $loop->add( IO::Async::Signal->new(
        name => '',
	on_receipt => sub { ... },
    ) );

=head1 METHODS

=head2 new

=cut

sub new
{
	my $class = shift;
	my ( %args ) = @_;

	my $kq = IO::KQueue->new() or croak "Cannot create kqueue handle - $!";

	my $self = $class->SUPER::__new( %args );
	$self->{kqueue} = $kq;

	return $self;
}

=head2 $count = $loop->loop_once( $timeout )

This method calls the kevent method, using the given timeout and processes 
the results of that call. It returns the total number of C<IO::Async::Notifier> 
callbacks invoked.

=cut

sub loop_once
{
	my $self = shift;
	my ( $timeout ) = @_;

	$self->_adjust_timeout( \$timeout );

	my $msec = defined $timeout ? $timeout * 1000 : -1;

	my @events = $self->{kqueue}->kevent($msec);

	my $count = 0;
	local $self->{cancellations} = \my %cancellations;

	foreach my $ev ( @events )
	{
		next if $cancellations{$ev->[KQ_FILTER]."/".$ev->[KQ_IDENT]};

		$ev->[KQ_UDATA]->();

		$count++;
	}

	$count += $self->_manage_queues;

	return $count;
}

sub watch_io
{
	my $self = shift;
	my %params = @_;

	my $kqueue = $self->{kqueue};

	my $handle = $params{handle};
	my $fd = $handle->fileno;

	if( my $cb = $params{on_read_ready} ) {
	       	$kqueue->EV_SET($fd, EVFILT_READ, EV_ADD, 0, 0, $cb);
	}

	if( my $cb = $params{on_write_ready} ) {
		$kqueue->EV_SET($fd, EVFILT_WRITE, EV_ADD, 0, 0, $cb);
	}
}

sub unwatch_io
{
	my $self = shift;
	my %params = @_;

	my $kqueue = $self->{kqueue};

	my $handle = $params{handle};
	my $fd = $handle->fileno;

	# Just ignore errors from EV_SET; doesn't matter if we fail to delete
	# because it wasn't there
	if( $params{on_read_ready} ) {
		eval { $kqueue->EV_SET($fd, EVFILT_READ, EV_DELETE) };
		$self->{cancellations}{EVFILT_READ."/$fd"}++ if $self->{cancellations};
	}

	if( $params{on_write_ready} ) {
		eval { $kqueue->EV_SET($fd, EVFILT_WRITE, EV_DELETE) };
		$self->{cancellations}{EVFILT_WRITE."/$fd"}++ if $self->{cancellations};
	}
}

=head1 AUTHOR

Squeeks, C<< <squeek at cpan.org> >>

=head1 BUGS

Please report any bugs or feature requests to C<bug-io-async-loop-kqueue at rt.cpan.org>, or through
the web interface at L<http://rt.cpan.org/NoAuth/ReportBug.html?Queue=IO-Async-Loop-KQueue>.  I will be notified, and then you'll
automatically be notified of progress on your bug as I make changes.

=head1 SUPPORT

You can find documentation for this module with the perldoc command.

    perldoc IO::Async::Loop::KQueue

You can also look for information at:

=over 4

=item * RT: CPAN's request tracker

L<http://rt.cpan.org/NoAuth/Bugs.html?Dist=IO-Async-Loop-KQueue>

=item * AnnoCPAN: Annotated CPAN documentation

L<http://annocpan.org/dist/IO-Async-Loop-KQueue>

=item * CPAN Ratings

L<http://cpanratings.perl.org/d/IO-Async-Loop-KQueue>

=item * Search CPAN

L<http://search.cpan.org/dist/IO-Async-Loop-KQueue/>

=back


=head1 ACKNOWLEDGEMENTS

Paul Evans (LeoNerd) for doing all the hard work. 

=head1 LICENSE AND COPYRIGHT

Copyright 2010 Squeeks.

This program is free software; you can redistribute it and/or modify it
under the terms of either: the GNU General Public License as published
by the Free Software Foundation; or the Artistic License.

See http://dev.perl.org/licenses/ for more information.


=cut

1; # End of IO::Async::Loop::KQueue