2015-01-02 36 views
5

Cerco di imparare come usare le pixmap di memoria condivisa nella libreria xcb. Qualcuno di voi ha esperienza con questo e desidera condividere codici di esempio e/o informazioni? Questo sarebbe molto utile.Come usare shm pixmap con xcb?

Grazie

risposta

6

Dopo alcune ricerche ho scoperto come utilizzare pixmaps di memoria condivisa in xcb.

Ecco il mio testcode:

#include <stdlib.h> 
#include <stdio.h> 

#include <sys/ipc.h> 
#include <sys/shm.h> 

#include <xcb/xcb.h> 
#include <xcb/shm.h> 
#include <xcb/xcb_image.h> 


#define WID 512 
#define HEI 512 



int main(){ 
    xcb_connection_t*  connection; 
    xcb_window_t   window; 
    xcb_screen_t*   screen; 
    xcb_gcontext_t   gcontext; 
    xcb_generic_event_t* event; 

    uint32_t value_mask; 
    uint32_t value_list[2]; 

    //connect to the X server and get screen 

    connection = xcb_connect(NULL, NULL); 
    screen = xcb_setup_roots_iterator(xcb_get_setup(connection)).data; 

    //create a window 

    value_mask = XCB_CW_BACK_PIXEL | XCB_CW_EVENT_MASK; 
    value_list[0] = screen->black_pixel; 
    value_list[1] = XCB_EVENT_MASK_EXPOSURE; 

    window = xcb_generate_id(connection); 

    xcb_create_window(
     connection,    
     screen->root_depth, 
     window,  
     screen->root,  
     0, 0,   
     WID, HEI,  
     0,    
     XCB_WINDOW_CLASS_INPUT_OUTPUT, 
     screen->root_visual,  
     value_mask, value_list 
    ); 

    //create a graphic context 

    value_mask = XCB_GC_FOREGROUND | XCB_GC_GRAPHICS_EXPOSURES; 
    value_list[0] = screen->black_pixel; 
    value_list[1] = 0; 

    gcontext = xcb_generate_id(connection); 
    xcb_create_gc(connection, gcontext, window, value_mask, value_list); 

    //map the window onto the screen 

    xcb_map_window(connection, window); 
    xcb_flush(connection); 


    //Shm test 
    xcb_shm_query_version_reply_t* reply; 
    xcb_shm_segment_info_t   info; 

    reply = xcb_shm_query_version_reply(
     connection, 
     xcb_shm_query_version(connection), 
     NULL 
    ); 

    if(!reply || !reply->shared_pixmaps){ 
     printf("Shm error...\n"); 
     exit(0); 
    } 

    info.shmid = shmget(IPC_PRIVATE, WID*HEI*4, IPC_CREAT | 0777); 
    info.shmaddr = shmat(info.shmid, 0, 0); 

    info.shmseg = xcb_generate_id(connection); 
    xcb_shm_attach(connection, info.shmseg, info.shmid, 0); 
    shmctl(info.shmid, IPC_RMID, 0); 

    uint32_t* data = info.shmaddr; 

    xcb_pixmap_t pix = xcb_generate_id(connection); 
    xcb_shm_create_pixmap(
     connection, 
     pix, 
     window, 
     WID, HEI, 
     screen->root_depth, 
     info.shmseg, 
     0 
    ); 

    int i = 0; 
    while(1){ 
     usleep(10000); 

     data[i] = 0xFFFFFF; 
     i++; 

     xcb_copy_area(
      connection, 
      pix, 
      window, 
      gcontext, 
      0, 0, 0, 0, 
      WID, HEI 
     ); 

     xcb_flush(connection); 
    } 

    xcb_shm_detach(connection, info.shmseg); 
    shmdt(info.shmaddr); 

    xcb_free_pixmap(connection, pix); 

    xcb_destroy_window(connection, window); 
    xcb_disconnect(connection); 

    return 0; 
} 
+1

'shmctl (..., IPC_RMID, ...)' dovrebbe essere chiamato dopo che il server X si attacca, su alcuni sistemi non Linux non è possibile allegare una segmento di memoria condivisa dopo averlo contrassegnato per la cancellazione. 'xcb_shm_attach' aggiunge solo la richiesta alla coda dei messaggi in uscita xcb, quindi il server X probabilmente non si è ancora connesso quando' xcb_shm_attach' restituisce. Suggerirei di spostare la chiamata 'shmctl' al codice cleanup. Vedere http://man7.org/linux/man-pages/man2/shmctl.2.html#NOTES – programmerjake

+0

fare attenzione con le autorizzazioni di accesso sul segmento di memoria condivisa; è meglio mantenerlo limitato solo all'utente corrente: 'shmget (IPC_PRIVATE, WID * HEI * 4, IPC_CREAT | 0600);' –