aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--lang/nqc/Makefile10
-rw-r--r--lang/nqc/distinfo2
-rw-r--r--lang/nqc/files/RCX_USBTowerPipe_fbsd.cpp320
-rw-r--r--lang/nqc/files/patch-aa21
-rw-r--r--lang/nqc/pkg-comment1
5 files changed, 340 insertions, 14 deletions
diff --git a/lang/nqc/Makefile b/lang/nqc/Makefile
index 9830c9fd3eab..be2d3508e91f 100644
--- a/lang/nqc/Makefile
+++ b/lang/nqc/Makefile
@@ -6,18 +6,22 @@
#
PORTNAME= nqc
-PORTVERSION= 2.4.r3
+PORTVERSION= 2.5.a5
CATEGORIES= lang
-MASTER_SITES= http://www.baumfamily.org/nqc/release/
+MASTER_SITES= http://www.baumfamily.org/nqc/beta/
EXTRACT_SUFX= .tgz
-MAINTAINER= kbyanc@FreeBSD.org
+MAINTAINER= jhay@FreeBSD.org
+COMMENT= A compiler for writing programs for the Lego RCX
USE_GMAKE= yes
CC= ${CXX}
MAN1= nqc.1
+pre-configure:
+ ${CP} ${FILESDIR}/RCX_USBTowerPipe_fbsd.cpp ${WRKSRC}/rcxlib/
+
do-install:
${INSTALL_PROGRAM} ${WRKSRC}/bin/nqc ${PREFIX}/bin
${INSTALL_MAN} ${WRKSRC}/nqc-man-2.1r1-0.man ${PREFIX}/man/man1/nqc.1
diff --git a/lang/nqc/distinfo b/lang/nqc/distinfo
index 62c03a49f299..759ef7f6a5ac 100644
--- a/lang/nqc/distinfo
+++ b/lang/nqc/distinfo
@@ -1 +1 @@
-MD5 (nqc-2.4.r3.tgz) = 5fc032b990dc065d523285d9ccab91b4
+MD5 (nqc-2.5.a5.tgz) = cfd711ad16d43c3cf959a932f6dc19c5
diff --git a/lang/nqc/files/RCX_USBTowerPipe_fbsd.cpp b/lang/nqc/files/RCX_USBTowerPipe_fbsd.cpp
new file mode 100644
index 000000000000..fd0eaacf61aa
--- /dev/null
+++ b/lang/nqc/files/RCX_USBTowerPipe_fbsd.cpp
@@ -0,0 +1,320 @@
+
+#include <sys/types.h>
+#include <sys/ioctl.h>
+#include <errno.h>
+#include <fcntl.h>
+#include <poll.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <unistd.h>
+#include <dev/usb/usb.h>
+
+#include "RCX_Pipe.h"
+
+#include "PDebug.h"
+
+#define kVendorID 1684
+#define kProductID 1
+#define kConfiguration 0
+#define kReadPipe 1
+#define kWritePipe 2
+
+// these should come from a LEGO include
+#define LTW_REQ_GET_PARM 1
+#define LTW_REQ_SET_PARM 2
+#define LTW_PARM_RANGE 2
+#define LTW_RANGE_SHORT 1
+#define LTW_RANGE_MEDIUM 2
+
+
+#define LTW_REQ_SET_TX_SPEED 0xEF
+#define LTW_REQ_SET_RX_SPEED 0xF1
+
+#define SPEED_COMM_BAUD_2400 8
+#define SPEED_COMM_BAUD_4800 0x10
+
+#define LTW_REQ_SET_TX_CARRIER_FREQUENCY 0xF4
+
+#define UInt32 uint32_t
+#define UInt16 uint16_t
+#define UInt8 uint8_t
+#define IOReturn int
+
+typedef struct LTW_REQ_REPLY_HEADER {
+ UInt16 wNoOfBytes; // Number of bytes in the reply
+ UInt8 bErrCode; // Request return code
+ UInt8 bValue; // Request return value
+} LTW_REQ_REPLY_HEADER;
+
+typedef LTW_REQ_REPLY_HEADER LTW_REQ_GET_SET_PARM_REPLY;
+
+class RCX_USBTowerPipe_fbsd : public RCX_Pipe
+{
+public:
+ RCX_USBTowerPipe_fbsd();
+ ~RCX_USBTowerPipe_fbsd() { Close(); }
+
+ virtual RCX_Result Open(const char *name, int mode);
+ virtual void Close();
+
+ virtual int GetCapabilities() const;
+ virtual RCX_Result SetMode(int mode);
+
+ virtual long Read(void *ptr, long count, long timeout_ms);
+ virtual long Write(const void *ptr, long count);
+
+private:
+ enum {
+ kReadPacketSize = 8
+ };
+
+ IOReturn OpenDevice(short vendorID, short productID);
+ IOReturn OpenInterface();
+ IOReturn ControlRequest(UInt8 request, UInt16 value);
+ IOReturn ControlRequest(UInt8 request, UInt8 loByte, UInt8 hiByte) {
+ return ControlRequest(request, loByte + (hiByte << 8));
+ }
+
+ int fdCntrl;
+ int fdRd;
+ int fdWr;
+
+ char *devname;
+};
+
+
+RCX_Pipe* RCX_NewUSBTowerPipe()
+{
+ return new RCX_USBTowerPipe_fbsd();
+}
+
+
+RCX_USBTowerPipe_fbsd::RCX_USBTowerPipe_fbsd()
+ : fdCntrl(0), devname(0)
+{
+}
+
+
+RCX_Result RCX_USBTowerPipe_fbsd::Open(const char *name, int mode)
+{
+ IOReturn err;
+
+ if (mode != kNormalIrMode) return kRCX_PipeModeError;
+
+ err = OpenDevice(kVendorID, kProductID);
+ PREQUIRENOT(err, Fail_OpenDevice);
+
+ err = OpenInterface();
+ PREQUIRENOT(err, Fail_OpenInterface);
+
+ UInt8 range;
+ range = (strcmp(name, "short")==0) ? LTW_RANGE_SHORT : LTW_RANGE_MEDIUM;
+ ControlRequest(LTW_REQ_SET_PARM, LTW_PARM_RANGE,range);
+
+ return 0;
+
+Fail_OpenInterface:
+Fail_OpenDevice:
+ Close();
+ return kRCX_OpenSerialError;
+}
+
+
+void RCX_USBTowerPipe_fbsd::Close()
+{
+ if (devname != NULL) {
+ close(fdWr);
+ close(fdRd);
+ close(fdCntrl);
+ devname = NULL;
+ }
+}
+
+
+int RCX_USBTowerPipe_fbsd::GetCapabilities() const
+{
+ return kNormalIrMode | kFastIrMode | kFastOddParityFlag | kAbsorb55Flag;
+}
+
+
+RCX_Result RCX_USBTowerPipe_fbsd::SetMode(int mode)
+{
+ switch(mode) {
+ case kNormalIrMode:
+ ControlRequest(LTW_REQ_SET_TX_SPEED, SPEED_COMM_BAUD_2400);
+ ControlRequest(LTW_REQ_SET_RX_SPEED, SPEED_COMM_BAUD_2400);
+ return kRCX_OK;
+ case kFastIrMode:
+ ControlRequest(LTW_REQ_SET_PARM, LTW_PARM_RANGE, LTW_RANGE_SHORT);
+ ControlRequest(LTW_REQ_SET_TX_SPEED, SPEED_COMM_BAUD_4800);
+ ControlRequest(LTW_REQ_SET_RX_SPEED, SPEED_COMM_BAUD_4800);
+ ControlRequest(LTW_REQ_SET_TX_CARRIER_FREQUENCY, 38);
+ return kRCX_OK;
+ default:
+ return kRCX_PipeModeError;
+ }
+}
+
+
+#define MAX_PACKET 200
+
+long RCX_USBTowerPipe_fbsd::Write(const void *ptr, long length)
+{
+ const unsigned char *data = (const unsigned char *)ptr;
+
+ int total = 0;
+
+// printf("Write %ld\n", length);
+
+ while(length > 0) {
+ IOReturn err;
+ int count = length;
+ if (count > MAX_PACKET)
+ count = MAX_PACKET;
+ err = write(fdWr, (void *)data, count);
+ if (err == count)
+ err = 0;
+ PREQUIRENOT(err, Fail_WritePipe);
+
+ length -= count;
+ data += count;
+ total += count;
+ }
+
+Fail_WritePipe:
+// printf("Done %d\n", total);
+ return total;
+}
+
+
+
+long RCX_USBTowerPipe_fbsd::Read(void *data, long length, long timeout_ms)
+{
+ const unsigned char *rdptr = (const unsigned char *)data;
+ IOReturn err;
+ int count;
+ int total = 0;
+ struct pollfd pfd;
+
+// printf("Read %ld, timeout %ld\n", length, timeout_ms);
+ while(length > 0) {
+ pfd.fd = fdRd;
+ pfd.events = POLLIN;
+ err = poll(&pfd, 1, timeout_ms);
+ if (err == 0)
+ break;
+ if (err == -1) {
+ if(errno == EINTR)
+ continue;
+ else
+ break;
+ }
+ count = length;
+ if (count > MAX_PACKET)
+ count = MAX_PACKET;
+ err = read(fdRd, (void *)rdptr, count);
+ if (err == -1)
+ break;
+
+// printf("err %d\n", err);
+ if (err == 0 || err == -1)
+ goto Fail_ReadPipe;
+
+ count = err;
+ length -= count;
+ rdptr += count;
+ total += count;
+ }
+
+Fail_ReadPipe:
+// printf("Got %d\n", total);
+ return total;
+}
+
+
+IOReturn RCX_USBTowerPipe_fbsd::OpenDevice(short vendorID, short productID)
+{
+ char *dn = "/dev/ugen0";
+ int err, fd;
+ usb_device_descriptor_t usbdd;
+
+ fd = open(dn, O_RDWR);
+ if (fd == -1)
+ return errno;
+ err = ioctl(fd, USB_GET_DEVICE_DESC, &usbdd);
+ if (err == -1) {
+ perror("ioctl error");
+ return errno;
+ }
+// printf("idVendor %d\n", UGETW(usbdd.idVendor));
+// printf("idProduct %d\n", UGETW(usbdd.idProduct));
+ if (UGETW(usbdd.idVendor) != kVendorID ||
+ UGETW(usbdd.idProduct) != kProductID) {
+ fprintf(stderr, "This is not an USB IR Tower!\n");
+ return -1;
+ }
+
+ devname = dn;
+ fdCntrl = fd;
+
+ return 0;
+}
+
+
+IOReturn RCX_USBTowerPipe_fbsd::OpenInterface()
+{
+ char *dn;
+ int err, iov;
+
+ dn = (char *)malloc(strlen(devname) + 2 + 1);
+ strcpy(dn, devname);
+ strcat(dn, ".1");
+ fdRd = open(dn, O_RDONLY|O_NONBLOCK);
+ if (fdRd == -1) {
+ perror(dn);
+ return -1;
+ }
+ iov = 1;
+ err = ioctl(fdRd, USB_SET_SHORT_XFER, &iov);
+ if (fdRd == -1) {
+ perror("ioctl USB_SET_SHORT_XFER");
+ return -1;
+ }
+ iov = 5000;
+ err = ioctl(fdRd, USB_SET_TIMEOUT, &iov);
+ if (fdRd == -1) {
+ perror("ioctl USB_SET_TIMEOUT");
+ return -1;
+ }
+
+ strcpy(dn, devname);
+ strcat(dn, ".2");
+ fdWr = open(dn, O_WRONLY);
+ iov = 5000;
+// ioctl(fdWr, USB_SET_TIMEOUT, &iov);
+
+ return 0;
+}
+
+
+IOReturn RCX_USBTowerPipe_fbsd::ControlRequest(UInt8 request, UInt16 value)
+{
+ IOReturn err;
+ struct usb_ctl_request req;
+ LTW_REQ_GET_SET_PARM_REPLY reply;
+
+ req.ucr_request.bmRequestType = UT_READ_VENDOR_DEVICE;
+ req.ucr_request.bRequest = request;
+ USETW(req.ucr_request.wValue, value);
+ USETW(req.ucr_request.wIndex, 0);
+ USETW(req.ucr_request.wLength, sizeof(reply));
+ req.ucr_data = &reply;
+
+ err = ioctl(fdCntrl, USB_DO_REQUEST, &req);
+
+ // size of actual reply in req.wLenDone
+ //printf("%d %d %d\n", reply.wNoOfBytes, reply.bErrCode, reply.bValue);
+
+ return err;
+}
diff --git a/lang/nqc/files/patch-aa b/lang/nqc/files/patch-aa
index 6f12464af46c..338eb7133622 100644
--- a/lang/nqc/files/patch-aa
+++ b/lang/nqc/files/patch-aa
@@ -1,5 +1,5 @@
---- Makefile.orig Sun Mar 24 23:34:39 2002
-+++ Makefile Tue Nov 19 13:35:48 2002
+--- Makefile.orig Tue Jan 28 06:24:21 2003
++++ Makefile Thu Jan 30 20:19:25 2003
@@ -28,20 +28,20 @@
#
# Pick your C++ compiler.
@@ -25,7 +25,7 @@
# Link in any necessary C++ libraries
#
-@@ -50,19 +50,20 @@
+@@ -50,21 +50,22 @@
# installation information
@@ -51,8 +51,11 @@
-CFLAGS = -pipe -Iplatform -Ircxlib -Inqc -Icompiler -Wall -Wstrict-prototypes -Wmissing-prototypes
+CFLAGS += -Iplatform -Ircxlib -Inqc -Icompiler
- USBOBJ = rcxlib/RCX_USBTowerPipe_none.o
+-USBOBJ = rcxlib/RCX_USBTowerPipe_none.o
++USBOBJ = rcxlib/RCX_USBTowerPipe_fbsd.o
+ #
+ # Platform specific tweaks
@@ -78,9 +79,6 @@
ifeq ($TARGET),solaris)
# Solaris
@@ -63,7 +66,7 @@
endif
endif
-@@ -89,7 +87,7 @@
+@@ -88,7 +86,7 @@
# wants to redefine the default serial name
#
ifndef DEFAULT_SERIAL_NAME
@@ -72,20 +75,20 @@
endif
CFLAGS += -DDEFAULT_SERIAL_NAME='$(DEFAULT_SERIAL_NAME)'
-@@ -139,10 +137,10 @@
+@@ -140,10 +138,10 @@
$(MKDIR) bin
- bin/nqc : compiler/parse.cpp $(OBJ) bin
+ bin/nqc : compiler/parse.cpp $(OBJ)
- $(CC) -o $@ $(OBJ) $(LIBS)
+ $(CXX) -o $@ $(OBJ) $(LIBS)
- bin/mkdata : mkdata/mkdata.cpp nqc/SRecord.cpp bin
+ bin/mkdata : mkdata/mkdata.cpp nqc/SRecord.cpp
- $(CC) -o bin/mkdata -Inqc/ -Iplatform/ mkdata/mkdata.cpp nqc/SRecord.cpp
+ $(CXX) -o bin/mkdata -Inqc/ -Iplatform/ mkdata/mkdata.cpp nqc/SRecord.cpp
#
# clean up stuff
-@@ -202,7 +200,7 @@
+@@ -203,7 +201,7 @@
# general rule for compiling
#
.cpp.o:
diff --git a/lang/nqc/pkg-comment b/lang/nqc/pkg-comment
deleted file mode 100644
index 90dccf63d95e..000000000000
--- a/lang/nqc/pkg-comment
+++ /dev/null
@@ -1 +0,0 @@
-A compiler for writing programs for the Lego RCX