if not modules then modules = { } end modules ['node-res'] = {
version = 1.001,
comment = "companion to node-ini.mkiv",
author = "Hans Hagen, PRAGMA-ADE, Hasselt NL",
copyright = "PRAGMA ADE / ConTeXt Development Team",
license = "see context related readme files"
}
local gmatch, format = string.gmatch, string.format
local copy_node, free_node, free_list, new_node, node_type, node_id = node.copy, node.free, node.flush_list, node.new, node.type, node.id
local tonumber, round = tonumber, math.round
local glyph_node = node_id("glyph")
--[[ldx--
The next function is not that much needed but in we use
for debugging node management.
--ldx]]--
nodes = nodes or { }
nodes.whatsits = { } -- table.swapped(node.whatsits())
local reserved = { }
local whatsits = nodes.whatsits
for k, v in pairs(node.whatsits()) do
whatsits[k], whatsits[v] = v, k -- two way
end
local function register_node(n)
reserved[#reserved+1] = n
return n
end
nodes.register = register_node
function nodes.cleanup_reserved(nofboxes) -- todo
nodes.tracers.steppers.reset() -- todo: make a registration subsystem
local nr, nl = #reserved, 0
for i=1,nr do
local ri = reserved[i]
-- if not (ri.id == glue_spec and not ri.is_writable) then
free_node(reserved[i])
-- end
end
if nofboxes then
local tb = tex.box
for i=0,nofboxes do
local l = tb[i]
if l then
free_node(tb[i])
nl = nl + 1
end
end
end
reserved = { }
return nr, nl, nofboxes -- can be nil
end
function nodes.usage()
local t = { }
for n, tag in gmatch(status.node_mem_usage,"(%d+) ([a-z_]+)") do
t[tag] = n
end
return t
end
local disc = register_node(new_node("disc"))
local kern = register_node(new_node("kern",1))
local penalty = register_node(new_node("penalty"))
local glue = register_node(new_node("glue")) -- glue.spec = nil
local glue_spec = register_node(new_node("glue_spec"))
local glyph = register_node(new_node("glyph",0))
local textdir = register_node(new_node("whatsit",whatsits.dir)) -- 7 (6 is local par node)
local rule = register_node(new_node("rule"))
local latelua = register_node(new_node("whatsit",whatsits.late_lua)) -- 35
local user_n = register_node(new_node("whatsit",whatsits.user_defined)) user_n.type = 100 -- 44
local user_l = register_node(new_node("whatsit",whatsits.user_defined)) user_l.type = 110 -- 44
local user_s = register_node(new_node("whatsit",whatsits.user_defined)) user_s.type = 115 -- 44
local user_t = register_node(new_node("whatsit",whatsits.user_defined)) user_t.type = 116 -- 44
local left_margin_kern = register_node(new_node("margin_kern",0))
local right_margin_kern = register_node(new_node("margin_kern",1))
local lineskip = register_node(new_node("glue",1))
local baselineskip = register_node(new_node("glue",2))
local leftskip = register_node(new_node("glue",8))
local rightskip = register_node(new_node("glue",9))
local temp = register_node(new_node("temp",0))
function nodes.zeroglue(n)
local s = n.spec
return not writable or (
s.width == 0
and s.stretch == 0
and s.shrink == 0
and s.stretch_order == 0
and s.shrink_order == 0
)
end
function nodes.glyph(fnt,chr)
local n = copy_node(glyph)
if fnt then n.font = fnt end
if chr then n.char = chr end
return n
end
function nodes.penalty(p)
local n = copy_node(penalty)
n.penalty = p
return n
end
function nodes.kern(k)
local n = copy_node(kern)
n.kern = k
return n
end
function nodes.glue_spec(width,stretch,shrink)
local s = copy_node(glue_spec)
s.width, s.stretch, s.shrink = width, stretch, shrink
return s
end
local function someskip(skip,width,stretch,shrink)
local n = copy_node(skip)
if not width then
-- no spec
elseif tonumber(width) then
local s = copy_node(glue_spec)
s.width, s.stretch, s.shrink = width, stretch, shrink
n.spec = s
else
-- shared
n.spec = copy_node(width)
end
return n
end
function nodes.glue(width,stretch,shrink)
return someskip(glue,width,stretch,shrink)
end
function nodes.leftskip(width,stretch,shrink)
return someskip(leftskip,width,stretch,shrink)
end
function nodes.rightskip(width,stretch,shrink)
return someskip(rightskip,width,stretch,shrink)
end
function nodes.lineskip(width,stretch,shrink)
return someskip(lineskip,width,stretch,shrink)
end
function nodes.baselineskip(width,stretch,shrink)
return someskip(baselineskip,width,stretch,shrink)
end
function nodes.disc()
return copy_node(disc)
end
function nodes.textdir(dir)
local t = copy_node(textdir)
t.dir = dir
return t
end
function nodes.rule(width,height,depth,dir)
local n = copy_node(rule)
if width then n.width = width end
if height then n.height = height end
if depth then n.depth = depth end
if dir then n.dir = dir end
return n
end
function nodes.latelua(code)
local n = copy_node(latelua)
n.data = code
return n
end
function nodes.leftmarginkern(glyph,width)
local n = copy_node(left_margin_kern)
if not glyph then
logs.fatal("nodes","invalid pointer to left margin glyph node")
elseif glyph.id ~= glyph_node then
logs.fatal("nodes","invalid node type %s for left margin glyph node",node_type(glyph))
else
n.glyph = glyph
end
if width then
n.width = width
end
return n
end
function nodes.rightmarginkern(glyph,width)
local n = copy_node(right_margin_kern)
if not glyph then
logs.fatal("nodes","invalid pointer to right margin glyph node")
elseif glyph.id ~= glyph_node then
logs.fatal("nodes","invalid node type %s for right margin glyph node",node_type(p))
else
n.glyph = glyph
end
if width then
n.width = width
end
return n
end
function nodes.temp()
return copy_node(temp)
end
--[[
At some point we ran into a problem that the glue specification
of the zeropoint dimension was overwritten when adapting a glue spec
node. This is a side effect of glue specs being shared. After a
couple of hours tracing and debugging Taco and I came to the
conclusion that it made no sense to complicate the spec allocator
and settled on a writable flag. This all is a side effect of the
fact that some glues use reserved memory slots (with the zeropoint
glue being a noticeable one). So, next we wrap this into a function
and hide it for the user. And yes, LuaTeX now gives a warning as
well.
]]--
if tex.luatexversion > 51 then
function nodes.writable_spec(n)
local spec = n.spec
if not spec then
spec = copy_node(glue_spec)
n.spec = spec
elseif not spec.writable then
spec = copy_node(spec)
n.spec = spec
end
return spec
end
else
function nodes.writable_spec(n)
local spec = n.spec
if not spec then
spec = copy_node(glue_spec)
else
spec = copy_node(spec)
end
n.spec = spec
return spec
end
end
local cache = { }
function nodes.usernumber(num)
local n = cache[num]
if n then
return copy_node(n)
else
local n = copy_node(user_n)
if num then n.value = num end
return n
end
end
function nodes.userlist(list)
local n = copy_node(user_l)
if list then n.value = list end
return n
end
local cache = { } -- we could use the same cache
function nodes.userstring(str)
local n = cache[str]
if n then
return copy_node(n)
else
local n = copy_node(user_s)
n.type = 115
if str then n.value = str end
return n
end
end
function nodes.usertokens(tokens)
local n = copy_node(user_t)
if tokens then n.value = tokens end
return n
end
statistics.register("cleaned up reserved nodes", function()
return format("%s nodes, %s lists of %s", nodes.cleanup_reserved(tex.count["lastallocatedbox"]))
end) -- \topofboxstack
statistics.register("node memory usage", function() -- comes after cleanup !
return status.node_mem_usage
end)