Initial sketch
diff --git a/pyotlss.py b/pyotlss.py
new file mode 100755
index 0000000..8c8b880
--- /dev/null
+++ b/pyotlss.py
@@ -0,0 +1,199 @@
+#!/usr/bin/python
+
+# Python OpenType Layout Subsetter
+# Writte by: Behdad Esfahbod
+
+import fontTools.ttx
+import sys
+import types
+
+if len (sys.argv) < 3:
+	print >>sys.stderr, "usage: pyotlss.py font-file glyph..."
+	sys.exit (1)
+
+fontfile = sys.argv[1]
+glyphs   = sys.argv[2:]
+
+font = fontTools.ttx.TTFont (fontfile)
+
+names = font.getGlyphNames()
+# Convert to glyph names
+glyphs = [g if g in names else font.getGlyphName(int(g)) for g in glyphs]
+
+def add_method (clazz):
+	def wrapper(method):
+		setattr (clazz, method.func_name, method)
+	return wrapper
+
+#
+# Subset
+#
+
+@add_method(fontTools.ttLib.tables.otTables.Coverage)
+def subset (self, glyphs):
+	indices = [i for (i,g) in enumerate (self.glyphs) if g in glyphs]
+	self.glyphs = [g for g in self.glyphs if g in glyphs]
+	return indices
+
+@add_method(fontTools.ttLib.tables.otTables.ClassDef)
+def subset (self, glyphs):
+	self.classDefs = {g:v for g,v in self.classDefs.items() if g in glyphs}
+
+@add_method(fontTools.ttLib.tables.otTables.SingleSubst)
+def subset (self, glyphs):
+	if self.Format in [1, 2]:
+		self.mapping = {g:v for g,v in self.mapping.items() if g in glyphs}
+	else:
+		assert 0, "unknown format: %s" % self.Format
+
+@add_method(fontTools.ttLib.tables.otTables.MultipleSubst)
+def subset (self, glyphs):
+	if self.Format == 1:
+		indices = self.Coverage.subset (glyphs)
+		self.Sequence = [self.Sequence[i] for i in indices]
+	else:
+		assert 0, "unknown format: %s" % self.Format
+
+@add_method(fontTools.ttLib.tables.otTables.AlternateSubst)
+def subset (self, glyphs):
+	if self.Format == 1:
+		self.alternates = {g:v for g,v in self.alternates.items() if g in glyphs}
+	else:
+		assert 0, "unknown format: %s" % self.Format
+
+@add_method(fontTools.ttLib.tables.otTables.LigatureSubst)
+def subset (self, glyphs):
+	self.ligatures = {g:v for g,v in self.ligatures.items() if g in glyphs}
+	self.ligatures = {g:[seq for seq in seqs if all(c in glyphs for c in seq.Component)]
+			  for g,seqs in self.ligatures.items()}
+	self.ligatures = {g:v for g,v in self.ligatures.items() if v}
+
+@add_method(fontTools.ttLib.tables.otTables.ContextSubst)
+def subset (self, glyphs):
+	pass # XXX
+
+@add_method(fontTools.ttLib.tables.otTables.ChainContextSubst)
+def subset (self, glyphs):
+	pass # XXX
+
+@add_method(fontTools.ttLib.tables.otTables.ExtensionSubst)
+def subset (self, glyphs):
+	if self.Format == 1:
+		self.ExtSubTable.subset (glyphs)
+	else:
+		assert 0, "unknown format: %s" % self.Format
+
+@add_method(fontTools.ttLib.tables.otTables.ReverseChainSingleSubst)
+def subset (self, glyphs):
+	if self.Format == 1:
+		indices = self.Coverage.subset (glyphs)
+		self.Substitute = [self.Substitute[i] for i in indices]
+		self.GlyphCount = len (self.Substitute)
+		for c in self.LookAheadCoverage:
+			c.subset (glyphs)
+		for c in self.BacktrackCoverage:
+			c.subset (glyphs)
+	else:
+		assert 0, "unknown format: %s" % self.Format
+
+@add_method(fontTools.ttLib.tables.otTables.SinglePos)
+def subset (self, glyphs):
+	if self.Format == 1:
+		self.Coverage.subset (glyphs)
+	elif self.Format == 2:
+		indices = self.Coverage.subset (glyphs)
+		self.Value = [self.Value[i] for i in indices]
+		self.ValueCount = len (self.Value)
+	else:
+		assert 0, "unknown format: %s" % self.Format
+
+@add_method(fontTools.ttLib.tables.otTables.PairPos)
+def subset (self, glyphs):
+	if self.Format == 1:
+		indices = self.Coverage.subset (glyphs)
+		self.PairSet = [self.PairSet[i] for i in indices]
+		for p in self.PairSet:
+			p.PairValueRecord = [r for r in p.PairValueRecord if r.SecondGlyph in glyphs]
+			p.PairValueCount = len (p.PairValueRecord)
+		# TODO Prune empty rules
+		self.PairSetCount = len (self.PairSet)
+	elif self.Format == 2:
+		self.Coverage.subset (glyphs)
+		self.ClassDef1.subset (glyphs)
+		self.ClassDef2.subset (glyphs)
+		# TODO Prune empty classes
+	else:
+		assert 0, "unknown format: %s" % self.Format
+
+@add_method(fontTools.ttLib.tables.otTables.CursivePos)
+def subset (self, glyphs):
+	if self.Format == 1:
+		indices = self.Coverage.subset (glyphs)
+		self.EntryExitRecord = [self.EntryExitRecord[i] for i in indices]
+	else:
+		assert 0, "unknown format: %s" % self.Format
+
+@add_method(fontTools.ttLib.tables.otTables.MarkBasePos)
+def subset (self, glyphs):
+	if self.Format == 1:
+		mark_indices = self.MarkCoverage.subset (glyphs)
+		self.MarkArray.MarkRecord = [self.MarkArray.MarkRecord[i] for i in mark_indices]
+		self.MarkArray.MarkCount = len (self.MarkArray.MarkRecord)
+		base_indices = self.BaseCoverage.subset (glyphs)
+		self.BaseArray.BaseRecord = [self.BaseArray.BaseRecord[i] for i in base_indices]
+		self.BaseArray.BaseCount = len (self.BaseArray.BaseRecord)
+		# TODO Prune empty classes
+	else:
+		assert 0, "unknown format: %s" % self.Format
+
+@add_method(fontTools.ttLib.tables.otTables.MarkLigPos)
+def subset (self, glyphs):
+	if self.Format == 1:
+		mark_indices = self.MarkCoverage.subset (glyphs)
+		self.MarkArray.MarkRecord = [self.MarkArray.MarkRecord[i] for i in mark_indices]
+		self.MarkArray.MarkCount = len (self.MarkArray.MarkRecord)
+		ligature_indices = self.LigatureCoverage.subset (glyphs)
+		self.LigatureArray.LigatureAttach = [self.LigatureArray.LigatureAttach[i] for i in ligature_indices]
+		self.LigatureArray.LigatureCount = len (self.LigatureArray.LigatureAttach)
+		# TODO Prune empty classes
+	else:
+		assert 0, "unknown format: %s" % self.Format
+
+@add_method(fontTools.ttLib.tables.otTables.MarkMarkPos)
+def subset (self, glyphs):
+	if self.Format == 1:
+		mark1_indices = self.Mark1Coverage.subset (glyphs)
+		self.Mark1Array.MarkRecord = [self.Mark1Array.MarkRecord[i] for i in mark1_indices]
+		self.Mark1Array.MarkCount = len (self.Mark1Array.MarkRecord)
+		mark2_indices = self.Mark2Coverage.subset (glyphs)
+		self.Mark2Array.Mark2Record = [self.Mark2Array.Mark2Record[i] for i in mark2_indices]
+		self.Mark2Array.MarkCount = len (self.Mark2Array.Mark2Record)
+		# TODO Prune empty classes
+	else:
+		assert 0, "unknown format: %s" % self.Format
+
+@add_method(fontTools.ttLib.tables.otTables.ContextPos)
+def subset (self, glyphs):
+	pass # XXX
+
+@add_method(fontTools.ttLib.tables.otTables.ChainContextPos)
+def subset (self, glyphs):
+	pass # XXX
+
+@add_method(fontTools.ttLib.tables.otTables.ExtensionPos)
+def subset (self, glyphs):
+	if self.Format == 1:
+		self.ExtSubTable.subset (glyphs)
+	else:
+		assert 0, "unknown format: %s" % self.Format
+
+for Gtag in ['GSUB', 'GPOS']:
+	if Gtag not in font:
+		continue
+	G = font[Gtag]
+	for lookup in G.table.LookupList.Lookup:
+		for subtable in lookup.SubTable:
+			print subtable
+			subtable.subset (glyphs)
+
+#font['GSUB'].toXML(sys.stdout, font)