diff options
| -rw-r--r-- | src/fontloader/misc/fontloader-basics.tex | 70 | ||||
| -rw-r--r-- | src/fontloader/misc/fontloader-font-map.lua | 53 | ||||
| -rw-r--r-- | src/fontloader/misc/fontloader-font-otf.lua | 343 | ||||
| -rw-r--r-- | src/fontloader/misc/fontloader-font-otp.lua | 2 | ||||
| -rw-r--r-- | src/fontloader/misc/fontloader-fonts-cbk.lua | 42 | ||||
| -rw-r--r-- | src/fontloader/misc/fontloader-fonts-otn.lua | 20 | ||||
| -rw-r--r-- | src/fontloader/misc/fontloader-l-table.lua | 280 | ||||
| -rw-r--r-- | src/fontloader/misc/fontloader-mplib.tex | 1 | ||||
| -rw-r--r-- | src/fontloader/misc/fontloader-plain.tex | 1 | ||||
| -rw-r--r-- | src/fontloader/misc/fontloader-util-str.lua | 2 | ||||
| -rw-r--r-- | src/fontloader/runtime/fontloader-fontloader.lua | 473 | 
11 files changed, 841 insertions, 446 deletions
| diff --git a/src/fontloader/misc/fontloader-basics.tex b/src/fontloader/misc/fontloader-basics.tex index abe4989..1180c68 100644 --- a/src/fontloader/misc/fontloader-basics.tex +++ b/src/fontloader/misc/fontloader-basics.tex @@ -20,4 +20,74 @@    {\global\advance\lastallocatedattribute 1     \attributedef#1\lastallocatedattribute} +% maybe we will have luatex-basics.lua some day for instance when more +% (pdf) primitives have moved to macros) + +\directlua { + +    gadgets = gadgets or { } % reserved namespace + +    gadgets.functions = { } +    local registered  = {} + +    function gadgets.functions.reverve() +        local numb = newtoken.scan_int() +        local name = newtoken.scan_string() +        local okay = string.gsub(name,"[\string\\ ]","") +        registered[okay] = numb +        texio.write_nl("reserving lua function '"..okay.."' with number "..numb) +    end + +    function gadgets.functions.register(name,f) +        local okay = string.gsub(name,"[\string\\ ]","") +        local numb = registered[okay] +        if numb then +            texio.write_nl("registering lua function '"..okay.."' with number "..numb) +            lua.get_functions_table()[numb] = f +        else +            texio.write_nl("lua function '"..okay.."' is not reserved") +        end +    end + +} + +\newcount\lastallocatedluafunction + +\def\newluafunction#1% +  {\ifdefined#1\else +     \global\advance\lastallocatedluafunction 1 +     \global\chardef#1\lastallocatedluafunction +     \directlua{gadgets.functions.reserve()}#1{\detokenize{#1}}% +   \fi} + +% an example of usage (if we ever support it it will go to the plain gadgets module): +% +% \directlua { +% +%     local cct = nil +%     local chr = nil +% +%     gadgets.functions.register("UcharcatLuaOne",function() +%         chr = newtoken.scan_int() +%         cct = tex.getcatcode(chr) +%         tex.setcatcode(chr,newtoken.scan_int()) +%         tex.sprint(unicode.utf8.char(chr)) +%     end) +% +%     gadgets.functions.register("UcharcatLuaTwo",function() +%         tex.setcatcode(chr,cct) +%     end) +% +% } +% +% \def\Ucharcat +%   {\expandafter\expandafter\expandafter\luafunction +%    \expandafter\expandafter\expandafter\UcharcatLuaTwo +%    \luafunction\UcharcatLuaOne} +% +% A:\the\catcode65:\Ucharcat 65 11:A:\the\catcode65\par +% A:\the\catcode65:\Ucharcat 65  5:A:\the\catcode65\par +% A:\the\catcode65:\Ucharcat 65 11:A:\the\catcode65\par + +  \endinput diff --git a/src/fontloader/misc/fontloader-font-map.lua b/src/fontloader/misc/fontloader-font-map.lua index 449a00f..69474ba 100644 --- a/src/fontloader/misc/fontloader-font-map.lua +++ b/src/fontloader/misc/fontloader-font-map.lua @@ -313,6 +313,59 @@ function mappings.addtounicode(data,filename)              -- The next time I look into this, I'll add an extra analysis step to the otf loader (we can              -- resolve some tounicodes by looking into the gsub data tables that are bound to glyphs.              -- +-- a real tricky last resort: +-- +--             local lookups = glyph.lookups +--             if lookups then +--                 for _, lookup in next, lookups do -- assume consistency else we need to sort +--                     for i=1,#lookup do +--                         local l = lookup[i] +--                         if l.type == "ligature" then +--                             local s = l.specification +--                             if s.char == glyph.name then +--                                 local components = s.components +--                                 if components then +--                                     local t, n = { }, 0 +--                                     unicode = true +--                                     for l=1,#components do +--                                         local base = components[l] +--                                         local u = unicodes[base] or unicodevector[base] +--                                         if not u then +--                                             break +--                                         elseif type(u) == "table" then +--                                             if u[1] >= private then +--                                                 unicode = false +--                                                 break +--                                             end +--                                             n = n + 1 +--                                             t[n] = u[1] +--                                         else +--                                             if u >= private then +--                                                 unicode = false +--                                                 break +--                                             end +--                                             n = n + 1 +--                                             t[n] = u +--                                         end +--                                     end +--                                     if n == 0 then -- done then +--                                         -- nothing +--                                     elseif n == 1 then +--                                         glyph.unicode = t[1] +--                                     else +--                                         glyph.unicode = t +--                                     end +--                                     nl = nl + 1 +--                                     break +--                                 end +--                             end +--                         end +--                     end +--                     if unicode then +--                         break +--                     end +--                 end +--             end              if not unicode or unicode == "" then                  local split = lpegmatch(namesplitter,name)                  local nsplit = split and #split or 0 diff --git a/src/fontloader/misc/fontloader-font-otf.lua b/src/fontloader/misc/fontloader-font-otf.lua index 85eebaa..e7a97c6 100644 --- a/src/fontloader/misc/fontloader-font-otf.lua +++ b/src/fontloader/misc/fontloader-font-otf.lua @@ -36,6 +36,7 @@ local elapsedtime        = statistics.elapsedtime  local findbinfile        = resolvers.findbinfile  local trace_private      = false  registertracker("otf.private",        function(v) trace_private   = v end) +local trace_subfonts     = false  registertracker("otf.subfonts",       function(v) trace_subfonts  = v end)  local trace_loading      = false  registertracker("otf.loading",        function(v) trace_loading   = v end)  local trace_features     = false  registertracker("otf.features",       function(v) trace_features  = v end)  local trace_dynamics     = false  registertracker("otf.dynamics",       function(v) trace_dynamics  = v end) @@ -53,7 +54,7 @@ local otf                = fonts.handlers.otf  otf.glists               = { "gsub", "gpos" } -otf.version              = 2.803 -- beware: also sync font-mis.lua +otf.version              = 2.812 -- beware: also sync font-mis.lua  otf.cache                = containers.define("fonts", "otf", otf.version, true)  local hashes             = fonts.hashes @@ -699,6 +700,7 @@ end  -- not setting hasitalics and class (when nil) during table cronstruction can save some mem  actions["prepare glyphs"] = function(data,filename,raw) +    local tableversion = tonumber(raw.table_version) or 0      local rawglyphs    = raw.glyphs      local rawsubfonts  = raw.subfonts      local rawcidinfo   = raw.cidinfo @@ -722,81 +724,139 @@ actions["prepare glyphs"] = function(data,filename,raw)              local cidmap = fonts.cid.getmap(rawcidinfo)              if cidmap then                  rawcidinfo.usedname = cidmap.usedname -                local nofnames, nofunicodes = 0, 0 -                local cidunicodes, cidnames = cidmap.unicodes, cidmap.names +                local nofnames    = 0 +                local nofunicodes = 0 +                local cidunicodes = cidmap.unicodes +                local cidnames    = cidmap.names +                local cidtotal    = 0 +                local unique      = trace_subfonts and { }                  for cidindex=1,#rawsubfonts do                      local subfont   = rawsubfonts[cidindex]                      local cidglyphs = subfont.glyphs                      if includesubfonts then                          metadata.subfonts[cidindex] = somecopy(subfont)                      end -                    -- we have delayed loading so we cannot use next -                    for index=0,subfont.glyphcnt-1 do -- we could take the previous glyphcnt instead of 0 -                        local glyph = cidglyphs[index] -                        if glyph then -                            local unicode = glyph.unicode -                            if     unicode >= 0x00E000 and unicode <= 0x00F8FF then -                                unicode = -1 -                            elseif unicode >= 0x0F0000 and unicode <= 0x0FFFFD then -                                unicode = -1 -                            elseif unicode >= 0x100000 and unicode <= 0x10FFFD then -                                unicode = -1 -                            end -                            local name = glyph.name or cidnames[index] -                            if not unicode or unicode == -1 then -- or unicode >= criterium then -                                unicode = cidunicodes[index] -                            end -                            if unicode and descriptions[unicode] then -                                if trace_private then -                                    report_otf("preventing glyph %a at index %H to overload unicode %U",name or "noname",index,unicode) +                    local cidcnt, cidmin, cidmax +                    if tableversion > 0.3 then +                        -- we have delayed loading so we cannot use next +                        cidcnt = subfont.glyphcnt +                        cidmin = subfont.glyphmin +                        cidmax = subfont.glyphmax +                    else +                        cidcnt = subfont.glyphcnt +                        cidmin = 0 +                        cidmax = cidcnt - 1 +                    end +                    if trace_subfonts then +                        local cidtot = cidmax - cidmin + 1 +                        cidtotal = cidtotal + cidtot +                        report_otf("subfont: %i, min: %i, max: %i, cnt: %i, n: %i",cidindex,cidmin,cidmax,cidtot,cidcnt) +                    end +                    if cidcnt > 0 then +                        for cidslot=cidmin,cidmax do +                            local glyph = cidglyphs[cidslot] +                            if glyph then +                                local index = tableversion > 0.3 and glyph.orig_pos or cidslot +                                if trace_subfonts then +                                    unique[index] = true                                  end -                                unicode = -1 -                            end -                            if not unicode or unicode == -1 then -- or unicode >= criterium then -                                if not name then -                                    name = format("u%06X.ctx",private) +                                local unicode = glyph.unicode +                                if     unicode >= 0x00E000 and unicode <= 0x00F8FF then +                                    unicode = -1 +                                elseif unicode >= 0x0F0000 and unicode <= 0x0FFFFD then +                                    unicode = -1 +                                elseif unicode >= 0x100000 and unicode <= 0x10FFFD then +                                    unicode = -1                                  end -                                unicode = private -                                unicodes[name] = private -                                if trace_private then -                                    report_otf("glyph %a at index %H is moved to private unicode slot %U",name,index,private) +                                local name = glyph.name or cidnames[index] +                                if not unicode or unicode == -1 then -- or unicode >= criterium then +                                    unicode = cidunicodes[index]                                  end -                                private = private + 1 -                                nofnames = nofnames + 1 -                            else -                             -- if unicode > criterium then -                             --     local taken = descriptions[unicode] -                             --     if taken then -                             --         private = private + 1 -                             --         descriptions[private] = taken -                             --         unicodes[taken.name] = private -                             --         indices[taken.index] = private -                             --         if trace_private then -                             --             report_otf("slot %U is moved to %U due to private in font",unicode) -                             --         end -                             --     end -                             -- end -                                if not name then -                                    name = format("u%06X.ctx",unicode) +                                if unicode and descriptions[unicode] then +                                    if trace_private then +                                        report_otf("preventing glyph %a at index %H to overload unicode %U",name or "noname",index,unicode) +                                    end +                                    unicode = -1 +                                end +                                if not unicode or unicode == -1 then -- or unicode >= criterium then +                                    if not name then +                                        name = format("u%06X.ctx",private) +                                    end +                                    unicode = private +                                    unicodes[name] = private +                                    if trace_private then +                                        report_otf("glyph %a at index %H is moved to private unicode slot %U",name,index,private) +                                    end +                                    private = private + 1 +                                    nofnames = nofnames + 1 +                                else +                                 -- if unicode > criterium then +                                 --     local taken = descriptions[unicode] +                                 --     if taken then +                                 --         private = private + 1 +                                 --         descriptions[private] = taken +                                 --         unicodes[taken.name] = private +                                 --         indices[taken.index] = private +                                 --         if trace_private then +                                 --             report_otf("slot %U is moved to %U due to private in font",unicode) +                                 --         end +                                 --     end +                                 -- end +                                    if not name then +                                        name = format("u%06X.ctx",unicode) +                                    end +                                    unicodes[name] = unicode +                                    nofunicodes = nofunicodes + 1                                  end -                                unicodes[name] = unicode -                                nofunicodes = nofunicodes + 1 +                                indices[index] = unicode -- each index is unique (at least now) +                                local description = { +                                 -- width       = glyph.width, +                                    boundingbox = glyph.boundingbox, +                                 -- name        = glyph.name or name or "unknown", -- uniXXXX +                                    name        = name or "unknown", -- uniXXXX +                                    cidindex    = cidindex, +                                    index       = cidslot, +                                    glyph       = glyph, +                                } +                                descriptions[unicode] = description +local altuni = glyph.altuni +if altuni then + -- local d +    for i=1,#altuni do +        local a = altuni[i] +        local u = a.unicode +        if u ~= unicode then +            local v = a.variant +            if v then +                -- tricky: no addition to d? needs checking but in practice such dups are either very simple +                -- shapes or e.g cjk with not that many features +                local vv = variants[v] +                if vv then +                    vv[u] = unicode +                else -- xits-math has some: +                    vv = { [u] = unicode } +                    variants[v] = vv +                end +         -- elseif d then +         --     d[#d+1] = u +         -- else +         --     d = { u } +            end +        end +    end + -- if d then + --     duplicates[unicode] = d -- is this needed ? + -- end +end                              end -                            indices[index] = unicode -- each index is unique (at least now) -                            local description = { -                             -- width       = glyph.width, -                                boundingbox = glyph.boundingbox, -                                name        = glyph.name or name or "unknown", -- uniXXXX -                                cidindex    = cidindex, -                                index       = index, -                                glyph       = glyph, -                            } -                            descriptions[unicode] = description -                        else -                         -- report_otf("potential problem: glyph %U is used but empty",index)                          end +                    else +                        report_otf("potential problem: no glyphs found in subfont %i",cidindex)                      end                  end +                if trace_subfonts then +                    report_otf("nofglyphs: %i, unique: %i",cidtotal,table.count(unique)) +                end                  if trace_loading then                      report_otf("cid font remapped, %s unicode points, %s symbolic names, %s glyphs",nofunicodes, nofnames, nofunicodes+nofnames)                  end @@ -809,87 +869,97 @@ actions["prepare glyphs"] = function(data,filename,raw)      else -        for index=0,raw.glyphcnt-1 do -- not raw.glyphmax-1 (as that will crash) -            local glyph = rawglyphs[index] -            if glyph then -                local unicode = glyph.unicode -                local name    = glyph.name -                if not unicode or unicode == -1 then -- or unicode >= criterium then -                    unicode = private -                    unicodes[name] = private -                    if trace_private then -                        report_otf("glyph %a at index %H is moved to private unicode slot %U",name,index,private) -                    end -                    private = private + 1 -                else -                    -- We have a font that uses and exposes the private area. As this is rather unreliable it's -                    -- advised no to trust slots here (better use glyphnames). Anyway, we need a double check: -                    -- we need to move already moved entries and we also need to bump the next private to after -                    -- the (currently) last slot. This could leave us with a hole but we have holes anyway. -                    if unicode > criterium then -                        -- \definedfont[file:HANBatang-LVT.ttf] \fontchar{uF0135} \char"F0135 -                        local taken = descriptions[unicode] -                        if taken then -                            if unicode >= private then -                                private = unicode + 1 -- restart private (so we can have mixed now) +        local cnt = raw.glyphcnt or 0 +        local min = tableversion > 0.3 and raw.glyphmin or 0 +        local max = tableversion > 0.3 and raw.glyphmax or (raw.glyphcnt - 1) +        if cnt > 0 then +--             for index=0,cnt-1 do +            for index=min,max do +                local glyph = rawglyphs[index] +                if glyph then +                    local unicode = glyph.unicode +                    local name    = glyph.name +                    if not unicode or unicode == -1 then -- or unicode >= criterium then +                        unicode = private +                        unicodes[name] = private +                        if trace_private then +                            report_otf("glyph %a at index %H is moved to private unicode slot %U",name,index,private) +                        end +                        private = private + 1 +                    else +                        -- We have a font that uses and exposes the private area. As this is rather unreliable it's +                        -- advised no to trust slots here (better use glyphnames). Anyway, we need a double check: +                        -- we need to move already moved entries and we also need to bump the next private to after +                        -- the (currently) last slot. This could leave us with a hole but we have holes anyway. +                        if unicode > criterium then +                            -- \definedfont[file:HANBatang-LVT.ttf] \fontchar{uF0135} \char"F0135 +                            local taken = descriptions[unicode] +                            if taken then +                                if unicode >= private then +                                    private = unicode + 1 -- restart private (so we can have mixed now) +                                else +                                    private = private + 1 -- move on +                                end +                                descriptions[private] = taken +                                unicodes[taken.name] = private +                                indices[taken.index] = private +                                if trace_private then +                                    report_otf("slot %U is moved to %U due to private in font",unicode) +                                end                              else -                                private = private + 1 -- move on -                            end -                            descriptions[private] = taken -                            unicodes[taken.name] = private -                            indices[taken.index] = private -                            if trace_private then -                                report_otf("slot %U is moved to %U due to private in font",unicode) -                            end -                        else -                            if unicode >= private then -                                private = unicode + 1 -- restart (so we can have mixed now) +                                if unicode >= private then +                                    private = unicode + 1 -- restart (so we can have mixed now) +                                end                              end                          end +                        unicodes[name] = unicode                      end -                    unicodes[name] = unicode -                end -                indices[index] = unicode -             -- if not name then -             --     name = format("u%06X",unicode) -- u%06X.ctx -             -- end -                descriptions[unicode] = { -                 -- width       = glyph.width, -                    boundingbox = glyph.boundingbox, -                    name        = name, -                    index       = index, -                    glyph       = glyph, -                } -                local altuni = glyph.altuni -                if altuni then -                 -- local d -                    for i=1,#altuni do -                        local a = altuni[i] -                        local u = a.unicode -                        local v = a.variant -                        if v then -                            -- tricky: no addition to d? needs checking but in practice such dups are either very simple -                            -- shapes or e.g cjk with not that many features -                            local vv = variants[v] -                            if vv then -                                vv[u] = unicode -                            else -- xits-math has some: -                                vv = { [u] = unicode } -                                variants[v] = vv +                    indices[index] = unicode +                 -- if not name then +                 --     name = format("u%06X",unicode) -- u%06X.ctx +                 -- end +                    descriptions[unicode] = { +                     -- width       = glyph.width, +                        boundingbox = glyph.boundingbox, +                        name        = name, +                        index       = index, +                        glyph       = glyph, +                    } +                    local altuni = glyph.altuni +                    if altuni then +                     -- local d +                        for i=1,#altuni do +                            local a = altuni[i] +                            local u = a.unicode +                            if u ~= unicode then +                                local v = a.variant +                                if v then +                                    -- tricky: no addition to d? needs checking but in practice such dups are either very simple +                                    -- shapes or e.g cjk with not that many features +                                    local vv = variants[v] +                                    if vv then +                                        vv[u] = unicode +                                    else -- xits-math has some: +                                        vv = { [u] = unicode } +                                        variants[v] = vv +                                    end +                             -- elseif d then +                             --     d[#d+1] = u +                             -- else +                             --     d = { u } +                                end                              end -                     -- elseif d then -                     --     d[#d+1] = u -                     -- else -                     --     d = { u }                          end +                     -- if d then +                     --     duplicates[unicode] = d -- is this needed ? +                     -- end                      end -                 -- if d then -                 --     duplicates[unicode] = d -- is this needed ? -                 -- end +                else +                    report_otf("potential problem: glyph %U is used but empty",index)                  end -            else -                report_otf("potential problem: glyph %U is used but empty",index)              end +        else +            report_otf("potential problem: no glyphs found")          end      end @@ -974,8 +1044,8 @@ actions["check encoding"] = function(data,filename,raw)      end      if mapdata then -        mapdata.map     = { } -- clear some memory -        mapdata.backmap = { } -- clear some memory +        mapdata.map     = { } -- clear some memory (virtual and created each time anyway) +        mapdata.backmap = { } -- clear some memory (virtual and created each time anyway)      end  end @@ -991,7 +1061,6 @@ actions["add duplicates"] = function(data,filename,raw)      local unicodes     = resources.unicodes -- name to unicode      local indices      = resources.indices  -- index to unicodes      local duplicates   = resources.duplicates -      for unicode, d in next, duplicates do          local nofduplicates = #d          if nofduplicates > 4 then diff --git a/src/fontloader/misc/fontloader-font-otp.lua b/src/fontloader/misc/fontloader-font-otp.lua index 63e4184..ebf36ed 100644 --- a/src/fontloader/misc/fontloader-font-otp.lua +++ b/src/fontloader/misc/fontloader-font-otp.lua @@ -9,6 +9,8 @@ if not modules then modules = { } end modules ['font-otp'] = {  -- todo: pack math (but not that much to share)  --  -- pitfall 5.2: hashed tables can suddenly become indexed with nil slots +-- +-- unless we sort all hashes we can get a different pack order (no big deal but size can differ)  local next, type = next, type  local sort, concat = table.sort, table.concat diff --git a/src/fontloader/misc/fontloader-fonts-cbk.lua b/src/fontloader/misc/fontloader-fonts-cbk.lua index ce19c88..81b5b6e 100644 --- a/src/fontloader/misc/fontloader-fonts-cbk.lua +++ b/src/fontloader/misc/fontloader-fonts-cbk.lua @@ -17,6 +17,9 @@ local nodes = nodes  -- Fonts: (might move to node-gef.lua)  local traverse_id = node.traverse_id +local free_node   = node.free +local remove_node = node.remove +  local glyph_code  = nodes.nodecodes.glyph  local disc_code   = nodes.nodecodes.disc @@ -57,6 +60,8 @@ function nodes.handlers.nodepass(head)          local basefonts = { }          local prevfont  = nil          local basefont  = nil +        local variants  = nil +        local redundant = nil          for n in traverse_id(glyph_code,head) do              local font = n.font              if font ~= prevfont then @@ -78,9 +83,46 @@ function nodes.handlers.nodepass(head)                                  basefonts[#basefonts+1] = basefont                              end                          end +                        local resources = tfmdata.resources +                        variants  = resources and resources.variants +                        variants = variants and next(variants) and variants or false +                    end +                else +                    local tfmdata = fontdata[prevfont] +                    if tfmdata then +                        local resources = tfmdata.resources +                        variants = resources and resources.variants +                        variants = variants and next(variants) and variants or false                      end                  end              end +            if variants then +                local char = n.char +                if char >= 0xFE00 and (char <= 0xFE0F or (char >= 0xE0100 and char <= 0xE01EF)) then +                    local hash = variants[char] +                    if hash then +                        local p = n.prev +                        if p and p.id == glyph_code then +                            local variant = hash[p.char] +                            if variant then +                                p.char = variant +                                if not redundant then +                                    redundant = { n } +                                else +                                    redundant[#redundant+1] = n +                                end +                            end +                        end +                    end +                end +            end +        end +        if redundant then +            for i=1,#redundant do +                local n = redundant[i] +                remove_node(head,n) +                free_node(n) +            end          end          for d in traverse_id(disc_code,head) do              local r = d.replace diff --git a/src/fontloader/misc/fontloader-fonts-otn.lua b/src/fontloader/misc/fontloader-fonts-otn.lua index 343e182..dd3aa61 100644 --- a/src/fontloader/misc/fontloader-fonts-otn.lua +++ b/src/fontloader/misc/fontloader-fonts-otn.lua @@ -376,7 +376,7 @@ local function copy_glyph(g) -- next and prev are untouched !      end  end ---  +--  -- start is a mark and we need to keep that one @@ -1834,15 +1834,8 @@ local function normal_handle_contextchain(head,start,kind,chainname,contexts,seq                              break                          end                      end -                elseif f == 2 then -                    match = seq[1][32]                  else -                    for n=f-1,1 do -                        if not seq[n][32] then -                            match = false -                            break -                        end -                    end +                    match = false                  end              end              -- after @@ -1895,15 +1888,8 @@ local function normal_handle_contextchain(head,start,kind,chainname,contexts,seq                              break                          end                      end -                elseif s-l == 1 then -                    match = seq[s][32]                  else -                    for n=l+1,s do -                        if not seq[n][32] then -                            match = false -                            break -                        end -                    end +                    match = false                  end              end          end diff --git a/src/fontloader/misc/fontloader-l-table.lua b/src/fontloader/misc/fontloader-l-table.lua index 97e0441..b02f210 100644 --- a/src/fontloader/misc/fontloader-l-table.lua +++ b/src/fontloader/misc/fontloader-l-table.lua @@ -39,7 +39,7 @@ end  function table.keys(t)      if t then          local keys, k = { }, 0 -        for key, _ in next, t do +        for key in next, t do              k = k + 1              keys[k] = key          end @@ -50,44 +50,126 @@ function table.keys(t)  end  -- local function compare(a,b) ---     local ta, tb = type(a), type(b) -- needed, else 11 < 2 ---     if ta == tb then +--     local ta = type(a) -- needed, else 11 < 2 +--     local tb = type(b) -- needed, else 11 < 2 +--     if ta == tb and ta == "number" then  --         return a < b  --     else  --         return tostring(a) < tostring(b) -- not that efficient  --     end  -- end +-- local function compare(a,b) +--     local ta = type(a) -- needed, else 11 < 2 +--     local tb = type(b) -- needed, else 11 < 2 +--     if ta == tb and (ta == "number" or ta == "string") then +--         return a < b +--     else +--         return tostring(a) < tostring(b) -- not that efficient +--     end +-- end + +-- local function sortedkeys(tab) +--     if tab then +--         local srt, category, s = { }, 0, 0 -- 0=unknown 1=string, 2=number 3=mixed +--         for key in next, tab do +--             s = s + 1 +--             srt[s] = key +--             if category == 3 then +--                 -- no further check +--             else +--                 local tkey = type(key) +--                 if tkey == "string" then +--                     category = (category == 2 and 3) or 1 +--                 elseif tkey == "number" then +--                     category = (category == 1 and 3) or 2 +--                 else +--                     category = 3 +--                 end +--             end +--         end +--         if category == 0 or category == 3 then +--             sort(srt,compare) +--         else +--             sort(srt) +--         end +--         return srt +--     else +--         return { } +--     end +-- end + +-- local function compare(a,b) +--     local ta = type(a) -- needed, else 11 < 2 +--     local tb = type(b) -- needed, else 11 < 2 +--     if ta == tb and (ta == "number" or ta == "string") then +--         return a < b +--     else +--         return tostring(a) < tostring(b) -- not that efficient +--     end +-- end + +-- local function compare(a,b) +--     local ta = type(a) -- needed, else 11 < 2 +--     if ta == "number" or ta == "string" then +--         local tb = type(b) -- needed, else 11 < 2 +--         if ta == tb then +--             return a < b +--         end +--     end +--     return tostring(a) < tostring(b) -- not that efficient +-- end +  local function compare(a,b)      local ta = type(a) -- needed, else 11 < 2 -    local tb = type(b) -- needed, else 11 < 2 -    if ta == tb and ta == "number" then -        return a < b -    else -        return tostring(a) < tostring(b) -- not that efficient +    if ta == "number" then +        local tb = type(b) -- needed, else 11 < 2 +        if ta == tb then +            return a < b +        elseif tb == "string" then +            return tostring(a) < b +        end +    elseif ta == "string" then +        local tb = type(b) -- needed, else 11 < 2 +        if ta == tb then +            return a < b +        else +            return a < tostring(b) +        end      end +    return tostring(a) < tostring(b) -- not that efficient  end  local function sortedkeys(tab)      if tab then          local srt, category, s = { }, 0, 0 -- 0=unknown 1=string, 2=number 3=mixed -        for key,_ in next, tab do +        for key in next, tab do              s = s + 1              srt[s] = key              if category == 3 then                  -- no further check +            elseif category == 1 then +                if type(key) ~= "string" then +                    category = 3 +                end +            elseif category == 2 then +                if type(key) ~= "number" then +                    category = 3 +                end              else                  local tkey = type(key)                  if tkey == "string" then -                    category = (category == 2 and 3) or 1 +                    category = 1                  elseif tkey == "number" then -                    category = (category == 1 and 3) or 2 +                    category = 2                  else                      category = 3                  end              end          end -        if category == 0 or category == 3 then +        if s < 2 then +            -- nothing to sort +        elseif category == 3 then              sort(srt,compare)          else              sort(srt) @@ -101,13 +183,15 @@ end  local function sortedhashonly(tab)      if tab then          local srt, s = { }, 0 -        for key,_ in next, tab do +        for key in next, tab do              if type(key) == "string" then                  s = s + 1                  srt[s] = key              end          end -        sort(srt) +        if s > 1 then +            sort(srt) +        end          return srt      else          return { } @@ -117,13 +201,15 @@ end  local function sortedindexonly(tab)      if tab then          local srt, s = { }, 0 -        for key,_ in next, tab do +        for key in next, tab do              if type(key) == "number" then                  s = s + 1                  srt[s] = key              end          end -        sort(srt) +        if s > 1 then +            sort(srt) +        end          return srt      else          return { } @@ -133,13 +219,15 @@ end  local function sortedhashkeys(tab,cmp) -- fast one      if tab then          local srt, s = { }, 0 -        for key,_ in next, tab do +        for key in next, tab do              if key then                  s= s + 1                  srt[s] = key              end          end -        sort(srt,cmp) +        if s > 1 then +            sort(srt,cmp) +        end          return srt      else          return { } @@ -149,7 +237,7 @@ end  function table.allkeys(t)      local keys = { }      for k, v in next, t do -        for k, v in next, v do +        for k in next, v do              keys[k] = true          end      end @@ -172,19 +260,21 @@ local function sortedhash(t,cmp)          else              s = sortedkeys(t) -- the robust one          end -        local n = 0          local m = #s -        local function kv() -- (s) -            if n < m then -                n = n + 1 -                local k = s[n] -                return k, t[k] +        if m == 1 then +            return next, t +        elseif m > 0 then +            local n = 0 +            return function() +                if n < m then +                    n = n + 1 +                    local k = s[n] +                    return k, t[k] +                end              end          end -        return kv -- , s -    else -        return nothing      end +    return nothing  end  table.sortedhash  = sortedhash @@ -328,7 +418,7 @@ end  local function copy(t, tables) -- taken from lua wiki, slightly adapted      tables = tables or { } -    local tcopy = {} +    local tcopy = { }      if not tables[t] then          tables[t] = tcopy      end @@ -388,7 +478,7 @@ function table.fromhash(t)      return hsh  end -local noquotes, hexify, handle, reduce, compact, inline, functions +local noquotes, hexify, handle, compact, inline, functions  local reserved = table.tohash { -- intercept a language inconvenience: no reserved words as key      'and', 'break', 'do', 'else', 'elseif', 'end', 'false', 'for', 'function', 'if', @@ -396,33 +486,67 @@ local reserved = table.tohash { -- intercept a language inconvenience: no reserv      'NaN', 'goto',  } +-- local function simple_table(t) +--     if #t > 0 then +--         local n = 0 +--         for _,v in next, t do +--             n = n + 1 +--         end +--         if n == #t then +--             local tt, nt = { }, 0 +--             for i=1,#t do +--                 local v = t[i] +--                 local tv = type(v) +--                 if tv == "number" then +--                     nt = nt + 1 +--                     if hexify then +--                         tt[nt] = format("0x%X",v) +--                     else +--                         tt[nt] = tostring(v) -- tostring not needed +--                     end +--                 elseif tv == "string" then +--                     nt = nt + 1 +--                     tt[nt] = format("%q",v) +--                 elseif tv == "boolean" then +--                     nt = nt + 1 +--                     tt[nt] = v and "true" or "false" +--                 else +--                     return nil +--                 end +--             end +--             return tt +--         end +--     end +--     return nil +-- end +  local function simple_table(t) -    if #t > 0 then +    local nt = #t +    if nt > 0 then          local n = 0          for _,v in next, t do              n = n + 1 +         -- if type(v) == "table" then +         --     return nil +         -- end          end -        if n == #t then -            local tt, nt = { }, 0 -            for i=1,#t do +        if n == nt then +            local tt = { } +            for i=1,nt do                  local v = t[i]                  local tv = type(v)                  if tv == "number" then -                    nt = nt + 1                      if hexify then -                        tt[nt] = format("0x%X",v) +                        tt[i] = format("0x%X",v)                      else -                        tt[nt] = tostring(v) -- tostring not needed +                        tt[i] = tostring(v) -- tostring not needed                      end                  elseif tv == "string" then -                    nt = nt + 1 -                    tt[nt] = format("%q",v) +                    tt[i] = format("%q",v)                  elseif tv == "boolean" then -                    nt = nt + 1 -                    tt[nt] = v and "true" or "false" +                    tt[i] = v and "true" or "false"                  else -                    tt = nil -                    break +                    return nil                  end              end              return tt @@ -480,14 +604,6 @@ local function do_serialize(root,name,depth,level,indexed)      end      -- we could check for k (index) being number (cardinal)      if root and next(root) ~= nil then -     -- local first, last = nil, 0 -- #root cannot be trusted here (will be ok in 5.2 when ipairs is gone) -     -- if compact then -     --     -- NOT: for k=1,#root do (we need to quit at nil) -     --     for k,v in ipairs(root) do -- can we use next? -     --         if not first then first = k end -     --         last = last + 1 -     --     end -     -- end          local first, last = nil, 0          if compact then              last = #root @@ -503,12 +619,10 @@ local function do_serialize(root,name,depth,level,indexed)          end          local sk = sortedkeys(root)          for i=1,#sk do -            local k = sk[i] -            local v = root[k] -            --~ if v == root then -                -- circular -            --~ else -            local tv, tk = type(v), type(k) +            local k  = sk[i] +            local v  = root[k] +            local tv = type(v) +            local tk = type(k)              if compact and first and tk == "number" and k >= first and k <= last then                  if tv == "number" then                      if hexify then @@ -517,11 +631,7 @@ local function do_serialize(root,name,depth,level,indexed)                          handle(format("%s %s,",depth,v)) -- %.99g                      end                  elseif tv == "string" then -                    if reduce and tonumber(v) then -                        handle(format("%s %s,",depth,v)) -                    else -                        handle(format("%s %q,",depth,v)) -                    end +                    handle(format("%s %q,",depth,v))                  elseif tv == "table" then                      if next(v) == nil then                          handle(format("%s {},",depth)) @@ -577,34 +687,18 @@ local function do_serialize(root,name,depth,level,indexed)                      end                  end              elseif tv == "string" then -                if reduce and tonumber(v) then -                    if tk == "number" then -                        if hexify then -                            handle(format("%s [0x%X]=%s,",depth,k,v)) -                        else -                            handle(format("%s [%s]=%s,",depth,k,v)) -                        end -                    elseif tk == "boolean" then -                        handle(format("%s [%s]=%s,",depth,k and "true" or "false",v)) -                    elseif noquotes and not reserved[k] and lpegmatch(propername,k) then -                        handle(format("%s %s=%s,",depth,k,v)) +                if tk == "number" then +                    if hexify then +                        handle(format("%s [0x%X]=%q,",depth,k,v))                      else -                        handle(format("%s [%q]=%s,",depth,k,v)) +                        handle(format("%s [%s]=%q,",depth,k,v))                      end +                elseif tk == "boolean" then +                    handle(format("%s [%s]=%q,",depth,k and "true" or "false",v)) +                elseif noquotes and not reserved[k] and lpegmatch(propername,k) then +                    handle(format("%s %s=%q,",depth,k,v))                  else -                    if tk == "number" then -                        if hexify then -                            handle(format("%s [0x%X]=%q,",depth,k,v)) -                        else -                            handle(format("%s [%s]=%q,",depth,k,v)) -                        end -                    elseif tk == "boolean" then -                        handle(format("%s [%s]=%q,",depth,k and "true" or "false",v)) -                    elseif noquotes and not reserved[k] and lpegmatch(propername,k) then -                        handle(format("%s %s=%q,",depth,k,v)) -                    else -                        handle(format("%s [%q]=%q,",depth,k,v)) -                    end +                    handle(format("%s [%q]=%q,",depth,k,v))                  end              elseif tv == "table" then                  if next(v) == nil then @@ -690,7 +784,6 @@ local function do_serialize(root,name,depth,level,indexed)                      handle(format("%s [%q]=%q,",depth,k,tostring(v)))                  end              end -            --~ end          end      end      if level > 0 then @@ -707,7 +800,6 @@ local function serialize(_handle,root,name,specification) -- handle wins          noquotes  = specification.noquotes          hexify    = specification.hexify          handle    = _handle or specification.handle or print -        reduce    = specification.reduce or false          functions = specification.functions          compact   = specification.compact          inline    = specification.inline and compact @@ -724,7 +816,6 @@ local function serialize(_handle,root,name,specification) -- handle wins          noquotes  = false          hexify    = false          handle    = _handle or print -        reduce    = false          compact   = true          inline    = true          functions = true @@ -798,15 +889,6 @@ end  table.tohandle = serialize --- sometimes tables are real use (zapfino extra pro is some 85M) in which --- case a stepwise serialization is nice; actually, we could consider: --- --- for line in table.serializer(root,name,reduce,noquotes) do ---    ...(line) --- end --- --- so this is on the todo list -  local maxtab = 2*1024  function table.tofile(filename,root,name,specification) diff --git a/src/fontloader/misc/fontloader-mplib.tex b/src/fontloader/misc/fontloader-mplib.tex index 8af9f2d..09dd179 100644 --- a/src/fontloader/misc/fontloader-mplib.tex +++ b/src/fontloader/misc/fontloader-mplib.tex @@ -61,6 +61,7 @@  %D Now load the needed \LUA\ code.  \directlua{dofile(kpse.find_file('luatex-mplib.lua'))} +% \directlua{dofile(resolvers.findfile('luatex-mplib.lua'))}  %D The following code takes care of encapsulating the literals: diff --git a/src/fontloader/misc/fontloader-plain.tex b/src/fontloader/misc/fontloader-plain.tex index 1ea8558..c9a9e36 100644 --- a/src/fontloader/misc/fontloader-plain.tex +++ b/src/fontloader/misc/fontloader-plain.tex @@ -20,6 +20,7 @@      \input {luatex-math}%      \input {luatex-languages}%      \input {luatex-mplib}% +  % \input {luatex-gadgets}%  }  \edef\fmtversion{\fmtversion+luatex} diff --git a/src/fontloader/misc/fontloader-util-str.lua b/src/fontloader/misc/fontloader-util-str.lua index de4a87e..c2139b1 100644 --- a/src/fontloader/misc/fontloader-util-str.lua +++ b/src/fontloader/misc/fontloader-util-str.lua @@ -1118,7 +1118,7 @@ function string.optionalquoted(str)      return lpegmatch(pattern,str) or str  end -local pattern = Cs((newline / os.newline + 1)^0) +local pattern = Cs((newline / (os.newline or "\r") + 1)^0)  function string.replacenewlines(str)      return lpegmatch(pattern,str) diff --git a/src/fontloader/runtime/fontloader-fontloader.lua b/src/fontloader/runtime/fontloader-fontloader.lua index ead4e2a..d8095a2 100644 --- a/src/fontloader/runtime/fontloader-fontloader.lua +++ b/src/fontloader/runtime/fontloader-fontloader.lua @@ -1,6 +1,6 @@  -- merged file : luatex-fonts-merged.lua  -- parent file : luatex-fonts.lua --- merge date  : 05/12/15 17:57:50 +-- merge date  : 05/24/15 12:42:55  do -- begin closure to overcome local limits and interference @@ -1041,7 +1041,7 @@ end  function table.keys(t)    if t then      local keys,k={},0 -    for key,_ in next,t do +    for key in next,t do        k=k+1        keys[k]=key      end @@ -1052,32 +1052,51 @@ function table.keys(t)  end  local function compare(a,b)    local ta=type(a)  -  local tb=type(b)  -  if ta==tb and ta=="number" then -    return a<b -  else -    return tostring(a)<tostring(b)  +  if ta=="number" then +    local tb=type(b)  +    if ta==tb then +      return a<b +    elseif tb=="string" then +      return tostring(a)<b +    end +  elseif ta=="string" then +    local tb=type(b)  +    if ta==tb then +      return a<b +    else +      return a<tostring(b) +    end    end +  return tostring(a)<tostring(b)   end  local function sortedkeys(tab)    if tab then      local srt,category,s={},0,0  -    for key,_ in next,tab do +    for key in next,tab do        s=s+1        srt[s]=key        if category==3 then +      elseif category==1 then +        if type(key)~="string" then +          category=3 +        end +      elseif category==2 then +        if type(key)~="number" then +          category=3 +        end        else          local tkey=type(key)          if tkey=="string" then -          category=(category==2 and 3) or 1 +          category=1          elseif tkey=="number" then -          category=(category==1 and 3) or 2 +          category=2          else            category=3          end        end      end -    if category==0 or category==3 then +    if s<2 then +    elseif category==3 then        sort(srt,compare)      else        sort(srt) @@ -1090,13 +1109,15 @@ end  local function sortedhashonly(tab)    if tab then      local srt,s={},0 -    for key,_ in next,tab do +    for key in next,tab do        if type(key)=="string" then          s=s+1          srt[s]=key        end      end -    sort(srt) +    if s>1 then +      sort(srt) +    end      return srt    else      return {} @@ -1105,13 +1126,15 @@ end  local function sortedindexonly(tab)    if tab then      local srt,s={},0 -    for key,_ in next,tab do +    for key in next,tab do        if type(key)=="number" then          s=s+1          srt[s]=key        end      end -    sort(srt) +    if s>1 then +      sort(srt) +    end      return srt    else      return {} @@ -1120,13 +1143,15 @@ end  local function sortedhashkeys(tab,cmp)     if tab then      local srt,s={},0 -    for key,_ in next,tab do +    for key in next,tab do        if key then          s=s+1          srt[s]=key        end      end -    sort(srt,cmp) +    if s>1 then +      sort(srt,cmp) +    end      return srt    else      return {} @@ -1135,7 +1160,7 @@ end  function table.allkeys(t)    local keys={}    for k,v in next,t do -    for k,v in next,v do +    for k in next,v do        keys[k]=true      end    end @@ -1154,19 +1179,21 @@ local function sortedhash(t,cmp)      else        s=sortedkeys(t)       end -    local n=0      local m=#s -    local function kv()  -      if n<m then -        n=n+1 -        local k=s[n] -        return k,t[k] +    if m==1 then +      return next,t +    elseif m>0 then +      local n=0 +      return function() +        if n<m then +          n=n+1 +          local k=s[n] +          return k,t[k] +        end        end      end -    return kv  -  else -    return nothing    end +  return nothing  end  table.sortedhash=sortedhash  table.sortedpairs=sortedhash  @@ -1308,39 +1335,36 @@ function table.fromhash(t)    end    return hsh  end -local noquotes,hexify,handle,reduce,compact,inline,functions +local noquotes,hexify,handle,compact,inline,functions  local reserved=table.tohash {     'and','break','do','else','elseif','end','false','for','function','if',    'in','local','nil','not','or','repeat','return','then','true','until','while',    'NaN','goto',  }  local function simple_table(t) -  if #t>0 then +  local nt=#t +  if nt>0 then      local n=0      for _,v in next,t do        n=n+1      end -    if n==#t then -      local tt,nt={},0 -      for i=1,#t do +    if n==nt then +      local tt={} +      for i=1,nt do          local v=t[i]          local tv=type(v)          if tv=="number" then -          nt=nt+1            if hexify then -            tt[nt]=format("0x%X",v) +            tt[i]=format("0x%X",v)            else -            tt[nt]=tostring(v)  +            tt[i]=tostring(v)             end          elseif tv=="string" then -          nt=nt+1 -          tt[nt]=format("%q",v) +          tt[i]=format("%q",v)          elseif tv=="boolean" then -          nt=nt+1 -          tt[nt]=v and "true" or "false" +          tt[i]=v and "true" or "false"          else -          tt=nil -          break +          return nil          end        end        return tt @@ -1394,7 +1418,8 @@ local function do_serialize(root,name,depth,level,indexed)      for i=1,#sk do        local k=sk[i]        local v=root[k] -      local tv,tk=type(v),type(k) +      local tv=type(v) +      local tk=type(k)        if compact and first and tk=="number" and k>=first and k<=last then          if tv=="number" then            if hexify then @@ -1403,11 +1428,7 @@ local function do_serialize(root,name,depth,level,indexed)              handle(format("%s %s,",depth,v))             end          elseif tv=="string" then -          if reduce and tonumber(v) then -            handle(format("%s %s,",depth,v)) -          else -            handle(format("%s %q,",depth,v)) -          end +          handle(format("%s %q,",depth,v))          elseif tv=="table" then            if next(v)==nil then              handle(format("%s {},",depth)) @@ -1463,34 +1484,18 @@ local function do_serialize(root,name,depth,level,indexed)            end          end        elseif tv=="string" then -        if reduce and tonumber(v) then -          if tk=="number" then -            if hexify then -              handle(format("%s [0x%X]=%s,",depth,k,v)) -            else -              handle(format("%s [%s]=%s,",depth,k,v)) -            end -          elseif tk=="boolean" then -            handle(format("%s [%s]=%s,",depth,k and "true" or "false",v)) -          elseif noquotes and not reserved[k] and lpegmatch(propername,k) then -            handle(format("%s %s=%s,",depth,k,v)) +        if tk=="number" then +          if hexify then +            handle(format("%s [0x%X]=%q,",depth,k,v))            else -            handle(format("%s [%q]=%s,",depth,k,v)) +            handle(format("%s [%s]=%q,",depth,k,v))            end +        elseif tk=="boolean" then +          handle(format("%s [%s]=%q,",depth,k and "true" or "false",v)) +        elseif noquotes and not reserved[k] and lpegmatch(propername,k) then +          handle(format("%s %s=%q,",depth,k,v))          else -          if tk=="number" then -            if hexify then -              handle(format("%s [0x%X]=%q,",depth,k,v)) -            else -              handle(format("%s [%s]=%q,",depth,k,v)) -            end -          elseif tk=="boolean" then -            handle(format("%s [%s]=%q,",depth,k and "true" or "false",v)) -          elseif noquotes and not reserved[k] and lpegmatch(propername,k) then -            handle(format("%s %s=%q,",depth,k,v)) -          else -            handle(format("%s [%q]=%q,",depth,k,v)) -          end +          handle(format("%s [%q]=%q,",depth,k,v))          end        elseif tv=="table" then          if next(v)==nil then @@ -1587,7 +1592,6 @@ local function serialize(_handle,root,name,specification)      noquotes=specification.noquotes      hexify=specification.hexify      handle=_handle or specification.handle or print -    reduce=specification.reduce or false      functions=specification.functions      compact=specification.compact      inline=specification.inline and compact @@ -1604,7 +1608,6 @@ local function serialize(_handle,root,name,specification)      noquotes=false      hexify=false      handle=_handle or print -    reduce=false      compact=true      inline=true      functions=true @@ -3439,7 +3442,7 @@ local pattern=Cs(dquote*(equote-P(-2))^0*dquote)  function string.optionalquoted(str)    return lpegmatch(pattern,str) or str  end -local pattern=Cs((newline/os.newline+1)^0) +local pattern=Cs((newline/(os.newline or "\r")+1)^0)  function string.replacenewlines(str)    return lpegmatch(pattern,str)  end @@ -7160,6 +7163,7 @@ local stoptiming=statistics.stoptiming  local elapsedtime=statistics.elapsedtime  local findbinfile=resolvers.findbinfile  local trace_private=false registertracker("otf.private",function(v) trace_private=v end) +local trace_subfonts=false registertracker("otf.subfonts",function(v) trace_subfonts=v end)  local trace_loading=false registertracker("otf.loading",function(v) trace_loading=v end)  local trace_features=false registertracker("otf.features",function(v) trace_features=v end)  local trace_dynamics=false registertracker("otf.dynamics",function(v) trace_dynamics=v end) @@ -7172,7 +7176,7 @@ local report_otf=logs.reporter("fonts","otf loading")  local fonts=fonts  local otf=fonts.handlers.otf  otf.glists={ "gsub","gpos" } -otf.version=2.803  +otf.version=2.812   otf.cache=containers.define("fonts","otf",otf.version,true)  local hashes=fonts.hashes  local definers=fonts.definers @@ -7705,6 +7709,7 @@ local function somecopy(old)    end  end  actions["prepare glyphs"]=function(data,filename,raw) +  local tableversion=tonumber(raw.table_version) or 0    local rawglyphs=raw.glyphs    local rawsubfonts=raw.subfonts    local rawcidinfo=raw.cidinfo @@ -7725,66 +7730,114 @@ actions["prepare glyphs"]=function(data,filename,raw)        local cidmap=fonts.cid.getmap(rawcidinfo)        if cidmap then          rawcidinfo.usedname=cidmap.usedname -        local nofnames,nofunicodes=0,0 -        local cidunicodes,cidnames=cidmap.unicodes,cidmap.names +        local nofnames=0 +        local nofunicodes=0 +        local cidunicodes=cidmap.unicodes +        local cidnames=cidmap.names +        local cidtotal=0 +        local unique=trace_subfonts and {}          for cidindex=1,#rawsubfonts do            local subfont=rawsubfonts[cidindex]            local cidglyphs=subfont.glyphs            if includesubfonts then              metadata.subfonts[cidindex]=somecopy(subfont)            end -          for index=0,subfont.glyphcnt-1 do  -            local glyph=cidglyphs[index] -            if glyph then -              local unicode=glyph.unicode -              if   unicode>=0x00E000 and unicode<=0x00F8FF then -                unicode=-1 -              elseif unicode>=0x0F0000 and unicode<=0x0FFFFD then -                unicode=-1 -              elseif unicode>=0x100000 and unicode<=0x10FFFD then -                unicode=-1 -              end -              local name=glyph.name or cidnames[index] -              if not unicode or unicode==-1 then  -                unicode=cidunicodes[index] -              end -              if unicode and descriptions[unicode] then -                if trace_private then -                  report_otf("preventing glyph %a at index %H to overload unicode %U",name or "noname",index,unicode) +          local cidcnt,cidmin,cidmax +          if tableversion>0.3 then +            cidcnt=subfont.glyphcnt +            cidmin=subfont.glyphmin +            cidmax=subfont.glyphmax +          else +            cidcnt=subfont.glyphcnt +            cidmin=0 +            cidmax=cidcnt-1 +          end +          if trace_subfonts then +            local cidtot=cidmax-cidmin+1 +            cidtotal=cidtotal+cidtot +            report_otf("subfont: %i, min: %i, max: %i, cnt: %i, n: %i",cidindex,cidmin,cidmax,cidtot,cidcnt) +          end +          if cidcnt>0 then +            for cidslot=cidmin,cidmax do +              local glyph=cidglyphs[cidslot] +              if glyph then +                local index=tableversion>0.3 and glyph.orig_pos or cidslot +                if trace_subfonts then +                  unique[index]=true                  end -                unicode=-1 -              end -              if not unicode or unicode==-1 then  -                if not name then -                  name=format("u%06X.ctx",private) +                local unicode=glyph.unicode +                if   unicode>=0x00E000 and unicode<=0x00F8FF then +                  unicode=-1 +                elseif unicode>=0x0F0000 and unicode<=0x0FFFFD then +                  unicode=-1 +                elseif unicode>=0x100000 and unicode<=0x10FFFD then +                  unicode=-1                  end -                unicode=private -                unicodes[name]=private -                if trace_private then -                  report_otf("glyph %a at index %H is moved to private unicode slot %U",name,index,private) +                local name=glyph.name or cidnames[index] +                if not unicode or unicode==-1 then  +                  unicode=cidunicodes[index]                  end -                private=private+1 -                nofnames=nofnames+1 -              else -                if not name then -                  name=format("u%06X.ctx",unicode) +                if unicode and descriptions[unicode] then +                  if trace_private then +                    report_otf("preventing glyph %a at index %H to overload unicode %U",name or "noname",index,unicode) +                  end +                  unicode=-1                  end -                unicodes[name]=unicode -                nofunicodes=nofunicodes+1 -              end -              indices[index]=unicode  -              local description={ -                boundingbox=glyph.boundingbox, -                name=glyph.name or name or "unknown", -                cidindex=cidindex, -                index=index, -                glyph=glyph, -              } -              descriptions[unicode]=description -            else +                if not unicode or unicode==-1 then  +                  if not name then +                    name=format("u%06X.ctx",private) +                  end +                  unicode=private +                  unicodes[name]=private +                  if trace_private then +                    report_otf("glyph %a at index %H is moved to private unicode slot %U",name,index,private) +                  end +                  private=private+1 +                  nofnames=nofnames+1 +                else +                  if not name then +                    name=format("u%06X.ctx",unicode) +                  end +                  unicodes[name]=unicode +                  nofunicodes=nofunicodes+1 +                end +                indices[index]=unicode  +                local description={ +                  boundingbox=glyph.boundingbox, +                  name=name or "unknown", +                  cidindex=cidindex, +                  index=cidslot, +                  glyph=glyph, +                } +                descriptions[unicode]=description +local altuni=glyph.altuni +if altuni then +  for i=1,#altuni do +    local a=altuni[i] +    local u=a.unicode +    if u~=unicode then +      local v=a.variant +      if v then +        local vv=variants[v] +        if vv then +          vv[u]=unicode +        else  +          vv={ [u]=unicode } +          variants[v]=vv +        end +      end +    end +  end +end +              end              end +          else +            report_otf("potential problem: no glyphs found in subfont %i",cidindex)            end          end +        if trace_subfonts then +          report_otf("nofglyphs: %i, unique: %i",cidtotal,table.count(unique)) +        end          if trace_loading then            report_otf("cid font remapped, %s unicode points, %s symbolic names, %s glyphs",nofunicodes,nofnames,nofunicodes+nofnames)          end @@ -7795,68 +7848,77 @@ actions["prepare glyphs"]=function(data,filename,raw)        report_otf("font %a has no glyphs",filename)      end    else -    for index=0,raw.glyphcnt-1 do  -      local glyph=rawglyphs[index] -      if glyph then -        local unicode=glyph.unicode -        local name=glyph.name -        if not unicode or unicode==-1 then  -          unicode=private -          unicodes[name]=private -          if trace_private then -            report_otf("glyph %a at index %H is moved to private unicode slot %U",name,index,private) -          end -          private=private+1 -        else -          if unicode>criterium then -            local taken=descriptions[unicode] -            if taken then -              if unicode>=private then -                private=unicode+1  +    local cnt=raw.glyphcnt or 0 +    local min=tableversion>0.3 and raw.glyphmin or 0 +    local max=tableversion>0.3 and raw.glyphmax or (raw.glyphcnt-1) +    if cnt>0 then +      for index=min,max do +        local glyph=rawglyphs[index] +        if glyph then +          local unicode=glyph.unicode +          local name=glyph.name +          if not unicode or unicode==-1 then  +            unicode=private +            unicodes[name]=private +            if trace_private then +              report_otf("glyph %a at index %H is moved to private unicode slot %U",name,index,private) +            end +            private=private+1 +          else +            if unicode>criterium then +              local taken=descriptions[unicode] +              if taken then +                if unicode>=private then +                  private=unicode+1  +                else +                  private=private+1  +                end +                descriptions[private]=taken +                unicodes[taken.name]=private +                indices[taken.index]=private +                if trace_private then +                  report_otf("slot %U is moved to %U due to private in font",unicode) +                end                else -                private=private+1  -              end -              descriptions[private]=taken -              unicodes[taken.name]=private -              indices[taken.index]=private -              if trace_private then -                report_otf("slot %U is moved to %U due to private in font",unicode) -              end -            else -              if unicode>=private then -                private=unicode+1  +                if unicode>=private then +                  private=unicode+1  +                end                end              end +            unicodes[name]=unicode            end -          unicodes[name]=unicode -        end -        indices[index]=unicode -        descriptions[unicode]={ -          boundingbox=glyph.boundingbox, -          name=name, -          index=index, -          glyph=glyph, -        } -        local altuni=glyph.altuni -        if altuni then -          for i=1,#altuni do -            local a=altuni[i] -            local u=a.unicode -            local v=a.variant -            if v then -              local vv=variants[v] -              if vv then -                vv[u]=unicode -              else  -                vv={ [u]=unicode } -                variants[v]=vv +          indices[index]=unicode +          descriptions[unicode]={ +            boundingbox=glyph.boundingbox, +            name=name, +            index=index, +            glyph=glyph, +          } +          local altuni=glyph.altuni +          if altuni then +            for i=1,#altuni do +              local a=altuni[i] +              local u=a.unicode +              if u~=unicode then +                local v=a.variant +                if v then +                  local vv=variants[v] +                  if vv then +                    vv[u]=unicode +                  else  +                    vv={ [u]=unicode } +                    variants[v]=vv +                  end +                end                end              end            end +        else +          report_otf("potential problem: glyph %U is used but empty",index)          end -      else -        report_otf("potential problem: glyph %U is used but empty",index)        end +    else +      report_otf("potential problem: no glyphs found")      end    end    resources.private=private @@ -12719,15 +12781,8 @@ local function normal_handle_contextchain(head,start,kind,chainname,contexts,seq                break              end            end -        elseif f==2 then -          match=seq[1][32]          else -          for n=f-1,1 do -            if not seq[n][32] then -              match=false -              break -            end -          end +          match=false          end        end        if match and s>l then @@ -12777,15 +12832,8 @@ local function normal_handle_contextchain(head,start,kind,chainname,contexts,seq                break              end            end -        elseif s-l==1 then -          match=seq[s][32]          else -          for n=l+1,s do -            if not seq[n][32] then -              match=false -              break -            end -          end +          match=false          end        end      end @@ -15162,6 +15210,8 @@ end  local fonts=fonts  local nodes=nodes  local traverse_id=node.traverse_id +local free_node=node.free +local remove_node=node.remove  local glyph_code=nodes.nodecodes.glyph  local disc_code=nodes.nodecodes.disc  local ligaturing=node.ligaturing @@ -15191,6 +15241,8 @@ function nodes.handlers.nodepass(head)      local basefonts={}      local prevfont=nil      local basefont=nil +    local variants=nil +    local redundant=nil      for n in traverse_id(glyph_code,head) do        local font=n.font        if font~=prevfont then @@ -15212,10 +15264,47 @@ function nodes.handlers.nodepass(head)                  basefonts[#basefonts+1]=basefont                end              end +            local resources=tfmdata.resources +            variants=resources and resources.variants +            variants=variants and next(variants) and variants or false +          end +        else +          local tfmdata=fontdata[prevfont] +          if tfmdata then +            local resources=tfmdata.resources +            variants=resources and resources.variants +            variants=variants and next(variants) and variants or false +          end +        end +      end +      if variants then +        local char=n.char +        if char>=0xFE00 and (char<=0xFE0F or (char>=0xE0100 and char<=0xE01EF)) then +          local hash=variants[char] +          if hash then +            local p=n.prev +            if p and p.id==glyph_code then +              local variant=hash[p.char] +              if variant then +                p.char=variant +                if not redundant then +                  redundant={ n } +                else +                  redundant[#redundant+1]=n +                end +              end +            end            end          end        end      end +    if redundant then +      for i=1,#redundant do +        local n=redundant[i] +        remove_node(head,n) +        free_node(n) +      end +    end      for d in traverse_id(disc_code,head) do        local r=d.replace        if r then | 
