nuttx-apps/examples/camera/camera_bkgd.c

340 lines
9.4 KiB
C
Raw Normal View History

/****************************************************************************
* 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;
NXHANDLE 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.
****************************************************************************/
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.
*/
(void)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)
{
(void) sem_wait(&g_nximage.sem);
}
printf("nximage_initialize: Screen resolution (%d,%d)\n",
g_nximage.xres, g_nximage.yres);
return 0;
}
/****************************************************************************
* Name: nximage_image
*
* 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((NXWINDOW)g_nximage.hbkgd, &dest, src, &origin,
g_nximage.xres * sizeof(nxgl_mxpixel_t));
if (ret < 0)
{
printf("nximage_image: nx_bitmapwindow failed: %d\n", errno);
}
}
/****************************************************************************
* Name: nximage_finalize()
*
* Description:
* Finalize NX server.
****************************************************************************/
void nximage_finalize(void)
{
nx_disconnect(g_nximage.hnx);
}