#!/usr/bin/env texlua

-- The MUSIXFLXVERSION below is the one checked against.

MUSIXFLXVERSION = "0.83"
VERSION = "0.83.3.lua7" 

--[[

Line breaking program for MusiXTeX.

 (c) Copyright Ross Mitchell 1992-1997 ross.mitchell@csiro.au
 (c) Copyright J. Hunsberger 1997 jhunsberger@i2k.co
 (c) Copyright 2009, 2010 Peter Breitenlohner <tex-live@tug.org>
 (c) Copyright 2011 Nikhil Helferty  6nh14@queensu.ca 
 (c) Copyright 2011 Bob Tennent rdt@cs.queensu.ca

This program is free software; you can redistribute it and/or modify it
under the terms of the GNU General Public License as published by the
Free Software Foundation; either version 2 of the License, or (at your
option) any later version.

This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
Public License for more details.

You should have received a copy of the GNU General Public License along
with this program; if not, write to the Free Software Foundation, Inc.,
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.

--]]

--[[

Conversion to Lua by Nikhil Helferty, 2011. 
Minor corrections by Bob Tennent rdt@cs.queensu.ca.

NH: Comments marked NH (such as this one) are comments new to the Lua
version. Most have to do with explaining any additions or changes made
necessary due to differences between Lua and C. Comments not so marked
are retained from C version.

Almost all comments from the C version have been put in this version as
well. Note that Lua variables are dynamically typed, and Lua has only
one data structure called "tables" which, though more flexible than
arrays, are essentially just used as arrays here.

--]]

--[[  

  ChangeLog:

  lua7   2011/04/21   RDT
         Decode option argument for debugging.
         Revert to accepting .mx1 or .tex filenames.
         Rationalize file-name variables.

  lua6   2011/04/18   RDT
         Revert to stand-alone use.
 
  lua5   2011/04/13   RDT
         Allow .mx1 file to have a version number only.

  lua4   2011/04/12   RDT
         Adaptation to being called from musixtex.tex

  lua3   2011/04/11   RDT
         Replaced body of eightfivefloat by string.format("%8.5f", ... )

  lua2   2011/04/11   RDT
         Change PRINT to print for dbug output.

  lua1   2011/04/08   RDT
         Strip off pt for L line.
   
--]]

-- max sizes
-- NH: these aren't needed for array sizes in Lua (tables grow dynamically)
-- but I am keeping the checks in case the maximums are necessary for
-- TeX processing

MAX_SIGNS = 128; MAX_SECTIONS = 128; MAX_BARS = 2048

function GETLINE()
  linebuf = infile:read()
  return linebuf
end

-- NH: declare tables

-- NH: the following tables were all arrays of type int in C
-- the following tables were size MAX_BARS in C (2048)
zbar = {}; lr_repeat = {}; raggedline = {}
l_repeat = {}; barno = {}

-- NH: the following tables were size MAX_SECTIONS in C (128)
autolines = {}; bars = {}; mulooseness = {}

-- NH: the following was of size MAX_SIGNS in C (128)
signchange = {}

-- NH: the following were counters declared for xbar sign change detection
-- xbar of size MAX_BARS, linegoal MAX_SECTIONS
-- xbar_count, xbar_flag, softspace_count were integers declared here
-- as they aren't tables they'll be declared upon use
xbar = {}; linegoal = {}

-- NH: the following tables were all arrays of type double in C
-- the following were size MAX_BARS
hardbarlength = {}; softbarlength = {}
width_leftrightrepeat = {}; width_leftrepeat = {}

-- NH: the following were size MAX_SECTIONS
eff_hardlength = {}; eff_softlength = {}

-- NH: the following were size MAX_SIGNS
oldsignskip = {}; signskip = {}; futuresignskip = {}


-- NH: (no need to add last \n as print does this)
function error_exit (error_number)
  -- NH: the following if/elseif/else construction mirrors case structure

  if error_number == 0 then
    print("\nFile error.")
  elseif error_number == 1 then
    print("\nUsage: [texlua] musixflx.lua basename[.tex | .mx1] [ d | m | f | s ]")
  elseif error_number == 2 then
    print("\nThis shouldn't happen ! Too few bars or \\mulooseness too large ?")
  elseif error_number == 3 then
    print("\nThis shouldn't happen ! Too few bars in section !")
  elseif error_number == 4 then
    print("\nMissing endmark ! Forgotten \\stop[end]piece ?")
  elseif error_number == 5 then
    print("\nDivision by zero ! Ask a wizard !")
  elseif error_number == 6 then
    print("\nVersion mis-match:")
    print("MusiXTeX version " .. linebuf)
    print("musixflx expected version " .. MUSIXFLXVERSION)
  elseif error_number == 7 then
    print("\nFile not found: " .. infilename)
  elseif error_number == 8 then
    print("\nCorrupted " .. infilename)
  elseif error_number == 9 then
    print("\nToo many sections, maximum number of sections: " .. MAX_SECTIONS)
  elseif error_number == 10 then
    print("\nError in " .. infilename .. " at line " .. currentline)
  elseif error_number == 11 then
    print("\nToo many bars, maximum number of bars: " ..  MAX_BARS)
  else
    print("!!! Can't go on !!!")
  end
  os.exit(3)

