X-Git-Url: http://www.chiark.greenend.org.uk/ucgi/~yarrgweb/git?p=ypp-sc-tools.main.git;a=blobdiff_plain;f=pctb%2Fpages.c;h=482c959c24261cf541ca0ea4e99b63ddf7e51e4e;hp=50a652c5946a02ffd1a82fc60780e19620217dbf;hb=ad71c2a5bfcc29289b2f94324544c3d1a186751b;hpb=a16126118a4da57568f926c7082dcc28aac9b8f3 diff --git a/pctb/pages.c b/pctb/pages.c index 50a652c..482c959 100644 --- a/pctb/pages.c +++ b/pctb/pages.c @@ -45,7 +45,7 @@ CanonImage *page_images[MAX_PAGES]; int npages; RgbImage *page0_rgbimage; -char *ocean, *pirate; +const char *ocean, *pirate; static XWindowAttributes attr; static Window id; @@ -243,6 +243,7 @@ static void snapshot(Snapshot **output) { double begin= timestamp(); if (shmim) { + rtimestamp(&begin, "XShmGetImage before"); xassert( XShmGetImage(disp,id,shmim, 0,0, AllPlanes) ); rtimestamp(&begin, "XShmGetImage"); @@ -253,6 +254,7 @@ static void snapshot(Snapshot **output) { memcpy(im_free->data, shmim->data, dsz); rtimestamp(&begin, "mmalloc/memcpy"); } else { + rtimestamp(&begin, "XGetImage before"); xassert( im_use= im_free= XGetImage(disp,id, 0,0, wwidth,wheight, AllPlanes, ZPixmap) ); rtimestamp(&begin, "XGetImage"); @@ -299,7 +301,7 @@ static int identical(const Snapshot *a, const Snapshot *b) { return 0; int compare_to= a->h; - if (max_relevant_y && compare_to > max_relevant_y) + if (max_relevant_y>=0 && compare_to > max_relevant_y) compare_to= max_relevant_y; return !memcmp(a->data, b->data, a->w * 3 * compare_to); @@ -319,46 +321,57 @@ static void wait_for_stability(Snapshot **output, va_start(al,fmt); Snapshot *last=0; + int nidentical=0; /* waits longer if we're going to return an image identical to previously * if previously==0, all images are considered identical to it */ - debugf("PAGING wait_for_stability" - " last_input=%f previously=%p\n", - last_input, previously); - char *doing; - sysassert( vasprintf(&doing,fmt,al) >=0); + sysassert( vasprintf(&doing,fmt,al) >=0 ); - progress("%s",doing); + debugf("PAGING wait_for_stability" + " last_input=%f previously=%p `%s'\n", + last_input, previously, doing); + double min_interval= 25000; /*us*/ for (;;) { - double at_snapshot= timestamp(); - double need_sleep= min_update_allowance - (at_snapshot - last_input); - if (need_sleep > 0) { delay(need_sleep); continue; } + progress_spinner("%s",doing); + + double since_last_input= timestamp() - last_input; + double this_interval= min_interval - since_last_input; - snapshot(output); + if (this_interval >= 0) + usleep(this_interval); - if (!with_keypress && - !(previously && identical(*output,previously))) { - debugf("PAGING wait_for_stability simple\n"); - break; - } + snapshot(output); - if (last && identical(*output,last)) { - debugf("PAGING wait_for_stability stabilised\n"); - break; + if (!last) { + debugf("PAGING wait_for_stability first...\n"); + last=*output; *output=0; + } else if (!identical(*output,last)) { + debugf("PAGING wait_for_stability changed...\n"); + free(last); last=*output; *output=0; + nidentical=0; + if (!with_keypress) { + min_interval *= 3.0; + min_interval += 0.5; + } + } else { + nidentical++; + int threshold= + !previously ? 3 : + identical(*output,previously) ? 5 + : 1; + debugf("PAGING wait_for_stability nidentical=%d threshold=%d\n", + nidentical, threshold); + if (nidentical >= threshold) + break; + + min_interval += 0.5; + min_interval *= 2.0; } - - progress_spinner("%s",doing); - - debugf("PAGING wait_for_stability retry\n"); - - free_snapshot(&last); last=*output; *output=0; if (with_keypress) with_keypress(); - - delay(0.5); } free_snapshot(&last); @@ -402,8 +415,10 @@ static void raise_and_get_details(void) { check_client_window_all_on_screen(); - int shm= XShmQueryExtension(disp); - shm=0; + Bool shmpixmaps=0; + int major=0,minor=0; + int shm= XShmQueryVersion(disp, &major,&minor,&shmpixmaps); + debugf("PAGING shm=%d %d.%d pixmaps=%d\n",shm,major,minor,shmpixmaps); if (shm) { xassert( shmim= XShmCreateImage(disp, attr.visual, attr.depth, ZPixmap, 0,&shminfo, wwidth,wheight) ); @@ -482,24 +497,18 @@ static void set_focus_commodity(void) { static CanonImage *convert_page(Snapshot *sn, RgbImage *ri) { CanonImage *im; - fprintf(screenshot_file, - "P6\n" - "%d %d\n" - "255\n", sn->w, sn->h); + fwrite_ppmraw(screenshot_file, sn); + if (ri) memcpy(ri->data, sn->data, ri->h * ri->w * 3); + unsigned char *pixel= sn->data; CANONICALISE_IMAGE(im, sn->w, sn->h, { - int i; - rgb= 0; - for (i=0; i<3; i++) { - rgb <<= 8; - unsigned long sample= RI_PIXEL(sn,x,y)[i]; - rgb |= sample; - fputc(sample, screenshot_file); - } - if (ri) - CANONIMG_ALSO_STORERGB(ri); + rgb= + (pixel[0] << 16) | + (pixel[1] << 8) | + (pixel[2] ); + pixel += 3; }); - + sysassert(!ferror(screenshot_file)); sysassert(!fflush(screenshot_file)); @@ -514,28 +523,18 @@ static void prepare_ypp_client(void) { raise_and_get_details(); wait_for_stability(¤t,0,0, "checking current YPP client screen..."); -#if 1 -timestamp(); -snapshot(¤t); -snapshot(¤t); -snapshot(¤t); -snapshot(¤t); -snapshot(¤t); -snapshot(¤t); -snapshot(¤t); -snapshot(¤t); -snapshot(¤t); -snapshot(¤t); -timestamp(); -#endif - test= convert_page(current, 0); find_structure(test, &max_relevant_y); check_correct_commodities(); + Rect sunshine= find_sunshine_widget(); + + progress("poking client..."); + mouse_1_updown((sunshine.tl.x + sunshine.br.x) / 2, + (sunshine.tl.y*9 + sunshine.br.y) / 10); + free(test); - free_snapshot(¤t); - progress("requesting status information..."); + wait_for_stability(¤t,0,0, "checking basic YPP client screen..."); mouse_1_updown(250, wheight-10); mouse_1_updown_here(); mouse_1_updown_here(); @@ -545,10 +544,16 @@ timestamp(); send_key(XK_w); send_key(XK_Return); sync_after_input(); + + Snapshot *status=0; + wait_for_stability(&status,current,0, "awaiting status information..."); + free_snapshot(¤t); + free_snapshot(&status); } void take_screenshots(void) { Snapshot *current=0, *last=0; + RgbImage *page0_store; prepare_ypp_client(); @@ -557,6 +562,8 @@ void take_screenshots(void) { wait_for_stability(¤t,0, send_pgup_many, "paging up to top of commodity list..."); + page0_rgbimage= page0_store= alloc_rgb_image(current->w, current->h); + /* now to actually page down */ for (;;) { debugf("paging page %d\n",npages); @@ -565,8 +572,8 @@ void take_screenshots(void) { fatal("Paging down seems to generate too many pages - max is %d.", MAX_PAGES); - page_images[npages]= convert_page(current, 0); - free_snapshot(&last); last=current; current=0; + page_images[npages]= convert_page(current, page0_store); + free_snapshot(&last); last=current; current=0; page0_store=0; debugf("PAGING page %d converted\n",npages); @@ -588,6 +595,7 @@ void take_screenshots(void) { debugf("PAGING all done.\n"); progress_log("collected %d screenshots.",npages); + assert(!page0_store); } void take_one_screenshot(void) { @@ -596,7 +604,7 @@ void take_one_screenshot(void) { prepare_ypp_client(); wait_for_stability(¤t,0,0, "taking screenshot..."); page0_rgbimage= alloc_rgb_image(current->w, current->h); - page_images[0]= convert_page(current, 0); + page_images[0]= convert_page(current, page0_rgbimage); npages= 1; progress_log("collected single screenshot."); } @@ -607,114 +615,122 @@ void set_yppclient_window(unsigned long wul) { DEBUG_DEFINE_SOME_DEBUGF(findypp,debugfind) +static int nfound; +static Atom wm_name; +static int screen; + +static void findypp_recurse(int depth, int targetdepth, Window w) { + unsigned int nchildren; + int i; + Window *children=0; + Window gotroot, gotparent; + + static const char prefix[]= "Puzzle Pirates - "; + static const char onthe[]= " on the "; + static const char suffix[]= " ocean"; +#define S(x) (sizeof((x))-1) + + debugfind("FINDYPP %d/%d screen %d %*s %lx", + depth,targetdepth,screen, + depth,"",(unsigned long)w); + + if (depth!=targetdepth) { + xassert( XQueryTree(disp,w, + &gotroot,&gotparent, + &children,&nchildren) ); + debugfind(" nchildren=%d\n",nchildren); + + for (i=0; i=' ' && c<=126) fputc(c,debug); + else fprintf(debug,"\\x%02x",c & 0xff); + } + fputs("\": ",debug); + } + +#define REQUIRE(pred) \ + if (!(pred)) { debugfind(" failed test %s\n", #pred); return; } \ + else + + REQUIRE( gottype!=None ); + REQUIRE( len ); + REQUIRE( gotfmt==8 ); + + REQUIRE( len >= S(prefix) + 1 + S(onthe) + 1 + S(suffix) ); + + char *spc1= strchr( title + S(prefix), ' '); REQUIRE(spc1); + char *spc2= strrchr((title + len) - S(suffix), ' '); REQUIRE(spc2); + + REQUIRE( (title + len) - spc1 >= S(onthe) + S(suffix) ); + REQUIRE( spc2 - title >= S(prefix) + S(onthe) ); + + REQUIRE( !memcmp(title, prefix, S(prefix)) ); + REQUIRE( !memcmp(title + len - S(suffix), suffix, S(suffix)) ); + REQUIRE( !memcmp(spc1, onthe, S(onthe)) ); + +#define ASSIGN(what, start, end) \ + what= masprintf("%.*s", (end)-(start), start); \ + if (o_##what) REQUIRE( !strcasecmp(o_##what, what) ); \ + else + + ASSIGN(ocean, spc1 + S(onthe), (title + len) - S(suffix)); + ASSIGN(pirate, title + S(prefix), spc1); + + debugfind(" YES!\n"); + id= w; + nfound++; + progress_log("found YPP client (0x%lx):" + " %s ocean - %s.", + (unsigned long)id, ocean, pirate); +} + void find_yppclient_window(void) { - Window root, gotroot, gotparent; - int screen; - int nfound=0; + int targetdepth; + + nfound=0; if (id) return; progress("looking for YPP client window..."); - static const char prefix[]= "Puzzle Pirates - "; - static const char onthe[]= " on the "; - static const char suffix[]= " ocean"; -#define S(x) (sizeof((x))-1) + xassert( (wm_name= XInternAtom(disp,"WM_NAME",True)) != None); - Atom wm_name= XInternAtom(disp,"WM_NAME",True); - xassert(wm_name != None); - - for (screen=0; screen=' ' && c<=126) fputc(c,debug); - else fprintf(debug,"\\x%02x",c & 0xff); - } - fputs("\": ",debug); - } - -#define REQUIRE(pred) \ - if (!(pred)) { debugfind(" failed test %s\n", #pred); continue; } \ - else - - REQUIRE( gottype!=None ); - REQUIRE( len ); - REQUIRE( gotfmt==8 ); - - REQUIRE( len >= S(prefix) + 1 + S(onthe) + 1 + S(suffix) ); - - char *spc1= strchr( title + S(prefix), ' '); REQUIRE(spc1); - char *spc2= strrchr((title + len) - S(suffix), ' '); REQUIRE(spc2); - - REQUIRE( (title + len) - spc1 >= S(onthe) + S(suffix) ); - REQUIRE( spc2 - title >= S(prefix) + S(onthe) ); - - REQUIRE( !memcmp(title, prefix, S(prefix)) ); - REQUIRE( !memcmp(title + len - S(suffix), suffix, S(suffix)) ); - REQUIRE( !memcmp(spc1, onthe, S(onthe)) ); - -#define ASSIGN(what, start, end) do { \ - sysassert( asprintf(&what, "%.*s", (end)-(start), start) >0 ); \ - }while(0) - ASSIGN(pirate, title + S(prefix), spc1); - ASSIGN(ocean, spc1 + S(onthe), (title + len) - S(suffix)); - - debugfind(" YES!\n"); - id= w2; - nfound++; - progress_log("found YPP client (0x%lx):" - " %s ocean - %s.", - (unsigned long)id, ocean, pirate); - } - if (children2) XFree(children2); + for (targetdepth=1; targetdepth<4; targetdepth++) { + for (screen=0; screen1) - fatal("Found several YPP clients." - " Close one, or specify the windowid with --window-id.\n"); + fatal("Found several possible YPP clients. Close one,\n" + " disambiguate with --pirate or --ocean," + " or specify --window-id.\n"); if (nfound<1) - fatal("Did not find YPP client." - " Use --window-id and/or report this as a fault.\n"); + fatal("Did not find %sYPP client." + " Use --window-id and/or report this as a fault.\n", + o_ocean || o_pirate ? "matching ": ""); }