# Primitives for the 9P2000 protocol.
#
# See http://cm.bell-labs.com/sys/man/5/INDEX.html
# See http://swtch.com/plan9port/man/man9/
module Rumai
module IXP
# define constants for easier bit manipulation of 9P2000 field values
# uchar (1 byte), ushort (2 bytes), uint32 (4 bytes), uint64 (8 bytes)
4.times do |n|
bytes = 2 ** n
bits = 8 * bytes
limit = 2 ** bits
mask = limit - 1
const_set "BYTE#{bytes}_BITS", bits
const_set "BYTE#{bytes}_LIMIT", limit
const_set "BYTE#{bytes}_MASK", mask
end
##
# A 9P2000 byte stream.
#
module Stream
# uchar, ushort, uint32 (all of them little-endian)
PACKING_FLAGS = { 1 => 'C', 2 => 'v', 4 => 'V' }.freeze
##
# Unpacks the given number of bytes from this 9P2000 byte stream.
#
def read_9p num_bytes
read(num_bytes).unpack(PACKING_FLAGS[num_bytes])[0]
end
end
##
# A common container for exceptions concerning IXP.
#
class Error < StandardError
end
##
# A serializable 9P2000 data structure.
#
module Struct
attr_reader :fields
##
# Allows field values to be initialized via the constructor.
#
# @param field_values
# a mapping from field name to field value
#
def initialize field_values = {}
@fields = self.class.fields
@values = field_values
end
##
# Transforms this object into a string of 9P2000 bytes.
#
def to_9p
@fields.inject([]) {|s,f| s << f.to_9p(@values) }.join
end
##
# Populates this object with information
# from the given 9P2000 byte stream.
#
def load_9p stream
@fields.each do |f|
f.load_9p stream, @values
end
end
##
# Provides a convenient DSL (for defining fields)
# to all objects which *include* this module.
#
def self.included target
class << target
##
# Returns a list of fields which compose this Struct.
#
def fields
@fields ||=
if superclass.respond_to? :fields
superclass.fields.dup
else
[]
end
end
##
# Defines a new field in this Struct.
#
# @param args
# arguments for {Field.new}
#
def field name, format = nil, *args
klass = Field.factory(format)
field = klass.new(name.to_sym, format, *args)
# register field as being part of this structure
fields << field
# provide accessor methods to field values
self.class_eval <<-EOS, __FILE__, __LINE__
def #{field.name}
@values[#{field.name.inspect}]
end
def #{field.name}= value
@values[#{field.name.inspect}] = value
end
EOS
field
end
##
# Creates a new instance of this class from the
# given 9P2000 byte stream and returns the instance.
#
def from_9p stream, msg_class = self
msg = msg_class.new
msg.load_9p(stream)
msg
end
end
end
##
# A field inside a Struct.
#
# A field's value is considered to be either:
# * array of format when counter && format.is_a? Class
# * raw byte string when counter && format.nil?
#
class Field
attr_reader :name, :format, :counter, :countee
##
# @param name
# unique (among all fields in a struct) name for the field
#
# @param format
# number of bytes, a class, or nil
#
# @param [Field] counter
# field which counts the length of this field's value
#
def initialize name, format = nil, counter = nil
@name = name
@format = format
@countee = nil
self.counter = counter
end
##
# Sets the counter for this field (implying that the
# length of this field is counted by the given field).
#
def counter= field
if @counter = field
extend CounteeField
@counter.countee = self
end
end
##
# Sets the countee for this field (implying that
# this field counts the length of the given field).
#
def countee= field
if @countee = field
extend CounterField
end
end
##
# Returns a Field class that best represents the given format.
#
def self.factory format
if format == String
StringField
elsif format.is_a? Class
ClassField
elsif format == 8
Integer8Field
else
Field
end
end
##
# Transforms this object into a string of 9P2000 bytes.
#
def to_9p field_values
value_to_9p field_values[@name]
end
##
# Populates this object with information
# taken from the given 9P2000 byte stream.
#
def load_9p stream, field_values
field_values[@name] = value_from_9p stream
end
private
##
# Converts the given value, according to the format
# of this field, into a string of 9P2000 bytes.
#
def value_to_9p value
value.to_i.to_9p @format.to_i
end
##
# Parses a value, according to the format of
# this field, from the given 9P2000 byte stream.
#
def value_from_9p stream
stream.read_9p @format.to_i
end
##
# Methods for a field that counts the length of another field.
#
module CounterField
def to_9p field_values
value = field_values[@countee.name]
count =
case value
when String then value.bytesize
else value.length
end
value_to_9p count
end
end
##
# Methods for a field whose length is counted by another field.
#
module CounteeField
def to_9p field_values
value = field_values[@name]
if @format
value.map {|v| value_to_9p v}.join
else
value.to_s # raw byte sequence
end
end
def load_9p stream, field_values
count = field_values[@counter.name].to_i
field_values[@name] =
if @format
Array.new(count) { value_from_9p stream }
else
stream.read(count) # raw byte sequence
end
end
end
end
##
# A field whose value knows how to convert itself to and from 9p.
#
class ClassField < Field
def value_to_9p value
value.to_9p
end
def value_from_9p stream
@format.from_9p stream
end
end
##
# A field whose value is a string.
#
class StringField < ClassField
def value_to_9p value
value.to_s.to_9p
end
end
##
# A field whose value is a 8-byte integer.
#
class Integer8Field < Field
def value_to_9p value
v = value.to_i
(BYTE4_MASK & v).to_9p(4) << # lower bytes
(BYTE4_MASK & (v >> BYTE4_BITS)).to_9p(4) # higher bytes
end
def value_from_9p stream
stream.read_9p(4) | (stream.read_9p(4) << BYTE4_BITS)
end
end
end
##
# Holds information about a file being accessed on a 9P2000 server.
#
# See http://cm.bell-labs.com/magic/man2html/5/intro
#
class Qid
include Struct
# type[1] version[4] path[8]
field :type , 1
field :version , 4
field :path , 8
##
# The following constant definitions originate from:
# http://swtch.com/usr/local/plan9/include/libc.h
#
QTDIR = 0x80 # type bit for directories
QTAPPEND = 0x40 # type bit for append only files
QTEXCL = 0x20 # type bit for exclusive use files
QTMOUNT = 0x10 # type bit for mounted channel
QTAUTH = 0x08 # type bit for authentication file
QTTMP = 0x04 # type bit for non-backed-up file
QTSYMLINK = 0x02 # type bit for symbolic link
QTFILE = 0x00 # type bits for plain file
end
##
# Holds information about a file on a 9P2000 server.
#
# See http://cm.bell-labs.com/magic/man2html/5/stat
#
class Stat
include Struct
field :size , 2
field :type , 2
field :dev , 4
field :qid , Qid
field :mode , 4
field :atime , Time
field :mtime , Time
field :length , 8
field :name , String
field :uid , String
field :gid , String
field :muid , String
##
# The following constant definitions originate from:
# http://swtch.com/usr/local/plan9/include/libc.h
#
DMDIR = 0x80000000 # mode bit for directories
DMAPPEND = 0x40000000 # mode bit for append only files
DMEXCL = 0x20000000 # mode bit for exclusive use files
DMMOUNT = 0x10000000 # mode bit for mounted channel
DMAUTH = 0x08000000 # mode bit for authentication file
DMTMP = 0x04000000 # mode bit for non-backed-up file
DMSYMLINK = 0x02000000 # mode bit for symbolic link (Unix, 9P2000.u)
DMDEVICE = 0x00800000 # mode bit for device file (Unix, 9P2000.u)
DMNAMEDPIPE = 0x00200000 # mode bit for named pipe (Unix, 9P2000.u)
DMSOCKET = 0x00100000 # mode bit for socket (Unix, 9P2000.u)
DMSETUID = 0x00080000 # mode bit for setuid (Unix, 9P2000.u)
DMSETGID = 0x00040000 # mode bit for setgid (Unix, 9P2000.u)
DMREAD = 0x4 # mode bit for read permission
DMWRITE = 0x2 # mode bit for write permission
DMEXEC = 0x1 # mode bit for execute permission
##
# Tests if this file is a directory.
#
def directory?
mode & DMDIR > 0
end
end
##
# Fcall is the basic unit of communication in the 9P2000 protocol.
# It is analogous to a "packet" in the Internetwork Protocol (IP).
#
# See http://cm.bell-labs.com/magic/man2html/2/fcall
#
class Fcall
include Struct
# The first two fields are disabled because they are automatically
# calculated by the Fcall#to_9p and Fcall::from_9p methods below:
#
# field :size , 4 # disabled
# field :type , 1 # disabled
#
field :tag , 2
##
# Transforms this object into a string of 9P2000 bytes.
#
def to_9p
data = type.to_9p(1) << super
size = (data.bytesize + 4).to_9p(4)
size << data
end
class << self
alias __from_9p__ from_9p
undef from_9p
##
# Creates a new instance of this class from the
# given 9P2000 byte stream and returns the instance.
#
def from_9p stream
size = stream.read_9p(4)
type = stream.read_9p(1)
unless fcall = TYPE_TO_CLASS[type]
raise Error, "illegal fcall type: #{type}"
end
__from_9p__ stream, fcall
end
end
NOTAG = BYTE2_MASK # (ushort)
NOFID = BYTE4_MASK # (uint32)
end
# size[4] Tversion tag[2] msize[4] version[s]
class Tversion < Fcall
field :msize , 4
field :version , String
VERSION = '9P2000'.freeze
MSIZE = 8192 # magic number defined in Plan9
# for [TR]version and [TR]read
end
# size[4] Rversion tag[2] msize[4] version[s]
class Rversion < Fcall
field :msize , 4
field :version , String
end
# size[4] Tauth tag[2] afid[4] uname[s] aname[s]
class Tauth < Fcall
field :afid , 4
field :uname , String
field :aname , String
end
# size[4] Rauth tag[2] aqid[13]
class Rauth < Fcall
field :aqid , Qid
end
# size[4] Tattach tag[2] fid[4] afid[4] uname[s] aname[s]
class Tattach < Fcall
field :fid , 4
field :afid , 4
field :uname , String
field :aname , String
end
# size[4] Rattach tag[2] qid[13]
class Rattach < Fcall
field :qid , Qid
end
# illegal
class Terror < Fcall
def to_9p
raise Error, 'the Terror fcall cannot be transmitted'
end
end
# size[4] Rerror tag[2] ename[s]
class Rerror < Fcall
field :ename , String
end
# size[4] Tflush tag[2] oldtag[2]
class Tflush < Fcall
field :oldtag , 2
end
# size[4] Rflush tag[2]
class Rflush < Fcall
end
# size[4] Twalk tag[2] fid[4] newfid[4] nwname[2] nwname*(wname[s])
class Twalk < Fcall
field :fid , 4
field :newfid , 4
c = field :nwname , 2
field :wname , String , c
end
# size[4] Rwalk tag[2] nwqid[2] nwqid*(wqid[13])
class Rwalk < Fcall
c = field :nwqid , 2
field :wqid , Qid , c
end
# size[4] Topen tag[2] fid[4] mode[1]
class Topen < Fcall
field :fid , 4
field :mode , 1
##
# The following constant definitions originate from:
# http://swtch.com/usr/local/plan9/include/libc.h
#
OREAD = 0 # open for read
OWRITE = 1 # write
ORDWR = 2 # read and write
OEXEC = 3 # execute, == read but check execute permission
OTRUNC = 16 # or'ed in (except for exec), truncate file first
OCEXEC = 32 # or'ed in, close on exec
ORCLOSE = 64 # or'ed in, remove on close
ODIRECT = 128 # or'ed in, direct access
ONONBLOCK = 256 # or'ed in, non-blocking call
OEXCL = 0x1000 # or'ed in, exclusive use (create only)
OLOCK = 0x2000 # or'ed in, lock after opening
OAPPEND = 0x4000 # or'ed in, append only
end
# size[4] Ropen tag[2] qid[13] iounit[4]
class Ropen < Fcall
field :qid , Qid
field :iounit , 4
end
# size[4] Tcreate tag[2] fid[4] name[s] perm[4] mode[1]
class Tcreate < Fcall
field :fid , 4
field :name , String
field :perm , 4
field :mode , 1
end
# size[4] Rcreate tag[2] qid[13] iounit[4]
class Rcreate < Fcall
field :qid , Qid
field :iounit , 4
end
# size[4] Tread tag[2] fid[4] offset[8] count[4]
class Tread < Fcall
field :fid , 4
field :offset , 8
field :count , 4
end
# size[4] Rread tag[2] count[4] data[count]
class Rread < Fcall
c = field :count , 4
field :data , nil , c
end
# size[4] Twrite tag[2] fid[4] offset[8] count[4] data[count]
class Twrite < Fcall
field :fid , 4
field :offset , 8
c = field :count , 4
field :data , nil , c
end
# size[4] Rwrite tag[2] count[4]
class Rwrite < Fcall
field :count , 4
end
# size[4] Tclunk tag[2] fid[4]
class Tclunk < Fcall
field :fid , 4
end
# size[4] Rclunk tag[2]
class Rclunk < Fcall
end
# size[4] Tremove tag[2] fid[4]
class Tremove < Fcall
field :fid , 4
end
# size[4] Rremove tag[2]
class Rremove < Fcall
end
# size[4] Tstat tag[2] fid[4]
class Tstat < Fcall
field :fid , 4
end
# size[4] Rstat tag[2] stat[n]
class Rstat < Fcall
field :nstat , 2
field :stat , Stat
end
# size[4] Twstat tag[2] fid[4] stat[n]
class Twstat < Fcall
field :fid , 4
field :nstat , 2
field :stat , Stat
end
# size[4] Rwstat tag[2]
class Rwstat < Fcall
end
##
# A remote function call (fcall).
#
class Fcall
CLASS_TO_TYPE = {
Tversion => 100,
Rversion => 101,
Tauth => 102,
Rauth => 103,
Tattach => 104,
Rattach => 105,
Terror => 106,
Rerror => 107,
Tflush => 108,
Rflush => 109,
Twalk => 110,
Rwalk => 111,
Topen => 112,
Ropen => 113,
Tcreate => 114,
Rcreate => 115,
Tread => 116,
Rread => 117,
Twrite => 118,
Rwrite => 119,
Tclunk => 120,
Rclunk => 121,
Tremove => 122,
Rremove => 123,
Tstat => 124,
Rstat => 125,
Twstat => 126,
Rwstat => 127,
}.freeze
TYPE_TO_CLASS = CLASS_TO_TYPE.invert.freeze
##
# Returns the value of the 'type' field for this fcall.
#
def self.type
CLASS_TO_TYPE[self]
end
##
# Returns the value of the 'type' field for this fcall.
#
def type
self.class.type
end
end
end
end
class Integer
##
# Transforms this object into a string of 9P2000 bytes.
#
def to_9p num_bytes
[self].pack Rumai::IXP::Stream::PACKING_FLAGS[num_bytes]
end
end
# count[2] s[count]
class String
##
# Transforms this object into a string of 9P2000 bytes.
#
def to_9p
count = [bytesize, Rumai::IXP::BYTE2_MASK].min
count.to_9p(2) << byteslice(0, count)
end
##
# Creates a new instance of this class from the
# given 9P2000 byte stream and returns the instance.
#
def self.from_9p stream
stream.read stream.read_9p(2)
end
unless method_defined? :byteslice
##
# Does the same thing as String#slice but
# operates on bytes instead of characters.
#
def byteslice(*args)
unpack('C*').slice(*args).pack('C*')
end
end
end
class Time
##
# Transforms this object into a string of 9P2000 bytes.
#
def to_9p
to_i.to_9p(4)
end
##
# Creates a new instance of this class from the
# given 9P2000 byte stream and returns the instance.
#
def self.from_9p stream
at stream.read_9p(4)
end
end
class IO
include Rumai::IXP::Stream
end
require 'stringio'
class StringIO
include Rumai::IXP::Stream
end