summaryrefslogtreecommitdiffstats
path: root/firmware/target
diff options
context:
space:
mode:
authorUdo Schläpfer <rockbox-2014.10@desktopwarrior.net>2015-02-02 21:44:29 +0100
committerUdo Schläpfer <rockbox-2014.10@desktopwarrior.net>2015-02-02 21:57:55 +0100
commitdbabd0d9c34a33bc0c51243ec37f230d117db955 (patch)
tree46de348929ce739702a230a2587fdb5539585753 /firmware/target
parentcef17e3d59ad93f766e8ee23b1610540a33dfe5e (diff)
downloadrockbox-dbabd0d9c34a33bc0c51243ec37f230d117db955.tar.gz
rockbox-dbabd0d9c34a33bc0c51243ec37f230d117db955.zip
iBasso DX50/DX90: Major code cleanup and reorganization.
Reorganization - Separated iBasso devices from PLATFORM_ANDROID. These are now standlone hosted targets. Most device specific code is in the firmware/target/hosted/ibasso directory. - No dependency on Android SDK, only the Android NDK is needed. 32 bit Android NDK and Android API Level 16. - Separate implementation for each device where feasible. Code cleanup - Rewrite of existing code, from simple reformat to complete reimplementation. - New backlight interface, seperating backlight from touchscreen. - Rewrite of device button handler, removing unneeded code and fixing memory leaks. - New Debug messages interface logging to Android adb logcat (DEBUGF, panicf, logf). - Rewrite of lcd device handler, removing unneeded code and fixing memory leaks. - Rewrite of audiohw device handler/pcm interface, removing unneeded code and fixing memory leaks, enabling 44.1/48kHz pthreaded playback. - Rewrite of power and powermng, proper shutdown, using batterylog results (see http://gerrit.rockbox.org/r/#/c/1047/). - Rewrite of configure (Android NDK) and device specific config. - Rewrite of the Android NDK specific Makefile. Misc - All plugins/games/demos activated. - Update tinyalsa to latest from https://github.com/tinyalsa/tinyalsa. Includes - http://gerrit.rockbox.org/r/#/c/993/ - http://gerrit.rockbox.org/r/#/c/1010/ - http://gerrit.rockbox.org/r/#/c/1035/ Does not include http://gerrit.rockbox.org/r/#/c/1007/ due to new backlight interface and new option for hold switch, touchscreen, physical button interaction. Rockbox needs the iBasso DX50/DX90 loader for startup, see http://gerrit.rockbox.org/r/#/c/1099/ The loader expects Rockbox to be installed in /mnt/sdcard/.rockbox/. If /mnt/sdcard/ is accessed as USB mass storage device, Rockbox will exit gracefully and the loader will restart Rockbox on USB disconnect. Tested on iBasso DX50. Compiled (not tested) for iBasso DX90. Compiled (not tested) for PLATFORM_ANDROID. Change-Id: I5f5e22e68f5b4cf29c28e2b40b2c265f2beb7ab7
Diffstat (limited to 'firmware/target')
-rw-r--r--firmware/target/hosted/android/dx50/adc-target.h0
-rw-r--r--firmware/target/hosted/android/dx50/backlight-dx50.c76
-rw-r--r--firmware/target/hosted/android/dx50/button-dx50.c316
-rw-r--r--firmware/target/hosted/android/dx50/lcd-dx50.c120
-rw-r--r--firmware/target/hosted/android/dx50/pcm-dx50.c364
-rw-r--r--firmware/target/hosted/android/dx50/powermgmt-dx50.c97
-rw-r--r--firmware/target/hosted/android/system-android.c54
-rw-r--r--firmware/target/hosted/filesystem-app.c2
-rw-r--r--firmware/target/hosted/ibasso/android_ndk.make49
-rw-r--r--firmware/target/hosted/ibasso/audiohw-ibasso.c81
-rw-r--r--firmware/target/hosted/ibasso/backlight-ibasso.c132
-rw-r--r--firmware/target/hosted/ibasso/backlight-target.h (renamed from firmware/target/hosted/android/dx50/backlight-target.h)20
-rw-r--r--firmware/target/hosted/ibasso/button-ibasso.c420
-rw-r--r--firmware/target/hosted/ibasso/button-ibasso.h61
-rw-r--r--firmware/target/hosted/ibasso/button-target.h (renamed from firmware/target/hosted/android/dx50/button-target.h)47
-rw-r--r--firmware/target/hosted/ibasso/debug-ibasso.c70
-rw-r--r--firmware/target/hosted/ibasso/debug-ibasso.h38
-rw-r--r--firmware/target/hosted/ibasso/dx50/audiohw-dx50.c68
-rw-r--r--firmware/target/hosted/ibasso/dx50/button-dx50.c96
-rw-r--r--firmware/target/hosted/ibasso/dx50/codec-dx50.h51
-rw-r--r--firmware/target/hosted/ibasso/dx90/audiohw-dx90.c63
-rw-r--r--firmware/target/hosted/ibasso/dx90/button-dx90.c104
-rw-r--r--firmware/target/hosted/ibasso/dx90/codec-dx90.h35
-rw-r--r--firmware/target/hosted/ibasso/hostfs-ibasso.c47
-rw-r--r--firmware/target/hosted/ibasso/lcd-ibasso.c195
-rw-r--r--firmware/target/hosted/ibasso/lcd-target.h44
-rw-r--r--firmware/target/hosted/ibasso/pcm-ibasso.c488
-rw-r--r--firmware/target/hosted/ibasso/pcm-ibasso.h33
-rw-r--r--firmware/target/hosted/ibasso/power-ibasso.c97
-rw-r--r--firmware/target/hosted/ibasso/powermgmt-ibasso.c122
-rw-r--r--firmware/target/hosted/ibasso/sysfs-ibasso.c404
-rw-r--r--firmware/target/hosted/ibasso/sysfs-ibasso.h111
-rw-r--r--firmware/target/hosted/ibasso/system-ibasso.c101
-rw-r--r--firmware/target/hosted/ibasso/system-target.h (renamed from firmware/target/hosted/android/dx50/lcd-target.h)24
-rw-r--r--firmware/target/hosted/ibasso/tinyalsa/include/sound/asound.h (renamed from firmware/target/hosted/android/dx50/tinyalsa/asound.h)1
-rw-r--r--firmware/target/hosted/ibasso/tinyalsa/include/tinyalsa/asoundlib.h (renamed from firmware/target/hosted/android/dx50/tinyalsa/asoundlib.h)45
-rw-r--r--firmware/target/hosted/ibasso/tinyalsa/mixer.c (renamed from firmware/target/hosted/android/dx50/tinyalsa/mixer.c)23
-rw-r--r--firmware/target/hosted/ibasso/tinyalsa/pcm.c (renamed from firmware/target/hosted/android/dx50/tinyalsa/pcm.c)106
-rw-r--r--firmware/target/hosted/ibasso/usb-ibasso.c92
-rw-r--r--firmware/target/hosted/ibasso/usb-ibasso.h54
-rw-r--r--firmware/target/hosted/ibasso/vold-ibasso.c203
-rw-r--r--firmware/target/hosted/ibasso/vold-ibasso.h42
42 files changed, 3480 insertions, 1116 deletions
diff --git a/firmware/target/hosted/android/dx50/adc-target.h b/firmware/target/hosted/android/dx50/adc-target.h
deleted file mode 100644
index e69de29bb2..0000000000
--- a/firmware/target/hosted/android/dx50/adc-target.h
+++ /dev/null
diff --git a/firmware/target/hosted/android/dx50/backlight-dx50.c b/firmware/target/hosted/android/dx50/backlight-dx50.c
deleted file mode 100644
index 8eb4c58191..0000000000
--- a/firmware/target/hosted/android/dx50/backlight-dx50.c
+++ /dev/null
@@ -1,76 +0,0 @@
-/***************************************************************************
- * __________ __ ___.
- * Open \______ \ ____ ____ | | _\_ |__ _______ ___
- * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
- * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
- * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
- * \/ \/ \/ \/ \/
- *
- * Copyright (C) 2011 by Lorenzo Miori
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU General Public License
- * as published by the Free Software Foundation; either version 2
- * of the License, or (at your option) any later version.
- *
- * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
- * KIND, either express or implied.
- *
- ****************************************************************************/
-#include "config.h"
-#include "system.h"
-#include "backlight.h"
-#include "backlight-target.h"
-#include "lcd.h"
-#include "lcd-target.h"
-#include <fcntl.h>
-#include <stdio.h>
-#include "unistd.h"
-
-bool backlight_hw_init(void)
-{
- /* We have nothing to do */
- return true;
-}
-
-void backlight_hw_on(void)
-{
- FILE *f = fopen("/sys/power/state", "w");
- fputs("on", f);
- fclose(f);
- lcd_enable(true);
-}
-
-void backlight_hw_off(void)
-{
- FILE * f;
-
- /* deny the player to sleep deep */
- f = fopen("/sys/power/wake_lock", "w");
- fputs("player", f);
- fclose(f);
-
- /* deny the player to mute */
- f = fopen("/sys/class/codec/wm8740_mute", "w");
- fputc(0, f);
- fclose(f);
-
- /* turn off backlight */
- f = fopen("/sys/power/state", "w");
- fputs("mem", f);
- fclose(f);
-
-}
-
-void backlight_hw_brightness(int brightness)
-{
- /* Just another check... */
- if (brightness > MAX_BRIGHTNESS_SETTING)
- brightness = MAX_BRIGHTNESS_SETTING;
- if (brightness < MIN_BRIGHTNESS_SETTING)
- brightness = MIN_BRIGHTNESS_SETTING;
-
- FILE *f = fopen("/sys/devices/platform/rk29_backlight/backlight/rk28_bl/brightness", "w");
- fprintf(f, "%d", brightness);
- fclose(f);
-}
diff --git a/firmware/target/hosted/android/dx50/button-dx50.c b/firmware/target/hosted/android/dx50/button-dx50.c
deleted file mode 100644
index 250b448491..0000000000
--- a/firmware/target/hosted/android/dx50/button-dx50.c
+++ /dev/null
@@ -1,316 +0,0 @@
-/***************************************************************************
- * __________ __ ___.
- * Open \______ \ ____ ____ | | _\_ |__ _______ ___
- * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
- * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
- * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
- * \/ \/ \/ \/ \/
- * $Id$
- *
- * Copyright (c) 2010 Thomas Martitz
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU General Public License
- * as published by the Free Software Foundation; either version 2
- * of the License, or (at your option) any later version.
- *
- * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
- * KIND, either express or implied.
- *
- ****************************************************************************/
-
-
-#include <stdbool.h>
-#include "button.h"
-#include "buttonmap.h"
-#include "config.h"
-#include "kernel.h"
-#include "system.h"
-#include "touchscreen.h"
-#include "powermgmt.h"
-#include "backlight.h"
-
-#include <linux/input.h>
-#include <stdio.h>
-#include <stdlib.h>
-#include <string.h>
-#include <stdint.h>
-#include <dirent.h>
-#include <fcntl.h>
-#include <sys/ioctl.h>
-#include <sys/inotify.h>
-#include <sys/limits.h>
-#include <sys/poll.h>
-#include <errno.h>
-
-
-static struct pollfd *ufds;
-static char **device_names;
-static int nfds;
-
-enum {
- PRINT_DEVICE_ERRORS = 1U << 0,
- PRINT_DEVICE = 1U << 1,
- PRINT_DEVICE_NAME = 1U << 2,
- PRINT_DEVICE_INFO = 1U << 3,
- PRINT_VERSION = 1U << 4,
- PRINT_POSSIBLE_EVENTS = 1U << 5,
- PRINT_INPUT_PROPS = 1U << 6,
- PRINT_HID_DESCRIPTOR = 1U << 7,
-
- PRINT_ALL_INFO = (1U << 8) - 1,
-
- PRINT_LABELS = 1U << 16,
-};
-
-static int last_y, last_x;
-static int last_btns;
-
-static enum {
- STATE_UNKNOWN,
- STATE_UP,
- STATE_DOWN
-} last_touch_state = STATE_UNKNOWN;
-
-
-static int open_device(const char *device, int print_flags)
-{
- int fd;
- struct pollfd *new_ufds;
- char **new_device_names;
-
- fd = open(device, O_RDWR);
- if(fd < 0) {
- if(print_flags & PRINT_DEVICE_ERRORS)
- fprintf(stderr, "could not open %s, %s\n", device, strerror(errno));
- return -1;
- }
-
- new_ufds = realloc(ufds, sizeof(ufds[0]) * (nfds + 1));
- if(new_ufds == NULL) {
- fprintf(stderr, "out of memory\n");
- close(fd);
- return -1;
- }
- ufds = new_ufds;
- new_device_names = realloc(device_names, sizeof(device_names[0]) * (nfds + 1));
- if(new_device_names == NULL) {
- fprintf(stderr, "out of memory\n");
- close(fd);
- return -1;
- }
- device_names = new_device_names;
-
- ufds[nfds].fd = fd;
- ufds[nfds].events = POLLIN;
- device_names[nfds] = strdup(device);
- nfds++;
-
- return 0;
-}
-
-
-
-static int scan_dir(const char *dirname, int print_flags)
-{
- char devname[PATH_MAX];
- char *filename;
- DIR *dir;
- struct dirent *de;
- dir = opendir(dirname);
- if(dir == NULL)
- return -1;
- strcpy(devname, dirname);
- filename = devname + strlen(devname);
- *filename++ = '/';
- while((de = readdir(dir))) {
- if(de->d_name[0] == '.' &&
- (de->d_name[1] == '\0' ||
- (de->d_name[1] == '.' && de->d_name[2] == '\0')))
- continue;
- strcpy(filename, de->d_name);
- open_device(devname, print_flags);
- }
- closedir(dir);
- return 0;
-}
-
-bool _hold;
-
-bool button_hold()
-{
- FILE *f = fopen("/sys/class/axppower/holdkey", "r");
- char x;
- fscanf(f, "%c", &x);
- fclose(f);
- _hold = !(x&STATE_UNLOCKED);
- return _hold;
-}
-
-
-void button_init_device(void)
-{
- int res;
- int print_flags = 0;
- const char *device = NULL;
- const char *device_path = "/dev/input";
-
- nfds = 1;
- ufds = calloc(1, sizeof(ufds[0]));
- ufds[0].fd = inotify_init();
- ufds[0].events = POLLIN;
- if(device) {
- res = open_device(device, print_flags);
- if(res < 0) {
- // return 1;
- }
- } else {
- res = inotify_add_watch(ufds[0].fd, device_path, IN_DELETE | IN_CREATE);
- if(res < 0) {
- fprintf(stderr, "could not add watch for %s, %s\n", device_path, strerror(errno));
- // return 1;
- }
- res = scan_dir(device_path, print_flags);
- if(res < 0) {
- fprintf(stderr, "scan dir failed for %s\n", device_path);
- // return 1;
- }
- }
-
- button_hold(); //store state
-
- set_rockbox_ready();
-}
-
-void touchscreen_enable_device(bool en)
-{
- (void)en; /* FIXME: do something smart */
-}
-
-
-int keycode_to_button(int keyboard_key)
-{
- switch(keyboard_key){
- case KEYCODE_PWR:
- return BUTTON_POWER;
-
- case KEYCODE_PWR_LONG:
- return BUTTON_POWER_LONG;
-
- case KEYCODE_VOLPLUS:
- return BUTTON_VOL_UP;
-
- case KEYCODE_VOLMINUS:
- return BUTTON_VOL_DOWN;
-
- case KEYCODE_PREV:
- return BUTTON_LEFT;
-
- case KEYCODE_NEXT:
- return BUTTON_RIGHT;
-
- case KEYCODE_PLAY:
- return BUTTON_PLAY;
-
- case KEYCODE_HOLD:
- button_hold(); /* store state */
- backlight_hold_changed(_hold);
- return BUTTON_NONE;
-
- default:
- return BUTTON_NONE;
- }
-}
-
-
-int button_read_device(int *data)
-{
- int i;
- int res;
- struct input_event event;
- int read_more;
- unsigned button = 0;
-
- if(last_btns & BUTTON_POWER_LONG)
- {
- return last_btns; /* simulate repeat */
- }
-
- do {
- read_more = 0;
- poll(ufds, nfds, 10);
- for(i = 1; i < nfds; i++) {
- if(ufds[i].revents & POLLIN) {
- res = read(ufds[i].fd, &event, sizeof(event));
- if(res < (int)sizeof(event)) {
- fprintf(stderr, "could not get event\n");
- }
-
- switch(event.type)
- {
- case 1: /* HW-Button */
- button = keycode_to_button(event.code);
- if (_hold) /* we have to wait for keycode_to_button() first to maybe clear hold state */
- break;
- if (button == BUTTON_NONE)
- {
- last_btns = button;
- break;
- }
-/* workaround for a wrong feedback, only present with DX90 */
-#if defined(DX90)
- if (button == BUTTON_RIGHT && (last_btns & BUTTON_LEFT == BUTTON_LEFT) && !event.value)
- {
- button = BUTTON_LEFT;
- }
-#endif
- if (event.value)
- last_btns |= button;
- else
- last_btns &= (~button);
-
- break;
-
- case 3: /* Touchscreen */
- if(_hold)
- break;
-
- switch(event.code)
- {
- case 53: /* x -> next will be y */
- last_x = event.value;
- read_more = 1;
- break;
- case 54: /* y */
- last_y = event.value;
- break;
- case 57: /* press -> next will be x */
- if(event.value==1)
- {
- last_touch_state = STATE_DOWN;
- read_more = 1;
- }
- else
- last_touch_state = STATE_UP;
- break;
- }
- break;
- }
- }
- }
- } while(read_more);
-
-
- /* Get grid button/coordinates based on the current touchscreen mode
- *
- * Caveat: the caller seemingly depends on *data always being filled with
- * the last known touchscreen position, so always call
- * touchscreen_to_pixels() */
- int touch = touchscreen_to_pixels(last_x, last_y, data);
-
- if (last_touch_state == STATE_DOWN)
- return last_btns | touch;
-
- return last_btns;
-}
-
diff --git a/firmware/target/hosted/android/dx50/lcd-dx50.c b/firmware/target/hosted/android/dx50/lcd-dx50.c
deleted file mode 100644
index 4d78baaf00..0000000000
--- a/firmware/target/hosted/android/dx50/lcd-dx50.c
+++ /dev/null
@@ -1,120 +0,0 @@
-/***************************************************************************
- * __________ __ ___.
- * Open \______ \ ____ ____ | | _\_ |__ _______ ___
- * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
- * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
- * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
- * \/ \/ \/ \/ \/
- * $Id: lcd-bitmap.c 29248 2011-02-08 20:05:25Z thomasjfox $
- *
- * Copyright (C) 2011 Lorenzo Miori, Thomas Martitz
- * Copyright (C) 2013 Lorenzo Miori
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU General Public License
- * as published by the Free Software Foundation; either version 2
- * of the License, or (at your option) any later version.
- *
- * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
- * KIND, either express or implied.
- *
- ****************************************************************************/
-
-#include <stdlib.h>
-#include <unistd.h>
-#include <stdio.h>
-#include <linux/fb.h>
-#include <sys/mman.h>
-#include <sys/ioctl.h>
-#include "config.h"
-#include "file.h"
-#include "debug.h"
-#include "system.h"
-#include "screendump.h"
-#include "lcd.h"
-#include "lcd-target.h"
-
-static int dev_fd = 0;
-fb_data *dev_fb = 0;
-
-#ifdef HAVE_LCD_SHUTDOWN
-void lcd_shutdown(void)
-{
- printf("FB closed.");
- munmap(dev_fb, FRAMEBUFFER_SIZE);
- close(dev_fd);
-}
-#endif
-
-void lcd_init_device(void)
-{
- size_t screensize;
- struct fb_var_screeninfo vinfo;
- struct fb_fix_screeninfo finfo;
-
- /* Open the framebuffer device */
- dev_fd = open("/dev/graphics/fb0", O_RDWR);
- if (dev_fd == -1) {
- perror("Error: cannot open framebuffer device");
- exit(1);
- }
-
- /* Get the fixed properties */
- if (ioctl(dev_fd, FBIOGET_FSCREENINFO, &finfo) == -1) {
- perror("Error reading fixed information");
- exit(2);
- }
-
-
- /* Now we get the settable settings, and we set 16 bit bpp */
- if (ioctl(dev_fd, FBIOGET_VSCREENINFO, &vinfo) == -1) {
- perror("Error reading variable information");
- exit(3);
- }
- /* framebuffer does not fit the screen, a bug of iBassos Firmware, not rockbox.
- cannot be solved with parameters */
- vinfo.bits_per_pixel = LCD_DEPTH;
- vinfo.xres = vinfo.xres_virtual = vinfo.width = LCD_WIDTH;
- vinfo.yres = vinfo.yres_virtual = vinfo.height = LCD_HEIGHT;
-
- if (ioctl(dev_fd, FBIOPUT_VSCREENINFO, &vinfo)) {
- perror("fbset(ioctl)");
- exit(4);
- }
-
- /* Figure out the size of the screen in bytes */
- screensize = vinfo.xres * vinfo.yres * vinfo.bits_per_pixel / 8;
- if (screensize != FRAMEBUFFER_SIZE) {
- exit(4);
- perror("Display and framebuffer mismatch!\n");
- }
-
- /* Map the device to memory */
- dev_fb = mmap(0, screensize, PROT_READ | PROT_WRITE, MAP_SHARED, dev_fd, 0);
- if ((int)dev_fb == -1) {
- perror("Error: failed to map framebuffer device to memory");
- exit(4);
- }
-
- /* Be sure to turn on display at startup */
- ioctl(dev_fd, FBIOBLANK, VESA_NO_BLANKING);
-#ifdef HAVE_LCD_ENABLE
- lcd_set_active(true);
-#endif
-}
-
-#ifdef HAVE_LCD_ENABLE
-void lcd_enable(bool enable)
- {
- if (lcd_active() == enable)
- return;
-
- lcd_set_active(enable);
-
- /* Turn on or off the display using Linux interface */
- ioctl(dev_fd, FBIOBLANK, enable ? VESA_NO_BLANKING : VESA_POWERDOWN);
-
- if (enable)
- send_event(LCD_EVENT_ACTIVATION, NULL);
-}
-#endif
diff --git a/firmware/target/hosted/android/dx50/pcm-dx50.c b/firmware/target/hosted/android/dx50/pcm-dx50.c
deleted file mode 100644
index e7695873a0..0000000000
--- a/firmware/target/hosted/android/dx50/pcm-dx50.c
+++ /dev/null
@@ -1,364 +0,0 @@
-/***************************************************************************
- * __________ __ ___.
- * Open \______ \ ____ ____ | | _\_ |__ _______ ___
- * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
- * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
- * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
- * \/ \/ \/ \/ \/
- * $Id$
- *
- * Copyright (C) 2010 Thomas Martitz
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU General Public License
- * as published by the Free Software Foundation; either version 2
- * of the License, or (at your option) any later version.
- *
- * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
- * KIND, either express or implied.
- *
- ****************************************************************************/
-
-
-/*
- * Based, but heavily modified, on the example given at
- * http://www.alsa-project.org/alsa-doc/alsa-lib/_2test_2pcm_8c-example.html
- *
- * This driver uses the so-called unsafe async callback method and hardcoded device
- * names. It fails when the audio device is busy by other apps.
- *
- * To make the async callback safer, an alternative stack is installed, since
- * it's run from a signal hanlder (which otherwise uses the user stack). If
- * tick tasks are run from a signal handler too, please install
- * an alternative stack for it too.
- *
- * TODO: Rewrite this to do it properly with multithreading
- *
- * Alternatively, a version using polling in a tick task is provided. While
- * supposedly safer, it appears to use more CPU (however I didn't measure it
- * accurately, only looked at htop). At least, in this mode the "default"
- * device works which doesnt break with other apps running.
- */
-
-
-#include "autoconf.h"
-
-#include <stdlib.h>
-#include <stdbool.h>
-#include <stdio.h>
-#include <errno.h>
-#include "tinyalsa/asoundlib.h"
-#include "tinyalsa/asound.h"
-#include "system.h"
-#include "debug.h"
-#include "kernel.h"
-
-#include "pcm.h"
-#include "pcm-internal.h"
-#include "pcm_mixer.h"
-#include "pcm_sampr.h"
-#include "audiohw.h"
-
-#include <pthread.h>
-#include <signal.h>
-
-static const snd_pcm_format_t format = PCM_FORMAT_S16_LE; /* sample format */
-static const int channels = 2; /* count of channels */
-static unsigned int rate = 44100; /* stream rate */
-
-typedef struct pcm snd_pcm_t;
-static snd_pcm_t *handle;
-struct pcm_config config;
-
-static snd_pcm_sframes_t period_size = 512; /* if set to >= 1024, all timers become even slower */
-static char *frames;
-
-static const void *pcm_data = 0;
-static size_t pcm_size = 0;
-
-static int recursion;
-
-static int set_hwparams(snd_pcm_t *handle)
-{
- int err;
- if (!frames)
- frames = malloc(pcm_frames_to_bytes(handle, pcm_get_buffer_size(handle)));
- err = 0; /* success */
- return err;
-}
-
-
-/* copy pcm samples to a spare buffer, suitable for snd_pcm_writei() */
-static bool fill_frames(void)
-{
- size_t copy_n, frames_left = period_size;
- bool new_buffer = false;
-
- while (frames_left > 0)
- {
- if (!pcm_size)
- {
- new_buffer = true;
- if (!pcm_play_dma_complete_callback(PCM_DMAST_OK, &pcm_data,
- &pcm_size))
- {
- return false;
- }
- }
- copy_n = MIN((size_t)pcm_size, pcm_frames_to_bytes(handle, frames_left));
- memcpy(&frames[pcm_frames_to_bytes(handle, period_size-frames_left)], pcm_data, copy_n);
-
- pcm_data += copy_n;
- pcm_size -= copy_n;
- frames_left -= pcm_bytes_to_frames(handle, copy_n);
-
- if (new_buffer)
- {
- new_buffer = false;
- pcm_play_dma_status_callback(PCM_DMAST_STARTED);
- }
- }
- return true;
-}
-
-
-static void pcm_tick(void)
-{
- if (fill_frames())
- {
- if (pcm_write(handle, frames, pcm_frames_to_bytes(handle, period_size))) {
- printf("Error playing sample\n");
- return;//break;
- }
-
- }
- else
- {
- DEBUGF("%s: No Data.\n", __func__);
- return;//break;
- }
-}
-
-static int async_rw(snd_pcm_t *handle)
-{
- int err;
- snd_pcm_sframes_t sample_size;
- char *samples;
-
- /* fill buffer with silence to initiate playback without noisy click */
- sample_size = pcm_frames_to_bytes(handle, pcm_get_buffer_size(handle));
- samples = malloc(sample_size);
-
- memset(samples, 0, sample_size);
-
- err = pcm_write(handle, samples, sample_size);
- free(samples);
-
- if (err != 0)
- {
- DEBUGF("Initial write error: %d\n", err);
- return err;
- }
- if (pcm_state(handle) == PCM_STATE_PREPARED)
- {
- err = pcm_start(handle);
- if (err < 0)
- {
- DEBUGF("Start error: %d\n", err);
- return err;
- }
- }
- return 0;
-}
-
-void cleanup(void)
-{
- free(frames);
- frames = NULL;
- pcm_close(handle);
-}
-
-void pcm_play_dma_init(void)
-{
- config.channels = channels;
- config.rate = rate;
- config.period_size = period_size;
- config.period_count = 4;
- config.format = format;
- config.start_threshold = 0;
- config.stop_threshold = 0;
- config.silence_threshold = 0;
-
-
- handle = pcm_open(0, 0, PCM_OUT, &config);
- if (!handle || !pcm_is_ready(handle)) {
- printf("Unable to open PCM device: %s\n", pcm_get_error(handle));
- return;
- }
-
- pcm_dma_apply_settings();
-
- tick_add_task(pcm_tick);
-
- atexit(cleanup);
- return;
-}
-
-
-void pcm_play_lock(void)
-{
- if (recursion++ == 0)
- tick_remove_task(pcm_tick);
-}
-
-void pcm_play_unlock(void)
-{
- if (--recursion == 0)
- tick_add_task(pcm_tick);
-}
-
-static void pcm_dma_apply_settings_nolock(void)
-{
- set_hwparams(handle);
-}
-
-void pcm_dma_apply_settings(void)
-{
- pcm_play_lock();
- pcm_dma_apply_settings_nolock();
- pcm_play_unlock();
-}
-
-
-void pcm_play_dma_pause(bool pause)
-{
- (void)pause;
-}
-
-
-void pcm_play_dma_stop(void)
-{
- pcm_stop(handle);
-}
-
-void pcm_play_dma_start(const void *addr, size_t size)
-{
-#if defined(DX50) || defined(DX90)
- /* headphone output relay: if this is done at startup already, a loud click is audible on headphones. Here, some time later,
- the output caps are charged a bit and the click is much softer */
- system("/system/bin/muteopen");
-#endif
- pcm_dma_apply_settings_nolock();
-
- pcm_data = addr;
- pcm_size = size;
-
- while (1)
- {
- snd_pcm_state_t state = pcm_state(handle);
- switch (state)
- {
- case PCM_STATE_RUNNING:
- return;
- case PCM_STATE_XRUN:
- {
- printf("No handler for STATE_XRUN!\n");
- continue;
- }
- case PCM_STATE_PREPARED:
- { /* prepared state, we need to fill the buffer with silence before
- * starting */
- int err = async_rw(handle);
- if (err < 0)
- printf("Start error: %d\n", err);
- return;
- }
- case PCM_STATE_PAUSED:
- { /* paused, simply resume */
- pcm_play_dma_pause(0);
- return;
- }
- case PCM_STATE_DRAINING:
- /* run until drained */
- continue;
- default:
- DEBUGF("Unhandled state: %d\n", state);
- return;
- }
- }
-}
-
-size_t pcm_get_bytes_waiting(void)
-{
- return pcm_size;
-}
-
-const void * pcm_play_dma_get_peak_buffer(int *count)
-{
- uintptr_t addr = (uintptr_t)pcm_data;
- *count = pcm_size / 4;
- return (void *)((addr + 3) & ~3);
-}
-
-void pcm_play_dma_postinit(void)
-{
- return;
-}
-
-void pcm_set_mixer_volume(int volume)
-{
-#if defined(DX50) || defined(DX90)
- /* -990 to 0 -> 0 to 255 */
- int val = (volume+990)*255/990;
-#if defined(DX50)
- FILE *f = fopen("/dev/codec_volume", "w");
-#else /* DX90 */
- FILE *f = fopen("/sys/class/codec/es9018_volume", "w");
-#endif /* DX50 */
- fprintf(f, "%d", val);
- fclose(f);
-#else
- (void)volume;
-#endif /* DX50 || DX90 */
-}
-
-#ifdef HAVE_RECORDING
-void pcm_rec_lock(void)
-{
-}
-
-void pcm_rec_unlock(void)
-{
-}
-
-void pcm_rec_dma_init(void)
-{
-}
-
-void pcm_rec_dma_close(void)
-{
-}
-
-void pcm_rec_dma_start(void *start, size_t size)
-{
- (void)start;
- (void)size;
-}
-
-void pcm_rec_dma_stop(void)
-{
-}
-
-const void * pcm_rec_dma_get_peak_buffer(void)
-{
- return NULL;
-}
-
-void audiohw_set_recvol(int left, int right, int type)
-{
- (void)left;
- (void)right;
- (void)type;
-}
-
-#endif /* HAVE_RECORDING */
diff --git a/firmware/target/hosted/android/dx50/powermgmt-dx50.c b/firmware/target/hosted/android/dx50/powermgmt-dx50.c
deleted file mode 100644
index 713e8a977e..0000000000
--- a/firmware/target/hosted/android/dx50/powermgmt-dx50.c
+++ /dev/null
@@ -1,97 +0,0 @@
-/***************************************************************************
- * __________ __ ___.
- * Open \______ \ ____ ____ | | _\_ |__ _______ ___
- * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
- * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
- * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
- * \/ \/ \/ \/ \/
- * $Id$
- *
- * Copyright (C) 2010 by Thomas Martitz
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU General Public License
- * as published by the Free Software Foundation; either version 2
- * of the License, or (at your option) any later version.
- *
- * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
- * KIND, either express or implied.
- *
- ****************************************************************************/
-
-#include <stdbool.h>
-#include <stdio.h>
-#include <stdlib.h>
-#include "config.h"
-#include "power.h"
-#include "powermgmt.h"
-
-
-unsigned int power_input_status(void)
-{
- int val;
- FILE *f = fopen("/sys/class/power_supply/ac/present", "r");
- fscanf(f, "%d", &val);
- fclose(f);
- return val?POWER_INPUT_MAIN_CHARGER:POWER_INPUT_NONE;
-}
-
-
-/* Returns true, if battery is charging, false else. */
-bool charging_state( void )
-{
- /* Full, Charging, Discharging */
- char state[9];
-
- /* true if charging. */
- bool charging = false;
-
- FILE *f = fopen( "/sys/class/power_supply/battery/status", "r" );
- if( f != NULL )
- {
- if( fgets( state, 9, f ) != NULL )
- {
- charging = ( strcmp( state, "Charging" ) == 0 );
- }
- }
- fclose( f );
-
- return charging;
-}
-
-
-/* Values for stock PISEN battery. TODO: Needs optimization */
-const unsigned short battery_level_dangerous[BATTERY_TYPES_COUNT] =
-{
- 3380
-};
-
-const unsigned short battery_level_shutoff[BATTERY_TYPES_COUNT] =
-{
- 3100
-};
-
-/* voltages (millivolt) of 0%, 10%, ... 100% when charging disabled */
-const unsigned short percent_to_volt_discharge[BATTERY_TYPES_COUNT][11] =
-{
- { 3370, 3650, 3700, 3740, 3780, 3820, 3870, 3930, 4000, 4090, 4190 }
-};
-
-/* Voltages (millivolt) of 0%, 10%, ... 100% when charging is enabled. */
-const unsigned short percent_to_volt_charge[11] =
-{
- 3370, 3650, 3700, 3740, 3780, 3820, 3870, 3930, 4000, 4090, 4190
-};
-
-
-/* Returns battery voltage from android measurement [millivolts] */
-int _battery_voltage(void)
-{
- int val;
- FILE *f = fopen("/sys/class/power_supply/battery/voltage_now", "r");
- fscanf(f, "%d", &val);
- fclose(f);
- return (val/1000);
-}
-
-
diff --git a/firmware/target/hosted/android/system-android.c b/firmware/target/hosted/android/system-android.c
index 6279504e48..d13b8d6462 100644
--- a/firmware/target/hosted/android/system-android.c
+++ b/firmware/target/hosted/android/system-android.c
@@ -23,12 +23,6 @@
#include <setjmp.h>
#include <jni.h>
#include <pthread.h>
-#if defined(DX50) || defined(DX90)
-#include <stdlib.h>
-#include <sys/reboot.h>
-#include <sys/stat.h>
-#include <stdio.h>
-#endif /* DX50 || DX90 */
#include <unistd.h>
#include "config.h"
#include "system.h"
@@ -37,83 +31,40 @@
-#if !defined(DX50) && !defined(DX90)
/* global fields for use with various JNI calls */
static JavaVM *vm_ptr;
JNIEnv *env_ptr;
jobject RockboxService_instance;
jclass RockboxService_class;
-#endif /* !DX50 && !DX90 */
uintptr_t *stackbegin;
uintptr_t *stackend;
extern int main(void);
-#if !defined(DX50) && !defined(DX90)
extern void telephony_init_device(void);
-#endif
+
void system_exception_wait(void)
{
-#if defined(DX50) || defined(DX90)
- while(1);
-#else
intptr_t dummy = 0;
while(button_read_device(&dummy) != BUTTON_BACK);
-#endif /* DX50 || DX90 */
}
void system_reboot(void)
{
-#if defined(DX50) || defined(DX90)
- reboot(RB_AUTOBOOT);
-#else
power_off();
-#endif /* DX50 || DX90 */
}
-#if !defined(DX50) && !defined(DX90)
/* this is used to return from the entry point of the native library. */
static jmp_buf poweroff_buf;
-#endif
-
void power_off(void)
{
-#if defined(DX50) || defined(DX90)
- reboot(RB_POWER_OFF);
-#else
longjmp(poweroff_buf, 1);
-#endif /* DX50 || DX90 */
}
void system_init(void)
{
-#if defined(DX50) || defined(DX90)
- volatile uintptr_t stack = 0;
- stackbegin = stackend = (uintptr_t*) &stack;
-
- struct stat m1, m2;
- stat("/mnt/", &m1);
- do
- {
- /* waiting for storage to get mounted */
- stat("/sdcard/", &m2);
- usleep(100000);
- }
- while(m1.st_dev == m2.st_dev);
-/* here would be the correct place for 'system("/system/bin/muteopen");' (headphone-out relay) but in pcm-dx50.c, pcm_play_dma_start()
- the output capacitors are charged already a bit and the click of the headphone-connection-relay is softer */
-
-#if defined(DX90)
- /* DAC needs to be unmuted on DX90 */
- FILE * f = fopen("/sys/class/codec/wm8740_mute", "w");
- fputc(0, f);
- fclose(f);
-#endif /* DX90 */
-
-#else
/* no better place yet */
telephony_init_device();
-#endif /* DX50 || DX90 */
}
int hostfs_init(void)
@@ -128,7 +79,6 @@ int hostfs_flush(void)
return 0;
}
-#if !defined(DX50) && !defined(DX90)
JNIEXPORT jint JNICALL
JNI_OnLoad(JavaVM *vm, void* reserved)
{
@@ -169,7 +119,7 @@ Java_org_rockbox_RockboxService_main(JNIEnv *env, jobject this)
/* simply return here. this will allow the VM to clean up objects and do
* garbage collection */
}
-#endif /* !DX50 && !DX90 */
+
/* below is the facility for external (from other java threads) to safely call
* into our snative code. When extracting rockbox.zip the main function is
diff --git a/firmware/target/hosted/filesystem-app.c b/firmware/target/hosted/filesystem-app.c
index 0c9943b635..826ab5bbb1 100644
--- a/firmware/target/hosted/filesystem-app.c
+++ b/firmware/target/hosted/filesystem-app.c
@@ -47,7 +47,7 @@ static const char *rbhome;
static const char rbhome[] = HOME_DIR;
#endif
-#if !(defined(SAMSUNG_YPR0) || defined(SAMSUNG_YPR1)) && !defined(__PCTOOL__)
+#if !(defined(SAMSUNG_YPR0) || defined(SAMSUNG_YPR1) || defined(DX50) || defined(DX90)) && !defined(__PCTOOL__)
/* Special dirs are user-accessible (and user-writable) dirs which take priority
* over the ones where Rockbox is installed to. Classic example would be
* $HOME/.config/rockbox.org vs /usr/share/rockbox */
diff --git a/firmware/target/hosted/ibasso/android_ndk.make b/firmware/target/hosted/ibasso/android_ndk.make
new file mode 100644
index 0000000000..e2f5c93db8
--- /dev/null
+++ b/firmware/target/hosted/ibasso/android_ndk.make
@@ -0,0 +1,49 @@
+# __________ __ ___
+# Open \______ \ ____ ____ | | _\_ |__ _______ ___
+# Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+# Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+# Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+# \/ \/ \/ \/ \/
+#
+# Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+# Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+# Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+# Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+#
+# This program is free software; you can redistribute it and/or
+# modify it under the terms of the GNU General Public License
+# as published by the Free Software Foundation; either version 2
+# of the License, or (at your option) any later version.
+#
+# This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+# KIND, either express or implied.
+
+
+# This is a glibc compatibility hack to provide a get_nprocs() replacement.
+# The NDK ships cpu-features.c which has a compatible function android_getCpuCount()
+CPUFEAT = $(ANDROID_NDK_PATH)/sources/android/cpufeatures
+CPUFEAT_BUILD = $(BUILDDIR)/android-ndk/sources/android/cpufeatures
+INCLUDES += -I$(CPUFEAT)
+OTHER_SRC += $(CPUFEAT)/cpu-features.c
+CLEANOBJS += $(CPUFEAT_BUILD)/cpu-features.o
+$(CPUFEAT_BUILD)/cpu-features.o: $(CPUFEAT)/cpu-features.c
+ $(SILENT)mkdir -p $(dir $@)
+ $(call PRINTS,CC $(subst $(CPUFEAT)/,,$<))$(CC) -o $@ -c $(CPUFEAT)/cpu-features.c $(GCCOPTS) -Wno-unused
+
+.SECONDEXPANSION:
+.PHONY: clean dirs
+
+DIRS += $(CPUFEAT_BUILD)
+
+.PHONY:
+$(BUILDDIR)/$(BINARY): $$(OBJ) $(FIRMLIB) $(VOICESPEEXLIB) $(CORE_LIBS) $(CPUFEAT_BUILD)/cpu-features.o
+ $(call PRINTS,LD $(BINARY))$(CC) -o $@ $^ $(LDOPTS) $(GLOBAL_LDOPTS) -Wl,-Map,$(BUILDDIR)/rockbox.map
+ $(call PRINTS,OC $(@F))$(call objcopy,$@,$@)
+
+$(DIRS):
+ $(SILENT)mkdir -p $@
+
+dirs: $(DIRS)
+
+clean::
+ $(SILENT)rm -rf $(BUILDDIR)/android-ndk
diff --git a/firmware/target/hosted/ibasso/audiohw-ibasso.c b/firmware/target/hosted/ibasso/audiohw-ibasso.c
new file mode 100644
index 0000000000..447e133eba
--- /dev/null
+++ b/firmware/target/hosted/ibasso/audiohw-ibasso.c
@@ -0,0 +1,81 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#include "config.h"
+#include "debug.h"
+#include "pcm_sw_volume.h"
+#include "settings.h"
+
+#include "debug-ibasso.h"
+#include "pcm-ibasso.h"
+
+
+void audiohw_close(void)
+{
+ TRACE;
+
+ pcm_close_device();
+}
+
+
+void set_software_volume(void)
+{
+ /* -73dB (?) minimum software volume in decibels. See pcm-internal.h. */
+ static const int SW_VOLUME_MIN = 730;
+
+ int sw_volume_l = 0;
+ int sw_volume_r = 0;
+
+ if(global_settings.balance > 0)
+ {
+ if(global_settings.balance == 100)
+ {
+ sw_volume_l = PCM_MUTE_LEVEL;
+ }
+ else
+ {
+ sw_volume_l -= SW_VOLUME_MIN * global_settings.balance / 100;
+ }
+ }
+ else if(global_settings.balance < 0)
+ {
+ if(global_settings.balance == -100)
+ {
+ sw_volume_r = PCM_MUTE_LEVEL;
+ }
+ else
+ {
+ sw_volume_r = SW_VOLUME_MIN * global_settings.balance / 100;
+ }
+ }
+
+ DEBUGF("DEBUG %s: global_settings.balance: %d, sw_volume_l: %d, sw_volume_r: %d.",
+ __func__,
+ global_settings.balance,
+ sw_volume_l,
+ sw_volume_r);
+
+ /* Emulate balance with software volume. */
+ pcm_set_master_volume(sw_volume_l, sw_volume_r);
+}
diff --git a/firmware/target/hosted/ibasso/backlight-ibasso.c b/firmware/target/hosted/ibasso/backlight-ibasso.c
new file mode 100644
index 0000000000..907980e01a
--- /dev/null
+++ b/firmware/target/hosted/ibasso/backlight-ibasso.c
@@ -0,0 +1,132 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#include <stdbool.h>
+
+#include "config.h"
+#include "debug.h"
+#include "lcd.h"
+#include "panic.h"
+
+#include "debug-ibasso.h"
+#include "sysfs-ibasso.h"
+
+
+/*
+ Prevent excessive backlight_hw_on usage.
+ Required for proper seeking.
+*/
+static bool _backlight_enabled = false;
+
+
+bool backlight_hw_init(void)
+{
+ TRACE;
+
+ /*
+ /sys/devices/platform/rk29_backlight/backlight/rk28_bl/bl_power
+ 0: backlight on
+ */
+ if(! sysfs_set_int(SYSFS_BACKLIGHT_POWER, 0))
+ {
+ DEBUGF("ERROR %s: Can not enable backlight.", __func__);
+ panicf("ERROR %s: Can not enable backlight.", __func__);
+ return false;
+ }
+
+ _backlight_enabled = true;
+
+ return true;
+}
+
+
+void backlight_hw_on(void)
+{
+ if(! _backlight_enabled)
+ {
+ backlight_hw_init();
+ lcd_enable(true);
+ }
+}
+
+
+void backlight_hw_off(void)
+{
+ TRACE;
+
+ /*
+ /sys/devices/platform/rk29_backlight/backlight/rk28_bl/bl_power
+ 1: backlight off
+ */
+ if(! sysfs_set_int(SYSFS_BACKLIGHT_POWER, 1))
+ {
+ DEBUGF("ERROR %s: Can not disable backlight.", __func__);
+ return;
+ }
+
+ lcd_enable(false);
+
+ _backlight_enabled = false;
+}
+
+
+/*
+ Prevent excessive backlight_hw_brightness usage.
+ Required for proper seeking.
+*/
+static int _current_brightness = -1;
+
+
+void backlight_hw_brightness(int brightness)
+{
+ if(brightness > MAX_BRIGHTNESS_SETTING)
+ {
+ DEBUGF("DEBUG %s: Adjusting brightness from %d to MAX.", __func__, brightness);
+ brightness = MAX_BRIGHTNESS_SETTING;
+ }
+ if(brightness < MIN_BRIGHTNESS_SETTING)
+ {
+ DEBUGF("DEBUG %s: Adjusting brightness from %d to MIN.", __func__, brightness);
+ brightness = MIN_BRIGHTNESS_SETTING;
+ }
+
+ if(_current_brightness == brightness)
+ {
+ return;
+ }
+
+ TRACE;
+
+ _current_brightness = brightness;
+
+ /*
+ /sys/devices/platform/rk29_backlight/backlight/rk28_bl/max_brightness
+ 0 ... 255
+ */
+ if(! sysfs_set_int(SYSFS_BACKLIGHT_BRIGHTNESS, _current_brightness))
+ {
+ DEBUGF("ERROR %s: Can not set brightness.", __func__);
+ return;
+ }
+}
diff --git a/firmware/target/hosted/android/dx50/backlight-target.h b/firmware/target/hosted/ibasso/backlight-target.h
index 0dc7ce387a..aa8fafab04 100644
--- a/firmware/target/hosted/android/dx50/backlight-target.h
+++ b/firmware/target/hosted/ibasso/backlight-target.h
@@ -1,13 +1,15 @@
/***************************************************************************
- * __________ __ ___.
+ * __________ __ ___
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
- * $Id$
*
- * Copyright (C) 2008 by Maurus Cuelenaere
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
@@ -18,14 +20,20 @@
* KIND, either express or implied.
*
****************************************************************************/
-#ifndef BACKLIGHT_TARGET_H
-#define BACKLIGHT_TARGET_H
+
+
+#ifndef _BACKLIGHT_TARGET_H_
+#define _BACKLIGHT_TARGET_H_
+
#include <stdbool.h>
+
+/* See backlight.c */
bool backlight_hw_init(void);
void backlight_hw_on(void);
void backlight_hw_off(void);
void backlight_hw_brightness(int brightness);
-#endif /* BACKLIGHT_TARGET_H */
+
+#endif
diff --git a/firmware/target/hosted/ibasso/button-ibasso.c b/firmware/target/hosted/ibasso/button-ibasso.c
new file mode 100644
index 0000000000..1694992ea4
--- /dev/null
+++ b/firmware/target/hosted/ibasso/button-ibasso.c
@@ -0,0 +1,420 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#include <dirent.h>
+#include <errno.h>
+#include <fcntl.h>
+#include <stdbool.h>
+#include <stdlib.h>
+#include <string.h>
+#include <unistd.h>
+#include <linux/input.h>
+#include <sys/inotify.h>
+#include <sys/limits.h>
+#include <sys/poll.h>
+
+#include "config.h"
+#include "backlight.h"
+#include "button.h"
+#include "debug.h"
+#include "panic.h"
+#include "settings.h"
+#include "touchscreen.h"
+
+#include "button-ibasso.h"
+#include "button-target.h"
+#include "debug-ibasso.h"
+#include "sysfs-ibasso.h"
+
+
+#define EVENT_TYPE_BUTTON 1
+
+/* /dev/input/event0 */
+#define EVENT_CODE_BUTTON_LINEOUT 113
+#define EVENT_CODE_BUTTON_SPDIF 114
+#define EVENT_CODE_BUTTON_HOLD 115
+
+/* /dev/input/event1 */
+#define EVENT_CODE_BUTTON_SDCARD 143
+
+
+#define EVENT_TYPE_TOUCHSCREEN 3
+
+/* /dev/input/event2 */
+#define EVENT_CODE_TOUCHSCREEN_X 53
+#define EVENT_CODE_TOUCHSCREEN_Y 54
+#define EVENT_CODE_TOUCHSCREEN 57
+
+#define EVENT_VALUE_TOUCHSCREEN_PRESS 1
+#define EVENT_VALUE_TOUCHSCREEN_RELEASE -1
+
+
+/*
+ Changing bit, when hold switch is toggled.
+ Bit is off when hold switch is engaged.
+*/
+#define HOLD_SWITCH_BIT 16
+
+/*
+ Changing bit, when coaxial out is plugged.
+ Bit is off when coaxial out is plugged in.
+*/
+#define COAX_BIT 32
+
+/*
+ Changing bit, when line out is plugged.
+ Bit is off when line out is plugged in.
+*/
+#define SPDIF_BIT 64
+
+
+/* State of the hold switch; true: hold switch engaged. */
+static bool _hold = false;
+
+
+/* See button.h. */
+bool button_hold(void)
+{
+ char hold_state;
+ if(! sysfs_get_char(SYSFS_HOLDKEY, &hold_state))
+ {
+ DEBUGF("ERROR %s: Can not get hold switch state.", __func__);
+ hold_state = HOLD_SWITCH_BIT;
+ }
+
+ /*DEBUGF("%s: hold_state: %d, %c.", __func__, hold_state, hold_state);*/
+
+ /*bool coax_connected = ! (hold_state & COAX_BIT);
+ bool spdif_connected = ! (hold_state & SPDIF_BIT);*/
+
+ _hold = ! (hold_state & HOLD_SWITCH_BIT);
+
+ /*DEBUGF("%s: _hold: %d, coax_connected: %d, spdif_connected: %d.", __func__, _hold, coax_connected, spdif_connected);*/
+
+ return _hold;
+}
+
+
+/* Input devices monitored with poll API. */
+static struct pollfd* _fds = NULL;
+
+
+/* Number of input devices monitored with poll API. */
+static nfds_t _nfds = 0;
+
+
+/* The names of the devices in _fds. */
+static char** _device_names = NULL;
+
+
+/* Open device device_name and add it to the list of polled devices. */
+static bool open_device(const char* device_name)
+{
+ int fd = open(device_name, O_RDONLY);
+ if(fd == -1)
+ {
+ DEBUGF("ERROR %s: open failed on %s.", __func__, device_name);
+ return false;
+ }
+
+ struct pollfd* new_fds = realloc(_fds, sizeof(struct pollfd) * (_nfds + 1));
+ if(new_fds == NULL)
+ {
+ DEBUGF("ERROR %s: realloc for _fds failed.", __func__);
+ panicf("ERROR %s: realloc for _fds failed.", __func__);
+ return false;
+ }
+
+ _fds = new_fds;
+ _fds[_nfds].fd = fd;
+ _fds[_nfds].events = POLLIN;
+
+ char** new_device_names = realloc(_device_names, sizeof(char*) * (_nfds + 1));
+ if(new_device_names == NULL)
+ {
+ DEBUGF("ERROR %s: realloc for _device_names failed.", __func__);
+ panicf("ERROR %s: realloc for _device_names failed.", __func__);
+ return false;
+ }
+
+ _device_names = new_device_names;
+ _device_names[_nfds] = strdup(device_name);
+ if(_device_names[_nfds] == NULL)
+ {
+ DEBUGF("ERROR %s: strdup failed.", __func__);
+ panicf("ERROR %s: strdup failed.", __func__);
+ return false;
+ }
+
+ ++_nfds;
+
+ DEBUGF("DEBUG %s: Opened device %s.", __func__, device_name);
+
+ return true;
+}
+
+
+/* See button.h. */
+void button_init_device(void)
+{
+ TRACE;
+
+ if((_fds != NULL) || (_nfds != 0) || (_device_names != NULL))
+ {
+ DEBUGF("ERROR %s: Allready initialized.", __func__);
+ panicf("ERROR %s: Allready initialized.", __func__);
+ return;
+ }
+
+ /* The input device directory. */
+ static const char device_path[] = "/dev/input";
+
+ /* Path delimeter. */
+ static const char delimeter[] = "/";
+
+ /* Open all devices in device_path. */
+ DIR* dir = opendir(device_path);
+ if(dir == NULL)
+ {
+ DEBUGF("ERROR %s: opendir failed: errno: %d.", __func__, errno);
+ panicf("ERROR %s: opendir failed: errno: %d.", __func__, errno);
+ return;
+ }
+
+ char device_name[PATH_MAX];
+ strcpy(device_name, device_path);
+ strcat(device_name, delimeter);
+ char* device_name_idx = device_name + strlen(device_name);
+
+ struct dirent* dir_entry;
+ while((dir_entry = readdir(dir)))
+ {
+ if( ((dir_entry->d_name[0] == '.') && (dir_entry->d_name[1] == '\0'))
+ || ((dir_entry->d_name[0] == '.') && (dir_entry->d_name[1] == '.') && (dir_entry->d_name[2] == '\0')))
+ {
+ continue;
+ }
+
+ strcpy(device_name_idx, dir_entry->d_name);
+
+ /* Open and add device to _fds. */
+ open_device(device_name);
+ }
+
+ closedir(dir);
+
+ /* Sanity check. */
+ if(_nfds < 2)
+ {
+ DEBUGF("ERROR %s: No input devices.", __func__);
+ panicf("ERROR %s: No input devices.", __func__);
+ return;
+ }
+
+ /*
+ Hold switch has a separate interface for its state.
+ Input events just report that it has been toggled, but not the state.
+ */
+ button_hold();
+}
+
+
+/* Last known touchscreen coordinates. */
+static int _last_x = 0;
+static int _last_y = 0;
+
+
+/* Last known touchscreen state. */
+static enum
+{
+ TOUCHSCREEN_STATE_UNKNOWN = 0,
+ TOUCHSCREEN_STATE_UP,
+ TOUCHSCREEN_STATE_DOWN
+} _last_touch_state = TOUCHSCREEN_STATE_UNKNOWN;
+
+
+static bool handle_touchscreen_event(__u16 code, __s32 value)
+{
+ bool read_more = false;
+
+ switch(code)
+ {
+ case EVENT_CODE_TOUCHSCREEN_X:
+ {
+ _last_x = value;
+
+ /* x -> next will be y. */
+ read_more = true;
+
+ break;
+ }
+
+ case EVENT_CODE_TOUCHSCREEN_Y:
+ {
+ _last_y = value;
+ break;
+ }
+
+ case EVENT_CODE_TOUCHSCREEN:
+ {
+ if(value == EVENT_VALUE_TOUCHSCREEN_PRESS)
+ {
+ _last_touch_state = TOUCHSCREEN_STATE_DOWN;
+
+ /* Press -> next will be x. */
+ read_more = true;
+ }
+ else
+ {
+ _last_touch_state = TOUCHSCREEN_STATE_UP;
+ }
+ break;
+ }
+ }
+
+ return read_more;
+}
+
+
+/* Last known hardware buttons pressed. */
+static int _last_btns = BUTTON_NONE;
+
+
+/* See button.h. */
+int button_read_device(int *data)
+{
+ bool read_more = true;
+ while(read_more)
+ {
+ read_more = false;
+
+ /* Poll all input devices. */
+ poll(_fds, _nfds, 0);
+
+ for(nfds_t fds_idx = 0; fds_idx < _nfds; ++fds_idx)
+ {
+ if(! (_fds[fds_idx].revents & POLLIN))
+ {
+ continue;
+ }
+
+ struct input_event event;
+ if(read(_fds[fds_idx].fd, &event, sizeof(event)) < (int) sizeof(event))
+ {
+ DEBUGF("ERROR %s: Read of input devices failed.", __func__);
+ continue;
+ }
+
+ /*DEBUGF("DEBUG %s: device: %s, event.type: %d, event.code: %d, event.value: %d", __func__, _device_names[fds_idx], event.type, event.code, event.value);*/
+
+ switch(event.type)
+ {
+ case EVENT_TYPE_BUTTON:
+ {
+ if(event.code == EVENT_CODE_BUTTON_HOLD)
+ {
+ /* Hold switch toggled, update hold switch state. */
+ button_hold();
+ backlight_hold_changed(_hold);
+
+ _last_btns = BUTTON_NONE;
+ break;
+ }
+
+ _last_btns = handle_button_event(event.code, event.value, _last_btns);
+
+ if(_hold)
+ {
+ /* Hold switch engaged. Ignore all button events. */
+ _last_btns = BUTTON_NONE;
+ }
+
+ /*DEBUGF("DEBUG %s: _last_btns: %#8.8x", __func__, _last_btns);*/
+ break;
+ }
+
+ case EVENT_TYPE_TOUCHSCREEN:
+ {
+ if(_hold)
+ {
+ /* Hold switch engaged, ignore all touchscreen events. */
+ _last_touch_state = TOUCHSCREEN_STATE_UNKNOWN;
+ _last_btns = BUTTON_NONE;
+ }
+ else
+ {
+ read_more = handle_touchscreen_event(event.code, event.value);
+ /*DEBUGF("DEBUG %s: _last_touch_state: %d, _last_x: %d, _last_y: %d, read_more: %s", __func__, _last_touch_state, _last_x, _last_y, read_more ? "true" : "false");*/
+ }
+ break;
+ }
+ }
+ }
+ }
+
+ /*
+ Get grid button/coordinates based on the current touchscreen mode
+ Caveat: The caller seemingly depends on *data always being filled with
+ the last known touchscreen position, so always call
+ touchscreen_to_pixels().
+ */
+ int touch = touchscreen_to_pixels(_last_x, _last_y, data);
+
+ if(_last_touch_state == TOUCHSCREEN_STATE_DOWN)
+ {
+ return _last_btns | touch;
+ }
+
+ /*DEBUGF("DEBUG %s: _last_btns: %#8.8x.", __func__, _last_btns);*/
+
+ return _last_btns;
+}
+
+
+void button_close_device(void)
+{
+ TRACE;
+
+ if(_fds)
+ {
+ for(nfds_t fds_idx = 0; fds_idx < _nfds; ++fds_idx)
+ {
+ close(_fds[fds_idx].fd);
+ }
+ free(_fds);
+ _fds = NULL;
+ }
+
+ if(_device_names)
+ {
+ for(nfds_t fds_idx = 0; fds_idx < _nfds; ++fds_idx)
+ {
+ free(_device_names[fds_idx]);
+ }
+ free(_device_names);
+ _device_names = NULL;
+ }
+
+ _nfds = 0;
+}
diff --git a/firmware/target/hosted/ibasso/button-ibasso.h b/firmware/target/hosted/ibasso/button-ibasso.h
new file mode 100644
index 0000000000..09c09e7c83
--- /dev/null
+++ b/firmware/target/hosted/ibasso/button-ibasso.h
@@ -0,0 +1,61 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#ifndef _BUTTON_IBASSO_H_
+#define _BUTTON_IBASSO_H_
+
+
+#include <sys/types.h>
+
+
+/* /dev/input/event0 */
+#define EVENT_CODE_BUTTON_PWR 116
+#define EVENT_CODE_BUTTON_PWR_LONG 117
+
+/* /dev/input/event1 */
+#define EVENT_CODE_BUTTON_VOLPLUS 158
+#define EVENT_CODE_BUTTON_VOLMINUS 159
+#define EVENT_CODE_BUTTON_REV 160
+#define EVENT_CODE_BUTTON_PLAY 161
+#define EVENT_CODE_BUTTON_NEXT 162
+
+#define EVENT_VALUE_BUTTON_PRESS 1
+#define EVENT_VALUE_BUTTON_RELEASE 0
+
+
+/*
+ Handle hardware button events.
+ code: Input event code.
+ value: Input event value.
+ last_btns: Last known pressed buttons.
+ Returns: Currently pressed buttons as bitmask (BUTTON_ values in button-target.h).
+*/
+int handle_button_event(__u16 code, __s32 value, int last_btns);
+
+
+/* Clean up the button device handler. */
+void button_close_device(void);
+
+
+#endif
diff --git a/firmware/target/hosted/android/dx50/button-target.h b/firmware/target/hosted/ibasso/button-target.h
index adc9cf6bfd..d1b3c8a8de 100644
--- a/firmware/target/hosted/android/dx50/button-target.h
+++ b/firmware/target/hosted/ibasso/button-target.h
@@ -1,32 +1,32 @@
/***************************************************************************
- * __________ __ ___.
+ * __________ __ ___
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
- * $Id$
*
- * Copyright (C) 2007 by Rob Purchase
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
- * of the License, or (at your option) any later version.r
+ * of the License, or (at your option) any later version.
*
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
* KIND, either express or implied.
*
****************************************************************************/
+
#ifndef _BUTTON_TARGET_H_
#define _BUTTON_TARGET_H_
-#include <stdbool.h>
-
-#define HAS_BUTTON_HOLD
-/* Main unit's buttons */
+/* Hardware buttons. */
#define BUTTON_LEFT 0x00000001
#define BUTTON_RIGHT 0x00000002
#define BUTTON_PLAY 0x00000004
@@ -35,25 +35,15 @@
#define BUTTON_VOL_DOWN 0x00000020
#define BUTTON_POWER_LONG 0x00000040
-#define BUTTON_MAIN (BUTTON_LEFT|BUTTON_VOL_UP|BUTTON_VOL_DOWN\
- |BUTTON_RIGHT|BUTTON_PLAY|BUTTON_POWER|BUTTON_POWER_LONG)
+#define BUTTON_MAIN ( BUTTON_LEFT | BUTTON_VOL_UP | BUTTON_VOL_DOWN | BUTTON_RIGHT \
+ | BUTTON_PLAY | BUTTON_POWER | BUTTON_POWER_LONG)
+
-#define KEYCODE_LINEOUT 113
-#define KEYCODE_SPDIF 114
-#define KEYCODE_HOLD 115
-#define KEYCODE_PWR 116
-#define KEYCODE_PWR_LONG 117
-#define KEYCODE_SD 143
-#define KEYCODE_VOLPLUS 158
-#define KEYCODE_VOLMINUS 159
-#define KEYCODE_PREV 160
-#define KEYCODE_NEXT 162
-#define KEYCODE_PLAY 161
-#define STATE_UNLOCKED 16
-#define STATE_SPDIF_UNPLUGGED 32
+#define STATE_SPDIF_UNPLUGGED 32
#define STATE_LINEOUT_UNPLUGGED 64
-/* Touch Screen Area Buttons */
+
+/* Touchscreen area buttons 3x3 grid. */
#define BUTTON_TOPLEFT 0x00001000
#define BUTTON_TOPMIDDLE 0x00002000
#define BUTTON_TOPRIGHT 0x00004000
@@ -65,8 +55,9 @@
#define BUTTON_BOTTOMRIGHT 0x00100000
-/* Software power-off */
-#define POWEROFF_BUTTON BUTTON_POWER_LONG
-#define POWEROFF_COUNT 0
+/* Power-off */
+#define POWEROFF_BUTTON BUTTON_POWER_LONG
+#define POWEROFF_COUNT 0
+
-#endif /* _BUTTON_TARGET_H_ */
+#endif
diff --git a/firmware/target/hosted/ibasso/debug-ibasso.c b/firmware/target/hosted/ibasso/debug-ibasso.c
new file mode 100644
index 0000000000..6295de1f6c
--- /dev/null
+++ b/firmware/target/hosted/ibasso/debug-ibasso.c
@@ -0,0 +1,70 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#include <stdarg.h>
+#include <stdio.h>
+#include <string.h>
+#include <android/log.h>
+
+#include "config.h"
+#include "debug.h"
+
+#include "debug-ibasso.h"
+
+
+static const char log_tag[] = "Rockbox";
+
+
+void debug_init(void)
+{}
+
+
+void debugf(const char *fmt, ...)
+{
+ va_list ap;
+ va_start(ap, fmt);
+ __android_log_vprint(ANDROID_LOG_DEBUG, log_tag, fmt, ap);
+ va_end(ap);
+}
+
+
+void ldebugf(const char* file, int line, const char *fmt, ...)
+{
+ va_list ap;
+ /* 13: 5 literal chars and 8 chars for the line number. */
+ char buf[strlen(file) + strlen(fmt) + 13];
+ snprintf(buf, sizeof(buf), "%s (%d): %s", file, line, fmt);
+ va_start(ap, fmt);
+ __android_log_vprint(ANDROID_LOG_DEBUG, log_tag, buf, ap);
+ va_end(ap);
+}
+
+
+void debug_trace(const char* function)
+{
+ static const char trace_tag[] = "TRACE: ";
+ char msg[strlen(trace_tag) + strlen(function) + 1];
+ snprintf(msg, sizeof(msg), "%s%s", trace_tag, function);
+ __android_log_write(ANDROID_LOG_DEBUG, log_tag, msg);
+}
diff --git a/firmware/target/hosted/ibasso/debug-ibasso.h b/firmware/target/hosted/ibasso/debug-ibasso.h
new file mode 100644
index 0000000000..456f189a5a
--- /dev/null
+++ b/firmware/target/hosted/ibasso/debug-ibasso.h
@@ -0,0 +1,38 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#ifndef _DEBUG_IBASSO_H_
+#define _DEBUG_IBASSO_H_
+
+
+void debug_trace(const char* function);
+
+
+#ifdef DEBUG
+#define TRACE debug_trace(__func__)
+#else
+#define TRACE
+#endif
+
+#endif \ No newline at end of file
diff --git a/firmware/target/hosted/ibasso/dx50/audiohw-dx50.c b/firmware/target/hosted/ibasso/dx50/audiohw-dx50.c
new file mode 100644
index 0000000000..5e61348c8d
--- /dev/null
+++ b/firmware/target/hosted/ibasso/dx50/audiohw-dx50.c
@@ -0,0 +1,68 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#include "config.h"
+#include "debug.h"
+
+#include "debug-ibasso.h"
+#include "sysfs-ibasso.h"
+
+
+extern void set_software_volume(void);
+
+
+void audiohw_set_volume(int volume)
+{
+ set_software_volume();
+
+ /*
+ See codec-dx50.h.
+ -128db -> -1 (adjusted to 0, mute)
+ -127dB to 0dB -> 1 to 255 in steps of 2
+ volume is in centibels (tenth-decibels).
+ */
+ int volume_adjusted = (volume / 10) * 2 + 255;
+
+ DEBUGF("DEBUG %s: volume: %d, volume_adjusted: %d.", __func__, volume, volume_adjusted);
+
+ if(volume_adjusted > 255)
+ {
+ DEBUGF("DEBUG %s: Adjusting volume from %d to 255.", __func__, volume);
+ volume_adjusted = 255;
+ }
+ if(volume_adjusted < 0)
+ {
+ DEBUGF("DEBUG %s: Adjusting volume from %d to 0.", __func__, volume);
+ volume_adjusted = 0;
+ }
+
+ /*
+ /dev/codec_volume
+ 0 ... 255
+ */
+ if(! sysfs_set_int(SYSFS_DX50_CODEC_VOLUME, volume_adjusted))
+ {
+ DEBUGF("ERROR %s: Can not set volume.", __func__);
+ }
+}
diff --git a/firmware/target/hosted/ibasso/dx50/button-dx50.c b/firmware/target/hosted/ibasso/dx50/button-dx50.c
new file mode 100644
index 0000000000..b4f6952d44
--- /dev/null
+++ b/firmware/target/hosted/ibasso/dx50/button-dx50.c
@@ -0,0 +1,96 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#include "config.h"
+#include "button.h"
+
+#include "button-ibasso.h"
+
+
+int handle_button_event(__u16 code, __s32 value, int last_btns)
+{
+ int button = BUTTON_NONE;
+
+ switch(code)
+ {
+ case EVENT_CODE_BUTTON_PWR:
+ {
+ button = BUTTON_POWER;
+ break;
+ }
+
+ case EVENT_CODE_BUTTON_PWR_LONG:
+ {
+ button = BUTTON_POWER_LONG;
+ break;
+ }
+
+ case EVENT_CODE_BUTTON_VOLPLUS:
+ {
+ button = BUTTON_VOL_UP;
+ break;
+ }
+
+ case EVENT_CODE_BUTTON_VOLMINUS:
+ {
+ button = BUTTON_VOL_DOWN;
+ break;
+ }
+
+ case EVENT_CODE_BUTTON_REV:
+ {
+ button = BUTTON_LEFT;
+ break;
+ }
+
+ case EVENT_CODE_BUTTON_PLAY:
+ {
+ button = BUTTON_PLAY;
+ break;
+ }
+
+ case EVENT_CODE_BUTTON_NEXT:
+ {
+ button = BUTTON_RIGHT;
+ break;
+ }
+
+ default:
+ {
+ return BUTTON_NONE;
+ }
+ }
+
+ int buttons = last_btns;
+ if(value == EVENT_VALUE_BUTTON_PRESS)
+ {
+ buttons = (last_btns | button);
+ }
+ else
+ {
+ buttons = (last_btns & (~ button));
+ }
+
+ return buttons;
+}
diff --git a/firmware/target/hosted/ibasso/dx50/codec-dx50.h b/firmware/target/hosted/ibasso/dx50/codec-dx50.h
new file mode 100644
index 0000000000..89a1a3f1c4
--- /dev/null
+++ b/firmware/target/hosted/ibasso/dx50/codec-dx50.h
@@ -0,0 +1,51 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#ifndef _CODEC_DX50_H_
+#define _CODEC_DX50_H_
+
+
+#define AUDIOHW_CAPS MONO_VOL_CAP
+
+
+/*
+ http://www.wolfsonmicro.com/media/76425/WM8740.pdf
+
+ 0.5 * ( x - 255 ) = ydB 1 <= x <= 255
+ mute x = 0
+
+ x = 255 -> 0dB
+ .
+ .
+ .
+ x = 2 -> -126.5dB
+ x = 1 -> -127dB
+ x = 0 -> -128dB
+
+ See audiohw.h, sound.c.
+*/
+AUDIOHW_SETTING(VOLUME, "dB", 0, 1, -128, 0, -30)
+
+
+#endif
diff --git a/firmware/target/hosted/ibasso/dx90/audiohw-dx90.c b/firmware/target/hosted/ibasso/dx90/audiohw-dx90.c
new file mode 100644
index 0000000000..ef18aae4bd
--- /dev/null
+++ b/firmware/target/hosted/ibasso/dx90/audiohw-dx90.c
@@ -0,0 +1,63 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#include "config.h"
+#include "debug.h"
+
+#include "debug-ibasso.h"
+#include "sysfs-ibasso.h"
+
+
+extern void set_software_volume(void);
+
+
+void audiohw_set_volume(int volume)
+{
+ set_software_volume();
+
+ /* See codec-dx90.h. -2550 to 0 -> 0 to 255 */
+ int volume_adjusted = ((volume + 2550) / 10);
+
+ DEBUGF("DEBUG %s: volume: %d, volume_adjusted: %d.", __func__, volume, volume_adjusted);
+
+ if(volume_adjusted > 255)
+ {
+ DEBUGF("DEBUG %s: Adjusting volume from %d to 255.", __func__, volume);
+ volume_adjusted = 255;
+ }
+ if(volume_adjusted < 0)
+ {
+ DEBUGF("DEBUG %s: Adjusting volume from %d to 0.", __func__, volume);
+ volume_adjusted = 0;
+ }
+
+ /*
+ /sys/class/codec/es9018_volume
+ 0 ... 255
+ */
+ if(! sysfs_set_int(SYSFS_DX90_ES9018_VOLUME, volume_adjusted))
+ {
+ DEBUGF("ERROR %s: Can not set volume.", __func__);
+ }
+}
diff --git a/firmware/target/hosted/ibasso/dx90/button-dx90.c b/firmware/target/hosted/ibasso/dx90/button-dx90.c
new file mode 100644
index 0000000000..27e4be0c1e
--- /dev/null
+++ b/firmware/target/hosted/ibasso/dx90/button-dx90.c
@@ -0,0 +1,104 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#include "config.h"
+#include "button.h"
+
+#include "button-ibasso.h"
+
+
+int handle_button_event(__u16 code, __s32 value, int last_btns)
+{
+ int button = BUTTON_NONE;
+
+ switch(code)
+ {
+ case EVENT_CODE_BUTTON_PWR:
+ {
+ button = BUTTON_POWER;
+ break;
+ }
+
+ case EVENT_CODE_BUTTON_PWR_LONG:
+ {
+ button = BUTTON_POWER_LONG;
+ break;
+ }
+
+ case EVENT_CODE_BUTTON_VOLPLUS:
+ {
+ button = BUTTON_VOL_UP;
+ break;
+ }
+
+ case EVENT_CODE_BUTTON_VOLMINUS:
+ {
+ button = BUTTON_VOL_DOWN;
+ break;
+ }
+
+ case EVENT_CODE_BUTTON_REV:
+ {
+ button = BUTTON_LEFT;
+ break;
+ }
+
+ case EVENT_CODE_BUTTON_PLAY:
+ {
+ button = BUTTON_PLAY;
+ break;
+ }
+
+ case EVENT_CODE_BUTTON_NEXT:
+ {
+ button = BUTTON_RIGHT;
+ break;
+ }
+
+ default:
+ {
+ return BUTTON_NONE;
+ }
+ }
+
+ if( (button == BUTTON_RIGHT)
+ && ((last_btns & BUTTON_LEFT) == BUTTON_LEFT)
+ && (value == EVENT_VALUE_BUTTON_RELEASE))
+ {
+ /* Workaround for a wrong feedback, only present with DX90. */
+ button = BUTTON_LEFT;
+ }
+
+ int buttons = last_btns;
+ if(value == EVENT_VALUE_BUTTON_PRESS)
+ {
+ buttons = (last_btns | button);
+ }
+ else
+ {
+ buttons = (last_btns & (~button));
+ }
+
+ return buttons;
+}
diff --git a/firmware/target/hosted/ibasso/dx90/codec-dx90.h b/firmware/target/hosted/ibasso/dx90/codec-dx90.h
new file mode 100644
index 0000000000..b96377dfec
--- /dev/null
+++ b/firmware/target/hosted/ibasso/dx90/codec-dx90.h
@@ -0,0 +1,35 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#ifndef _CODEC_DX90_H_
+#define _CODEC_DX90_H_
+
+
+#define AUDIOHW_CAPS MONO_VOL_CAP
+
+
+AUDIOHW_SETTING(VOLUME, "", 0, 1, -255, 0, -128)
+
+
+#endif
diff --git a/firmware/target/hosted/ibasso/hostfs-ibasso.c b/firmware/target/hosted/ibasso/hostfs-ibasso.c
new file mode 100644
index 0000000000..3970d06987
--- /dev/null
+++ b/firmware/target/hosted/ibasso/hostfs-ibasso.c
@@ -0,0 +1,47 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#include <unistd.h>
+
+#include "debug-ibasso.h"
+
+
+/* See hostfs.h. */
+
+
+int hostfs_init(void)
+{
+ TRACE;
+
+ return 0;
+}
+
+
+int hostfs_flush(void)
+{
+ TRACE;
+
+ sync();
+ return 0;
+}
diff --git a/firmware/target/hosted/ibasso/lcd-ibasso.c b/firmware/target/hosted/ibasso/lcd-ibasso.c
new file mode 100644
index 0000000000..4e03ba7e50
--- /dev/null
+++ b/firmware/target/hosted/ibasso/lcd-ibasso.c
@@ -0,0 +1,195 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#include <errno.h>
+#include <fcntl.h>
+#include <stdbool.h>
+#include <stdlib.h>
+#include <linux/fb.h>
+#include <sys/ioctl.h>
+#include <sys/mman.h>
+#include <sys/stat.h>
+#include <sys/types.h>
+
+#include "config.h"
+#include "debug.h"
+#include "events.h"
+#include "panic.h"
+
+#include "debug-ibasso.h"
+#include "lcd-target.h"
+#include "sysfs-ibasso.h"
+
+
+fb_data *dev_fb = 0;
+
+
+/* Framebuffer device handle. */
+static int dev_fd = 0;
+
+
+void lcd_init_device(void)
+{
+ TRACE;
+
+ dev_fd = open("/dev/graphics/fb0", O_RDWR);
+ if(dev_fd == -1)
+ {
+ DEBUGF("ERROR %s: open failed on /dev/graphics/fb0, errno: %d.", __func__, errno);
+ exit(errno);
+ }
+
+ /* Get the changeable information. */
+ struct fb_var_screeninfo vinfo;
+ if(ioctl(dev_fd, FBIOGET_VSCREENINFO, &vinfo) < 0)
+ {
+ DEBUGF("ERROR %s: ioctl FBIOGET_VSCREENINFO failed on /dev/graphics/fb0, errno: %d.", __func__, errno);
+ exit(errno);
+ }
+
+ DEBUGF("DEBUG %s: bits_per_pixel: %u, width: %u, height: %u.", __func__, vinfo.bits_per_pixel, vinfo.width, vinfo.height);
+
+ /*
+ Framebuffer does not fit the screen, a bug of iBassos Firmware, not Rockbox.
+ Cannot be solved with parameters.
+ */
+ /*vinfo.bits_per_pixel = LCD_DEPTH;
+ vinfo.xres = LCD_WIDTH;
+ vinfo.xres_virtual = LCD_WIDTH;
+ vinfo.width = LCD_WIDTH;
+ vinfo.yres = LCD_HEIGHT;
+ vinfo.yres_virtual = LCD_HEIGHT;
+ vinfo.height = LCD_HEIGHT;
+ vinfo.activate = FB_ACTIVATE_NOW;
+ if(ioctl(dev_fd, FBIOPUT_VSCREENINFO, &vinfo) == -1)
+ {
+ DEBUGF("ERROR %s: ioctl FBIOPUT_VSCREENINFO failed on /dev/graphics/fb0, errno: %d.", __func__, errno);
+ exit(EXIT_FAILURE);
+ }*/
+
+
+ /* Sanity check: Does framebuffer config match Rockbox config? */
+ size_t screensize = vinfo.xres * vinfo.yres * vinfo.bits_per_pixel / 8;
+ if(screensize != FRAMEBUFFER_SIZE)
+ {
+ DEBUGF("ERROR %s: Screen size does not match config: %d != %d.", __func__, screensize, FRAMEBUFFER_SIZE);
+ exit(EXIT_FAILURE);
+ }
+
+ /* Map the device to memory. */
+ dev_fb = mmap(0, FRAMEBUFFER_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, dev_fd, 0);
+ if(dev_fb == MAP_FAILED)
+ {
+ DEBUGF("ERROR %s: mmap failed on /dev/graphics/fb0, errno: %d.", __func__, errno);
+ exit(errno);
+ }
+
+ /* Activate Rockbox LCD. */
+ lcd_enable(true);
+}
+
+
+void lcd_shutdown(void)
+{
+ TRACE;
+
+ lcd_set_active(false);
+ munmap(dev_fb, FRAMEBUFFER_SIZE);
+ close(dev_fd) ;
+}
+
+
+/*
+ Left as reference. Unblanking does not work as expected, will not enable LCD after a few
+ seconds of power down.
+ Instead the backlight power is toggled.
+*/
+/*void lcd_power_on(void)
+{
+ TRACE;
+
+ if(ioctl(dev_fd, FBIOBLANK, VESA_NO_BLANKING) == -1)
+ {
+ DEBUGF("ERROR %s: ioctl FBIOBLANK failed on /dev/graphics/fb0, errno: %d.", __func__, errno);
+ panicf("ERROR %s: ioctl FBIOBLANK failed on /dev/graphics/fb0, errno: %d.", __func__, errno);
+ return;
+ }
+
+ lcd_set_active(true);
+ send_event(LCD_EVENT_ACTIVATION, NULL);
+}
+
+
+void lcd_power_off(void)
+{
+ TRACE;
+
+ lcd_set_active(false);
+
+ if(ioctl(dev_fd, FBIOBLANK, VESA_POWERDOWN) == -1)
+ {
+ DEBUGF("ERROR %s: ioctl FBIOBLANK failed on /dev/graphics/fb0, errno: %d.", __func__, errno);
+ panicf("ERROR %s: ioctl FBIOBLANK failed on /dev/graphics/fb0, errno: %d.", __func__, errno);
+ return;
+ }
+}*/
+
+
+void lcd_enable(bool on)
+{
+ TRACE;
+
+ lcd_set_active(on);
+
+ if(on)
+ {
+ /*
+ /sys/power/state
+ on: Cancel suspend.
+ */
+ if(! sysfs_set_string(SYSFS_POWER_STATE, "on"))
+ {
+ DEBUGF("ERROR %s: Can not set power state.", __func__);
+ }
+
+ send_event(LCD_EVENT_ACTIVATION, NULL);
+ }
+}
+
+
+void lcd_sleep(void)
+{
+ TRACE;
+
+ /*
+ See system_init(). Without suspend blocker und mute prevention this will interrupt playback.
+ Essentially, we are turning off the touch screen.
+ /sys/power/state
+ mem: Suspend to RAM.
+ */
+ if(! sysfs_set_string(SYSFS_POWER_STATE, "mem"))
+ {
+ DEBUGF("ERROR %s: Can not set power state.", __func__);
+ }
+}
diff --git a/firmware/target/hosted/ibasso/lcd-target.h b/firmware/target/hosted/ibasso/lcd-target.h
new file mode 100644
index 0000000000..50cc92599d
--- /dev/null
+++ b/firmware/target/hosted/ibasso/lcd-target.h
@@ -0,0 +1,44 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#ifndef __LCD_TARGET_H__
+#define __LCD_TARGET_H__
+
+
+#include "lcd.h"
+
+
+/*
+ Framebuffer device and framebuffer access.
+ See lcd-memframe.c
+*/
+extern fb_data *dev_fb;
+#define LCD_FRAMEBUF_ADDR(col, row) (dev_fb + row * LCD_WIDTH + col)
+
+
+/* See lcd-memframe.c */
+extern void lcd_set_active(bool active);
+
+
+#endif
diff --git a/firmware/target/hosted/ibasso/pcm-ibasso.c b/firmware/target/hosted/ibasso/pcm-ibasso.c
new file mode 100644
index 0000000000..14ef298af0
--- /dev/null
+++ b/firmware/target/hosted/ibasso/pcm-ibasso.c
@@ -0,0 +1,488 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#include <pthread.h>
+#include <stdbool.h>
+#include <unistd.h>
+
+#include "config.h"
+#include "debug.h"
+#include "panic.h"
+#include "pcm.h"
+#include "pcm-internal.h"
+
+#include "sound/asound.h"
+#include "tinyalsa/asoundlib.h"
+
+#include "debug-ibasso.h"
+#include "sysfs-ibasso.h"
+
+
+/* Tiny alsa handle. */
+static struct pcm* _alsa_handle = NULL;
+
+
+/* Bytes left in the Rockbox PCM frame buffer. */
+static size_t _pcm_buffer_size = 0;
+
+
+/* Rockbox PCM frame buffer. */
+static const void *_pcm_buffer = NULL;
+
+
+/*
+ 1: PCM thread suspended.
+ 0: PCM thread running.
+ These are used by pcm_play_[lock|unlock] or pcm_play_dma_[start|stop|pause]. These need to be
+ separated because of nested calls for locking and stopping.
+*/
+static volatile sig_atomic_t _dma_stopped = 1;
+static volatile sig_atomic_t _dma_locked = 1;
+
+
+/* Mutex for PCM thread suspend/unsuspend. */
+static pthread_mutex_t _dma_suspended_mtx = PTHREAD_MUTEX_INITIALIZER;
+
+
+/* Signal condition for PCM thread suspend/unsuspend. */
+static pthread_cond_t _dma_suspended_cond = PTHREAD_COND_INITIALIZER;
+
+
+static void* pcm_thread_run(void* nothing)
+{
+ (void) nothing;
+
+ DEBUGF("DEBUG %s: Thread start.", __func__);
+
+ while(true)
+ {
+ pthread_mutex_lock(&_dma_suspended_mtx);
+ while((_dma_stopped == 1) || (_dma_locked == 1))
+ {
+ DEBUGF("DEBUG %s: Playback suspended.", __func__);
+ pthread_cond_wait(&_dma_suspended_cond, &_dma_suspended_mtx);
+ DEBUGF("DEBUG %s: Playback resumed.", __func__);
+ }
+ pthread_mutex_unlock(&_dma_suspended_mtx);
+
+ if(_pcm_buffer_size == 0)
+ {
+ /* Retrive a new PCM buffer from Rockbox. */
+ if(! pcm_play_dma_complete_callback(PCM_DMAST_OK, &_pcm_buffer, &_pcm_buffer_size))
+ {
+ DEBUGF("DEBUG %s: No new buffer.", __func__);
+
+ usleep( 10000 );
+ continue;
+ }
+ }
+ pcm_play_dma_status_callback(PCM_DMAST_STARTED);
+
+ /* This relies on Rockbox PCM frame buffer size == ALSA PCM frame buffer size. */
+ if(pcm_write(_alsa_handle, _pcm_buffer, _pcm_buffer_size) != 0)
+ {
+ DEBUGF("ERROR %s: pcm_write failed: %s.", __func__, pcm_get_error(_alsa_handle));
+
+ usleep( 10000 );
+ continue;
+ }
+
+ _pcm_buffer_size = 0;
+
+ /*DEBUGF("DEBUG %s: Thread running.", __func__);*/
+ }
+
+ DEBUGF("DEBUG %s: Thread end.", __func__);
+
+ return 0;
+}
+
+
+#ifdef DEBUG
+
+/* https://github.com/tinyalsa/tinyalsa/blob/master/tinypcminfo.c */
+
+static const char* format_lookup[] =
+{
+ /*[0] =*/ "S8",
+ "U8",
+ "S16_LE",
+ "S16_BE",
+ "U16_LE",
+ "U16_BE",
+ "S24_LE",
+ "S24_BE",
+ "U24_LE",
+ "U24_BE",
+ "S32_LE",
+ "S32_BE",
+ "U32_LE",
+ "U32_BE",
+ "FLOAT_LE",
+ "FLOAT_BE",
+ "FLOAT64_LE",
+ "FLOAT64_BE",
+ "IEC958_SUBFRAME_LE",
+ "IEC958_SUBFRAME_BE",
+ "MU_LAW",
+ "A_LAW",
+ "IMA_ADPCM",
+ "MPEG",
+ /*[24] =*/ "GSM",
+ [31] = "SPECIAL",
+ "S24_3LE",
+ "S24_3BE",
+ "U24_3LE",
+ "U24_3BE",
+ "S20_3LE",
+ "S20_3BE",
+ "U20_3LE",
+ "U20_3BE",
+ "S18_3LE",
+ "S18_3BE",
+ "U18_3LE",
+ /*[43] =*/ "U18_3BE"
+};
+
+
+static const char* pcm_get_format_name(unsigned int bit_index)
+{
+ return(bit_index < 43 ? format_lookup[bit_index] : NULL);
+}
+
+#endif
+
+
+/* Thread that copies the Rockbox PCM buffer to ALSA. */
+static pthread_t _pcm_thread;
+
+
+/* ALSA card and device. */
+static const unsigned int CARD = 0;
+static const unsigned int DEVICE = 0;
+
+
+/* ALSA config. */
+static struct pcm_config _config;
+
+
+void pcm_play_dma_init(void)
+{
+ TRACE;
+
+#ifdef DEBUG
+
+ /*
+ DEBUG pcm_play_dma_init: Access: 0x000009
+ DEBUG pcm_play_dma_init: Format[0]: 0x000044
+ DEBUG pcm_play_dma_init: Format[1]: 0x000010
+ DEBUG pcm_play_dma_init: Format: S16_LE
+ DEBUG pcm_play_dma_init: Format: S24_LE
+ DEBUG pcm_play_dma_init: Format: S20_3LE
+ DEBUG pcm_play_dma_init: Subformat: 0x000001
+ DEBUG pcm_play_dma_init: Rate: min = 8000Hz, max = 192000Hz
+ DEBUG pcm_play_dma_init: Channels: min = 2, max = 2
+ DEBUG pcm_play_dma_init: Sample bits: min=16, max=32
+ DEBUG pcm_play_dma_init: Period size: min=8, max=10922
+ DEBUG pcm_play_dma_init: Period count: min=3, max=128
+ DEBUG pcm_play_dma_init: 0 mixer controls.
+ */
+
+ struct pcm_params* params = pcm_params_get(CARD, DEVICE, PCM_OUT);
+ if(params == NULL)
+ {
+ DEBUGF("ERROR %s: Card/device does not exist.", __func__);
+ panicf("ERROR %s: Card/device does not exist.", __func__);
+ return;
+ }
+
+ struct pcm_mask* m = pcm_params_get_mask(params, PCM_PARAM_ACCESS);
+ if(m)
+ {
+ DEBUGF("DEBUG %s: Access: %#08x", __func__, m->bits[0]);
+ }
+
+ m = pcm_params_get_mask(params, PCM_PARAM_FORMAT);
+ if(m)
+ {
+ DEBUGF("DEBUG %s: Format[0]: %#08x", __func__, m->bits[0]);
+ DEBUGF("DEBUG %s: Format[1]: %#08x", __func__, m->bits[1]);
+
+ unsigned int j;
+ unsigned int k;
+ const unsigned int bitcount = sizeof(m->bits[0]) * 8;
+ for(k = 0; k < 2; ++k)
+ {
+ for(j = 0; j < bitcount; ++j)
+ {
+ const char* name;
+ if(m->bits[k] & (1 << j))
+ {
+ name = pcm_get_format_name(j + (k * bitcount));
+ if(name)
+ {
+ DEBUGF("DEBUG %s: Format: %s", __func__, name);
+ }
+ }
+ }
+ }
+ }
+
+ m = pcm_params_get_mask(params, PCM_PARAM_SUBFORMAT);
+ if(m)
+ {
+ DEBUGF("DEBUG %s: Subformat: %#08x", __func__, m->bits[0]);
+ }
+
+ unsigned int min = pcm_params_get_min(params, PCM_PARAM_RATE);
+ unsigned int max = pcm_params_get_max(params, PCM_PARAM_RATE) ;
+ DEBUGF("DEBUG %s: Rate: min = %uHz, max = %uHz", __func__, min, max);
+
+ min = pcm_params_get_min(params, PCM_PARAM_CHANNELS);
+ max = pcm_params_get_max(params, PCM_PARAM_CHANNELS);
+ DEBUGF("DEBUG %s: Channels: min = %u, max = %u", __func__, min, max);
+
+ min = pcm_params_get_min(params, PCM_PARAM_SAMPLE_BITS);
+ max = pcm_params_get_max(params, PCM_PARAM_SAMPLE_BITS);
+ DEBUGF("DEBUG %s: Sample bits: min=%u, max=%u", __func__, min, max);
+
+ min = pcm_params_get_min(params, PCM_PARAM_PERIOD_SIZE);
+ max = pcm_params_get_max(params, PCM_PARAM_PERIOD_SIZE);
+ DEBUGF("DEBUG %s: Period size: min=%u, max=%u", __func__, min, max);
+
+ min = pcm_params_get_min(params, PCM_PARAM_PERIODS);
+ max = pcm_params_get_max(params, PCM_PARAM_PERIODS);
+ DEBUGF("DEBUG %s: Period count: min=%u, max=%u", __func__, min, max);
+
+ pcm_params_free(params);
+
+ struct mixer* mixer = mixer_open(CARD);
+ if(! mixer)
+ {
+ DEBUGF("ERROR %s: Failed to open mixer.", __func__);
+ }
+ else
+ {
+ int num_ctls = mixer_get_num_ctls(mixer);
+
+ DEBUGF("DEBUG %s: %d mixer controls.", __func__, num_ctls);
+
+ mixer_close(mixer);
+ }
+
+#endif
+
+ if(_alsa_handle != NULL)
+ {
+ DEBUGF("ERROR %s: Allready initialized.", __func__);
+ panicf("ERROR %s: Allready initialized.", __func__);
+ return;
+ }
+
+ /*
+ Rockbox outputs 16 Bit/44.1kHz stereo by default.
+
+ ALSA frame buffer size = config.period_count * config.period_size * config.channels * (16 \ 8)
+ = 4 * 256 * 2 * 2
+ = 4096
+ = Rockbox PCM buffer size
+ pcm_thread_run relies on this size match. See pcm_mixer.h.
+ */
+ _config.channels = 2;
+ _config.rate = 44100;
+ _config.period_size = 256;
+ _config.period_count = 4;
+ _config.format = PCM_FORMAT_S16_LE;
+ _config.start_threshold = 0;
+ _config.stop_threshold = 0;
+ _config.silence_threshold = 0;
+
+ _alsa_handle = pcm_open(CARD, DEVICE, PCM_OUT, &_config);
+ if(! pcm_is_ready(_alsa_handle))
+ {
+ DEBUGF("ERROR %s: pcm_open failed: %s.", __func__, pcm_get_error(_alsa_handle));
+ panicf("ERROR %s: pcm_open failed: %s.", __func__, pcm_get_error(_alsa_handle));
+ return;
+ }
+
+ DEBUGF("DEBUG %s: ALSA PCM frame buffer size: %d.", __func__, pcm_frames_to_bytes(_alsa_handle, pcm_get_buffer_size(_alsa_handle)));
+
+ /* Create pcm thread in the suspended state. */
+ pthread_mutex_lock(&_dma_suspended_mtx);
+ _dma_stopped = 1;
+ _dma_locked = 1;
+ pthread_create(&_pcm_thread, NULL, pcm_thread_run, NULL);
+ pthread_mutex_unlock(&_dma_suspended_mtx);
+}
+
+
+void pcm_play_dma_start(const void *addr, size_t size)
+{
+ TRACE;
+
+ /*
+ DX50
+ /sys/class/codec/mute
+ Mute: echo 'A' > /sys/class/codec/mute
+ Unmute: echo 'B' > /sys/class/codec/mute
+
+ DX90?
+ */
+ if(! sysfs_set_char(SYSFS_MUTE, 'B'))
+ {
+ DEBUGF("ERROR %s: Could not unmute.", __func__);
+ panicf("ERROR %s: Could not unmute.", __func__);
+ }
+
+ _pcm_buffer = addr;
+ _pcm_buffer_size = size;
+
+ pthread_mutex_lock(&_dma_suspended_mtx);
+ _dma_stopped = 0;
+ pthread_cond_signal(&_dma_suspended_cond);
+ pthread_mutex_unlock(&_dma_suspended_mtx);
+}
+
+
+/* TODO: Why is this in the API if it gets never called? */
+void pcm_play_dma_pause(bool pause)
+{
+ TRACE;
+
+ pthread_mutex_lock(&_dma_suspended_mtx);
+ _dma_stopped = pause ? 1 : 0;
+ if(_dma_stopped == 0)
+ {
+ pthread_cond_signal(&_dma_suspended_cond);
+ }
+ pthread_mutex_unlock(&_dma_suspended_mtx);
+}
+
+
+void pcm_play_dma_stop(void)
+{
+ TRACE;
+
+ pthread_mutex_lock(&_dma_suspended_mtx);
+ _dma_stopped = 1;
+ pcm_stop(_alsa_handle);
+ pthread_mutex_unlock(&_dma_suspended_mtx);
+}
+
+
+/* Unessecary play locks before pcm_play_dma_postinit. */
+static int _play_lock_recursion_count = -10000;
+
+
+void pcm_play_dma_postinit(void)
+{
+ TRACE;
+
+ _play_lock_recursion_count = 0;
+}
+
+
+void pcm_play_lock(void)
+{
+ TRACE;
+
+ ++_play_lock_recursion_count;
+
+ if(_play_lock_recursion_count == 1)
+ {
+ pthread_mutex_lock(&_dma_suspended_mtx);
+ _dma_locked = 1;
+ pthread_mutex_unlock(&_dma_suspended_mtx);
+ }
+}
+
+
+void pcm_play_unlock(void)
+{
+ TRACE;
+
+ --_play_lock_recursion_count;
+
+ if(_play_lock_recursion_count == 0)
+ {
+ pthread_mutex_lock(&_dma_suspended_mtx);
+ _dma_locked = 0;
+ pthread_cond_signal(&_dma_suspended_cond);
+ pthread_mutex_unlock(&_dma_suspended_mtx);
+ }
+}
+
+
+void pcm_dma_apply_settings(void)
+{
+ unsigned int rate = pcm_get_frequency();
+
+ DEBUGF("DEBUG %s: Current sample rate: %u, next sampe rate: %u.", __func__, _config.rate, rate);
+
+ if(( _config.rate != rate) && (rate >= 8000) && (rate <= 192000))
+ {
+ _config.rate = rate;
+
+ pcm_close(_alsa_handle);
+ _alsa_handle = pcm_open(CARD, DEVICE, PCM_OUT, &_config);
+
+ if(! pcm_is_ready(_alsa_handle))
+ {
+ DEBUGF("ERROR %s: pcm_open failed: %s.", __func__, pcm_get_error(_alsa_handle));
+ panicf("ERROR %s: pcm_open failed: %s.", __func__, pcm_get_error(_alsa_handle));
+ }
+ }
+}
+
+
+size_t pcm_get_bytes_waiting(void)
+{
+ TRACE;
+
+ return _pcm_buffer_size;
+}
+
+
+/* TODO: WTF */
+const void* pcm_play_dma_get_peak_buffer(int* count)
+{
+ TRACE;
+
+ uintptr_t addr = (uintptr_t) _pcm_buffer;
+ *count = _pcm_buffer_size / 4;
+ return (void*) ((addr + 3) & ~3);
+}
+
+
+void pcm_close_device(void)
+{
+ TRACE;
+
+ pthread_mutex_lock(&_dma_suspended_mtx);
+ _dma_stopped = 1;
+ pthread_mutex_unlock(&_dma_suspended_mtx);
+
+ pcm_close(_alsa_handle);
+ _alsa_handle = NULL;
+}
diff --git a/firmware/target/hosted/ibasso/pcm-ibasso.h b/firmware/target/hosted/ibasso/pcm-ibasso.h
new file mode 100644
index 0000000000..588c4dfb9b
--- /dev/null
+++ b/firmware/target/hosted/ibasso/pcm-ibasso.h
@@ -0,0 +1,33 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#ifndef __PCM_IBASSO_H__
+#define __PCM_IBASSO_H__
+
+
+/* Clean up the audio device handler. */
+void pcm_close_device(void);
+
+
+#endif
diff --git a/firmware/target/hosted/ibasso/power-ibasso.c b/firmware/target/hosted/ibasso/power-ibasso.c
new file mode 100644
index 0000000000..8257de5f33
--- /dev/null
+++ b/firmware/target/hosted/ibasso/power-ibasso.c
@@ -0,0 +1,97 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#include <stdbool.h>
+#include <stdlib.h>
+#include <sys/reboot.h>
+
+#include "config.h"
+#include "debug.h"
+#include "power.h"
+
+#include "button-ibasso.h"
+#include "debug-ibasso.h"
+#include "pcm-ibasso.h"
+#include "sysfs-ibasso.h"
+#include "vold-ibasso.h"
+
+
+unsigned int power_input_status(void)
+{
+ /*TRACE;*/
+
+ /*
+ /sys/class/power_supply/usb/present
+ 0: No external power supply connected.
+ 1: External power supply connected.
+ */
+ int val = 0;
+ if(! sysfs_get_int(SYSFS_USB_POWER_PRESENT, &val))
+ {
+ DEBUGF("ERROR %s: Can not get power supply status.", __func__);
+ return POWER_INPUT_NONE;
+ }
+
+ return val ? POWER_INPUT_USB_CHARGER : POWER_INPUT_NONE;
+}
+
+
+void power_off(void)
+{
+ TRACE;
+
+ button_close_device();
+
+ if(vold_monitor_forced_close_imminent())
+ {
+ /*
+ We are here, because Android Vold is going to kill Rockbox. Instead of powering off,
+ we exit into the loader.
+ */
+ DEBUGF("DEBUG %s: Exit Rockbox.", __func__);
+ exit(42);
+ }
+
+ reboot(RB_POWER_OFF);
+}
+
+
+/* Returns true, if battery is charging, false else. */
+bool charging_state(void)
+{
+ /*TRACE;*/
+
+ /*
+ /sys/class/power_supply/battery/status
+ "Full", "Charging", "Discharging"
+ */
+ char state[9];
+ if(! sysfs_get_string(SYSFS_BATTERY_STATUS, state, 9))
+ {
+ DEBUGF("ERROR %s: Can not get battery charging state.", __func__);
+ return false;
+ }
+
+ return(strcmp(state, "Charging") == 0);;
+}
diff --git a/firmware/target/hosted/ibasso/powermgmt-ibasso.c b/firmware/target/hosted/ibasso/powermgmt-ibasso.c
new file mode 100644
index 0000000000..7df0064097
--- /dev/null
+++ b/firmware/target/hosted/ibasso/powermgmt-ibasso.c
@@ -0,0 +1,122 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#include <stdio.h>
+
+#include "config.h"
+#include "debug.h"
+#include "panic.h"
+
+#include "debug-ibasso.h"
+#include "sysfs-ibasso.h"
+
+
+/* Based on batterymonitor with PISEN and Samsung SIII battery. */
+
+
+const unsigned short battery_level_dangerous[BATTERY_TYPES_COUNT] =
+{
+ 3600
+};
+
+
+const unsigned short battery_level_shutoff[BATTERY_TYPES_COUNT] =
+{
+ 3500
+};
+
+
+/*
+ Averages at percent of running time from five measuremnts with PISEN and Samsung SIII battery
+ during normal usage.
+
+ Mongo default values (?)
+ < 3660 (0%), < 3730 (1% - 10%), < 3780 (11% - 20%), < 3830 (21% - 40%), < 3950 (41% - 60%),
+ < 4080 (61% - 80%), > 4081 (81% - 100%)
+*/
+const unsigned short percent_to_volt_discharge[BATTERY_TYPES_COUNT][11] =
+{
+ { 3522, 3660, 3720, 3752, 3784, 3827, 3896, 3978, 4072, 4168, 4255 }
+};
+
+
+/* Copied from percent_to_volt_discharge. */
+const unsigned short percent_to_volt_charge[11] =
+{
+ 3500, 3544, 3578, 3623, 3660, 3773, 3782, 3853, 3980, 4130, 4360
+};
+
+
+static int _battery_present = -1;
+
+
+int _battery_voltage(void)
+{
+ /*TRACE;*/
+
+ if( (_battery_present == -1)
+ && (! sysfs_get_int(SYSFS_BATTERY_PRESENT, &_battery_present)))
+ {
+ /* This check is only done once at startup. */
+
+ DEBUGF("ERROR %s: Can not get current battery availabilty.", __func__);
+ _battery_present = 1;
+ }
+
+ int val;
+
+ if(_battery_present == 1)
+ {
+ /* Battery is present. */
+
+ /*
+ /sys/class/power_supply/battery/voltage_now
+ Voltage in microvolt.
+ */
+ if(! sysfs_get_int(SYSFS_BATTERY_VOLTAGE_NOW, &val))
+ {
+ DEBUGF("ERROR %s: Can not get current battery voltage.", __func__);
+ return 0;
+ }
+ }
+ else
+ {
+ /*
+ No battery, so we have to be running solely from USB power.
+ This will prevent Rockbox from forcing shutdown due to low power.
+ */
+
+ /*
+ /sys/class/power_supply/usb/voltage_now
+ Voltage in microvolt.
+ */
+ if(! sysfs_get_int(SYSFS_USB_POWER_VOLTAGE_NOW, &val))
+ {
+ DEBUGF("ERROR %s: Can not get current USB voltage.", __func__);
+ return 0;
+ }
+ }
+
+ return(val / 1000);
+}
diff --git a/firmware/target/hosted/ibasso/sysfs-ibasso.c b/firmware/target/hosted/ibasso/sysfs-ibasso.c
new file mode 100644
index 0000000000..8ca3edf387
--- /dev/null
+++ b/firmware/target/hosted/ibasso/sysfs-ibasso.c
@@ -0,0 +1,404 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#include <stdio.h>
+#include <string.h>
+
+#include "config.h"
+#include "debug.h"
+
+#include "debug-ibasso.h"
+#include "sysfs-ibasso.h"
+
+
+static const char* SYSFS_PATHS[] =
+{
+ /* SYSFS_DX50_CODEC_VOLUME */
+ "/dev/codec_volume",
+
+ /* SYSFS_HOLDKEY */
+ "/sys/class/axppower/holdkey",
+
+ /* SYSFS_DX90_ES9018_VOLUME */
+ "/sys/class/codec/es9018_volume",
+
+ /* SYSFS_MUTE */
+ "/sys/class/codec/mute",
+
+ /* SYSFS_WM8740_MUTE */
+ "/sys/class/codec/wm8740_mute",
+
+ /* SYSFS_BATTERY_CAPACITY */
+ "/sys/class/power_supply/battery/capacity",
+
+ /* SYSFS_BATTERY_CURRENT_NOW */
+ "/sys/class/power_supply/battery/current_now",
+
+ /* SYSFS_BATTERY_ENERGY_FULL_DESIGN */
+ "/sys/class/power_supply/battery/energy_full_design",
+
+ /* SYSFS_BATTERY_HEALTH */
+ "/sys/class/power_supply/battery/health",
+
+ /* SYSFS_BATTERY_MODEL_NAME */
+ "/sys/class/power_supply/battery/model_name",
+
+ /* SYSFS_BATTERY_ONLINE */
+ "/sys/class/power_supply/battery/online",
+
+ /* SYSFS_BATTERY_PRESENT */
+ "/sys/class/power_supply/battery/present",
+
+ /* SYSFS_BATTERY_STATUS */
+ "/sys/class/power_supply/battery/status",
+
+ /* SYSFS_BATTERY_TECHNOLOGY */
+ "/sys/class/power_supply/battery/technology",
+
+ /* SYSFS_BATTERY_TEMP */
+ "/sys/class/power_supply/battery/temp",
+
+ /* SYSFS_BATTERY_TYPE */
+ "/sys/class/power_supply/battery/type",
+
+ /* SYSFS_BATTERY_VOLTAGE_MAX_DESIGN */
+ "/sys/class/power_supply/battery/voltage_max_design",
+
+ /* SYSFS_BATTERY_VOLTAGE_MIN_DESIGN */
+ "/sys/class/power_supply/battery/voltage_min_design",
+
+ /* SYSFS_BATTERY_VOLTAGE_NOW */
+ "/sys/class/power_supply/battery/voltage_now",
+
+ /* SYSFS_USB_POWER_CURRENT_NOW */
+ "/sys/class/power_supply/usb/current_now",
+
+ /* SYSFS_USB_POWER_ONLINE */
+ "/sys/class/power_supply/usb/online",
+
+ /* SYSFS_USB_POWER_PRESENT */
+ "/sys/class/power_supply/usb/present",
+
+ /* SYSFS_USB_POWER_VOLTAGE_NOW */
+ "/sys/class/power_supply/usb/voltage_now",
+
+ /* SYSFS_BACKLIGHT_POWER */
+ "/sys/devices/platform/rk29_backlight/backlight/rk28_bl/bl_power",
+
+ /* SYSFS_BACKLIGHT_BRIGHTNESS */
+ "/sys/devices/platform/rk29_backlight/backlight/rk28_bl/brightness",
+
+ /* SYSFS_POWER_STATE */
+ "/sys/power/state",
+
+ /* SYSFS_POWER_WAKE_LOCK */
+ "/sys/power/wake_lock"
+};
+
+
+static FILE* open_read(const char* file_name)
+{
+ FILE *f = fopen(file_name, "r");
+ if(f == NULL)
+ {
+ DEBUGF("ERROR %s: Can not open %s for reading.", __func__, file_name);
+ }
+
+ return f;
+}
+
+
+static FILE* open_write(const char* file_name)
+{
+ FILE *f = fopen(file_name, "w");
+ if(f == NULL)
+ {
+ DEBUGF("ERROR %s: Can not open %s for writing.", __func__, file_name);
+ }
+
+ return f;
+}
+
+
+bool sysfs_get_int(enum sys_fs_interface_id id, int* value)
+{
+ *value = -1;
+
+ switch(id)
+ {
+ case SYSFS_BATTERY_CAPACITY:
+ case SYSFS_BATTERY_CURRENT_NOW:
+ case SYSFS_BATTERY_ENERGY_FULL_DESIGN:
+ case SYSFS_BATTERY_ONLINE:
+ case SYSFS_BATTERY_PRESENT:
+ case SYSFS_BATTERY_TEMP:
+ case SYSFS_BATTERY_VOLTAGE_MAX_DESIGN:
+ case SYSFS_BATTERY_VOLTAGE_MIN_DESIGN:
+ case SYSFS_BATTERY_VOLTAGE_NOW:
+ case SYSFS_USB_POWER_CURRENT_NOW:
+ case SYSFS_USB_POWER_VOLTAGE_NOW:
+ case SYSFS_USB_POWER_ONLINE:
+ case SYSFS_USB_POWER_PRESENT:
+ {
+ break;
+ }
+
+ default:
+ {
+ DEBUGF("ERROR %s: Unknown interface id: %d.", __func__, id);
+ return false;
+ }
+ }
+
+ const char* interface = SYSFS_PATHS[id];
+
+ /*DEBUGF("%s: interface: %s.", __func__, interface);*/
+
+ FILE *f = open_read(interface);
+ if(f == NULL)
+ {
+ return false;
+ }
+
+ bool success = true;
+ if(fscanf(f, "%d", value) == EOF)
+ {
+ DEBUGF("ERROR %s: Read failed for %s.", __func__, interface);
+ success = false;
+ }
+
+ fclose(f);
+ return success;
+}
+
+
+bool sysfs_set_int(enum sys_fs_interface_id id, int value)
+{
+ switch(id)
+ {
+ case SYSFS_BACKLIGHT_POWER:
+ case SYSFS_BACKLIGHT_BRIGHTNESS:
+ case SYSFS_DX50_CODEC_VOLUME:
+ case SYSFS_DX90_ES9018_VOLUME:
+ {
+ break;
+ }
+
+ default:
+ {
+ DEBUGF("ERROR %s: Unknown interface id: %d.", __func__, id);
+ return false;
+ }
+ }
+
+ const char* interface = SYSFS_PATHS[id];
+
+ /*DEBUGF("%s: interface: %s, value: %d.", __func__, interface, value);*/
+
+ FILE *f = open_write(interface);
+ if(f == NULL)
+ {
+ return false;
+ }
+
+ bool success = true;
+ if(fprintf(f, "%d", value) < 1)
+ {
+ DEBUGF("ERROR %s: Write failed for %s.", __func__, interface);
+ success = false;
+ }
+
+ fclose(f);
+ return success;
+}
+
+
+bool sysfs_get_char(enum sys_fs_interface_id id, char* value)
+{
+ *value = '\0';
+
+ switch(id)
+ {
+ case SYSFS_HOLDKEY:
+ {
+ break;
+ }
+
+ default:
+ {
+ DEBUGF("ERROR %s: Unknown interface id: %d.", __func__, id);
+ return false;
+ }
+ }
+
+ const char* interface = SYSFS_PATHS[id];
+
+ /*DEBUGF("%s: interface: %s.", __func__, interface);*/
+
+ FILE *f = open_read(interface);
+ if(f == NULL)
+ {
+ return false;
+ }
+
+ bool success = true;
+ if(fscanf(f, "%c", value) == EOF)
+ {
+ DEBUGF("ERROR %s: Read failed for %s.", __func__, interface);
+ success = false;
+ }
+
+ fclose(f);
+ return success;
+}
+
+
+bool sysfs_set_char(enum sys_fs_interface_id id, char value)
+{
+ switch(id)
+ {
+ case SYSFS_MUTE:
+ case SYSFS_WM8740_MUTE:
+ {
+ break;
+ }
+
+ default:
+ {
+ DEBUGF("ERROR %s: Unknown interface id: %d.", __func__, id);
+ return false;
+ }
+ }
+
+ const char* interface = SYSFS_PATHS[id];
+
+ /*DEBUGF("%s: interface: %s, value: %c.", __func__, interface, value);*/
+
+ FILE *f = open_write(interface);
+ if(f == NULL)
+ {
+ return false;
+ }
+
+ bool success = true;
+ if(fprintf(f, "%c", value) < 1)
+ {
+ DEBUGF("ERROR %s: Write failed for %s.", __func__, interface);
+ success = false;
+ }
+
+ fclose(f);
+ return success;
+}
+
+
+bool sysfs_get_string(enum sys_fs_interface_id id, char* value, int size)
+{
+ value[0] = '\0';
+
+ switch(id)
+ {
+ case SYSFS_BATTERY_STATUS:
+ case SYSFS_BATTERY_HEALTH:
+ case SYSFS_BATTERY_MODEL_NAME:
+ case SYSFS_BATTERY_TECHNOLOGY:
+ case SYSFS_BATTERY_TYPE:
+ {
+ break;
+ }
+
+ default:
+ {
+ DEBUGF("ERROR %s: Unknown interface id: %d.", __func__, id);
+ return false;
+ }
+ }
+
+ const char* interface = SYSFS_PATHS[id];
+
+ /*DEBUGF("%s: interface: %s, size: %d.", __func__, interface, size);*/
+
+ FILE *f = open_read(interface);
+ if(f == NULL)
+ {
+ return false;
+ }
+
+ bool success = true;
+ if(fgets(value, size, f) == NULL)
+ {
+ DEBUGF("ERROR %s: Read failed for %s.", __func__, interface);
+ success = false;
+ }
+ else
+ {
+ size_t length = strlen(value);
+ if((length > 0) && value[length - 1] == '\n')
+ {
+ value[length - 1] = '\0';
+ }
+ }
+
+ fclose(f);
+ return success;
+}
+
+
+bool sysfs_set_string(enum sys_fs_interface_id id, char* value)
+{
+ switch(id)
+ {
+ case SYSFS_POWER_STATE:
+ case SYSFS_POWER_WAKE_LOCK:
+ {
+ break;
+ }
+
+ default:
+ {
+ DEBUGF("ERROR %s: Unknown interface id: %d.", __func__, id);
+ return false;
+ }
+ }
+
+ const char* interface = SYSFS_PATHS[id];
+
+ /*DEBUGF("%s: interface: %s, value: %s.", __func__, interface, value);*/
+
+ FILE *f = open_write(interface);
+ if(f == NULL)
+ {
+ return false;
+ }
+
+ bool success = true;
+ if(fprintf(f, "%s", value) < 1)
+ {
+ DEBUGF("ERROR %s: Write failed for %s.", __func__, interface);
+ success = false;
+ }
+
+ fclose(f);
+ return success;
+}
diff --git a/firmware/target/hosted/ibasso/sysfs-ibasso.h b/firmware/target/hosted/ibasso/sysfs-ibasso.h
new file mode 100644
index 0000000000..fec8a082f9
--- /dev/null
+++ b/firmware/target/hosted/ibasso/sysfs-ibasso.h
@@ -0,0 +1,111 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#ifndef _SYSFS_IBASSO_H_
+#define _SYSFS_IBASSO_H_
+
+
+#include <stdbool.h>
+
+
+/*
+ Sys FS path identifiers.
+ See SYSFS_PATHS in sysfs-ibasso.c.
+*/
+enum sys_fs_interface_id
+{
+ SYSFS_DX50_CODEC_VOLUME = 0,
+ SYSFS_HOLDKEY,
+ SYSFS_DX90_ES9018_VOLUME,
+ SYSFS_MUTE,
+ SYSFS_WM8740_MUTE,
+ SYSFS_BATTERY_CAPACITY,
+ SYSFS_BATTERY_CURRENT_NOW,
+ SYSFS_BATTERY_ENERGY_FULL_DESIGN,
+ SYSFS_BATTERY_HEALTH,
+ SYSFS_BATTERY_MODEL_NAME,
+ SYSFS_BATTERY_ONLINE,
+ SYSFS_BATTERY_PRESENT,
+ SYSFS_BATTERY_STATUS,
+ SYSFS_BATTERY_TECHNOLOGY,
+ SYSFS_BATTERY_TEMP,
+ SYSFS_BATTERY_TYPE,
+ SYSFS_BATTERY_VOLTAGE_MAX_DESIGN,
+ SYSFS_BATTERY_VOLTAGE_MIN_DESIGN,
+ SYSFS_BATTERY_VOLTAGE_NOW,
+ SYSFS_USB_POWER_CURRENT_NOW,
+ SYSFS_USB_POWER_ONLINE,
+ SYSFS_USB_POWER_PRESENT,
+ SYSFS_USB_POWER_VOLTAGE_NOW,
+ SYSFS_BACKLIGHT_POWER,
+ SYSFS_BACKLIGHT_BRIGHTNESS,
+ SYSFS_POWER_STATE,
+ SYSFS_POWER_WAKE_LOCK
+};
+
+
+/*
+ Read a integer value from the sys fs interface given by id.
+ Returns true on success, false else.
+*/
+bool sysfs_get_int(enum sys_fs_interface_id id, int* value);
+
+
+/*
+ Write a integer value to the sys fs interface given by id.
+ Returns true on success, false else.
+*/
+bool sysfs_set_int(enum sys_fs_interface_id id, int value);
+
+
+/*
+ Read a char value from the sys fs interface given by id.
+ Returns true on success, false else.
+*/
+bool sysfs_get_char(enum sys_fs_interface_id id, char* value);
+
+
+/*
+ Write a char value to the sys fs interface given by id.
+ Returns true on success, false else.
+*/
+bool sysfs_set_char(enum sys_fs_interface_id id, char value);
+
+/*
+ Read a single line of text from the sys fs interface given by id.
+ A newline will be discarded.
+ size: The size of value.
+ Returns true on success, false else.
+*/
+bool sysfs_get_string(enum sys_fs_interface_id id, char* value, int size);
+
+
+/*
+ Write text to the sys fs interface given by id.
+ Returns true on success, false else.
+*/
+bool sysfs_set_string(enum sys_fs_interface_id id, char* value);
+
+
+#endif
diff --git a/firmware/target/hosted/ibasso/system-ibasso.c b/firmware/target/hosted/ibasso/system-ibasso.c
new file mode 100644
index 0000000000..00f8669ae0
--- /dev/null
+++ b/firmware/target/hosted/ibasso/system-ibasso.c
@@ -0,0 +1,101 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#include <stdint.h>
+#include <stdlib.h>
+#include <sys/reboot.h>
+
+#include "config.h"
+#include "cpufreq-linux.h"
+#include "debug.h"
+
+#include "button-ibasso.h"
+#include "debug-ibasso.h"
+#include "sysfs-ibasso.h"
+#include "usb-ibasso.h"
+#include "vold-ibasso.h"
+
+
+/* Fake stack. */
+uintptr_t* stackbegin;
+uintptr_t* stackend;
+
+
+void system_init(void)
+{
+ TRACE;
+
+ /* Fake stack. */
+ volatile uintptr_t stack = 0;
+ stackbegin = stackend = (uintptr_t*) &stack;
+
+ cpufreq_set_governor("powersave", CPUFREQ_ALL_CPUS);
+ vold_monitor_start();
+ ibasso_set_usb_mode(USB_MODE_MASS_STORAGE);
+
+ /*
+ Prevent device from deep sleeping, which will interrupt playback.
+ /sys/power/wake_lock
+ */
+ if(! sysfs_set_string(SYSFS_POWER_WAKE_LOCK, "rockbox"))
+ {
+ DEBUGF("ERROR %s: Can not set suspend blocker.", __func__);
+ }
+
+ /*
+ Prevent device to mute, which will cause tinyalsa pcm_writes to fail.
+ /sys/class/codec/wm8740_mute
+ */
+ if(! sysfs_set_char(SYSFS_WM8740_MUTE, '0'))
+ {
+ DEBUGF("ERROR %s: Can not set WM8740 lock.", __func__);
+ }
+}
+
+
+void system_reboot(void)
+{
+ TRACE;
+
+ button_close_device();
+
+ if(vold_monitor_forced_close_imminent())
+ {
+ /*
+ We are here, because Android Vold is going to kill Rockbox. Instead of powering off,
+ we exit into the loader.
+ */
+ exit(42);
+ }
+
+ reboot(RB_AUTOBOOT);
+}
+
+
+void system_exception_wait(void)
+{
+ TRACE;
+
+ while(1) {};
+}
diff --git a/firmware/target/hosted/android/dx50/lcd-target.h b/firmware/target/hosted/ibasso/system-target.h
index 900350eca2..17b1238380 100644
--- a/firmware/target/hosted/android/dx50/lcd-target.h
+++ b/firmware/target/hosted/ibasso/system-target.h
@@ -1,11 +1,15 @@
/***************************************************************************
- * __________ __ ___.
+ * __________ __ ___
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
- * $Id$
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
@@ -17,13 +21,13 @@
*
****************************************************************************/
-#ifndef __LCD_TARGET_H__
-#define __LCD_TARGET_H__
-extern fb_data *dev_fb;
-#define LCD_FRAMEBUF_ADDR(col, row) (dev_fb + row*LCD_WIDTH + col)
-#ifdef HAVE_LCD_ENABLE
-extern void lcd_set_active(bool active);
-extern void lcd_enable(bool enable);
-#endif
+#ifndef __SYSTEM_TARGET_H__
+#define __SYSTEM_TARGET_H__
+
+
+#include "kernel-unix.h"
+#include "system-hosted.h"
+
+
#endif
diff --git a/firmware/target/hosted/android/dx50/tinyalsa/asound.h b/firmware/target/hosted/ibasso/tinyalsa/include/sound/asound.h
index fc1e4f6d67..9dd66fe169 100644
--- a/firmware/target/hosted/android/dx50/tinyalsa/asound.h
+++ b/firmware/target/hosted/ibasso/tinyalsa/include/sound/asound.h
@@ -818,4 +818,3 @@ struct snd_ctl_event {
#define SNDRV_CTL_NAME_IEC958(expl,direction,what) "IEC958 " expl SNDRV_CTL_NAME_##direction SNDRV_CTL_NAME_IEC958_##what
#endif
-
diff --git a/firmware/target/hosted/android/dx50/tinyalsa/asoundlib.h b/firmware/target/hosted/ibasso/tinyalsa/include/tinyalsa/asoundlib.h
index 6aacae46d6..ba58bdcdfc 100644
--- a/firmware/target/hosted/android/dx50/tinyalsa/asoundlib.h
+++ b/firmware/target/hosted/ibasso/tinyalsa/include/tinyalsa/asoundlib.h
@@ -55,17 +55,14 @@ struct pcm;
* second call to pcm_write will attempt to
* restart the stream.
*/
+#define PCM_MONOTONIC 0x00000008 /* see pcm_get_htimestamp */
/* PCM runtime states */
-#define PCM_STATE_OPEN 0
-#define PCM_STATE_SETUP 1
-#define PCM_STATE_PREPARED 2
-#define PCM_STATE_RUNNING 3
-#define PCM_STATE_XRUN 4
-#define PCM_STATE_DRAINING 5
-#define PCM_STATE_PAUSED 6
-#define PCM_STATE_SUSPENDED 7
-#define PCM_STATE_DISCONNECTED 8
+#define PCM_STATE_RUNNING 3
+#define PCM_STATE_XRUN 4
+#define PCM_STATE_DRAINING 5
+#define PCM_STATE_SUSPENDED 7
+#define PCM_STATE_DISCONNECTED 8
/* Bit formats */
enum pcm_format {
@@ -77,6 +74,11 @@ enum pcm_format {
PCM_FORMAT_MAX,
};
+/* Bitmask has 256 bits (32 bytes) in asound.h */
+struct pcm_mask {
+ unsigned int bits[32 / sizeof(unsigned int)];
+};
+
/* Configuration for a stream */
struct pcm_config {
unsigned int channels;
@@ -101,6 +103,11 @@ struct pcm_config {
/* PCM parameters */
enum pcm_param
{
+ /* mask parameters */
+ PCM_PARAM_ACCESS,
+ PCM_PARAM_FORMAT,
+ PCM_PARAM_SUBFORMAT,
+ /* interval parameters */
PCM_PARAM_SAMPLE_BITS,
PCM_PARAM_FRAME_BITS,
PCM_PARAM_CHANNELS,
@@ -138,15 +145,14 @@ int pcm_is_ready(struct pcm *pcm);
struct pcm_params *pcm_params_get(unsigned int card, unsigned int device,
unsigned int flags);
void pcm_params_free(struct pcm_params *pcm_params);
+
+struct pcm_mask *pcm_params_get_mask(struct pcm_params *pcm_params,
+ enum pcm_param param);
unsigned int pcm_params_get_min(struct pcm_params *pcm_params,
enum pcm_param param);
unsigned int pcm_params_get_max(struct pcm_params *pcm_params,
enum pcm_param param);
-/* Set and get config */
-int pcm_get_config(struct pcm *pcm, struct pcm_config *config);
-int pcm_set_config(struct pcm *pcm, struct pcm_config *config);
-
/* Returns a human readable reason for the last error */
const char *pcm_get_error(struct pcm *pcm);
@@ -162,10 +168,9 @@ unsigned int pcm_get_buffer_size(struct pcm *pcm);
unsigned int pcm_frames_to_bytes(struct pcm *pcm, unsigned int frames);
unsigned int pcm_bytes_to_frames(struct pcm *pcm, unsigned int bytes);
-/* Returns the pcm latency in ms */
-unsigned int pcm_get_latency(struct pcm *pcm);
-
/* Returns available frames in pcm buffer and corresponding time stamp.
+ * The clock is CLOCK_MONOTONIC if flag PCM_MONOTONIC was specified in pcm_open,
+ * otherwise the clock is CLOCK_REALTIME.
* For an input stream, frames available are frames ready for the
* application to read.
* For an output stream, frames available are the number of empty frames available
@@ -185,11 +190,13 @@ int pcm_read(struct pcm *pcm, void *data, unsigned int count);
* mmap() support.
*/
int pcm_mmap_write(struct pcm *pcm, const void *data, unsigned int count);
+int pcm_mmap_read(struct pcm *pcm, void *data, unsigned int count);
int pcm_mmap_begin(struct pcm *pcm, void **areas, unsigned int *offset,
unsigned int *frames);
int pcm_mmap_commit(struct pcm *pcm, unsigned int offset, unsigned int frames);
-
+/* Prepare the PCM substream to be triggerable */
+int pcm_prepare(struct pcm *pcm);
/* Start and stop a PCM channel that doesn't transfer data */
int pcm_start(struct pcm *pcm);
int pcm_stop(struct pcm *pcm);
@@ -197,10 +204,6 @@ int pcm_stop(struct pcm *pcm);
/* Interrupt driven API */
int pcm_wait(struct pcm *pcm, int timeout);
-int pcm_avail_update(struct pcm *pcm);
-
-int pcm_state(struct pcm *pcm);
-
/*
* MIXER API
diff --git a/firmware/target/hosted/android/dx50/tinyalsa/mixer.c b/firmware/target/hosted/ibasso/tinyalsa/mixer.c
index f75dec488a..24e94f4f1d 100644
--- a/firmware/target/hosted/android/dx50/tinyalsa/mixer.c
+++ b/firmware/target/hosted/ibasso/tinyalsa/mixer.c
@@ -40,7 +40,7 @@
#define __force
#define __bitwise
#define __user
-#include <tinyalsa/asound.h>
+#include <sound/asound.h>
#include <tinyalsa/asoundlib.h>
@@ -259,14 +259,11 @@ unsigned int mixer_ctl_get_num_values(struct mixer_ctl *ctl)
static int percent_to_int(struct snd_ctl_elem_info *ei, int percent)
{
- int range;
-
- if (percent > 100)
- percent = 100;
- else if (percent < 0)
- percent = 0;
+ if ((percent > 100) || (percent < 0)) {
+ return -EINVAL;
+ }
- range = (ei->value.integer.max - ei->value.integer.min);
+ int range = (ei->value.integer.max - ei->value.integer.min);
return ei->value.integer.min + (range * percent) / 100;
}
@@ -389,6 +386,11 @@ int mixer_ctl_set_value(struct mixer_ctl *ctl, unsigned int id, int value)
break;
case SNDRV_CTL_ELEM_TYPE_INTEGER:
+ if ((value < mixer_ctl_get_range_min(ctl)) ||
+ (value > mixer_ctl_get_range_max(ctl))) {
+ return -EINVAL;
+ }
+
ev.value.integer.value[id] = value;
break;
@@ -396,6 +398,10 @@ int mixer_ctl_set_value(struct mixer_ctl *ctl, unsigned int id, int value)
ev.value.enumerated.item[id] = value;
break;
+ case SNDRV_CTL_ELEM_TYPE_BYTES:
+ ev.value.bytes.data[id] = value;
+ break;
+
default:
return -EINVAL;
}
@@ -494,4 +500,3 @@ int mixer_ctl_set_enum_by_string(struct mixer_ctl *ctl, const char *string)
return -EINVAL;
}
-
diff --git a/firmware/target/hosted/android/dx50/tinyalsa/pcm.c b/firmware/target/hosted/ibasso/tinyalsa/pcm.c
index bd44dce52f..0d2f0adf08 100644
--- a/firmware/target/hosted/android/dx50/tinyalsa/pcm.c
+++ b/firmware/target/hosted/ibasso/tinyalsa/pcm.c
@@ -44,7 +44,7 @@
#define __force
#define __bitwise
#define __user
-#include <tinyalsa/asound.h>
+#include <sound/asound.h>
#include <tinyalsa/asoundlib.h>
@@ -159,6 +159,7 @@ struct pcm {
int fd;
unsigned int flags;
int running:1;
+ int prepared:1;
int underruns;
unsigned int buffer_size;
unsigned int boundary;
@@ -300,7 +301,7 @@ static void pcm_hw_munmap_status(struct pcm *pcm) {
}
static int pcm_areas_copy(struct pcm *pcm, unsigned int pcm_offset,
- const char *src, unsigned int src_offset,
+ char *buf, unsigned int src_offset,
unsigned int frames)
{
int size_bytes = pcm_frames_to_bytes(pcm, frames);
@@ -308,12 +309,18 @@ static int pcm_areas_copy(struct pcm *pcm, unsigned int pcm_offset,
int src_offset_bytes = pcm_frames_to_bytes(pcm, src_offset);
/* interleaved only atm */
- memcpy((char*)pcm->mmap_buffer + pcm_offset_bytes,
- src + src_offset_bytes, size_bytes);
+ if (pcm->flags & PCM_IN)
+ memcpy(buf + src_offset_bytes,
+ (char*)pcm->mmap_buffer + pcm_offset_bytes,
+ size_bytes);
+ else
+ memcpy((char*)pcm->mmap_buffer + pcm_offset_bytes,
+ buf + src_offset_bytes,
+ size_bytes);
return 0;
}
-static int pcm_mmap_write_areas(struct pcm *pcm, const char *src,
+static int pcm_mmap_transfer_areas(struct pcm *pcm, char *buf,
unsigned int offset, unsigned int size)
{
void *pcm_areas;
@@ -323,7 +330,7 @@ static int pcm_mmap_write_areas(struct pcm *pcm, const char *src,
while (size > 0) {
frames = size;
pcm_mmap_begin(pcm, &pcm_areas, &pcm_offset, &frames);
- pcm_areas_copy(pcm, pcm_offset, src, offset, frames);
+ pcm_areas_copy(pcm, pcm_offset, buf, offset, frames);
commit = pcm_mmap_commit(pcm, pcm_offset, frames);
if (commit < 0) {
oops(pcm, commit, "failed to commit %d frames\n", frames);
@@ -386,14 +393,16 @@ int pcm_write(struct pcm *pcm, const void *data, unsigned int count)
for (;;) {
if (!pcm->running) {
- if (ioctl(pcm->fd, SNDRV_PCM_IOCTL_PREPARE))
- return oops(pcm, errno, "cannot prepare channel");
+ int prepare_error = pcm_prepare(pcm);
+ if (prepare_error)
+ return prepare_error;
if (ioctl(pcm->fd, SNDRV_PCM_IOCTL_WRITEI_FRAMES, &x))
return oops(pcm, errno, "cannot write initial data");
pcm->running = 1;
return 0;
}
if (ioctl(pcm->fd, SNDRV_PCM_IOCTL_WRITEI_FRAMES, &x)) {
+ pcm->prepared = 0;
pcm->running = 0;
if (errno == EPIPE) {
/* we failed to make our window -- try to restart if we are
@@ -429,6 +438,7 @@ int pcm_read(struct pcm *pcm, void *data, unsigned int count)
}
}
if (ioctl(pcm->fd, SNDRV_PCM_IOCTL_READI_FRAMES, &x)) {
+ pcm->prepared = 0;
pcm->running = 0;
if (errno == EPIPE) {
/* we failed to make our window -- try to restart */
@@ -494,6 +504,12 @@ void pcm_params_free(struct pcm_params *pcm_params)
static int pcm_param_to_alsa(enum pcm_param param)
{
switch (param) {
+ case PCM_PARAM_ACCESS:
+ return SNDRV_PCM_HW_PARAM_ACCESS;
+ case PCM_PARAM_FORMAT:
+ return SNDRV_PCM_HW_PARAM_FORMAT;
+ case PCM_PARAM_SUBFORMAT:
+ return SNDRV_PCM_HW_PARAM_SUBFORMAT;
case PCM_PARAM_SAMPLE_BITS:
return SNDRV_PCM_HW_PARAM_SAMPLE_BITS;
break;
@@ -536,6 +552,23 @@ static int pcm_param_to_alsa(enum pcm_param param)
}
}
+struct pcm_mask *pcm_params_get_mask(struct pcm_params *pcm_params,
+ enum pcm_param param)
+{
+ int p;
+ struct snd_pcm_hw_params *params = (struct snd_pcm_hw_params *)pcm_params;
+ if (params == NULL) {
+ return NULL;
+ }
+
+ p = pcm_param_to_alsa(param);
+ if (p < 0 || !param_is_mask(p)) {
+ return NULL;
+ }
+
+ return (struct pcm_mask *)param_to_mask(params, p);
+}
+
unsigned int pcm_params_get_min(struct pcm_params *pcm_params,
enum pcm_param param)
{
@@ -582,6 +615,7 @@ int pcm_close(struct pcm *pcm)
if (pcm->fd >= 0)
close(pcm->fd);
+ pcm->prepared = 0;
pcm->running = 0;
pcm->buffer_size = 0;
pcm->fd = -1;
@@ -706,7 +740,7 @@ struct pcm *pcm_open(unsigned int card, unsigned int device,
pcm->boundary = sparams.boundary = pcm->buffer_size;
while (pcm->boundary * 2 <= INT_MAX - pcm->buffer_size)
- pcm->boundary *= 2;
+ pcm->boundary *= 2;
if (ioctl(pcm->fd, SNDRV_PCM_IOCTL_SW_PARAMS, &sparams)) {
oops(pcm, errno, "cannot set sw params");
@@ -719,6 +753,17 @@ struct pcm *pcm_open(unsigned int card, unsigned int device,
goto fail;
}
+#ifdef SNDRV_PCM_IOCTL_TTSTAMP
+ if (pcm->flags & PCM_MONOTONIC) {
+ int arg = SNDRV_PCM_TSTAMP_TYPE_MONOTONIC;
+ rc = ioctl(pcm->fd, SNDRV_PCM_IOCTL_TTSTAMP, &arg);
+ if (rc < 0) {
+ oops(pcm, rc, "cannot set timestamp type");
+ goto fail;
+ }
+ }
+#endif
+
pcm->underruns = 0;
return pcm;
@@ -736,13 +781,26 @@ int pcm_is_ready(struct pcm *pcm)
return pcm->fd >= 0;
}
-int pcm_start(struct pcm *pcm)
+int pcm_prepare(struct pcm *pcm)
{
+ if (pcm->prepared)
+ return 0;
+
if (ioctl(pcm->fd, SNDRV_PCM_IOCTL_PREPARE) < 0)
return oops(pcm, errno, "cannot prepare channel");
+ pcm->prepared = 1;
+ return 0;
+}
+
+int pcm_start(struct pcm *pcm)
+{
+ int prepare_error = pcm_prepare(pcm);
+ if (prepare_error)
+ return prepare_error;
+
if (pcm->flags & PCM_MMAP)
- pcm_sync_ptr(pcm, 0);
+ pcm_sync_ptr(pcm, 0);
if (ioctl(pcm->fd, SNDRV_PCM_IOCTL_START) < 0)
return oops(pcm, errno, "cannot start channel");
@@ -756,6 +814,7 @@ int pcm_stop(struct pcm *pcm)
if (ioctl(pcm->fd, SNDRV_PCM_IOCTL_DROP) < 0)
return oops(pcm, errno, "cannot stop channel");
+ pcm->prepared = 0;
pcm->running = 0;
return 0;
}
@@ -831,7 +890,7 @@ int pcm_mmap_begin(struct pcm *pcm, void **areas, unsigned int *offset,
int pcm_mmap_commit(struct pcm *pcm, unsigned int offset, unsigned int frames)
{
- (void)offset;
+ (void) offset;
/* update the application pointer in userspace and kernel */
pcm_mmap_appl_forward(pcm, frames);
pcm_sync_ptr(pcm, 0);
@@ -895,7 +954,7 @@ int pcm_wait(struct pcm *pcm, int timeout)
return 1;
}
-int pcm_mmap_write(struct pcm *pcm, const void *buffer, unsigned int bytes)
+int pcm_mmap_transfer(struct pcm *pcm, const void *buffer, unsigned int bytes)
{
int err = 0, frames, avail;
unsigned int offset = 0, count;
@@ -915,7 +974,7 @@ int pcm_mmap_write(struct pcm *pcm, const void *buffer, unsigned int bytes)
}
/* start the audio if we reach the threshold */
- if (!pcm->running &&
+ if (!pcm->running &&
(pcm->buffer_size - avail) >= pcm->config.start_threshold) {
if (pcm_start(pcm) < 0) {
fprintf(stderr, "start error: hw 0x%x app 0x%x avail 0x%x\n",
@@ -937,6 +996,7 @@ int pcm_mmap_write(struct pcm *pcm, const void *buffer, unsigned int bytes)
err = pcm_wait(pcm, time);
if (err < 0) {
+ pcm->prepared = 0;
pcm->running = 0;
fprintf(stderr, "wait error: hw 0x%x app 0x%x avail 0x%x\n",
(unsigned int)pcm->mmap_status->hw_ptr,
@@ -956,7 +1016,7 @@ int pcm_mmap_write(struct pcm *pcm, const void *buffer, unsigned int bytes)
break;
/* copy frames from buffer */
- frames = pcm_mmap_write_areas(pcm, buffer, offset, frames);
+ frames = pcm_mmap_transfer_areas(pcm, (void *)buffer, offset, frames);
if (frames < 0) {
fprintf(stderr, "write error: hw 0x%x app 0x%x avail 0x%x\n",
(unsigned int)pcm->mmap_status->hw_ptr,
@@ -971,3 +1031,19 @@ int pcm_mmap_write(struct pcm *pcm, const void *buffer, unsigned int bytes)
return 0;
}
+
+int pcm_mmap_write(struct pcm *pcm, const void *data, unsigned int count)
+{
+ if ((~pcm->flags) & (PCM_OUT | PCM_MMAP))
+ return -ENOSYS;
+
+ return pcm_mmap_transfer(pcm, (void *)data, count);
+}
+
+int pcm_mmap_read(struct pcm *pcm, void *data, unsigned int count)
+{
+ if ((~pcm->flags) & (PCM_IN | PCM_MMAP))
+ return -ENOSYS;
+
+ return pcm_mmap_transfer(pcm, data, count);
+}
diff --git a/firmware/target/hosted/ibasso/usb-ibasso.c b/firmware/target/hosted/ibasso/usb-ibasso.c
new file mode 100644
index 0000000000..e1b134e545
--- /dev/null
+++ b/firmware/target/hosted/ibasso/usb-ibasso.c
@@ -0,0 +1,92 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#include <stdlib.h>
+
+#include "config.h"
+#include "debug.h"
+
+#include "debug-ibasso.h"
+#include "usb-ibasso.h"
+
+
+static void usb_enable_adb(void)
+{
+ TRACE;
+
+ if(system(NULL))
+ {
+ system("setprop persist.sys.usb.config adb");
+ system("setprop persist.usb.debug 1");
+ return;
+ }
+
+ DEBUGF("ERROR %s: No command processor available.", __func__);
+}
+
+
+static void usb_enable_mass_storage(void)
+{
+ TRACE;
+
+ if(system(NULL))
+ {
+ system("setprop persist.sys.usb.config mass_storage");
+ system("setprop persist.usb.debug 0");
+ return;
+ }
+
+ DEBUGF("ERROR %s: No command processor available.", __func__);
+}
+
+
+/* Default at boot not known. */
+static int _last_usb_mode = -1;
+
+
+void ibasso_set_usb_mode(int mode)
+{
+ DEBUGF("DEBUG %s: _last_usb_mode: %d, mode: %d.", __func__, _last_usb_mode, mode);
+
+ if(_last_usb_mode != mode)
+ {
+ switch(mode)
+ {
+ case USB_MODE_MASS_STORAGE:
+ {
+ _last_usb_mode = mode;
+ usb_enable_mass_storage();
+ break;
+ }
+
+ case USB_MODE_CHARGE: /* Work around. */
+ case USB_MODE_ADB:
+ {
+ _last_usb_mode = mode;
+ usb_enable_adb();
+ break;
+ }
+ }
+ }
+}
diff --git a/firmware/target/hosted/ibasso/usb-ibasso.h b/firmware/target/hosted/ibasso/usb-ibasso.h
new file mode 100644
index 0000000000..f509d43038
--- /dev/null
+++ b/firmware/target/hosted/ibasso/usb-ibasso.h
@@ -0,0 +1,54 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#ifndef _USB_DX50_H_
+#define _USB_DX50_H_
+
+
+/* Supported usb modes. */
+enum ibasso_usb_mode
+{
+ /*
+ USB mass storage mode. On USB connection, Rockbox will terminate and the internel and
+ external storage gets exported to the connected client.
+ */
+ USB_MODE_MASS_STORAGE = 0,
+
+ /*
+ Actually the same, since we to not have proper USB detection.
+ Starts the adb server and enables adb connection over USB. Rockbox will continue to run.
+ */
+ USB_MODE_CHARGE,
+ USB_MODE_ADB
+};
+
+
+/*
+ Set the usb mode.
+ mode: ibasso_usb_mode
+*/
+void ibasso_set_usb_mode(int mode);
+
+
+#endif
diff --git a/firmware/target/hosted/ibasso/vold-ibasso.c b/firmware/target/hosted/ibasso/vold-ibasso.c
new file mode 100644
index 0000000000..c92b86d364
--- /dev/null
+++ b/firmware/target/hosted/ibasso/vold-ibasso.c
@@ -0,0 +1,203 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#include <pthread.h>
+#include <stdbool.h>
+#include <string.h>
+#include <unistd.h>
+#include <sys/poll.h>
+#include <sys/socket.h>
+#include <sys/types.h>
+#include <sys/un.h>
+
+#include "config.h"
+#include "debug.h"
+#include "powermgmt.h"
+
+#include "debug-ibasso.h"
+
+
+/*
+ Without this socket iBasso Vold will not start.
+ iBasso Vold uses this to send status messages about storage devices.
+*/
+static const char VOLD_MONITOR_SOCKET_NAME[] = "UNIX_domain";
+static int _vold_monitor_socket_fd = -1;
+
+
+static void vold_monitor_open_socket(void)
+{
+ TRACE;
+
+ unlink(VOLD_MONITOR_SOCKET_NAME);
+
+ _vold_monitor_socket_fd = socket(AF_UNIX, SOCK_STREAM, 0);
+
+ if(_vold_monitor_socket_fd < 0)
+ {
+ _vold_monitor_socket_fd = -1;
+ return;
+ }
+
+ struct sockaddr_un addr;
+ memset(&addr, 0, sizeof(addr));
+ addr.sun_family = AF_UNIX;
+ strncpy(addr.sun_path, VOLD_MONITOR_SOCKET_NAME, sizeof(addr.sun_path) -1);
+
+ if(bind(_vold_monitor_socket_fd, (struct sockaddr*) &addr, sizeof(addr)) < 0)
+ {
+ close(_vold_monitor_socket_fd);
+ unlink(VOLD_MONITOR_SOCKET_NAME);
+ _vold_monitor_socket_fd = -1;
+ return;
+ }
+
+ if(listen(_vold_monitor_socket_fd, 1) < 0)
+ {
+ close(_vold_monitor_socket_fd);
+ unlink(VOLD_MONITOR_SOCKET_NAME);
+ _vold_monitor_socket_fd = -1;
+ return;
+ }
+}
+
+
+/*
+ bionic does not have pthread_cancel.
+ 0: Vold monitor thread stopped/ending.
+ 1: Vold monitor thread started/running.
+*/
+static volatile sig_atomic_t _vold_monitor_active = 0;
+
+
+/*
+ 1: /mnt/sdcard is unmounting
+ 0: else
+*/
+static volatile sig_atomic_t _vold_monitor_forced_close_imminent = 0;
+
+
+static void* vold_monitor_run(void* nothing)
+{
+ _vold_monitor_active = 1;
+
+ (void) nothing;
+
+ DEBUGF("DEBUG %s: Thread start.", __func__);
+
+ vold_monitor_open_socket();
+ if(_vold_monitor_socket_fd < 0)
+ {
+ DEBUGF("ERROR %s: Thread end: No socket.", __func__);
+
+ _vold_monitor_active = 0;
+ return 0;
+ }
+
+ struct pollfd fds[1];
+ fds[0].fd = _vold_monitor_socket_fd;
+ fds[0].events = POLLIN;
+
+ while(_vold_monitor_active == 1)
+ {
+ poll(fds, 1, 10);
+ if(! (fds[0].revents & POLLIN))
+ {
+ continue;
+ }
+
+ int socket_fd = accept(_vold_monitor_socket_fd, NULL, NULL);
+
+ if(socket_fd < 0)
+ {
+ DEBUGF("ERROR %s: accept failed.", __func__);
+
+ continue;
+ }
+
+ while(true)
+ {
+ char msg[1024];
+ memset(msg, 0, sizeof(msg));
+ int length = read(socket_fd, msg, sizeof(msg));
+
+ if(length <= 0)
+ {
+ close(socket_fd);
+ break;
+ }
+
+ DEBUGF("%s: msg: %s", __func__, msg);
+
+ if(strcmp(msg, "Volume flash /mnt/sdcard state changed from 4 (Mounted) to 5 (Unmounting)") == 0)
+ {
+ /* We are losing /mnt/sdcard, shutdown Rockbox before it is forced closed. */
+
+ _vold_monitor_forced_close_imminent = 1;
+ sys_poweroff();
+ _vold_monitor_active = 0;
+ }
+ else if(strcmp(msg, "Volume sdcard /mnt/external_sd state changed from 4 (Mounted) to 5 (Unmounting)") == 0)
+ {
+ /* We are loosing the external sdcard, inform Rockbox. */
+ }
+ else if(strcmp(msg, "Volume sdcard /mnt/external_sd state changed from 3 (Checking) to 4 (Mounted)") == 0)
+ {
+ /* The external sdcard is back, inform Rockbox. */
+ }
+ }
+ }
+
+ close(_vold_monitor_socket_fd);
+ unlink(VOLD_MONITOR_SOCKET_NAME);
+ _vold_monitor_socket_fd = -1;
+
+ DEBUGF("%s: Thread end.", __func__);
+
+ _vold_monitor_active = 0;
+ return 0;
+}
+
+
+/* Vold monitor thread. */
+static pthread_t _vold_monitor_thread;
+
+
+void vold_monitor_start(void)
+{
+ TRACE;
+
+ if(_vold_monitor_active == 0)
+ {
+ pthread_create(&_vold_monitor_thread, NULL, vold_monitor_run, NULL);
+ }
+}
+
+
+bool vold_monitor_forced_close_imminent(void)
+{
+ TRACE;
+
+ return(_vold_monitor_forced_close_imminent == 1);
+}
diff --git a/firmware/target/hosted/ibasso/vold-ibasso.h b/firmware/target/hosted/ibasso/vold-ibasso.h
new file mode 100644
index 0000000000..18012b7e16
--- /dev/null
+++ b/firmware/target/hosted/ibasso/vold-ibasso.h
@@ -0,0 +1,42 @@
+/***************************************************************************
+ * __________ __ ___
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ *
+ * Copyright (C) 2014 by Ilia Sergachev: Initial Rockbox port to iBasso DX50
+ * Copyright (C) 2014 by Mario Basister: iBasso DX90 port
+ * Copyright (C) 2014 by Simon Rothen: Initial Rockbox repository submission, additional features
+ * Copyright (C) 2014 by Udo Schläpfer: Code clean up, additional features
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+
+#ifndef __VOLD_IBASSO_H__
+#define __VOLD_IBASSO_H__
+
+
+/* Start the vold monitor thread. */
+void vold_monitor_start(void);
+
+
+/*
+ Used to change Rockbox shutdown from reboot/power off to program exit.
+ true: vold monitor has detected, that vold is remounting /mnt/sdcard for USB mass storage
+ access.
+ false: else.
+*/
+bool vold_monitor_forced_close_imminent(void);
+
+
+#endif