| /**************************************************************************** |
| * apps/examples/camera/camera_bkgd.c |
| * |
| * Licensed to the Apache Software Foundation (ASF) under one or more |
| * contributor license agreements. See the NOTICE file distributed with |
| * this work for additional information regarding copyright ownership. The |
| * ASF licenses this file to you under the Apache License, Version 2.0 (the |
| * "License"); you may not use this file except in compliance with the |
| * License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
| * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
| * License for the specific language governing permissions and limitations |
| * under the License. |
| * |
| ****************************************************************************/ |
| |
| /**************************************************************************** |
| * Included Files |
| ****************************************************************************/ |
| |
| #include <nuttx/config.h> |
| |
| #include <stdint.h> |
| #include <stdbool.h> |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <semaphore.h> |
| #include <errno.h> |
| #include <debug.h> |
| |
| #include <arch/board/board.h> |
| #include <nuttx/nx/nx.h> |
| #include <nuttx/nx/nxglib.h> |
| |
| #include "camera_bkgd.h" |
| |
| /**************************************************************************** |
| * Private Types |
| ****************************************************************************/ |
| |
| struct nximage_data_s |
| { |
| /* The NX handles */ |
| |
| NXHANDLE hnx; |
| NXWINDOW hbkgd; |
| bool connected; |
| |
| /* The screen resolution */ |
| |
| nxgl_coord_t xres; |
| nxgl_coord_t yres; |
| |
| volatile bool havepos; |
| sem_t sem; |
| }; |
| |
| /**************************************************************************** |
| * Private Function Prototypes |
| ****************************************************************************/ |
| |
| static void nximage_redraw(NXWINDOW hwnd, |
| FAR const struct nxgl_rect_s *rect, |
| bool more, |
| FAR void *arg); |
| static void nximage_position(NXWINDOW hwnd, |
| FAR const struct nxgl_size_s *size, |
| FAR const struct nxgl_point_s *pos, |
| FAR const struct nxgl_rect_s *bounds, |
| FAR void *arg); |
| |
| /**************************************************************************** |
| * Private Data |
| ****************************************************************************/ |
| |
| /* Background window call table */ |
| |
| static const struct nx_callback_s g_nximagecb = |
| { |
| nximage_redraw, /* redraw */ |
| nximage_position /* position */ |
| #ifdef CONFIG_NX_XYINPUT |
| , NULL /* mousein */ |
| #endif |
| #ifdef CONFIG_NX_KBD |
| , NULL /* my kbdin */ |
| #endif |
| }; |
| |
| /* To handle nx context, below variable is defined for this application. */ |
| |
| static struct nximage_data_s g_nximage = |
| { |
| NULL, /* hnx */ |
| NULL, /* hbkgd */ |
| false, /* connected */ |
| 0, /* xres */ |
| 0, /* yres */ |
| false, /* havpos */ |
| { 0 }, /* sem */ |
| }; |
| |
| /**************************************************************************** |
| * Private Functions |
| ****************************************************************************/ |
| |
| /**************************************************************************** |
| * Name: nximage_listener() |
| * |
| * Description: |
| * NX event listener for an event from NX server. |
| ****************************************************************************/ |
| |
| static FAR void *nximage_listener(FAR void *arg) |
| { |
| int ret; |
| |
| /* Process events forever */ |
| |
| for (; ; ) |
| { |
| /* Handle the next event. If we were configured blocking, then |
| * we will stay right here until the next event is received. Since |
| * we have dedicated a while thread to servicing events, it would |
| * be most natural to also select CONFIG_NX_BLOCKING -- if not, the |
| * following would be a tight infinite loop (unless we added addition |
| * logic with nx_eventnotify and sigwait to pace it). |
| */ |
| |
| ret = nx_eventhandler(g_nximage.hnx); |
| if (ret < 0) |
| { |
| /* An error occurred... assume that we have lost connection with |
| * the server. |
| */ |
| |
| printf("nximage_listener: Lost server connection: %d\n", errno); |
| exit(EXIT_FAILURE); |
| } |
| |
| /* If we received a message, we must be connected */ |
| |
| if (!g_nximage.connected) |
| { |
| g_nximage.connected = true; |
| sem_post(&g_nximage.sem); |
| printf("nximage_listener: Connected\n"); |
| } |
| } |
| } |
| |
| /**************************************************************************** |
| * Name: nximage_redraw |
| * |
| * Description: |
| * NX re-draw handler |
| * |
| ****************************************************************************/ |
| |
| static void nximage_redraw(NXWINDOW hwnd, FAR const struct nxgl_rect_s *rect, |
| bool more, FAR void *arg) |
| { |
| ginfo("hwnd=%p rect={(%d,%d),(%d,%d)} more=%s\n", |
| hwnd, rect->pt1.x, rect->pt1.y, rect->pt2.x, rect->pt2.y, |
| more ? "true" : "false"); |
| } |
| |
| /**************************************************************************** |
| * Name: nximage_position |
| * |
| * Description: |
| * NX position change handler |
| * |
| ****************************************************************************/ |
| |
| static void nximage_position(NXWINDOW hwnd, |
| FAR const struct nxgl_size_s *size, |
| FAR const struct nxgl_point_s *pos, |
| FAR const struct nxgl_rect_s *bounds, |
| FAR void *arg) |
| { |
| /* Report the position */ |
| |
| ginfo("hwnd=%p size=(%d,%d) pos=(%d,%d) bounds={(%d,%d),(%d,%d)}\n", |
| hwnd, size->w, size->h, pos->x, pos->y, |
| bounds->pt1.x, bounds->pt1.y, bounds->pt2.x, bounds->pt2.y); |
| |
| /* Have we picked off the window bounds yet? */ |
| |
| if (!g_nximage.havepos) |
| { |
| /* Save the background window handle */ |
| |
| g_nximage.hbkgd = hwnd; |
| |
| /* Save the window limits */ |
| |
| g_nximage.xres = bounds->pt2.x + 1; |
| g_nximage.yres = bounds->pt2.y + 1; |
| |
| g_nximage.havepos = true; |
| sem_post(&g_nximage.sem); |
| ginfo("Have xres=%d yres=%d\n", g_nximage.xres, g_nximage.yres); |
| } |
| } |
| |
| /**************************************************************************** |
| * Public Functions |
| ****************************************************************************/ |
| |
| /**************************************************************************** |
| * Name: nximage_initialize |
| * |
| * Description: |
| * Initialize NX graphics subsystem. |
| * |
| ****************************************************************************/ |
| |
| int nximage_initialize(void) |
| { |
| nxgl_mxpixel_t color; |
| pthread_t thread; |
| int ret; |
| |
| /* Start the NX server kernel thread */ |
| |
| ret = boardctl(BOARDIOC_NX_START, 0); |
| if (ret < 0) |
| { |
| printf("nximage_initialize: Failed to start the NX server: %d\n", |
| errno); |
| return ERROR; |
| } |
| |
| /* Connect to the server */ |
| |
| g_nximage.hnx = nx_connect(); |
| if (!g_nximage.hnx) |
| { |
| printf("nximage_initialize: nx_connect failed: %d\n", errno); |
| return ERROR; |
| } |
| |
| /* Start a separate thread to listen for server events. |
| * For simplicity, use defaul thread attribute. |
| */ |
| |
| ret = pthread_create(&thread, NULL, nximage_listener, NULL); |
| if (ret != 0) |
| { |
| printf("nximage_initialize: pthread_create failed: %d\n", ret); |
| return ERROR; |
| } |
| |
| /* Don't return until we are connected to the server */ |
| |
| while (!g_nximage.connected) |
| { |
| /* Wait for the listener thread to wake us up when we really |
| * are connected. |
| */ |
| |
| sem_wait(&g_nximage.sem); |
| } |
| |
| /* Set background color to black */ |
| |
| color = 0; |
| nx_setbgcolor(g_nximage.hnx, &color); |
| ret = nx_requestbkgd(g_nximage.hnx, &g_nximagecb, NULL); |
| if (ret < 0) |
| { |
| printf("nximage_initialize: nx_requestbkgd failed: %d\n", errno); |
| nx_disconnect(g_nximage.hnx); |
| return ERROR; |
| } |
| |
| while (!g_nximage.havepos) |
| { |
| sem_wait(&g_nximage.sem); |
| } |
| |
| printf("nximage_initialize: Screen resolution (%d,%d)\n", |
| g_nximage.xres, g_nximage.yres); |
| |
| return 0; |
| } |
| |
| /**************************************************************************** |
| * Name: nximage_draw |
| * |
| * Description: |
| * Put the NuttX logo in the center of the display. |
| * |
| ****************************************************************************/ |
| |
| void nximage_draw(FAR void *image, int w, int h) |
| { |
| FAR struct nxgl_point_s origin; |
| FAR struct nxgl_rect_s dest; |
| FAR const void *src[CONFIG_NX_NPLANES]; |
| int ret; |
| |
| origin.x = 0; |
| origin.y = 0; |
| |
| /* Set up the destination to whole LCD screen */ |
| |
| dest.pt1.x = 0; |
| dest.pt1.y = 0; |
| dest.pt2.x = g_nximage.xres - 1; |
| dest.pt2.y = g_nximage.yres - 1; |
| |
| src[0] = image; |
| |
| ret = nx_bitmap(g_nximage.hbkgd, &dest, src, &origin, |
| g_nximage.xres * sizeof(nxgl_mxpixel_t)); |
| if (ret < 0) |
| { |
| printf("nximage_image: nx_bitmap failed: %d\n", errno); |
| } |
| } |
| |
| /**************************************************************************** |
| * Name: nximage_finalize() |
| * |
| * Description: |
| * Finalize NX server. |
| ****************************************************************************/ |
| |
| void nximage_finalize(void) |
| { |
| nx_disconnect(g_nximage.hnx); |
| } |