From 6fcdc5fa69ea44d38e5505c23a6e9645efd35027 Mon Sep 17 00:00:00 2001 From: Martin Sustrik Date: Thu, 3 Dec 2009 10:14:07 +0100 Subject: common lisp binding & perf tests added --- bindings/Makefile.am | 8 +- bindings/cl/Makefile.am | 9 ++ bindings/cl/meta.lisp | 59 +++++++++++ bindings/cl/package.lisp | 108 ++++++++++++++++++++ bindings/cl/zeromq-api.lisp | 155 ++++++++++++++++++++++++++++ bindings/cl/zeromq.asd | 38 +++++++ bindings/cl/zeromq.lisp | 244 ++++++++++++++++++++++++++++++++++++++++++++ configure.in | 21 ++++ perf/Makefile.am | 2 +- perf/cl/Makefile.am | 2 + perf/cl/lat-parms.lisp | 22 ++++ perf/cl/local-lat-poll.lisp | 43 ++++++++ perf/cl/local-lat.lisp | 52 ++++++++++ perf/cl/local-thr.lisp | 54 ++++++++++ perf/cl/remote-lat.lisp | 50 +++++++++ perf/cl/remote-thr.lisp | 41 ++++++++ perf/cl/thr-parms.lisp | 25 +++++ 17 files changed, 930 insertions(+), 3 deletions(-) create mode 100644 bindings/cl/Makefile.am create mode 100644 bindings/cl/meta.lisp create mode 100644 bindings/cl/package.lisp create mode 100644 bindings/cl/zeromq-api.lisp create mode 100644 bindings/cl/zeromq.asd create mode 100644 bindings/cl/zeromq.lisp create mode 100644 perf/cl/Makefile.am create mode 100644 perf/cl/lat-parms.lisp create mode 100644 perf/cl/local-lat-poll.lisp create mode 100644 perf/cl/local-lat.lisp create mode 100644 perf/cl/local-thr.lisp create mode 100644 perf/cl/remote-lat.lisp create mode 100644 perf/cl/remote-thr.lisp create mode 100644 perf/cl/thr-parms.lisp diff --git a/bindings/Makefile.am b/bindings/Makefile.am index 77b4ec2..08e3659 100644 --- a/bindings/Makefile.am +++ b/bindings/Makefile.am @@ -10,6 +10,10 @@ if BUILD_RUBY DIR_R = ruby endif -SUBDIRS = $(DIR_J) $(DIR_P) $(DIR_R) -DIST_SUBDIRS = java python ruby +if BUILD_CL +DIR_R = cl +endif + +SUBDIRS = $(DIR_J) $(DIR_P) $(DIR_R) $(DIR_CL) +DIST_SUBDIRS = java python ruby cl diff --git a/bindings/cl/Makefile.am b/bindings/cl/Makefile.am new file mode 100644 index 0000000..034d1f4 --- /dev/null +++ b/bindings/cl/Makefile.am @@ -0,0 +1,9 @@ +sitedir=$(CLDIR)/../site/zeromq +zeromqasd=$(CLDIR)/zeromq.asd + +install-data-local: + if test -d $(sitedir); then rm -rdf $(sitedir); fi + mkdir $(sitedir) + chown --reference=$(sitedir)/.. $(sitedir) + cp *.lisp *.asd $(sitedir) + ln -sf $(sitedir)/zeromq.asd $(zeromqasd) diff --git a/bindings/cl/meta.lisp b/bindings/cl/meta.lisp new file mode 100644 index 0000000..751a089 --- /dev/null +++ b/bindings/cl/meta.lisp @@ -0,0 +1,59 @@ +;; Copyright (c) 2009 Vitaly Mayatskikh +;; +;; This file is part of 0MQ. +;; +;; 0MQ is free software; you can redistribute it and/or modify it under +;; the terms of the Lesser GNU General Public License as published by +;; the Free Software Foundation; either version 3 of the License, or +;; (at your option) any later version. +;; +;; 0MQ is distributed in the hope that it will be useful, +;; but WITHOUT ANY WARRANTY; without even the implied warranty of +;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +;; Lesser GNU General Public License for more details. +;; +;; You should have received a copy of the Lesser GNU General Public License +;; along with this program. If not, see . + +(in-package :zeromq) + +(define-condition error-again (error) + ((argument :reader error-again :initarg :argument)) + (:report (lambda (condition stream) + (write-string (convert-from-foreign + (%strerror (error-again condition)) + :string) + stream)))) + +(defmacro defcfun* (name-and-options return-type &body args) + (let* ((c-name (car name-and-options)) + (l-name (cadr name-and-options)) + (n-name (cffi::format-symbol t "%~A" l-name)) + (name (list c-name n-name)) + + (docstring (when (stringp (car args)) (pop args))) + (ret (gensym))) + (loop with opt + for i in args + unless (consp i) do (setq opt t) + else + collect i into args* + and if (not opt) collect (car i) into names + else collect (car i) into opts + and collect (list (car i) 0) into opts-init + end + finally (return + `(progn + (defcfun ,name ,return-type + ,@args*) + + (defun ,l-name (,@names &optional ,@opts-init) + ,docstring + (let ((,ret (,n-name ,@names ,@opts))) + (if ,(if (eq return-type :pointer) + `(zerop (pointer-address ,ret)) + `(not (zerop ,ret))) + (cond + ((eq *errno* isys:eagain) (error 'error-again :argument *errno*)) + (t (error (convert-from-foreign (%strerror *errno*) :string)))) + ,ret)))))))) diff --git a/bindings/cl/package.lisp b/bindings/cl/package.lisp new file mode 100644 index 0000000..89713b1 --- /dev/null +++ b/bindings/cl/package.lisp @@ -0,0 +1,108 @@ +;; Copyright (c) 2009 Vitaly Mayatskikh +;; +;; This file is part of 0MQ. +;; +;; 0MQ is free software; you can redistribute it and/or modify it under +;; the terms of the Lesser GNU General Public License as published by +;; the Free Software Foundation; either version 3 of the License, or +;; (at your option) any later version. +;; +;; 0MQ is distributed in the hope that it will be useful, +;; but WITHOUT ANY WARRANTY; without even the implied warranty of +;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +;; Lesser GNU General Public License for more details. +;; +;; You should have received a copy of the Lesser GNU General Public License +;; along with this program. If not, see . + +(defpackage #:zeromq + (:nicknames :zmq) + (:use :cl :cffi) + (:shadow #:sleep #:close) + (:export + ;; constants + #:affinity + #:delimiter + #:downstream + #:efsm + #:emthread + #:enocompatproto + #:hausnumero + #:hwm + #:identity + #:lwm + #:max-vsm-size + #:mcast-loop + #:noblock + #:noflush + #:p2p + #:poll + #:pollin + #:pollout + #:pub + #:rate + #:recovery-ivl + #:rep + #:req + #:sub + #:subscribe + #:swap + #:unsubscribe + #:upstream + #:vsm + + #:events + + ;; structures + #:msg + #:pollitem + + ;; functions + #:bind + #:close + #:connect + #:flush + #:init + #:msg-close + #:msg-copy + #:msg-data-as-array + #:msg-data-as-is + #:msg-data-as-string + #:msg-init + #:msg-init-data + #:msg-init-size + #:msg-move + #:msg-size + #:msg-type + #:poll + #:pollitem-events + #:pollitem-fd + #:pollitem-revents + #:pollitem-socket + #:recv + #:send + #:setsockopt + #:sleep + #:socket + #:stopwatch-start + #:stopwatch-stop + #:strerror + #:term + + ;; macros + #:with-context + #:with-polls + #:with-socket + #:with-stopwatch + + ;; conditions + #:error-again)) + +(in-package :zeromq) + +(eval-when (:compile-toplevel :load-toplevel :execute) + (define-foreign-library zeromq + (:unix (:or "libzmq.so.0.0.0" "libzmq.so")) + (t "libzmq"))) + +(use-foreign-library zeromq) diff --git a/bindings/cl/zeromq-api.lisp b/bindings/cl/zeromq-api.lisp new file mode 100644 index 0000000..953b98b --- /dev/null +++ b/bindings/cl/zeromq-api.lisp @@ -0,0 +1,155 @@ +;; Copyright (c) 2009 Vitaly Mayatskikh +;; +;; This file is part of 0MQ. +;; +;; 0MQ is free software; you can redistribute it and/or modify it under +;; the terms of the Lesser GNU General Public License as published by +;; the Free Software Foundation; either version 3 of the License, or +;; (at your option) any later version. +;; +;; 0MQ is distributed in the hope that it will be useful, +;; but WITHOUT ANY WARRANTY; without even the implied warranty of +;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +;; Lesser GNU General Public License for more details. +;; +;; You should have received a copy of the Lesser GNU General Public License +;; along with this program. If not, see . + +(in-package :zeromq) + +(defclass msg () + ((raw :accessor msg-raw :initform nil) + (shared :accessor msg-shared :initform 0 :initarg :shared))) + +(defmethod initialize-instance :after ((inst msg) &key size data) + (let ((obj (foreign-alloc 'msg))) + (with-slots (raw shared) inst + (setf raw obj) + (tg:finalize inst (lambda () + (%msg-close raw) + (foreign-free raw))) + (when shared + (setf (foreign-slot-value obj 'msg 'shared) (if shared 1 0))) + (cond (size (%msg-init-size raw size)) + (data + (multiple-value-bind (ptr len) + (etypecase data + (string (foreign-string-alloc data)) + (array (values (foreign-alloc :uchar :initial-contents data) + (length data)))) + (msg-init-data raw ptr len (callback zmq-free)))) + (t (msg-init raw)))))) + +(defclass pollitem () + ((raw :accessor pollitem-raw :initform nil) + (socket :accessor pollitem-socket :initform nil :initarg :socket) + (fd :accessor pollitem-fd :initform -1 :initarg :fd) + (events :accessor pollitem-events :initform 0 :initarg :events) + (revents :accessor pollitem-revents :initform 0))) + +(defmethod initialize-instance :after ((inst pollitem) &key) + (let ((obj (foreign-alloc 'pollitem))) + (setf (pollitem-raw inst) obj) + (tg:finalize inst (lambda () (foreign-free obj))))) + +(defun bind (s address) + (with-foreign-string (addr address) + (%bind s addr))) + +(defun connect (s address) + (with-foreign-string (addr address) + (%connect s addr))) + +(defmacro with-context ((context app-threads io-threads &optional flags) &body body) + `(let ((,context (init ,app-threads ,io-threads (or ,flags 0)))) + ,@body + (term ,context))) + +(defmacro with-socket ((socket context type) &body body) + `(let ((,socket (socket ,context ,type))) + ,@body + (close ,socket))) + +(defmacro with-stopwatch (&body body) + (let ((watch (gensym))) + `(with-foreign-object (,watch :long 2) + (setq ,watch (stopwatch-start)) + ,@body + (stopwatch-stop ,watch)))) + +(defun msg-data-as-is (msg) + (%msg-data (msg-raw msg))) + +(defun msg-data-as-string (msg) + (let ((data (%msg-data (msg-raw msg)))) + (unless (zerop (pointer-address data)) + (convert-from-foreign data :string)))) + +(defun msg-data-as-array (msg) + (let ((data (%msg-data (msg-raw msg)))) + (unless (zerop (pointer-address data)) + (let* ((len (msg-size msg)) + (arr (make-array len :element-type '(unsigned-byte)))) + (dotimes (i len) + (setf (aref arr i) (mem-aref data :uchar i))) + arr)))) + +(defun send (s msg &optional flags) + (%send s (msg-raw msg) (or flags 0))) + +(defun recv (s msg &optional flags) + (%recv s (msg-raw msg) (or flags 0))) + +(defun msg-init-size (msg size) + (%msg-init-size (msg-raw msg) size)) + +(defun msg-close (msg) + (%msg-close (msg-raw msg))) + +(defun msg-size (msg) + (%msg-size (msg-raw msg))) + +(defun msg-move (dst src) + (%msg-move (msg-raw dst) (msg-raw src))) + +(defun msg-copy (dst src) + (%msg-copy (msg-raw dst) (msg-raw src))) + +(defun setsockopt (socket option value) + (etypecase value + (string (with-foreign-string (string value) + (%setsockopt socket option string (length value)))) + (integer (with-foreign-object (int :long 2) + (setf (mem-aref int :long 0) value) + (%setsockopt socket option int (foreign-type-size :long)))))) + +(defun poll (items) + (let ((len (length items))) + (with-foreign-object (%items 'pollitem len) + (dotimes (i len) + (let ((item (nth i items)) + (%item (mem-aref %items 'pollitem i))) + (with-foreign-slots ((socket fd events revents) %item pollitem) + (setf socket (pollitem-socket item) + fd (pollitem-fd item) + events (pollitem-events item))))) + (let ((ret (%poll %items len))) + (if (> ret 0) + (loop for i below len + for revent = (foreign-slot-value (mem-aref %items 'pollitem i) + 'pollitem + 'revents) + collect (setf (pollitem-revents (nth i items)) revent)) + (error (convert-from-foreign (%strerror *errno*) :string))))))) + +(defmacro with-polls (list &body body) + `(let ,(loop for (name . polls) in list + collect `(,name + (list + ,@(loop for (socket . events) in polls + collect `(make-instance 'pollitem + :socket ,socket + :events ,events))))) + ,@body)) + +; diff --git a/bindings/cl/zeromq.asd b/bindings/cl/zeromq.asd new file mode 100644 index 0000000..3aa8d5e --- /dev/null +++ b/bindings/cl/zeromq.asd @@ -0,0 +1,38 @@ +;; Copyright (c) 2009 Vitaly Mayatskikh +;; +;; This file is part of 0MQ. +;; +;; 0MQ is free software; you can redistribute it and/or modify it under +;; the terms of the Lesser GNU General Public License as published by +;; the Free Software Foundation; either version 3 of the License, or +;; (at your option) any later version. +;; +;; 0MQ is distributed in the hope that it will be useful, +;; but WITHOUT ANY WARRANTY; without even the implied warranty of +;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +;; Lesser GNU General Public License for more details. +;; +;; You should have received a copy of the Lesser GNU General Public License +;; along with this program. If not, see . + +(cl:eval-when (:load-toplevel :execute) + (asdf:operate 'asdf:load-op :cffi) + (asdf:operate 'asdf:load-op :trivial-garbage) + (asdf:operate 'asdf:load-op :iolib.syscalls)) + +(defpackage #:zeromq-asd + (:use :cl :asdf)) + +(in-package #:zeromq-asd) + +(defsystem zeromq + :name "zeromq" + :version "0.1" + :author "Vitaly Mayatskikh " + :licence "LGPLv3" + :description "Zero MQ 2 bindings" + :serial t + :components ((:file "package") + (:file "meta") + (:file "zeromq") + (:file "zeromq-api"))) diff --git a/bindings/cl/zeromq.lisp b/bindings/cl/zeromq.lisp new file mode 100644 index 0000000..94f7672 --- /dev/null +++ b/bindings/cl/zeromq.lisp @@ -0,0 +1,244 @@ +;; Copyright (c) 2009 Vitaly Mayatskikh +;; +;; This file is part of 0MQ. +;; +;; 0MQ is free software; you can redistribute it and/or modify it under +;; the terms of the Lesser GNU General Public License as published by +;; the Free Software Foundation; either version 3 of the License, or +;; (at your option) any later version. +;; +;; 0MQ is distributed in the hope that it will be useful, +;; but WITHOUT ANY WARRANTY; without even the implied warranty of +;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +;; Lesser GNU General Public License for more details. +;; +;; You should have received a copy of the Lesser GNU General Public License +;; along with this program. If not, see . + +(in-package :zeromq) + +(defcvar "errno" :int) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; 0MQ errors. +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +(defconstant hausnumero 156384712) + +;; On Windows platform some of the standard POSIX errnos are not defined. +;; #ifndef ENOTSUP +;; #define ENOTSUP (ZMQ_HAUSNUMERO + 1) +;; #endif +;; #ifndef EPROTONOSUPPORT +;; #define EPROTONOSUPPORT (ZMQ_HAUSNUMERO + 2) +;; #endif +;; #ifndef ENOBUFS +;; #define ENOBUFS (ZMQ_HAUSNUMERO + 3) +;; #endif +;; #ifndef ENETDOWN +;; #define ENETDOWN (ZMQ_HAUSNUMERO + 4) +;; #endif +;; #ifndef EADDRINUSE +;; #define EADDRINUSE (ZMQ_HAUSNUMERO + 5) +;; #endif +;; #ifndef EADDRNOTAVAIL +;; #define EADDRNOTAVAIL (ZMQ_HAUSNUMERO + 6) +;; #endif + +;; Native 0MQ error codes. +(defconstant emthread (+ hausnumero 50)) +(defconstant efsm (+ hausnumero 51)) +(defconstant enocompatproto (+ hausnumero 52)) + +(defcfun ("zmq_strerror" %strerror) :pointer + (errnum :int)) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; 0MQ message definition. +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +(defconstant max-vsm-size 30) + +;; Message types. These integers may be stored in 'content' member of the +;; message instead of regular pointer to the data. +(defconstant delimiter 31) +(defconstant vsm 32) + +(defcstruct (msg) + (content :pointer) + (shared :uchar) + (vsm-size :uchar) + (vsm-data :uchar :count 30)) ;; FIXME max-vsm-size + +(defcfun ("zmq_msg_init" msg-init) :int + (msg msg)) + +(defcfun* ("zmq_msg_init_size" %msg-init-size) :int + (msg msg) + (size :long)) + +(defcallback zmq-free :void ((ptr :pointer)) + (foreign-free ptr)) + +;;typedef void (zmq_free_fn) (void *data); +(defcfun ("zmq_msg_init_data" msg-init-data) :int + (msg msg) + (data :pointer) + (size :long) + (ffn :pointer)) ; zmq_free_fn + +(defcfun* ("zmq_msg_close" %msg-close) :int + (msg msg)) + +(defcfun ("zmq_msg_move" %msg-move) :int + (dest msg) + (src msg)) + +(defcfun ("zmq_msg_copy" %msg-copy) :int + (dest msg) + (src msg)) + +(defcfun ("zmq_msg_data" %msg-data) :pointer + (msg msg)) + +(defcfun ("zmq_msg_size" %msg-size) :int + (msg msg)) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; 0MQ infrastructure (a.k.a. context) initialisation & termination. +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +(defconstant poll 1) + +(defcfun* ("zmq_init" init) :pointer + (app-threads :int) + (io-threads :int) + (flags :int)) + +(defcfun ("zmq_term" term) :int + (context :pointer)) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; 0MQ socket definition. +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +;; Creating a 0MQ socket. +;; ********************** + +(defconstant p2p 0) +(defconstant pub 1) +(defconstant sub 2) +(defconstant req 3) +(defconstant rep 4) +(defconstant upstream 5) +(defconstant downstream 6) + +(defcfun* ("zmq_socket" socket) :pointer + (context :pointer) + (type :int)) + +;; Destroying the socket. +;; ********************** + +(defcfun ("zmq_close" close) :int + (s :pointer)) + +;; Manipulating socket options. +;; **************************** + +;; Available socket options, their types and default values. + +(defconstant hwm 1) +(defconstant lwm 2) +(defconstant swap 3) +(defconstant affinity 4) +(defconstant identity 5) +(defconstant subscribe 6) +(defconstant unsubscribe 7) +(defconstant rate 8) +(defconstant recovery-ivl 9) +(defconstant mcast-loop 10) + +(defcfun* ("zmq_setsockopt" %setsockopt) :int + (s :pointer) + (option :int) + (optval :pointer) + (optvallen :long)) + +;; Creating connections. +;; ********************* + +;; Addresses are composed of the name of the protocol to use followed by :// +;; and a protocol-specific address. Available protocols: +;; +;; tcp - the address is composed of IP address and port delimited by colon +;; sign (:). The IP address can be a hostname (with 'connect') or +;; a network interface name (with 'bind'). Examples "tcp://eth0:5555", +;; "tcp://192.168.0.1:20000", "tcp://hq.mycompany.com:80". +;; +;; pgm & udp - both protocols have same address format. It's network interface +;; to use, semicolon (;), multicast group IP address, colon (:) and +;; port. Examples: "pgm://eth2;224.0.0.1:8000", +;; "udp://192.168.0.111;224.1.1.1:5555". + +(defcfun* ("zmq_bind" %bind) :int + (s :pointer) + (addr :pointer :char)) + +(defcfun* ("zmq_connect" %connect) :int + (s :pointer) + (addr :pointer :char)) + +;; Sending and receiving messages. +;; ******************************* + +(defconstant noblock 1) + +(defconstant noflush 2) + +(defcfun* ("zmq_send" %send) :int + (s :pointer) + (msg msg) + :optional + (flags :int)) + +(defcfun* ("zmq_flush" flush) :int + (s :pointer)) + +(defcfun* ("zmq_recv" %recv) :int + (s :pointer) + (msg msg) + :optional + (flags :int)) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; I/O multiplexing. +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +(defconstant pollin 1) +(defconstant pollout 2) + +(defcstruct pollitem + (socket :pointer) + (fd :int) + (events :short) + (revents :short)) + +(defcfun ("zmq_poll" %poll) :int + (items :pointer) + (nitems :int)) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Helper functions. +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +;; Helper functions used by perf tests so that they don't have to care +;; about minutiae of time-related functions on different OS platforms. + +(defcfun ("zmq_stopwatch_start" stopwatch-start) :pointer) + +(defcfun ("zmq_stopwatch_stop" stopwatch-stop) :ulong + (watch :pointer)) + +(defcfun ("zmq_sleep" sleep) :void + (seconds :int)) diff --git a/configure.in b/configure.in index 4051536..eba8188 100644 --- a/configure.in +++ b/configure.in @@ -391,6 +391,21 @@ else JAR=true fi +# Common Lisp +cldir="" +AC_ARG_WITH([cl], [AS_HELP_STRING([--with-cl], + [build Common Lisp language binding and install it into specified ASDF central registry [default=no]])], + [cldir="$withval";clzmq="yes"], [clzmq="no"]) + +if test "x$clzmq" = "xyes"; then + if test "x$cldir" = "xyes"; then + AC_MSG_ERROR([Please, specify ASDF central registry path]) + fi + + CLDIR=${cldir} + AC_SUBST([CLDIR]) +fi + # PGM extension pgm1_ext="no" pgm2_ext="no" @@ -631,6 +646,7 @@ AM_CONDITIONAL(BUILD_JAVA, test "x$jzmq" = "xyes") AM_CONDITIONAL(BUILD_PYTHON, test "x$pyzmq" = "xyes") AM_CONDITIONAL(BUILD_RUBY, test "x$rbzmq" = "xyes") AM_CONDITIONAL(BUILD_C, test "x$czmq" = "xyes") +AM_CONDITIONAL(BUILD_CL, test "x$clzmq" = "xyes") AM_CONDITIONAL(BUILD_CPP, test "x$cppzmq" = "xyes") AM_CONDITIONAL(BUILD_PGM1, test "x$pgm1_ext" = "xyes") AM_CONDITIONAL(BUILD_PGM2, test "x$pgm2_ext" = "xyes") @@ -657,6 +673,7 @@ AC_CHECK_FUNCS(perror gettimeofday memset socket getifaddrs freeifaddrs) AC_OUTPUT(Makefile src/Makefile man/Makefile bindings/python/Makefile \ bindings/python/setup.py bindings/ruby/Makefile \ + bindings/cl/Makefile perf/cl/Makefile \ bindings/java/Makefile perf/Makefile perf/c/Makefile perf/cpp/Makefile \ perf/python/Makefile perf/ruby/Makefile perf/java/Makefile src/libzmq.pc \ devices/Makefile devices/zmq_forwarder/Makefile \ @@ -676,6 +693,10 @@ AC_MSG_RESULT([ 0MQ install dir: $prefix]) AC_MSG_RESULT([ Language bindings:]) AC_MSG_RESULT([ C: $czmq]) AC_MSG_RESULT([ C++: $cppzmq]) +AC_MSG_RESULT([ Common Lisp: $clzmq]) +if test "x$clzmq" = "xyes"; then +AC_MSG_RESULT([ ASDF central registry: $cldir]) +fi AC_MSG_RESULT([ Java: $jzmq]) AC_MSG_RESULT([ Python: $pyzmq]) AC_MSG_RESULT([ Ruby: $rbzmq]) diff --git a/perf/Makefile.am b/perf/Makefile.am index a3c13ad..3cd29ff 100644 --- a/perf/Makefile.am +++ b/perf/Makefile.am @@ -20,4 +20,4 @@ endif SUBDIRS = $(PERF_DIR_C) $(PERF_DIR_CPP) $(PERF_DIR_P) \ $(PERF_DIR_J) $(PERF_DIR_R) -DIST_SUBDIRS = c cpp python java ruby +DIST_SUBDIRS = c cpp python java ruby cl diff --git a/perf/cl/Makefile.am b/perf/cl/Makefile.am new file mode 100644 index 0000000..72d2663 --- /dev/null +++ b/perf/cl/Makefile.am @@ -0,0 +1,2 @@ +dist_noinst_CL = local-lat.lisp local-lat-poll.lisp remote-lat.lisp \ +lat-parms.lisp local-thr.lisp remote-thr.lisp thr-params.lisp diff --git a/perf/cl/lat-parms.lisp b/perf/cl/lat-parms.lisp new file mode 100644 index 0000000..d821237 --- /dev/null +++ b/perf/cl/lat-parms.lisp @@ -0,0 +1,22 @@ +;; Copyright (c) 2009 Vitaly Mayatskikh +;; +;; This file is part of 0MQ. +;; +;; 0MQ is free software; you can redistribute it and/or modify it under +;; the terms of the Lesser GNU General Public License as published by +;; the Free Software Foundation; either version 3 of the License, or +;; (at your option) any later version. +;; +;; 0MQ is distributed in the hope that it will be useful, +;; but WITHOUT ANY WARRANTY; without even the implied warranty of +;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +;; Lesser GNU General Public License for more details. +;; +;; You should have received a copy of the Lesser GNU General Public License +;; along with this program. If not, see . + +(in-package :zeromq-test) + +(defvar *address* "tcp://127.0.0.1:5555/") +(defvar *roundtrip-count* 1000) +(defvar *message-size* 32) diff --git a/perf/cl/local-lat-poll.lisp b/perf/cl/local-lat-poll.lisp new file mode 100644 index 0000000..4fea85e --- /dev/null +++ b/perf/cl/local-lat-poll.lisp @@ -0,0 +1,43 @@ +;; Copyright (c) 2009 Vitaly Mayatskikh +;; +;; This file is part of 0MQ. +;; +;; 0MQ is free software; you can redistribute it and/or modify it under +;; the terms of the Lesser GNU General Public License as published by +;; the Free Software Foundation; either version 3 of the License, or +;; (at your option) any later version. +;; +;; 0MQ is distributed in the hope that it will be useful, +;; but WITHOUT ANY WARRANTY; without even the implied warranty of +;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +;; Lesser GNU General Public License for more details. +;; +;; You should have received a copy of the Lesser GNU General Public License +;; along with this program. If not, see . + +(asdf:oos 'asdf:load-op :zeromq) + +(defpackage :zeromq-test + (:use :cl)) + +(in-package :zeromq-test) + +(load "lat-parms") + +(zmq:with-context (ctx 1 1 zmq:poll) + (zmq:with-socket (s ctx zmq:rep) + (zmq:bind s *address*) + (let ((msg (make-instance 'zmq:msg))) + (zmq:with-polls ((poll-in . ((s . zmq:pollin))) + (poll-out . ((s . zmq:pollout)))) + (dotimes (i *roundtrip-count*) + (zmq:poll poll-in) + (zmq:recv s msg zmq:noblock) + (zmq:poll poll-out) + (zmq:send s msg zmq:noblock)))))) + +(tg:gc) +#+sbcl (sb-ext:quit) +#+clisp (ext:quit) + +; diff --git a/perf/cl/local-lat.lisp b/perf/cl/local-lat.lisp new file mode 100644 index 0000000..7de16a6 --- /dev/null +++ b/perf/cl/local-lat.lisp @@ -0,0 +1,52 @@ +;; Copyright (c) 2009 Vitaly Mayatskikh +;; +;; This file is part of 0MQ. +;; +;; 0MQ is free software; you can redistribute it and/or modify it under +;; the terms of the Lesser GNU General Public License as published by +;; the Free Software Foundation; either version 3 of the License, or +;; (at your option) any later version. +;; +;; 0MQ is distributed in the hope that it will be useful, +;; but WITHOUT ANY WARRANTY; without even the implied warranty of +;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +;; Lesser GNU General Public License for more details. +;; +;; You should have received a copy of the Lesser GNU General Public License +;; along with this program. If not, see . + +(asdf:oos 'asdf:load-op :zeromq) + +(defpackage :zeromq-test + (:use :cl)) + +(in-package :zeromq-test) + +(load "lat-parms") + +(zmq:with-context (ctx 1 1) + (zmq:with-socket (s ctx zmq:rep) + (zmq:bind s *address*) + (let ((msg (make-instance 'zmq:msg))) + (dotimes (i *roundtrip-count*) +;; non-blocking recv + #+nil + (tagbody retry + (handler-case + (progn + (zmq:recv s msg zmq:noblock) + (format t "size ~d, ~a~%" (zmq:msg-size msg) (zmq:msg-data-as-array msg))) + (zmq:error-again (c) + (declare (ignore c)) + (sleep 0.01) + (go retry)))) +;; blocking recv + (zmq:recv s msg) + (zmq:send s msg))) + (zmq:sleep 1))) + +(tg:gc) +#+sbcl (sb-ext:quit) +#+clisp (ext:quit) + +; diff --git a/perf/cl/local-thr.lisp b/perf/cl/local-thr.lisp new file mode 100644 index 0000000..1098c47 --- /dev/null +++ b/perf/cl/local-thr.lisp @@ -0,0 +1,54 @@ +;; Copyright (c) 2009 Vitaly Mayatskikh +;; +;; This file is part of 0MQ. +;; +;; 0MQ is free software; you can redistribute it and/or modify it under +;; the terms of the Lesser GNU General Public License as published by +;; the Free Software Foundation; either version 3 of the License, or +;; (at your option) any later version. +;; +;; 0MQ is distributed in the hope that it will be useful, +;; but WITHOUT ANY WARRANTY; without even the implied warranty of +;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +;; Lesser GNU General Public License for more details. +;; +;; You should have received a copy of the Lesser GNU General Public License +;; along with this program. If not, see . + +(asdf:oos 'asdf:load-op :zeromq) + +(defpackage :zeromq-test + (:use :cl)) + +(in-package :zeromq-test) + +(load "thr-parms") + +(defvar *elapsed* nil) +(defvar *throughput* nil) +(defvar *megabits* nil) + +(zmq::with-context (ctx 1 1) + (zmq:with-socket (s ctx zmq:sub) + (zmq:setsockopt s zmq:subscribe "*") + (zmq:setsockopt s zmq:rate *rate*) + (zmq:bind s *bind-address*) + (let ((msg (make-instance 'zmq:msg))) + (zmq:recv s msg) + (setf *elapsed* + (zmq:with-stopwatch + (dotimes (i (1- *message-count*)) + (zmq:recv s msg)))))) + (setq *throughput* (* (/ *message-count* *elapsed*) 1e6) + *megabits* (/ (* *throughput* *message-count* 8) 1e6)) + + (format t "message size: ~d [B]~%" *message-size*) + (format t "message count: ~d~%" *message-count*) + (format t "mean throughput: ~d [msg/s]~%" (round *throughput*)) + (format t "mean throughput: ~,3f [Mb/s]~%" *megabits*)) + +(tg:gc) +#+sbcl (sb-ext:quit) +#+clisp (ext:quit) + +; diff --git a/perf/cl/remote-lat.lisp b/perf/cl/remote-lat.lisp new file mode 100644 index 0000000..c0f0d8a --- /dev/null +++ b/perf/cl/remote-lat.lisp @@ -0,0 +1,50 @@ +;; Copyright (c) 2009 Vitaly Mayatskikh +;; +;; This file is part of 0MQ. +;; +;; 0MQ is free software; you can redistribute it and/or modify it under +;; the terms of the Lesser GNU General Public License as published by +;; the Free Software Foundation; either version 3 of the License, or +;; (at your option) any later version. +;; +;; 0MQ is distributed in the hope that it will be useful, +;; but WITHOUT ANY WARRANTY; without even the implied warranty of +;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +;; Lesser GNU General Public License for more details. +;; +;; You should have received a copy of the Lesser GNU General Public License +;; along with this program. If not, see . + +(asdf:oos 'asdf:load-op :zeromq) + +(defpackage :zeromq-test + (:use :cl)) + +(in-package :zeromq-test) + +(load "lat-parms") + +(defvar *elapsed* nil) +(defvar *latency* nil) + +(zmq::with-context (ctx 1 1) + (zmq:with-socket (s ctx zmq:req) + (zmq:connect s *address*) + (let ((msg (make-instance 'zmq:msg :size *message-size*))) + (setf *elapsed* + (zmq:with-stopwatch + (dotimes (i *roundtrip-count*) + (zmq:send s msg) + (zmq:recv s msg))))) + (zmq:sleep 1))) + +(setf *latency* (/ *elapsed* (* 2 *roundtrip-count*))) + +(format t "message size: ~d [B]~%" *message-size*) +(format t "roundtrip count: ~d~%" *roundtrip-count*) +(format t "average latency: ~f [us]~%" *latency*) + +(tg:gc) +#+sbcl (sb-ext:quit) +#+clisp (ext:quit) +; diff --git a/perf/cl/remote-thr.lisp b/perf/cl/remote-thr.lisp new file mode 100644 index 0000000..c50170e --- /dev/null +++ b/perf/cl/remote-thr.lisp @@ -0,0 +1,41 @@ +;; Copyright (c) 2009 Vitaly Mayatskikh +;; +;; This file is part of 0MQ. +;; +;; 0MQ is free software; you can redistribute it and/or modify it under +;; the terms of the Lesser GNU General Public License as published by +;; the Free Software Foundation; either version 3 of the License, or +;; (at your option) any later version. +;; +;; 0MQ is distributed in the hope that it will be useful, +;; but WITHOUT ANY WARRANTY; without even the implied warranty of +;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +;; Lesser GNU General Public License for more details. +;; +;; You should have received a copy of the Lesser GNU General Public License +;; along with this program. If not, see . + +(asdf:oos 'asdf:load-op :zeromq) + +(defpackage :zeromq-test + (:use :cl)) + +(in-package :zeromq-test) + +(load "thr-parms") + +(zmq::with-context (ctx 1 1) + (zmq:with-socket (s ctx zmq:pub) + (zmq:setsockopt s zmq:rate *rate*) + (zmq:connect s *connect-address*) + (let ((msg (make-instance 'zmq:msg))) + (dotimes (i *message-count*) + (zmq:msg-init-size msg *message-size*) + (zmq:send s msg) + (zmq:msg-close msg)) + (zmq:sleep 10)))) + +(tg:gc) +#+sbcl (sb-ext:quit) +#+clisp (ext:quit) +; diff --git a/perf/cl/thr-parms.lisp b/perf/cl/thr-parms.lisp new file mode 100644 index 0000000..e2693e8 --- /dev/null +++ b/perf/cl/thr-parms.lisp @@ -0,0 +1,25 @@ +;; Copyright (c) 2009 Vitaly Mayatskikh +;; +;; This file is part of 0MQ. +;; +;; 0MQ is free software; you can redistribute it and/or modify it under +;; the terms of the Lesser GNU General Public License as published by +;; the Free Software Foundation; either version 3 of the License, or +;; (at your option) any later version. +;; +;; 0MQ is distributed in the hope that it will be useful, +;; but WITHOUT ANY WARRANTY; without even the implied warranty of +;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +;; Lesser GNU General Public License for more details. +;; +;; You should have received a copy of the Lesser GNU General Public License +;; along with this program. If not, see . + +(in-package :zeromq-test) + +;(defvar *address* "pgm://lo;224.0.0.1:8000") +(defvar *bind-address* "tcp://lo:8000") +(defvar *connect-address* "tcp://localhost:8000") +(defvar *message-count* 1000) +(defvar *message-size* 256) +(defvar *rate* 256) -- cgit v1.2.3