Hi all,
I inadvertently discovered some bugs in the gray-streams implementation due to a stale flexi-streams that did not get upgraded properly.
There were quite a few things that needed tidying - the type checking using EQ is wrong. Also, after trying to hunt around CLHS, I think BYTE is not a CL type (though I may be mistaken). This could be a reflection of some Corman Lisp specific feature in the past (Yes! These bugs seem to have been around ABCL for that long!)
Some test examples (avoiding flexi-streams or other dependencies)
;; ----- setup
(require 'gray-streams)
(defclass test-gray-binary-input-stream (gray-streams:fundamental-binary-input-stream) ())
(defclass test-gray-character-input-stream (gray-streams:fundamental-character-input-stream) ())
(let ((bytes (list 65 66 67 68 69)) (pos -1)) (defmethod gray-streams::stream-read-byte ((stream test-gray-binary-input-stream)) (elt bytes (mod (incf pos) 5))))
(let ((chars (list #\A #\B #\C #\D #\E)) (pos -1)) (defmethod gray-streams::stream-read-char ((stream test-gray-character-input-stream)) (elt chars (mod (incf pos) 5))))
;; -----
(let ((s (make-instance 'test-gray-binary-input-stream))) (loop repeat 10 collect (read-byte s))) => (65 66 67 68 69 65 66 67 68 69)
(let ((arr (make-array 10 :initial-element nil))) (list (read-sequence arr (make-instance 'test-gray-binary-input-stream) :start 2 :end 7) arr))
Expected (7 #(NIL NIL 65 66 67 68 69 NIL NIL NIL)), but got an error instead, ... no applicable method...
I guess implementors of gray streams would normally define both read-byte & read-sequence. But ABCL's default/fallback method does try to support binary streams, it just isn't being dispatched on the right class.
Also, read-sequence should return the "final" array index, not the number of bytes/characters read. Similarly, write-sequence needs to do its work, then return the sequence itself.
Attached is a patch with what I hope are the required fixes, please review.
Two additional notes,
1. Even after loading my proposed fixes, I'd would still get this, which is ok, but the error message is not obvious enough. This is actually a method not found error, not a type error. There appears to be some type guessing going on, so perhaps the message should indicate failure to dispatch on STREAM or similar.
(gray-streams::stream-element-type (make-instance 'test-gray-binary-input-stream)) The value #<TEST-GRAY-BINARY-INPUT-STREAM {500BBBF9}> is not of type STREAM. [Condition of type TYPE-ERROR]
So we just need something like this,
(defmethod gray-streams::stream-element-type ((stream test-gray-binary-input-stream)) '(unsigned-byte 8))
2. I also don't understand this snippet in the original code, converting unsigned-bytes/integers (basically the wrong type) to a character?
(#+nil ccl:int-char code-char (elt sequence n))
Of course, after patching the above, I discovered that re-compiling flexi-streams got rid of my particular set of problems :-)
Yong