Skip to content

Instantly share code, notes, and snippets.

@dancek
Created July 5, 2017 12:12
Show Gist options
  • Save dancek/f1377390ed6cd7f45710b29e67fa6061 to your computer and use it in GitHub Desktop.
Save dancek/f1377390ed6cd7f45710b29e67fa6061 to your computer and use it in GitHub Desktop.
SHENZHEN I/O Weekly #14: DUAL PULSE SPECTRUM ANALYZER
-- The function get_name() should return a single string that is the name of the puzzle.
--
function get_name()
return "Dual Pulse Spectrum Analyzer"
end
-- The function get_description() should return an array of strings, where each string is
-- a line of description for the puzzle. Surrounding text with asterisks will cause it to
-- be rendered in bold, something we use when mentioning a signal by name.
--
-- By using the syntax that we use in our puzzle descriptions, you can also create tables:
--
-- "| This is a table. | This is a second column. | This is a column with "
-- "| | | multiple line of text."
-- "----------------------------------------------------------------------"
-- "| This is a second | | More text here! "
-- "| table row. | | "
--
-- Puzzle descriptions are not automatically paginated, so you can use the string "<PAGE>"
-- to start a new page of text. If you don't it will overflow and make Zach sad.
--
function get_description()
return {
"*waveform* is a simple input connected to a radio receiver.",
"*spectrum* is an XBus output.",
"There are up to two pulse signals in *waveform*. Measure the pulse interval of each of them, and send them to *spectrum* in ascending order.",
"Each signal has different amplitude. The pulse interval is at most 13."
}
end
-- The function get_board() allows you to specify an 18x7 "ASCII art" grid to customize
-- the layout of the board and the placement of the input and output terminals. Generally
-- speaking, inputs are placed on the left of boards, while outputs are placed on the right.
--
-- For empty space, use the '.' character.
-- For buildable board space, use the '#' character.
-- For an input or output terminal, use characters '0' through '9' no more than once each.
-- For the bottom-left corner of the radio, use the 'R' character.
-- For the bottom-left corners of dials, use characters 'A', 'B', and 'C'.
--
function get_board()
return [[
#################1
##################
##################
##################
##################
##################
0#################
]]
end
function listContains(ls, item)
for k,v in pairs(ls) do
if v == item then
return true
end
end
return false
end
-- The function get_data() is called both to establish information about the puzzle (such as what
-- the inputs and outputs are) and to generate the random test cases. Signal levels and XBus data
-- should change from call to call, but information like the names and types of terminals should
-- not change at all.
--
-- To create a standard input or output terminal, call create_terminal(). Valid terminal types are
-- TYPE_SIMPLE, TYPE_XBUS, and TYPE_XBUS_NONBLOCKING. Valid terminal directions are DIR_INPUT and
-- DIR_OUTPUT. Valid data for a simple I/O signal is an array of integers, 0 - 100 inclusive. Valid
-- data for an XBus signal is an array of integer arrays, each with values -999 to 999 inclusive.
--
-- create_terminal(name, board_character, type, direction, data)
--
-- To create a radio (C2S-RF901), call create_radio(). You may only create one radio in each puzzle.
-- Since radios are XBus-only, the only valid data for data_rx and data_tx are arrays of integer arrays,
-- each with values -999 to 999 inclusive. You cannot customize the signal names for a radio.
--
-- By default the radio will be placed in the bottom-left corner of the screen. However, if you use an
-- 'R' character in your board layout, the bottom-left corner of the radio will be placed there instead.
--
-- create_radio(data_rx, data_tx)
--
-- To create a dial (N4DL-1000), call create_dial(). You may create up to three dials in each puzzle.
-- The names of dials should be kept short, as there is not much room to display them visually. A valid
-- value is an integer between 0 and 99, inclusive.
--
-- By default dials will be placed in the bottom-left corner of the screen. However, if you use an
-- 'A', 'B', or 'C' character in your board layout, the bottom-left corners of the first, second, and
-- third dials will be placed there, respectively.
--
-- create_dial(name, value)
--
-- NOTE: To generate random values you should use math.random(). However, you SHOULD NOT seed
-- the random number generator with a new seed value, as that is how the game ensures that
-- the first test run is consistent for all users, and thus something that allows for the
-- comparison of cycle scores.
--
-- NOTE: Fun fact! Arrays in Lua are implemented as tables (dictionaries) with integer keys that
-- start at 1 by convention. Contrast this with nearly every other programming language, in
-- which arrays start with an index of 0. Because of this, the 60 "time slices" that make
-- up a test case are indexed from 1 to 60 inclusive.
--
function get_data()
input = {}
output = {}
time = 1
pulse_count = math.random(1,2)
pulse_period = {}
pulse_amplitude = {}
for i = 1, pulse_count do
period = 0
amplitude = 0
repeat
period = math.random(2, 13)
until listContains(pulse_period, period) == false
repeat
amplitude = math.random(1, 33)
until listContains(pulse_amplitude, amplitude) == false
pulse_amplitude[i] = amplitude
pulse_period[i] = period
end
table.sort(pulse_period)
-- avoid overlapping third sequence
if pulse_count == 3 then
overlap = math.fmod(pulse_period[3], pulse_period[1]) == 0 or math.fmod(pulse_period[3], pulse_period[2]) == 0
if overlap then
pulse_period[3] = 13 -- guaranteed no overlap
end
if pulse_amplitude[1] + pulse_amplitude[2] == pulse_amplitude[3] then
pulse_amplitude[3] = pulse_amplitude[3] + 1
end
end
while time < 60 do
time = time + 1
amplitude = 0
for i = 1, pulse_count do
if math.fmod(time-2, pulse_period[i]) == 0 then
amplitude = amplitude + pulse_amplitude[i]
end
end
input[time] = amplitude
end
output[60] = pulse_period
create_terminal("waveform", "0", TYPE_SIMPLE, DIR_INPUT, input)
create_terminal("spectrum", "1", TYPE_XBUS, DIR_OUTPUT, output)
end
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment