Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 1 | import getopt |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 2 | from gl import * |
| 3 | from GL import * |
| 4 | from DEVICE import * |
| 5 | import time |
| 6 | import sys |
| 7 | import al |
| 8 | import AL |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 9 | import colorsys |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 10 | |
| 11 | BUFFERSIZE = 32000 |
| 12 | |
| 13 | class Struct(): pass |
| 14 | epoch = Struct() |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 15 | epoch.correcttiming = 1 |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 16 | EndOfFile = 'End of file' |
| 17 | bye = 'bye' |
| 18 | |
| 19 | def openspkr(): |
| 20 | conf = al.newconfig() |
| 21 | conf.setqueuesize(BUFFERSIZE) |
| 22 | conf.setwidth(AL.SAMPLE_16) |
| 23 | conf.setchannels(AL.MONO) |
| 24 | return al.openport('spkr','w',conf) |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 25 | |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 26 | def openvideo(name): |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 27 | try: |
| 28 | f = open(name, 'r') |
| 29 | except: |
| 30 | sys.stderr.write(name + ': cannot open\n') |
| 31 | sys.exit(1) |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 32 | line = f.readline() |
| 33 | if not line: raise EndOfFile |
Guido van Rossum | 2c8bf9d | 1992-05-06 17:58:18 +0000 | [diff] [blame] | 34 | colorinfo = (8, 0, 0, 0) |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 35 | if line[:4] == 'CMIF': |
| 36 | if line[:14] == 'CMIF video 2.0': |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 37 | line = f.readline() |
| 38 | colorinfo = eval(line[:-1]) |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 39 | line = f.readline() |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 40 | x = eval(line[:-1]) |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 41 | if len(x) == 3: w, h, pf = x |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 42 | else: w, h = x; pf = 2 |
Guido van Rossum | 864cde2 | 1992-05-07 15:21:25 +0000 | [diff] [blame] | 43 | if pf and w/pf % 4 <> 0: |
| 44 | sys.stderr.write( \ |
| 45 | 'warning: stride not a multiple of 4 -- may not work on Indigo XS\n') |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 46 | return f, w, h, pf, colorinfo |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 47 | |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 48 | def loadframe(f,w,h,pf,af,spkr, (ybits,ibits,qbits,chrompack),mf): |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 49 | line = f.readline() |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 50 | if line == '': |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 51 | raise EndOfFile |
| 52 | x = eval(line[:-1]) |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 53 | if type(x) == type(0) or type(x) == type(0.0): |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 54 | tijd = x |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 55 | if pf == 0: |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 56 | size = w*h*4 |
| 57 | else: |
| 58 | size = (w/pf) * (h/pf) |
| 59 | else: |
| 60 | tijd, size = x |
| 61 | data = f.read(size) |
| 62 | if len(data) <> size: |
| 63 | raise EndOfFile |
| 64 | if pf: |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 65 | w = w/pf |
| 66 | h = h/pf |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 67 | if chrompack: |
| 68 | cw = (w+chrompack-1)/chrompack |
| 69 | ch = (h+chrompack-1)/chrompack |
| 70 | chromdata = f.read(2*cw*ch) |
| 71 | rectzoom(pf*chrompack*mf,pf*chrompack*mf) |
Guido van Rossum | 864cde2 | 1992-05-07 15:21:25 +0000 | [diff] [blame] | 72 | pixmode(PM_SIZE,16) |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 73 | writemask(0x7ff - ((1<<ybits)-1)) |
| 74 | lrectwrite(0,0,cw-1,ch-1,chromdata) |
| 75 | writemask((1<<ybits)-1) |
Guido van Rossum | 864cde2 | 1992-05-07 15:21:25 +0000 | [diff] [blame] | 76 | pixmode(PM_SIZE,8) |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 77 | if pf: |
| 78 | rectzoom(pf*mf, pf*mf) |
| 79 | elif mf <> 1: |
| 80 | rectzoom(mf,mf) |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 81 | lrectwrite(0,0,w-1,h-1,data) |
| 82 | # This is ugly here, but the only way to get the two |
| 83 | # channels started in sync |
| 84 | #if af <> None: |
| 85 | # playsound(af,spkr) |
| 86 | ct = time.millitimer() - epoch.epoch |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 87 | if epoch.correcttiming and tijd > 0 and ct < tijd: |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 88 | time.millisleep(tijd-ct) |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 89 | #swapbuffers() |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 90 | return tijd |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 91 | |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 92 | def initcmap(ybits,ibits,qbits,chrompack): |
| 93 | if ybits+ibits+qbits > 11: |
| 94 | raise 'Sorry, 11 bits max' |
| 95 | maxy = pow(2,ybits) |
| 96 | maxi = pow(2,ibits) |
| 97 | maxq = pow(2,qbits) |
| 98 | for i in range(2048,4096-256): |
| 99 | mapcolor(i, 0, 255, 0) |
| 100 | for y in range(maxy): |
| 101 | yv = float(y)/float(maxy-1) |
| 102 | for i in range(maxi): |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 103 | if maxi == 1: iv = 0 |
Guido van Rossum | 696f911 | 1991-12-03 17:25:52 +0000 | [diff] [blame] | 104 | else: iv = (float(i)/float(maxi-1))-0.5 |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 105 | for q in range(maxq): |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 106 | if maxq == 1: qv = 0 |
Guido van Rossum | 696f911 | 1991-12-03 17:25:52 +0000 | [diff] [blame] | 107 | else: qv = (float(q)/float(maxq-1))-0.5 |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 108 | index = 2048 + y + (i << ybits) + (q << (ybits+ibits)) |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 109 | rv,gv,bv = colorsys.yiq_to_rgb(yv,iv,qv) |
| 110 | r,g,b = int(rv*255.0), int(gv*255.0), int(bv*255.0) |
| 111 | if index < 4096 - 256: |
| 112 | mapcolor(index, r,g,b) |
| 113 | |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 114 | def playsound(af, spkr): |
| 115 | nsamp = spkr.getfillable() |
| 116 | data = af.read(nsamp*2) |
| 117 | spkr.writesamps(data) |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 118 | |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 119 | def main(): |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 120 | looping = 0 |
| 121 | packfactor = 0 |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 122 | magfactor = 1 |
| 123 | try: |
| 124 | opts, args = getopt.getopt(sys.argv[1:], 'm:p:lF') |
| 125 | except getopt.error: |
| 126 | sys.stderr.write('usage: video ' + \ |
| 127 | '[-l] [-p pf] [-m mag] [-F] [moviefile [soundfile [skipbytes]]]\n') |
| 128 | sys.exit(2) |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 129 | for opt, arg in opts: |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 130 | if opt == '-m': |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 131 | magfactor = int(eval(arg)) |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 132 | elif opt == '-p': |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 133 | packfactor = int(eval(arg)) |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 134 | elif opt == '-l': |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 135 | looping = 1 |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 136 | elif opt == '-F': |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 137 | epoch.correcttiming = 0 |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 138 | if args: |
| 139 | filename = args[0] |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 140 | else: |
| 141 | filename = 'film.video' |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 142 | f, w, h, pf, cinfo = openvideo(filename) |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 143 | if 0 < packfactor <> pf: |
| 144 | w = w/pf*packfactor |
| 145 | h = h/pf*packfactor |
| 146 | pf = packfactor |
| 147 | if args[1:]: |
| 148 | audiofilename = args[1] |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 149 | af = open(audiofilename, 'r') |
| 150 | spkr = openspkr() |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 151 | afskip = 0 |
Guido van Rossum | b3c493c | 1991-11-04 18:14:23 +0000 | [diff] [blame] | 152 | if args[2:]: |
| 153 | afskip = eval(args[2]) |
| 154 | af.seek(afskip) |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 155 | else: |
| 156 | af, spkr = None, None |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 157 | foreground() |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 158 | prefsize(w*magfactor,h*magfactor) |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 159 | win = winopen(filename) |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 160 | if pf: |
| 161 | #doublebuffer() |
| 162 | cmode() |
| 163 | else: |
| 164 | RGBmode() |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 165 | #doublebuffer() |
| 166 | gconfig() |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 167 | if pf: |
| 168 | initcmap(cinfo) |
| 169 | color(2048) |
| 170 | clear() |
| 171 | writemask(2047) |
Guido van Rossum | 864cde2 | 1992-05-07 15:21:25 +0000 | [diff] [blame] | 172 | pixmode(PM_SIZE,8) # 8 bit pixels |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 173 | qdevice(ESCKEY) |
| 174 | qdevice(WINSHUT) |
| 175 | qdevice(WINQUIT) |
| 176 | running = 1 |
| 177 | epoch.epoch = time.millitimer() |
| 178 | nframe = 0 |
| 179 | tijd = 1 |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 180 | if looping: |
| 181 | looping = f.tell() |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 182 | try: |
| 183 | while 1: |
| 184 | if running: |
| 185 | try: |
Guido van Rossum | a63f197 | 1991-11-22 14:04:01 +0000 | [diff] [blame] | 186 | tijd = loadframe(f, w, h, pf, af, spkr, cinfo,magfactor) |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 187 | nframe = nframe + 1 |
| 188 | except EndOfFile: |
| 189 | running = 0 |
| 190 | t = time.millitimer() |
| 191 | if tijd > 0: |
| 192 | print 'Recorded at', |
| 193 | print 0.1 * int(nframe * 10000.0 / tijd), |
| 194 | print 'frames/sec' |
| 195 | print 'Played', nframe, 'frames at', |
| 196 | print 0.1 * int(nframe * 10000.0 / (t-epoch.epoch)), |
| 197 | print 'frames/sec' |
Guido van Rossum | 5dafd91 | 1991-11-04 18:04:21 +0000 | [diff] [blame] | 198 | if looping: |
| 199 | f.seek(looping) |
| 200 | epoch.epoch = time.millitimer() |
| 201 | nframe = 0 |
| 202 | running = 1 |
| 203 | if af <> None: |
| 204 | af.seek(afskip) |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 205 | if af <> None: |
| 206 | playsound(af,spkr) |
| 207 | if not running or qtest(): |
| 208 | dev, val = qread() |
| 209 | if dev in (ESCKEY, WINSHUT, WINQUIT): |
| 210 | raise bye |
Guido van Rossum | f47d048 | 1992-02-11 14:50:22 +0000 | [diff] [blame] | 211 | elif dev == REDRAW: |
Guido van Rossum | b51afcc | 1991-11-04 14:29:27 +0000 | [diff] [blame] | 212 | reshapeviewport() |
| 213 | except bye: |
| 214 | pass |
Guido van Rossum | e4bddea | 1991-10-30 11:52:48 +0000 | [diff] [blame] | 215 | |
| 216 | main() |