end
-- end error_exit

-- NH: there is really no analogue to the strrchr function in C, which is frequently used
-- to locate the location of a last occurrence of a string
-- so I coded a similar function for Lua
-- takes two strings (str and pat), and returns everything after the last occurence of pat
-- in str - if no occurrences of pat, then returns nil
function strrchr(str, pat)
  notfound = true
  currentIndex = 0
  lastIndex = 0
  while notfound do
    currentIndex = string.find(str, pat, lastIndex+1, true)
    if (not currentIndex) then -- NH: returned nil - no occurence past lastIndex
      notfound = false
    else  -- NH: got a hit for an occurence - store this value as lastIndex
      lastIndex = currentIndex
    end
  end
  if lastIndex == 0 then return nil end -- NH: no occurence found
  return string.sub(str, lastIndex+1)
end


-- NH: start "main chunk"

io.write(string.format("Musixflx-%s", VERSION))

-- initialize variables
junk = -9999
dbug = false
dbug_lines = false
dbug_logfile = false
showresult = false
detectraggedline = false
currentline = 1
samechapter = true
line_number = 0
chapterno = 1

lthick = .4

if dbug then
  print("\n ... decoding command line")
end

filename = arg[1]
if not filename then
   error_exit(1)
end

extension = string.sub(filename, -4, -1)
if extension == ".tex" or extension == ".mx1" then
  basename = string.sub(filename, 1, -5)
else
  basename = filename
end

--[[

debugging options:
 d output debugging information to the screen
 m output line numbers to the screen 
 f output debugging information to file basename.mxl (not mx1)
 s output computed lines to the screen

 --]]

