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