ruby-x11/lib/X11/display.rb
Vidar Hokstad da2fe51596 Cleanups, more automated lookups of atoms to avoid needing to dereference constants.
More constants and requestsm including SendEvent, and
a helper for ClientMessage, w/example
2024-01-11 19:24:49 +00:00

693 lines
22 KiB
Ruby

# FIXME: Temp workaround
require 'stringio'
module X11
class DisplayError < X11Error; end
class ConnectionError < X11Error; end
class AuthorizationError < X11Error; end
class ProtocolError < X11Error; end
class Display
attr_accessor :socket
# Open a connection to the specified display (numbered from 0) on the specified host
def initialize(target = ENV['DISPLAY'])
target =~ /^([\w.-]*):(\d+)(?:.(\d+))?$/
host, display_id, screen_id = $1, $2, $3
family = nil
@debug = ENV["PUREX_DEBUG"].to_s.strip == "true"
if host.empty?
@socket = UNIXSocket.new("/tmp/.X11-unix/X#{display_id}")
family = :Local
host = nil
else
@socket = TCPSocket.new(host,6000+display_id)
family = :Internet
end
authorize(host, family, display_id)
@requestseq = 1
@rqueue = Queue.new # Read but not returned events
@wqueue = Queue.new
@extensions = {} # Known extensions
@atoms = {} # Interned atoms
start_io
end
def event_handler= block
@event_handler= block
end
def flush
while !@wqueue.empty?
sleep(0.01)
end
end
def display_info
@internal
end
def screens
@internal.screens.map do |s|
Screen.new(self, s)
end
end
##
# The resource-id-mask contains a single contiguous set of bits (at least 18).
# The client allocates resource IDs for types WINDOW, PIXMAP, CURSOR, FONT,
# GCONTEXT, and COLORMAP by choosing a value with only some subset of these
# bits set and ORing it with resource-id-base.
def new_id
id = (@xid_next ||= 0)
@xid_next += 1
(id & @internal.resource_id_mask) | @internal.resource_id_base
end
def read_error data
error = Form::Error.from_packet(StringIO.new(data))
# FIXME: Maybe make this configurable, as it means potentially
# keeping along really heavy requests. or alternative purge them
# more aggressively also when there are no errors, as otherwise
# the growth might be unbounded
error.request = @requests[error.sequence_number]
@requests.keys.find_all{|s| s <= error.sequence_number}.each do |s|
@requests.delete(s)
end
STDERR.puts "ERROR: #{error.inspect}"
error
end
def read_reply data
len = data.unpack("@4L")[0]
extra = len > 0 ? @socket.read(len*4) : ""
#STDERR.puts "REPLY: #{data.inspect}"
#STDERR.puts "EXTRA: #{extra.inspect}"
data + extra
end
def read_event type, data, event_class
io = StringIO.new(data)
case type
# 0 is error, not handled here
# 1 is reply, not handled here
when 2 then return Form::KeyPress.from_packet(io)
when 3 then return Form::KeyRelease.from_packet(io)
when 4 then return Form::ButtonPress.from_packet(io)
when 5 then return Form::ButtonRelease.from_packet(io)
when 6 then return Form::MotionNotify.from_packet(io)
when 7 then return Form::EnterNotify.from_packet(io)
when 8 then return Form::LeaveNotify.from_packet(io)
when 9 then return Form::FocusIn.from_packet(io)
when 10 then return Form::FocusOut.from_packet(io)
# FIXME 11: KeymapNotify
when 12 then return Form::Expose.from_packet(io)
# FIXME 13: GraphicsExposure
when 14 then return Form::NoExposure.from_packet(io)
# FIXME: 15: VisibilityNotify
when 16 then return Form::CreateNotify.from_packet(io)
when 17 then return Form::DestroyNotify.from_packet(io)
when 18 then return Form::UnmapNotify.from_packet(io)
when 19 then return Form::MapNotify.from_packet(io)
when 20 then return Form::MapRequest.from_packet(io)
when 21 then return Form::ReparentNotify.from_packet(io)
when 22 then return Form::ConfigureNotify.from_packet(io)
when 23 then return Form::ConfigureRequest.from_packet(io)
# FIXME: 24: GravityNotify
# FIXME: 25: ResizeRequest
# FIXME: 26: CirculateNotify
# FIXME: 27: CirculateRequest
when 28 then return Form::PropertyNotify.from_packet(io)
# FIXME: 29: SelectionClear
# FIXME: 30: SelectionRequest
# FIXME: 31: SelectionNotify
# FIXME: 32: ColormapNotify
when 33 then return Form::ClientMessage.from_packet(io)
# FIXME: 34: MappingNotify
else
STDERR.puts "FIXME: Event: #{type}"
STDERR.puts "EVENT: #{data.inspect}"
data
end
end
def read_full_packet(len = 32)
data = @socket.read(32)
return nil if data.nil?
while data.length < 32
IO.select([@socket],nil,nil,0.001)
data.concat(@socket.read_nonblock(32 - data.length))
end
return data
end
def read_packet
data = read_full_packet(32)
return nil if data.nil?
# FIXME: Make it configurable.
@requests.keys.find_all{|s| s <= @requestseq - 50}.each do |s|
@requests.delete(s)
end
# FIXME: What is bit 8 for? Synthentic?
type = data.unpack("C").first & 0x7f
case type
when 0 then read_error(data)
when 1 then read_reply(data)
when 2..34 then read_event(type, data, nil)
else
raise ProtocolError, "Unsupported reply type: #{type} #{data.inspect}"
end
end
def write_packet(*args)
pkt = args.join
pkt[2..3] = u16(pkt.length/4)
@wqueue << [nil,nil,pkt]
end
def write_request ob
data = ob.to_packet(self) if ob.respond_to?(:to_packet)
raise "BAD LENGTH for #{ob.inspect} (#{ob.request_length.to_i*4} ! #{data.size} " if ob.request_length && ob.request_length.to_i*4 != data.size
STDERR.puts "write_req: #{ob.inspect}" if @debug
@wqueue << [ob,nil,data]
end
def write_sync(ob, reply=nil)
data = ob.to_packet(self) if ob.respond_to?(:to_packet)
q = Queue.new
@wqueue << [ob,q,data]
STDERR.puts "write_sync_req: #{ob.inspect}" if @debug
pkt = q.shift
STDERR.puts "write_sync_rep: #{pkt.inspect}" if @debug
raise(pkt) if pkt.is_a?(X11::Form::Error)
return pkt if !pkt.is_a?(String)
reply ? reply.from_packet(StringIO.new(pkt)) : pkt
end
def peek_packet = !@rqueue.empty?
def next_packet = @rqueue.shift
def close = @rqueue.close
def start_io
@replies ||= {}
@requests ||= {}
# Read thread.
# FIXME: Drop the select.
rt = Thread.new do
while pkt = read_packet
#STDERR.puts "read: #{pkt.inspect}"
if !pkt
sleep 0.1
elsif pkt.is_a?(String)
# This is a reply. We need the sequence number.
#
seq = pkt.unpack1("@2S")
STDERR.puts " - seq= #{seq}" if @debug
STDERR.puts @replies.inspect if @debug
if @replies[seq]
q = @replies.delete(seq)
STDERR.puts " - reply to #{q}" if @debug
q << pkt
end
elsif pkt.is_a?(X11::Form::Error)
if @replies[pkt.sequence_number]
q = @replies.delete(pkt.sequence_number)
q << pkt
else
@rqueue << pkt
end
else
@rqueue << pkt
end
end
@rqueue.close
@replies.values.each(&:close)
end
# Write thread
wt = Thread.new do
while msg = @wqueue.shift
ob, q, data = *msg
@requests[@requestseq] = ob
@replies[@requestseq] = q if q
@requestseq = (@requestseq + 1) % 65536
@socket.write(data)
end
end
at_exit do
flush
@rqueue.close
@wqueue.close
# We kill this because it may be stuck in a read
# we'll never care about
Thread.kill(rt)
# We wait for this to finish because otherwise we may
# lose side-effects
wt.join
end
end
def run
loop do
pkt = next_packet
return if !pkt
yield(pkt)
end
end
def find_visual(screen, depth, qlass = 4)
self.display_info.screens[screen].depths.find{|d|
d.depth == depth }.visuals.find{|v| v.qlass = qlass }
end
def default_root = screens.first.root
# Requests
def create_window(x,y,w,h,
values: {},
depth: 32, parent: nil, border_width: 0, wclass: X11::Form::InputOutput, visual: nil
)
wid = new_id
parent ||= default_root
if visual.nil?
visual = find_visual(0, depth).visual_id
end
values[X11::Form::CWColorMap] ||= create_colormap(0, parent, visual)
values = values.sort_by{_1[0]}
mask = values.inject(0) {|acc,v| (acc | v[0]) }
values = values.map{_1[1]}
write_request(
X11::Form::CreateWindow.new(
depth, wid, parent,
x,y,w,h,border_width, wclass, visual, mask, values)
)
return wid
end
def get_window_attributes(wid)
write_sync( Form::GetWindowAttributes.new(wid), Form::WindowAttributes )
end
def change_window_attributes(wid,
values: {})
values = values.sort_by{_1[0]}
mask = values.inject(0) {|acc,v| (acc | v[0]) }
values = values.map{_1[1]}
write_request(Form::ChangeWindowAttributes.new(wid, mask, values))
end
def select_input(w, events) = change_window_attributes(w, values: {Form::CWEventMask => events})
def atom(name)
return name if name.is_a?(Integer) # Allow atom(atom_integer_or_symbol)
begin
return Form::Atoms.const_get(name.to_sym) if Form::Atoms.const_defined?(name.to_sym)
rescue
# const_defined? will throw if name isn't a valid constant name, but
# that's fine
end
name = name.to_sym
intern_atom(false, name) if !@atoms[name]
@atoms[name]
end
def query_extension(name)
r = write_sync(Form::QueryExtension.new(name), Form::QueryExtensionReply)
@extensions[name] = { major: r.major_opcode }
r
end
def major_opcode(name)
if !@extensions[name]
query_extension(name)
end
raise "No such extension '#{name}'" if !@extensions[name]
@extensions[name][:major]
end
def intern_atom(flag, name)
reply = write_sync(Form::InternAtom.new(flag, name.to_s),Form::InternAtomReply)
if reply
@atoms[name.to_sym] = reply.atom
end
end
def get_atom_name(atom) = write_sync(Form::GetAtomName.new(atom), Form::AtomName)&.name
def destroy_window(window) = write_request(Form::DestroyWindow.new(window))
def get_geometry(drawable) = write_sync(Form::GetGeometry.new(drawable), Form::Geometry)
def get_keyboard_mapping(min_keycode=display_info.min_keycode, count= display_info.max_keycode - min_keycode)
write_sync(Form::GetKeyboardMapping.new(min_keycode, count), Form::GetKeyboardMappingReply)
end
def create_colormap(alloc, window, visual)
mid = new_id
write_request(Form::CreateColormap.new(alloc, mid, window, visual))
mid
end
def get_property(window, property, type, offset: 0, length: 4, delete: false)
property = atom(property)
type = atom_enum(type)
result = write_sync(Form::GetProperty.new(
delete, window, property, type, offset, length
), Form::Property)
if result && result.format != 0
case result.format
when 16
result.value = result.value.unpack("v")
result.value = result.value.first if length == 2
when 32
result.value = result.value.unpack("V")
result.value = result.value.first if length == 4
end
elsif result
result.value = nil
end
result
end
def change_property(mode, window, property, type, format, data)
property = atom(property.to_sym) if property.is_a?(Symbol) || property.is_a?(String)
mode = open_enum(mode, {replace: 0, prepend: 1, append: 2})
type = atom_enum(type)
write_request(Form::ChangeProperty.new(mode, window, property, type, format, data))
end
def list_fonts(...) = write_sync(Form::ListFonts.new(...), Form::ListFontsReply)
def open_font(...) = write_request(Form::OpenFont.new(...))
def change_gc(...) = write_request(Form::ChangeGC.new(...))
def change_save_set(...)= write_request(Form::ChangeSaveSet.new(...))
def reparent_window(window, parent, x, y, save: true)
# You so almost always want this that it should've been a single request
change_save_set(0, window) if save
write_request(Form::ReparentWindow.new(window, parent, x,y))
end
def map_window(...) = write_request(Form::MapWindow.new(...))
def unmap_window(...) = write_request(Form::UnmapWindow.new(...))
def u8(*args) = args.pack("c*")
def u16(*args) = args.pack("v*")
def u32(*args) = args.pack("V*")
def atom_enum(val)
open_enum(val, {cardinal: Form::CardinalAtom, atom: Form::AtomAtom, window: Form::WindowAtom}) || atom(val)
end
def window(*args)
args.each {|a| raise "Window expected" if a.nil? }
u32(*args)
end
def open_enum(val, map) = (map[val].nil? ? val : map[val])
def set_input_focus(revert_to, focus, time=:now)
# FIXME: This is an experiment.
# Upside: Simpler. Downside: Doesn't work server-side.
# Probably a bad idea.
revert_to = open_enum(revert_to, {none: 0, pointer_root: 1, parent: 2})
focus = open_enum(focus, {none: 0, pointer_root: 1 })
time = open_enum(time, {current_time: 0, now: 0})
write_packet(u8(42,revert_to), u16(3), window(focus), u32(time))
end
def grab_key(owner_events, grab_window, modifiers, keycode, pointer_mode, keyboard_mode)
write_request(Form::GrabKey.new(
owner_events,
grab_window,
modifiers,
keycode,
pointer_mode == :async ? 1 : 0,
keyboard_mode == :async ? 1 : 0
))
end
def grab_button(owner_events, grab_window, event_mask, pointer_mode,
keyboard_mode, confine_to, cursor, button, modifiers)
write_request(Form::GrabButton.new(
owner_events, grab_window, event_mask,
pointer_mode == :async ? 1 : 0,
keyboard_mode == :async ? 1 : 0,
confine_to.to_i, cursor.to_i, button, modifiers)
)
end
def set_value(values, mask, x)
if x
values << x
mask
else
0
end
end
def configure_window(window, x: nil, y: nil, width: nil, height: nil,
border_width: nil, sibling: nil, stack_mode: nil)
values = []
mask = 0
mask |= set_value(values, 0x001, x)
mask |= set_value(values, 0x002, y)
mask |= set_value(values, 0x004, width)
mask |= set_value(values, 0x008, height)
mask |= set_value(values, 0x010, border_width)
mask |= set_value(values, 0x020, sibling)
if stack_mode
mask |= 0x040
values << case stack_mode
when :above then 0
when :below then 1
when :top_if then 2
when :bottom_if then 3
when :opposite then 4
else raise "Unknown stack_mode #{stack_mode.inspect}"
end
end
write_request(X11::Form::ConfigureWindow.new(window, mask, values))
end
def create_gc(window, foreground: nil, background: nil,
graphics_exposures: nil
)
mask = 0
args = []
# FIXME:
# The rest can be found here:
# https://tronche.com/gui/x/xlib/GC/manipulating.html#XGCValues
mask |= set_value(args, 0x04, foreground)
mask |= set_value(args, 0x08, background)
mask |= set_value(args, 0x10000, graphics_exposures)
gc = new_id
write_request(X11::Form::CreateGC.new(gc, window, mask, args))
gc
end
def send_event(...) = write_request(Form::SendEvent.new(...))
def client_message(window: default_root, type: :ClientMessage, format: 32, destination: default_root, mask: 0, data: [], propagate: true)
f = {8 => "C20", 16 => "S10", 32 => "L5"}[format]
p f
data = (Array(data).map{|item|atom(item)} +[0]*20).pack(f)
event = Form::ClientMessage.new(
format, 0, window, atom(type), data
)
event.code =33
pp event
send_event(propagate, destination, mask, event)
end
def put_image(*args) = write_request(X11::Form::PutImage.new(*args))
def clear_area(*args) = write_request(X11::Form::ClearArea.new(*args))
def copy_area(*args) = write_request(X11::Form::CopyArea.new(*args))
def image_text8(*args) = write_request(X11::Form::ImageText8.new(*args))
def image_text16(*args)= write_request(X11::Form::ImageText16.new(*args))
def poly_fill_rectangle(wid, gc, *rects)
rects = rects.map{|r| r.is_a?(Array) ? Form::Rectangle.new(*r) : r}
write_request(X11::Form::PolyFillRectangle.new(wid, gc, rects))
end
def create_pixmap(depth, drawable, w,h)
new_id.tap{|pid| write_request(Form::CreatePixmap.new(depth, pid, drawable, w,h)) }
end
# XRender
def render_opcode
return @render_opcode if @render_opcode
@render_opcode = major_opcode("RENDER")
if @render_opcode
@render_version = write_sync(X11::Form::XRenderQueryVersion.new(
@render_opcode,0,11),
X11::Form::XRenderQueryVersionReply
)
end
@render_opcode
end
def render_create_picture(drawable, format, vmask=0, vlist=[])
pid = new_id
write_request(X11::Form::XRenderCreatePicture.new(
render_opcode, pid, drawable, format, vmask, vlist))
pid
end
def render_query_pict_formats
@render_formats ||= write_sync(
Form::XRenderQueryPictFormats.new(render_opcode),
Form::XRenderQueryPictFormatsReply
)
end
def render_find_visual_format(visual)
# FIXME.
render_query_pict_formats.screens.map do |s|
s.depths.map do |d|
d.visuals.map {|v| v.visual == visual ? v : nil }
end
end.flatten.compact.first&.format
end
def render_find_standard_format(sym)
# A pox be on the people who made this necessary
formats = render_query_pict_formats
case sym
when :a8
@a8 ||= formats.formats.find do |f|
f.type == 1 &&
f.depth == 8 &&
f.direct.alpha_mask == 255
end
when :rgb24
@rgb24 ||= formats.formats.find do |f|
f.type == 1 &&
f.depth == 24 &&
f.direct.red == 16 &&
f.direct.green == 8 &&
f.direct.blue == 0
end
when :argb24
@argb24 ||= formats.formats.find do |f|
f.type == 1 &&
f.depth == 32 &&
f.direct.alpha == 24 &&
f.direct.red == 16 &&
f.direct.green == 8 &&
f.direct.blue == 0
end
else
raise "Unsupported format (a4/a1 by omission)"
end
end
def render_create_glyph_set(format)
glyphset = new_id
write_request(X11::Form::XRenderCreateGlyphSet.new(
major_opcode("RENDER"),glyphset, format))
glyphset
end
def render_add_glyphs(glyphset, glyphids, glyphinfos, data)
write_request(X11::Form::XRenderAddGlyphs.new(render_opcode,
glyphset, Array(glyphids), Array(glyphinfos), data))
end
def render_fill_rectangles(op, dst, color, rects)
color = Form::XRenderColor.new(*color) if color.is_a?(Array)
rects = rects.map{|r| r.is_a?(Array) ? Form::Rectangle.new(*r) : r}
write_request(Form::XRenderFillRectangles.new(render_opcode, op, dst, color, rects))
end
def render_composite_glyphs32(op, src, dst, fmt, glyphset, srcx,srcy, *elts)
write_request(X11::Form::XRenderCompositeGlyphs32.new(
render_opcode,
op, src, dst, fmt,
glyphset,
srcx, srcy,
elts.map {|e| e.is_a?(Array) ? Form::GlyphElt32.new(*e) : e }
))
end
def render_create_solid_fill(*color)
if color.length == 1 && color.is_a?(Form::XRenderColor)
color = color[0]
else
color = Form::XRenderColor.new(*color)
end
fill = new_id
write_request(Form::XRenderCreateSolidFill.new(render_opcode,fill,color))
fill
end
private
def authorize(host, family, display_id)
auth = Auth.new
auth_info = auth.get_by_hostname(host||"localhost", family, display_id)
if auth_info
auth_name, auth_data = auth_info.auth_name, auth_info.auth_data
else
auth_name = ""
auth_data = ""
end
handshake = Form::ClientHandshake.new(
Protocol::BYTE_ORDER,
Protocol::MAJOR,
Protocol::MINOR,
auth_name,
auth_data
)
@socket.write(handshake.to_packet(self))
data = @socket.read(1)
raise AuthorizationError, "Failed to read response from server" if !data
case data.unpack("w").first
when X11::Auth::FAILED
len, major, minor, xlen = @socket.read(7).unpack("CSSS")
reason = @socket.read(xlen * 4)
reason = reason[0..len]
raise AuthorizationError, "Connection to server failed -- (version #{major}.#{minor}) #{reason}"
when X11::Auth::AUTHENTICATE
raise AuthorizationError, "Connection requires authentication"
when X11::Auth::SUCCESS
@socket.read(7) # skip unused bytes
@internal = Form::DisplayInfo.from_packet(@socket)
else
raise AuthorizationError, "Received unknown opcode #{type}"
end
end
def to_s
"#<X11::Display:0x#{object_id.to_s(16)} screens=#{@internal.screens.size}>"
end
end
end