if (#arg == 2) then
  if (arg[2] == "d") then 
    dbug = true
  elseif (arg[2] == "m") then 
    dbug_lines = true
  elseif (arg[2] == "f") then
    dbug = true
    dbug_logfile = true
    dbug_lines = true
  elseif (arg[2] == "s") then
    showresult = true
  else
    error_exit(1)
  end
end

infilename = basename .. ".mx1"

-- open the .mx1 file containing bar length information

if dbug then
  print(" ... opening " .. infilename .. " for input")
-- NH: io.open analagous to fopen C function
-- will return nil value if not successful
-- nil is taken as false in if statements
end
infile = io.open(infilename, "r")
if (not infile) then 
  error_exit(7)
end
io.write(string.format(" (%s", infilename))
if showresult then
  print("")
end

currentline = currentline + 1
if GETLINE() and (linebuf ~= MUSIXFLXVERSION) then 
  error_exit(6)
end


-- Open the output file
outfilename = basename .. ".mx2"
if dbug then
  print(" ... opening " .. outfilename .. " for output")
end
outfile = io.open(outfilename, "w")
if (not outfile) then
  print("\nCan't create: " .. outfilename)
  os.exit(3)
end

-- Open the logfile
if dbug_logfile then
  logfilename = basename .. ".mxl"
  print(" ... open " .. logfilename .. " for debugging")
  logfile = io.open(logfilename, "w")
  if (not logfile) then
    print("\nCan't create: " .. logfilename)
    os.exit(3)
  end
  logfile:write("Version ", VERSION, "\n")
end

-- skip startindicator

if GETLINE() and (linebuf ~= "S") then
  error_exit(8)
end

--[[
do...while loop for
 moretimes call of \startpiece
 >>>>>>>>>>>>>>>>>>>>>>>>>>>

 Do while equivalent is repeat...until
 --]]

 GETLINE()

 repeat

-- reset all arrays

if dbug then print("\n------- Chapter " .. chapterno .. " -------\n"); end
if dbug_logfile then logfile:write("\n------- Chapter ", chapterno, " -------\n\n"); end
-- NH: copying C method of loops but this means will be initializing the max spaces every time
-- ... not taking advantage of dynamic sizes of tables in Lua
-- however necessary to initialize since the starting values are used in operations
-- note upper bounds for these loops are the max-1, as for syntax in lua will include the upper (<= instead of <)

for i = 0, MAX_SIGNS-1 do
  signchange[i] = junk
  oldsignskip[i] = 0
  futuresignskip[i] = 0 -- jh-1
  signskip[i] = 0
end

for i = 0, MAX_BARS-1 do
  hardbarlength[i] = 0
  softbarlength[i] = 0
  width_leftrightrepeat[i] = 0
  lr_repeat[i] = false
  width_leftrepeat[i] = 0
  xbar[i] = 0 -- jh-1
  l_repeat[i] = false
  zbar[i] = false
  raggedline[i] = false
  barno[i] = 0
end

for i = 0, MAX_SECTIONS-1 do
  eff_hardlength[i] = 0
  eff_softlength[i] = 0
  bars[i] = 0
  autolines[i] = false
  linegoal[i] = 0
  mulooseness[i] = 0
end

--[[

 Read and decode header items:

 1. Linewidth;
 2. Parindent;
 3. Beforeruleskip;
 4. Afterruleskip;
 5. Elemskip;
 6. Clefskip;
 7. Signskip;
 --]]

 -- NH: for those curious why it is necessary to take a substring of the line, it is to strip the "pt" off of the string
 -- to make them valid strings for coercion to numbers later on (something that the atof does automatically)
 -- RDT: linebuf tests are to avoid taking a substring of an empty string
if linebuf then linewidth = string.sub(linebuf, 1, -3); currentline=currentline+1 end
GETLINE(); if linebuf then parindent = string.sub(linebuf, 1, -3); currentline=currentline+1 end
GETLINE(); if linebuf then beforerule = string.sub(linebuf, 1, -3); currentline=currentline+1 end
GETLINE(); if linebuf then afterrule = string.sub(linebuf, 1, -3); currentline=currentline+1 end
GETLINE(); if linebuf then elemskip = string.sub(linebuf, 1, -3); currentline=currentline+1 end
GETLINE(); if linebuf then clefskip = string.sub(linebuf, 1, -3); currentline=currentline+1 end
GETLINE(); if linebuf then signskip[0] = string.sub(linebuf, 1, -3); currentline=currentline+1 end

futuresignskip[0] = signskip[0] -- Initialize for xbar signs, jh-1

--[[
 Read the records specifying contributions to bar length.
 Lengths are of two types:
 (a) Hard or unscaleable, eg barlines, clef or meter changes (typ=0).
 (b) Soft or scaleable, eg noteboxes which scale with \elemskip (typ=1).
 Note that \afterruleskip and \beforerulskip are considered soft.
 --]]

jbar = junk
i = 0
sign = 0
xbar_count = 0 -- Used to detect presence of xbars jh-1
xbar_flag = false -- True allows check for xbar jh-1
hardsign = 0 -- Accumulator for hard skip from sign change jh-1
softspace_count = 0 -- Watch for softspace after bars are posted jh-1

all_section = 0

if dbug then
  print(" ... reading")
end

while (samechapter and GETLINE()) do

  currentline = currentline + 1
  local firstChar = string.sub(linebuf, 1, 1)

  -- NH: use if/elseif/else to mirror switch/case statements

--[[
'\startpiece'
 stop reading,
 compute,
 write
 and start again
 --]]

  if (firstChar == "S") then
    samechapter = false

--[[
 End of section. Action:
 Right justify the material ending at the previous bar.
 Read the number following the *, which is the 'looseness'
 parameter of the section just ended.
 Reset the bar test integer to JUNK in case the bar number
 was reset between sections.
 --]]

  elseif (firstChar == "*") then
    local firstSpace = string.find(linebuf, " ")
    local secondSpace = string.find(linebuf, " ", firstSpace+1)
    if not secondSpace then -- NH: possibility that there is no second number in line
      mulooseness[all_section] = string.sub(linebuf, firstSpace+1)
    else
      mulooseness[all_section] = string.sub(linebuf, firstSpace+1, secondSpace-1)
    end
    -- NH: above firstSpace/secondSpace mathematics were to mimic function of atol use in original C function
    linegoal[all_section] = 0 -- jh-2 + WS-1
    if secondSpace then
      linegoal[all_section] = string.sub(linebuf, secondSpace+1) -- NH: this my interpretation of an sscanf line signed jh-2 + WS-1 in C
    end
    linegoal[all_section] = linegoal[all_section] + 0 -- NH: this is a fairly silly trick to force coercion for valid comparison to numbers, could also use tonumber function
    mulooseness[all_section] = mulooseness[all_section] + 0
    if ((mulooseness[all_section] ~= 0) and (linegoal[all_section] > 0)) then
      print("\\linegoal{" .. linegoal[all_section] .. "} ignored because \\mulooseness not equal to zero")
      print("   for section " .. all_section+1 .. " in chapter " .. chapterno)
      linegoal[all_section] = 0 -- Reset to ignore it. jh-2 + WS-1
    end
    all_section = all_section + 1
    if all_section > (MAX_SECTIONS - 1) then
      error_exit(9)
    end
    jbar=junk

--[[
 Right justify the material ending at the previous bar.
 Set flag.
 Reset the bar test integer to JUNK in case the bar number
 was reset between sections.
 ]]


  elseif (firstChar == "a") then
    autolines[all_section] = true
    all_section = all_section + 1
    if (all_section > (MAX_SECTIONS - 1)) then
      error_exit(9)
    end
    jbar=junk

--[[
 found a raggedline, let's set a flag
 I think, I'll hardly get a Nobel-Award for coding this
 What a pity! :-(
 but perhaps a Pulitzer-Award for my comments. :-)
--]]

  elseif (firstChar == "r") then
    raggedline[i+1] = true


  -- found \zbar, set flag and store barno
  elseif (firstChar == "z") then
    zbar[i] = true
    barno[i] = strrchr(linebuf, " ")
    xbar_count = xbar_count + 1 -- Help detect xbars, track zbar offsets. jh-1
    hardsign = 0 -- reset any sign change skip accumulated. jh-1
    softspace_count = 0 -- Reset for next bar. jh-1


--[[
 found a leftrightrepeat
 set a flag and store the different widths
 ]]

  elseif (firstChar == "l") then
    -- NH: there are some commented out printf lines in C code in this case
    -- I'm assuming they were used at one point for debugging purposes and
    -- am not translating them, but if should be reinstated they exist at lines
    -- 380-387 in C program
    lr_repeat[i] = true
    local firstSpace = string.find(linebuf, " ")
    local secondSpace = string.find(linebuf, " ", firstSpace + 1)
    width_leftrightrepeat[i] = string.sub(linebuf, firstSpace+1, secondSpace-3) -- NH: secondspace-3 as secondspace-1 will include the "pt"
    width_leftrepeat[i] = string.sub(linebuf, secondSpace+1, -3) -- NH: again, -3 to chop off the "pt"


--[[
 found a leftrepeat
 set a flag and store width
 ]]

  elseif (firstChar == "L") then
    l_repeat[i] = true
    local firstSpace = string.find(linebuf, " ")
    width_leftrepeat[i] = string.sub(linebuf, firstSpace+1, -3) -- RDT: again, -3 to chop off the "pt"


  -- store barno
  elseif (firstChar == "b") then
    barno[i] = strrchr(linebuf, " ")
    xbar_count = xbar_count+1 -- Track possible xbars. jh-1
    xbar_flag = true -- Allow check of next line to detect xbar. jh-1
    hardsign = 0 -- Reset accumulated hard sign skip. jh-1
    softspace_count = 0 -- Reset for next bar. jh-1

--[[
 enabling the use of 'hard' offsets
 advance current hardwith
 reduce current softwidth
--]]

  elseif (firstChar == "h") then
    x = string.sub(strrchr(linebuf, " "), 1, -3) -- -3 is to chop off "pt" in string to allow x to be valid for arithmetic
    softbarlength[i] = softbarlength[i] - x
    hardbarlength[i] = hardbarlength[i] + x
    eff_softlength[all_section] = eff_softlength[all_section] - x
    eff_hardlength[all_section] = eff_hardlength[all_section] + x


--[[
 This record began with 's' and specifies a key signature change
 store the signskip, s.b.
 ]]

  elseif (firstChar == "s") then -- Changes to detect xbar and signchange interaction, jh-1

    local firstSpace = string.find(linebuf, " ")
    local secondSpace = string.find(linebuf, " ", firstSpace+1)
    if not secondSpace then
      tempholdskip = string.sub(linebuf, firstSpace+1, -3)
    else
      tempholdskip = string.sub(linebuf, firstSpace+1, secondSpace-3)
    end
    -- NH: all of above to mimic atof statement
    -- wasn't sure if statements with s would have a second space and more info
    -- so accounted for either possibility

    -- We might be in the middle of an xbar setup, and we only want to increment
    -- the sign pointer if this is the first sign change. jh-1
    if (not (signchange[sign] == i)) then -- first time for this bar set jh-1
      sign = sign + 1
      signchange[sign] = i
      signskip[sign] = tempholdskip
      oldsignskip[sign] = hardsign -- Capture accumulated hard space. jh-1
      xbar[i] = xbar[i] + 1 -- Increment to detect xbars with sign changes. jh-1
      --[[
      Housekeeping is done... Now, one more condition to check. If there
      has been any soft space since the bar was declared, then musixtex
      will NOT publish a sign change notice at the end of the line for
      this sign change when there is a line break in this bar.
      In that case, signal that this should be treated as an xbar, which
      will effectively suppress the transfer of hardspace for the sign
      change notice. jh-1
      --]]
      if softspace_count > 0 then xbar[i] = 2 end-- Suppress space move jh-1
    end
    -- Always update the futuresignskip value in case this is the last
    futuresignskip[sign] = tempholdskip

  -- comment, do nothing
  elseif (firstChar == "%") then

  -- This is an 'ordinary' line, listing a contribution to the barlength.
  else
    if (not (tonumber(firstChar))) then
      error_exit(10)
    end
    local firstSpace = string.find(linebuf, " ")
    local secondSpace = string.find(linebuf, " ", firstSpace + 1)
    bar = string.sub(linebuf, 1, firstSpace-1)
    typ = string.sub(linebuf, firstSpace+1, secondSpace-1)
    x = string.sub(linebuf, secondSpace+1, -3) -- the -3 is to strip the "pt" off of the x
    if (typ ~= "0") then eff_softlength[all_section] = eff_softlength[all_section] + x
    else eff_hardlength[all_section] = eff_hardlength[all_section] + x
    end

--[[
 Increment bar number if the bar number
 read from the file has changed.
 Accumulate current bar length.
 --]]
    bar = bar + 0 -- NH: forcing coercion to number type from string
    if tonumber(bar) > jbar then
      i = i + 1
      bars[all_section] = bars[all_section] + 1
      if i > MAX_BARS then
        error_exit(11)
      end
    end

      -- At this point, can check if this is an xbar...Only check once. jh-1
    if xbar_flag then
      if ((xbar_count - bar - 1) == 0) then -- find an xbar. jh-1
        xbar_count = xbar_count-1 -- adjust offset to stay on track

        --[[
        To handle the special conditions caused by possible xbars and
        sign changes, the xbar logic state has to be examined and changed
        only if this is the first xbar, and not for subsequent ones. jh-1
        --]]
        if (xbar[i] == 0) then -- then this is the first xbar in the setup
          -- or there has already been a sign change
          xbar[i] = 1 -- jh-1
        end
      end
    end

    xbar_flag = false -- Reset to prevent checks until next bar. jh-1

    if (typ ~= "0")  then
      softbarlength[i] = softbarlength[i] + x
      -- Count softspace entries to help in sign/linebreak decisions jh-1
      softspace_count = softspace_count+1
      hardsign = 0 -- Safety - just be sure in case line break jh-1
    else -- jh-1
      hardbarlength[i] = hardbarlength[i] + x
      hardsign = hardsign + x -- accumulate hardspace, there may be a sign change
    end

    jbar = bar

  end -- end if statements

end -- end while loop

--[[
Decrement the number of sections if the final section is void.
 This will be the usual case where the input file ends with
 an end of section record.
 If this record has been omitted, stop going on to avoid
 'You can't use \raise....'.
 ]]

if dbug then
  print(" ... compute")
end

if (bars[all_section] == 0) then
  all_section = all_section - 1
else
  error_exit(4)
end

-- Summarize sectioning information

if dbug then
  print("\nNumber of sections        : "  .. (all_section+1) .."\n")

  for section = 0, all_section do
    if (autolines[section]) then
      print("---- autoline section ----")
    end
    print("Section                   : " .. (section+1))
    print("Number of bars in section : " .. bars[section])
    print("Length(hard) of section " .. (section+1) .. " : " .. eff_hardlength[section])
    print("Length(soft) of section " .. (section+1) .. " : " .. eff_softlength[section])
    if (linegoal[section] ~= 0) then -- linegoal applies, signed jh-2 in  C
      print("Section line goal was determined by \\linegoal value...")
      print("Line Goal for section     : " .. linegoal[section])
    else -- mulooseness applies, signed jh-2 in C
      print("Looseness of section      : " .. mulooseness[section])
    end

    io.read() -- NH: mimics getchar()
  end -- end for

end -- end if

if dbug_logfile then
  logfile:write("\nNumber of sections        : ", all_section+1, "\n\n")

  for section = 0, all_section do
    if (autolines[section]) then
      logfile:write("---- autoline section ----\n")
    end
    logfile:write("Section                   : ", section+1, "\n")
    logfile:write("Number of bars in section : ", bars[section], "\n")
    logfile:write("Length(hard) of section ", section+1, " : ", eff_hardlength[section], "\n")
    logfile:write("Length(soft) of section ", section+1, " : ", eff_softlength[section], "\n")
    if (linegoal[section] ~= 0) then -- line goal applies. signed jh-2 in C
      logfile:write("Section line goal was determined by \\linegoal value...\n")
      logfile:write("Line goal for section     : ", linegoal[section], "\n")
    else -- mulooseness applies, signed jh-2 in C
      logfile:write("Looseness of section      : ", mulooseness[section], "\n")
    end
  end
end

--[[
C Comments:
Loop over the sections defined in the input file.
 Each section must be right justified.
 LAST is the absolute number of the last bar
   in the current section.
--]]

sign = 0
mark = 0
lastbarnumber = 0


for section = 0, all_section do
  line_in_section = 1
  lastbarnumber = lastbarnumber + bars[section]

  -- Find number of lines to work towards
  lines = math.floor(((eff_hardlength[section] + eff_softlength[section] + parindent)/
        (linewidth-(clefskip+signskip[sign]))) + .5) -- need to use floor function as lua has no "integer" type, just numbers
  if (lines == 0) then lines = 1 end -- safety

  natural_lines = lines -- Keep this for debug report. Signed jh-2 in C
  lines = lines + mulooseness[section]
  if ((mulooseness[section] ~= 0) and (linegoal[section] > 0)) then -- signed jh-2 in C
    print("Unexpected line goal reset occured for section " .. section+1)
    linegoal[section] = 0 -- Zero it, Safety, should not happen, signed jh-2 in C
  end
  if (linegoal[section] > 0) then lines = linegoal[section] end -- Signed jh-2 in C
  if lines < 1 then
    lines = 1
    print("Don't stress \\mulooseness too much !!!")
  end

--[[
C comments:
 autolinesflag set in current section ?
 iftrue force number of lines to 1
 --]]

  if (autolines[section]) then lines = 1 end

  if dbug then
    print("Section number           : " .. section+1)
    print("Last bar in this section : " .. lastbarnumber)
    print("Number of bars           : " .. bars[section])
    print("Natural number of lines  : " .. natural_lines) -- signed jh-2 in C
    print("Chosen  number of lines  : " .. lines .. "\n")
  end

  if dbug_logfile then
    logfile:write("Section number           : ", section+1, "\n")
    logfile:write("Last bar in this section : ", lastbarnumber, "\n")
    logfile:write("Number of bars           : ", bars[section], "\n")
    logfile:write("Natural number of lines : ", natural_lines, "\n") -- signed jh-2 in C
    logfile:write("Chosen  number of lines : ", lines, "\n\n")
  end

  if (bars[section]<1) then error_exit(3) end

--[[
C comments:
 fill_length is the length of 'bar' material (ie excluding
 signature space) required to fill the remainder
 of the piece. This value will not be exact if there are
 sign changes within the section. However,
 fill_length is used only to keep track of the mean scale factor
 for the remainder of the piece, as opposed to individual lines.

 Loop over lines, working out number of bars
 and revised \elemskip for each line.
 added correct computing of fill_length
 --]]


  for j = 1, lines do
    line_number = line_number + 1
    fill_length = (lines-j+1) * (linewidth - (clefskip + signskip[sign]))


--[[
C comments:
 Work out mean element skip over remaining bars
 in the current section.
 EFFWID is the effective line width once
 key signature have been written.
 Set parindent to zero after it has been used for the
 first line of the first section.
 --]]

    if (eff_softlength[section] == 0) then error_exit(5) end
    spc_factor = (fill_length - eff_hardlength[section])/eff_softlength[section]
    if ((xbar[mark+1] > 1) and (mark>0)) then
      -- The bar is an bar+xbar with a sign change. Signed jh-1 in C
      eff_linewidth = linewidth - (clefskip+signskip[sign-1]) - parindent
    else -- This is a normal bar. Signed jh-1 in C
      eff_linewidth = linewidth - (clefskip+signskip[sign]) - parindent
    end -- Signed jh-1 in C

    signskip[sign] = futuresignskip[sign] -- Supports xbar signs. Signed jh-1 in C

    parindent = 0

    -- Fill the current line by adding bars until overflow.

    i = mark
    firstbarno = barno[mark+1]
    hardlength = 0
    softlength = 0
    x = 0
    lastbar = 0
    detect_end = false

    while (x < eff_linewidth) do
      if detect_end then break end
      i = i + 1
      -- Check for raggedline

      if raggedline[i] then detectraggedline = true end

      -- Check for key signature change at this bar.

      if (i == signchange[sign+1]) then sign = sign + 1 end

      lastbar = hardbarlength[i] + spc_factor*softbarlength[i]
      x = x + lastbar

      -- Enforce termination at last bar and last line

      if (i==lastbarnumber) then detect_end = true
      elseif (line_in_section == lines) then
        detect_end = false
        x = 0
      end

      hardlength = hardlength + hardbarlength[i]
      softlength = softlength + softbarlength[i]

    end

--[[
C comments:
 If the overhang is less than half the barlength,
 include the latest bar in the line,
 and shrink the line accordingly.
--]]

    if ((x-eff_linewidth) < (lastbar/2)) then
      barsinline = i - mark
      mark = i
      lastbarno = barno[mark]

--[[
C comments:
 last bar in line a zbar?
 if true -> add to the first bar in next line
 the amount of afterruleskip
--]]

      if zbar[mark] then
        softbarlength[i+1] = softbarlength[i+1] + afterrule
        eff_softlength[section] = eff_softlength[section] + afterrule
      end

--[[
C comments:
 last bar in line a leftrightrepeat?
 if true -> reduce hardwidth of current line
            advance the hardwidth of next bar
            advance the softwidth of next bar
--]]

      if lr_repeat[mark] then
        -- NH: there are some commented lines of code out here (printf statements) that I have not preserved
        -- if they should be reinstated then they are located at lines 744-746 of the original program
        hardlength = hardlength - (width_leftrightrepeat[i]  - width_leftrepeat[i])
        eff_hardlength[section] = eff_hardlength[section] + (width_leftrightrepeat[i] - width_leftrepeat[i])
        hardbarlength[i+1] = hardbarlength[i+1] + width_leftrepeat[i]
        softbarlength[i+1] = softbarlength[i+1] + (afterrule/2)
        eff_softlength[section] = eff_softlength[section] + (afterrule/2)
      end

--[[
C comments:
 last bar in line a leftrepeat?
 if true -> reduce hardwidth of current line
            advance the hardwidth of next bar
            advance the softwidth of next bar
--]]

      if l_repeat[mark] then
        hardlength = hardlength - (width_leftrepeat[i] - lthick)
        hardbarlength[i+1] = hardbarlength[i+1] + width_leftrepeat[i]
        softbarlength[i+1] = softbarlength[i+1] + (afterrule/2)
        eff_softlength[section] = eff_softlength[section] + (afterrule/2)
      end

      if (signchange[sign+1] == mark+1) then -- signed s.b. in C
        sign = sign + 1
--[[
C comments:
  Because the bar is staying here in the line, we look ahead
  to see if the upcoming bar is a sign change, and adjust space
  to account for the complimentary sign change notice that will
  be posted at the end of this line.  However, if the upcoming
  sign change bar is really a bar+xbar set, where the sign change
  is buried in the xbar, then we don't do the move because the
  change notice really won't be posted in this line.

  Signed jh-1 in C.

--]]
        if (xbar[mark+1] < 2) then -- okay to do the move. signed jh-1 in C
          hardlength = hardlength + oldsignskip[sign]
          hardbarlength[mark+1] = hardbarlength[mark+1] - oldsignskip[sign]
        end
      end

    -- Exclude the latest bar, and stretch the line.

    else
      barsinline=i-1-mark
      if (barsinline < 1) then error_exit(2) end
      mark = i - 1
      lastbarno = barno[mark]
      hardlength = hardlength - hardbarlength[i]
      softlength = softlength - softbarlength[i]

      if zbar[mark] then softbarlength[i] = softbarlength[i] + afterrule end

      if lr_repeat[mark] then
        hardlength = hardlength - (width_leftrightrepeat[i-1]-width_leftrepeat[i-1])
        eff_hardlength[section] = eff_hardlength[section] + (width_leftrightrepeat[i-1] - width_leftrepeat[i-1])
        hardbarlength[i] = hardbarlength[i] + width_leftrepeat[i-1]
        softbarlength[i] = softbarlength[i] + (afterrule/2)
        eff_softlength[section] = eff_softlength[section] + (afterrule/2)
      end

      if l_repeat[mark] then
        hardlength = hardlength - (width_leftrepeat[i-1] - lthick)
        hardbarlength[i] = hardbarlength[i] + width_leftrepeat[i-1]
        softbarlength[i] = softbarlength[i] + (afterrule/2)
        eff_softlength[section] = eff_softlength[section] + (afterrule/2)
      end

--[[
C comments:
 Error (o/u-hbox) occurs only when signature change start in next line
 -> look for signature change in next line
 if true then advance the hardwidth of current line
    reduce next hard barlength by signature change
--]]

      if (signchange[sign] == (mark+1)) then
--[[
C comments:
However, if the next bar is a bar+xbar set where the
sign change comes from the xbar, then don't do this
move, because the extra skip is not really there!

Signed jh-1 in C.
--]]
        if(xbar[mark+1] < 2) then -- alright, do the move. signed jh-1 in C
          hardlength = hardlength + oldsignskip[sign]
          hardbarlength[mark+1] = hardbarlength[mark+1] - oldsignskip[sign]
        end

      end

    end

--[[
C comments:
 Define a flex factor for this line as the ratio
 of soft part of the specified line width,
 to soft width in the approximate line.
 --]]

    if (softlength == 0) then error_exit(5) end
    flexit = (eff_linewidth - hardlength) / softlength
    if detectraggedline then
      flexit = 1
      detectraggedline = false
    end
    cor_elemskip = elemskip * flexit
    cor_afterrule = afterrule * flexit
    cor_beforerule = beforerule * flexit

    if dbug then
      print("Line number             : " .. line_number)
      print("Fill length             : " .. fill_length)
      print("Effective length        : " .. eff_softlength[section] + eff_hardlength[section])
      print("Mean space factor       : " .. spc_factor)
      print("Bars in line            : " .. barsinline)
      print("Effective linewidth     : " .. eff_linewidth)
      print("Uncorrected hard length : " .. hardlength)
      print("Uncorrected soft length : " .. softlength)
      print("fLex factor (soft)      : " .. flexit)
      print("Corrected elemskip      : " .. cor_elemskip)
      print("Corrected afterrule     : " .. cor_afterrule)
      print("Corrected beforerule    : " .. cor_beforerule)

      io.read() -- NH: mimics getchar()
    end

    if dbug_logfile then
      logfile:write("Line number             : ", line_number, "\n")
      logfile:write("Fill length             : ", fill_length, "\n")
      logfile:write("Effective length        : ", eff_softlength[section] + eff_hardlength[section], "\n")
      logfile:write("Mean space factor       : ", spc_factor, "\n")
      logfile:write("Bars in line            : ", barsinline, "\n")
      logfile:write("Effective linewidth     : ", eff_linewidth, "\n")
      logfile:write("Uncorrected hard length : ", hardlength, "\n")
      logfile:write("Uncorrected soft length : ", softlength, "\n")
      logfile:write("Flex factor (soft)      : ", flexit, "\n")
      logfile:write("Corrected elemskip      : ", cor_elemskip, "\n")
      logfile:write("Corrected afterrule     : ", cor_afterrule, "\n")
      logfile:write("Corrected beforerule    : ", cor_beforerule, "\n\n")
    end

    eff_hardlength[section] = eff_hardlength[section] - hardlength
    eff_softlength[section] = eff_softlength[section] - softlength
    fill_length = fill_length - eff_linewidth

    -- Write a record to the output file

    outfile:write(
      string.format("\\lineset{%3d}{%2d}{%8.5fpt}{%8.5fpt}{%8.5fpt}%% %d - %d\n",
      line_number, barsinline, cor_elemskip, cor_afterrule,
      cor_beforerule,
      firstbarno, lastbarno))



    if showresult then
      print(
        string.format("\\lineset{%3d}{%2d}{%8.5fpt}{%8.5fpt}{%8.5fpt}%% %d - %d",
        line_number, barsinline, cor_elemskip, cor_afterrule,
        cor_beforerule, 
        firstbarno, lastbarno))
    end

    if dbug_lines then print(" ... writing line : " .. line_number) end
    line_in_section = line_in_section + 1 -- NH: need to do this in loop, lua syntax doesn't allow for it in for loop spec
  end -- end lines for
end -- end sections for

if dbug_lines then print("") end


samechapter = true
chapterno = chapterno + 1
until (not GETLINE()) -- end repeat loop

-- closing files
infile:close()
io.write(")")
if dbug_logfile then
  logfile:close()
  if (not logfile) then error_exit(0) end
end
outfile:close()
if (not outfile) then error_exit(0) end

if dbug then
  print(" ... that's all, bye")
end
print("")
-- NH: end main chunk and program!