diff --git a/Makefile.common b/Makefile.common
index e8567377ab..a32e4819a0 100644
--- a/Makefile.common
+++ b/Makefile.common
@@ -93,6 +93,7 @@ OBJ += frontend/frontend.o \
 		file_path.o \
 		hash.o \
 		driver.o \
+		general.o \
 		settings.o \
 		settings_data.o \
 		dynamic.o \
@@ -254,7 +255,9 @@ ifeq ($(HAVE_GLUI), 1)
    DEFINES += -DHAVE_GLUI
 endif
 ifeq ($(HAVE_LAKKA), 1)
-   OBJ += frontend/menu/backend/menu_lakka_backend.o frontend/menu/disp/lakka.o
+   OBJ += frontend/menu/backend/menu_lakka_backend.o \
+          frontend/menu/disp/lakka.o \
+          frontend/menu/disp/tween.o
    DEFINES += -DHAVE_LAKKA
 endif
 endif
diff --git a/apple/common/RAGameView.m b/apple/common/RAGameView.m
index 6b24f09cb3..4e1e1824b8 100644
--- a/apple/common/RAGameView.m
+++ b/apple/common/RAGameView.m
@@ -14,6 +14,7 @@
  *  If not, see <http://www.gnu.org/licenses/>.
  */
 
+#import <Availability.h>
 #import "RetroArch_Apple.h"
 #include "../../general.h"
 
@@ -198,6 +199,11 @@ static GLContextClass* g_context;
          return (apple_frontend_settings.orientation_flags & UIInterfaceOrientationMaskLandscapeLeft);
       case UIInterfaceOrientationLandscapeRight:
          return (apple_frontend_settings.orientation_flags & UIInterfaceOrientationMaskLandscapeRight);
+
+#if __IPHONE_OS_VERSION_MAX_ALLOWED >= __IPHONE_8_0
+      case UIInterfaceOrientationUnknown:
+         return (apple_frontend_settings.orientation_flags & UIInterfaceOrientationMaskAll);
+#endif
    }
    
    return YES;
diff --git a/apple/common/contentview_camera_ios.m.inl b/apple/common/contentview_camera_ios.m.inl
index 7429189483..068afe2cb8 100644
--- a/apple/common/contentview_camera_ios.m.inl
+++ b/apple/common/contentview_camera_ios.m.inl
@@ -4,6 +4,10 @@ CVOpenGLESTextureCacheRef textureCache;
 GLuint outputTexture;
 static bool newFrame = false;
 
+#ifndef GL_BGRA
+#define GL_BGRA 0x80E1
+#endif
+
 void event_process_camera_frame(void* pixelBufferPtr)
 {
     CVOpenGLESTextureRef renderTexture;
diff --git a/apple/common/utility.m b/apple/common/utility.m
index 2ca230d4ee..15003e2612 100644
--- a/apple/common/utility.m
+++ b/apple/common/utility.m
@@ -64,7 +64,7 @@ void apple_display_alert(const char *message, const char *title)
 
 - (BOOL)isPartialStringValid:(NSString*)partialString newEditingString:(NSString**)newString errorDescription:(NSString**)error
 {
-   int i;
+   NSUInteger i;
    bool hasDot = false;
 
    if (partialString.length)
diff --git a/apple/iOS/RetroArch_iOS.xcodeproj/project.pbxproj b/apple/iOS/RetroArch_iOS.xcodeproj/project.pbxproj
index 7ad77e0196..fda405fa99 100644
--- a/apple/iOS/RetroArch_iOS.xcodeproj/project.pbxproj
+++ b/apple/iOS/RetroArch_iOS.xcodeproj/project.pbxproj
@@ -505,6 +505,8 @@
 					"-DRARCH_INTERNAL",
 					"-DHAVE_THREADS",
 					"-DHAVE_FILTERS_BUILTIN",
+					"-DHAVE_LAKKA",
+					"-DHAVE_GLUI",
 				);
 				PRODUCT_NAME = "$(TARGET_NAME)";
 				VALID_ARCHS = "armv7 armv7s";
@@ -558,6 +560,8 @@
 					"-DHAVE_THREADS",
 					"-DHAVE_FILTERS_BUILTIN",
 					"-DHAVE_7ZIP",
+					"-DHAVE_LAKKA",
+					"-DHAVE_GLUI",
 				);
 				"OTHER_CFLAGS[arch=*]" = (
 					"-DNS_BLOCK_ASSERTIONS=1",
@@ -580,14 +584,16 @@
 					"-DRARCH_MOBILE",
 					"-DHAVE_COREAUDIO",
 					"-DHAVE_DYNAMIC",
+					"-DRARCH_INTERNAL",
 					"-DHAVE_OVERLAY",
 					"-DHAVE_ZLIB",
 					"-DWANT_MINIZ",
 					"-DSINC_LOWER_QUALITY",
-					"-DRARCH_INTERNAL",
 					"-DHAVE_THREADS",
 					"-DHAVE_FILTERS_BUILTIN",
 					"-DHAVE_7ZIP",
+					"-DHAVE_LAKKA",
+					"-DHAVE_GLUI",
 				);
 				PRODUCT_NAME = "$(TARGET_NAME)";
 				PROVISIONING_PROFILE = "";
diff --git a/apple/iOS/RetroArch_iOS.xcodeproj/project.xcworkspace/xcshareddata/RetroArch_iOS.xccheckout b/apple/iOS/RetroArch_iOS.xcodeproj/project.xcworkspace/xcshareddata/RetroArch_iOS.xccheckout
index 2bc51df130..96f4777dbe 100644
--- a/apple/iOS/RetroArch_iOS.xcodeproj/project.xcworkspace/xcshareddata/RetroArch_iOS.xccheckout
+++ b/apple/iOS/RetroArch_iOS.xcodeproj/project.xcworkspace/xcshareddata/RetroArch_iOS.xccheckout
@@ -12,7 +12,9 @@
 	<dict>
 		<key>49FCA0A2-0FE0-4A7A-9C54-642C68ED726A</key>
 		<string>git://github.com/libretro/common-overlays.git</string>
-		<key>A5B07A9E-AFED-41CB-BE4B-7C0CDBC26E8C</key>
+		<key>76200F0D6584D865E96F58DE862E738E88B23A3C</key>
+		<string>https://github.com/libretro/libretro-super.git</string>
+		<key>C7C12374C7051F8843B3EFA1ACCAF2907102CCF7</key>
 		<string>https://github.com/libretro/RetroArch.git</string>
 	</dict>
 	<key>IDESourceControlProjectPath</key>
@@ -21,17 +23,27 @@
 	<dict>
 		<key>49FCA0A2-0FE0-4A7A-9C54-642C68ED726A</key>
 		<string>../../../media/overlays</string>
-		<key>A5B07A9E-AFED-41CB-BE4B-7C0CDBC26E8C</key>
+		<key>76200F0D6584D865E96F58DE862E738E88B23A3C</key>
+		<string>../../../..</string>
+		<key>C7C12374C7051F8843B3EFA1ACCAF2907102CCF7</key>
 		<string>../../..</string>
 	</dict>
 	<key>IDESourceControlProjectURL</key>
 	<string>https://github.com/libretro/RetroArch.git</string>
 	<key>IDESourceControlProjectVersion</key>
-	<integer>110</integer>
+	<integer>111</integer>
 	<key>IDESourceControlProjectWCCIdentifier</key>
-	<string>A5B07A9E-AFED-41CB-BE4B-7C0CDBC26E8C</string>
+	<string>C7C12374C7051F8843B3EFA1ACCAF2907102CCF7</string>
 	<key>IDESourceControlProjectWCConfigurations</key>
 	<array>
+		<dict>
+			<key>IDESourceControlRepositoryExtensionIdentifierKey</key>
+			<string>public.vcs.git</string>
+			<key>IDESourceControlWCCIdentifierKey</key>
+			<string>76200F0D6584D865E96F58DE862E738E88B23A3C</string>
+			<key>IDESourceControlWCCName</key>
+			<string></string>
+		</dict>
 		<dict>
 			<key>IDESourceControlRepositoryExtensionIdentifierKey</key>
 			<string>public.vcs.git</string>
@@ -44,7 +56,7 @@
 			<key>IDESourceControlRepositoryExtensionIdentifierKey</key>
 			<string>public.vcs.git</string>
 			<key>IDESourceControlWCCIdentifierKey</key>
-			<string>A5B07A9E-AFED-41CB-BE4B-7C0CDBC26E8C</string>
+			<string>C7C12374C7051F8843B3EFA1ACCAF2907102CCF7</string>
 			<key>IDESourceControlWCCName</key>
 			<string>retroarch</string>
 		</dict>
diff --git a/apple/iOS/browser.m b/apple/iOS/browser.m
index 94c3034e61..bde80cea4c 100644
--- a/apple/iOS/browser.m
+++ b/apple/iOS/browser.m
@@ -46,9 +46,9 @@ static bool zlib_extract_callback(const char *name,
    // Ignore directories
    if (name[strlen(name) - 1] == '/')
       return true;
-  
+
    fill_pathname_join(path, (const char*)userdata, name, sizeof(path));
-   
+
    switch (cmode)
    {
       case 0: // Uncompressed
@@ -80,7 +80,7 @@ static void file_action(enum file_action action, NSString* source, NSString* tar
    NSError* error = nil;
    bool result = false;
    NSFileManager* manager = [NSFileManager defaultManager];
-   
+
    switch (action)
    {
       case FA_DELETE:
@@ -123,16 +123,16 @@ static void file_action(enum file_action action, NSString* source, NSString* tar
 {
    static NSString* const cell_id = @"path_item";
    static NSString* const icon_types[2] = { @"ic_file", @"ic_dir" };
-   
+
    uint32_t type_id = self.isDirectory ? 1 : 0;
-   
+
    UITableViewCell* result = [tableView dequeueReusableCellWithIdentifier:cell_id];
    if (!result)
       result = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:cell_id];
 
    result.textLabel.text = [self.path lastPathComponent];
    result.imageView.image = [UIImage imageNamed:icon_types[type_id]];
-   
+
    return result;
 }
 
@@ -152,21 +152,17 @@ static void file_action(enum file_action action, NSString* source, NSString* tar
 {
    if ((self = [super initWithStyle:UITableViewStylePlain]))
    {
-      _path = path ? path : NSHomeDirectory();
-      _chooseAction = action;
-      _extensions = extensions ? BOXSTRING(extensions) : 0;
-
-      self = [super initWithStyle:UITableViewStylePlain];
+      self.path = path ? path : NSHomeDirectory();
+      self.chooseAction = action;
+      self.extensions = extensions ? BOXSTRING(extensions) : 0;
       self.hidesHeaders = YES;
-      
+
       self.navigationItem.leftBarButtonItem = [[UIBarButtonItem alloc] initWithTitle:@"Up" style:UIBarButtonItemStyleBordered target:self
                                                                        action:@selector(gotoParent)];
 
-
       self.navigationItem.rightBarButtonItem = [[UIBarButtonItem alloc] initWithBarButtonSystemItem:UIBarButtonSystemItemCancel target:self
                                                                         action:@selector(cancelBrowser)];
 
-      
       // NOTE: The "App" and "Root" buttons aren't really needed for non-jailbreak devices.
       NSMutableArray* toolbarButtons = [NSMutableArray arrayWithObjects:
          [[UIBarButtonItem alloc] initWithTitle:@"Home" style:UIBarButtonItemStyleBordered target:self
@@ -183,12 +179,11 @@ static void file_action(enum file_action action, NSString* source, NSString* tar
                                   action:@selector(createNewFolder)],
          nil
       ];
-      
+
       self.toolbarItems = toolbarButtons;
 
       [self.tableView addGestureRecognizer:[[UILongPressGestureRecognizer alloc] initWithTarget:self
                       action:@selector(fileAction:)]];
-      
    }
 
    return self;
@@ -201,7 +196,7 @@ static void file_action(enum file_action action, NSString* source, NSString* tar
 
 - (void)gotoParent
 {
-   [self browseTo:[_path stringByDeletingLastPathComponent]];
+   [self browseTo:[self.path stringByDeletingLastPathComponent]];
 }
 
 - (void)gotoHomeDir
@@ -221,27 +216,27 @@ static void file_action(enum file_action action, NSString* source, NSString* tar
 
 - (void)refresh
 {
-   [self browseTo:_path];
+   [self browseTo: self.path];
 }
 
 - (void)browseTo:(NSString*)path
 {
-   _path = path;
-   self.title = _path.lastPathComponent;
+   self.path = path;
+   self.title = self.path.lastPathComponent;
 
    // Need one array per section
    self.sections = [NSMutableArray array];
-   
+
    for (NSString* i in [self sectionIndexTitlesForTableView:self.tableView])
       [self.sections addObject:[NSMutableArray arrayWithObject:i]];
-   
+
    // List contents
-   struct string_list* contents = dir_list_new(_path.UTF8String, _extensions.UTF8String, true);
-   
+   struct string_list* contents = dir_list_new(self.path.UTF8String, self.extensions.UTF8String, true);
+
    if (contents)
    {
       RADirectoryList __weak* weakSelf = self;
-      
+
       if (self.allowBlank)
          [self.sections[0] addObject:[RAMenuItemBasic itemWithDescription:@"[ Use Empty Path ]"
                                                                    action:^{ weakSelf.chooseAction(weakSelf, nil); }]];
@@ -250,17 +245,17 @@ static void file_action(enum file_action action, NSString* source, NSString* tar
                                                                    action:^{ weakSelf.chooseAction(weakSelf, [RADirectoryItem directoryItemFromPath:path]); }]];
 
       dir_list_sort(contents, true);
-   
-      for (int i = 0; i < contents->size; i ++)
+
+      for (size_t i = 0; i < contents->size; i ++)
       {
          const char* basename = path_basename(contents->elems[i].data);
-      
+
          uint32_t section = isalpha(basename[0]) ? (toupper(basename[0]) - 'A') + 2 : 1;
          section = (contents->elems[i].attr.i == RARCH_DIRECTORY) ? 0 : section;
 
          [self.sections[section] addObject:[RADirectoryItem directoryItemFromElement:&contents->elems[i]]];
       }
-   
+
       dir_list_free(contents);
    }
    else
@@ -280,7 +275,7 @@ static void file_action(enum file_action action, NSString* source, NSString* tar
 - (void)viewWillAppear:(BOOL)animated
 {
    [super viewWillAppear:animated];
-   [self browseTo:_path];
+   [self browseTo: self.path];
 }
 
 - (NSArray*)sectionIndexTitlesForTableView:(UITableView*)tableView
@@ -312,7 +307,7 @@ static void file_action(enum file_action action, NSString* source, NSString* tar
    {
       CGPoint point = [gesture locationInView:self.tableView];
       NSIndexPath* indexPath = [self.tableView indexPathForRowAtPoint:point];
-      
+
       if (indexPath)
       {
          self.selectedItem = [self itemForIndexPath:indexPath];
@@ -320,7 +315,7 @@ static void file_action(enum file_action action, NSString* source, NSString* tar
 
          NSString* button4_name = (IOS_IS_VERSION_7_OR_HIGHER()) ? @"AirDrop" : @"Delete";
          NSString* button5_name = (IOS_IS_VERSION_7_OR_HIGHER()) ? @"Delete" : nil;
-         
+
          UIActionSheet* menu = [[UIActionSheet alloc] initWithTitle:self.selectedItem.path.lastPathComponent delegate:self
                                                       cancelButtonTitle:@"Cancel" destructiveButtonTitle:nil
                                                       otherButtonTitles:is_zip ? @"Unzip" : @"Zip", @"Move", @"Rename", button4_name, button5_name, nil];
@@ -335,7 +330,7 @@ static void file_action(enum file_action action, NSString* source, NSString* tar
 {
    NSString* target = self.selectedItem.path;
    NSString* action = [actionSheet buttonTitleAtIndex:buttonIndex];
-   
+
    if ([action isEqualToString:@"Unzip"])
    {
       UIAlertView* alertView = [[UIAlertView alloc] initWithTitle:@"Enter target directory" message:@"" delegate:self
@@ -360,11 +355,11 @@ static void file_action(enum file_action action, NSString* source, NSString* tar
    else if ([action isEqualToString:@"AirDrop"] && IOS_IS_VERSION_7_OR_HIGHER())
    {
       // TODO: Zip if not already zipped
-      
+
       NSURL* url = [NSURL fileURLWithPath:self.selectedItem.path isDirectory:self.selectedItem.isDirectory];
       NSArray* items = [NSArray arrayWithObject:url];
       UIActivityViewController* avc = [[UIActivityViewController alloc] initWithActivityItems:items applicationActivities:nil];
-      
+
       [self presentViewController:avc animated:YES completion:nil];
    }
 #endif
@@ -392,10 +387,10 @@ static void file_action(enum file_action action, NSString* source, NSString* tar
       NSString* text = [alertView textFieldAtIndex:0].text;
 
       if (text.length)
-         file_action((enum file_action)alertView.tag, self.selectedItem.path, [_path stringByAppendingPathComponent:text]);
+         file_action((enum file_action)alertView.tag, self.selectedItem.path, [self.path stringByAppendingPathComponent:text]);
    }
 
-   [self browseTo:_path];
+   [self browseTo: self.path];
 }
 
 @end
@@ -411,23 +406,23 @@ static void file_action(enum file_action action, NSString* source, NSString* tar
    if ((self = [super initWithStyle:UITableViewStyleGrouped]))
    {
       RAFoldersList* __weak weakSelf = self;
-      _path = path;
+      self.path = path;
 
       // Parent item
-      NSString* sourceItem = _path.stringByDeletingLastPathComponent;
-      
+      NSString* sourceItem = self.path.stringByDeletingLastPathComponent;
+
       RAMenuItemBasic* parentItem = [RAMenuItemBasic itemWithDescription:BOXSTRING("<Parent>") association:sourceItem.stringByDeletingLastPathComponent
          action:^(id userdata){ [weakSelf moveInto:userdata]; } detail:NULL];
       [self.sections addObject:@[BOXSTRING(""), parentItem]];
 
 
       // List contents
-      struct string_list* contents = dir_list_new([_path stringByDeletingLastPathComponent].UTF8String, 0, true);
+      struct string_list* contents = dir_list_new([self.path stringByDeletingLastPathComponent].UTF8String, 0, true);
       NSMutableArray* items = [NSMutableArray arrayWithObject:BOXSTRING("")];
-   
+
       if (contents)
       {
-         int i;
+         size_t i;
          dir_list_sort(contents, true);
 
          for (i = 0; i < contents->size; i ++)
@@ -435,18 +430,18 @@ static void file_action(enum file_action action, NSString* source, NSString* tar
             if (contents->elems[i].attr.i == RARCH_DIRECTORY)
             {
                const char* basename = path_basename(contents->elems[i].data);
-               
+
                RAMenuItemBasic* item = [RAMenuItemBasic itemWithDescription:BOXSTRING(basename) association:BOXSTRING(contents->elems[i].data)
                   action:^(id userdata){ [weakSelf moveInto:userdata]; } detail:NULL];
                [items addObject:item];
             }
          }
-   
+
          dir_list_free(contents);
       }
 
-      [self setTitle:[BOXSTRING("Move ") stringByAppendingString:_path.lastPathComponent]];
-      
+      [self setTitle:[BOXSTRING("Move ") stringByAppendingString: self.path.lastPathComponent]];
+
       [self.sections addObject:items];
    }
 
diff --git a/apple/iOS/menu.m b/apple/iOS/menu.m
index 3baff5d374..3f2e8cf09d 100644
--- a/apple/iOS/menu.m
+++ b/apple/iOS/menu.m
@@ -52,7 +52,7 @@ typedef void (^RAActionSheetCallback)(UIActionSheet*, NSInteger);
 
 static void RunActionSheet(const char* title, const struct string_list* items, UIView* parent, RAActionSheetCallback callback)
 {
-   int i;
+   size_t i;
    RARunActionSheetDelegate* delegate = [[RARunActionSheetDelegate alloc] initWithCallbackBlock:callback];
    UIActionSheet* actionSheet = [UIActionSheet new];
 
@@ -134,6 +134,10 @@ static void RunActionSheet(const char* title, const struct string_list* items, U
 /* selected.                                 */
 /*********************************************/
 @implementation RAMenuItemBasic
+@synthesize description;
+@synthesize userdata;
+@synthesize action;
+@synthesize detail;
 
 + (RAMenuItemBasic*)itemWithDescription:(NSString*)description action:(void (^)())action
 {
@@ -653,15 +657,15 @@ static void RunActionSheet(const char* title, const struct string_list* items, U
 
 - (void)dealloc
 {
-   if (_history)
-      content_playlist_free(_history);
+   if (self.history)
+      content_playlist_free(self.history);
 }
 
 - (id)initWithHistoryPath:(const char*)historyPath
 {
    if ((self = [super initWithStyle:UITableViewStylePlain]))
    {
-      _history = content_playlist_init(historyPath, 100);
+      self.history = content_playlist_init(historyPath, 100);
       [self reloadData];
       self.navigationItem.rightBarButtonItem = [[UIBarButtonItem alloc] initWithTitle:BOXSTRING("Clear History")
                                                 style:UIBarButtonItemStyleBordered target:self action:@selector(clearHistory)];
@@ -672,18 +676,18 @@ static void RunActionSheet(const char* title, const struct string_list* items, U
 
 - (void)clearHistory
 {
-   if (_history)
-      content_playlist_clear(_history);
+   if (self.history)
+      content_playlist_clear(self.history);
    [self reloadData];
 }
 
 - (void)willReloadData
 {
-   int i;
+   size_t i;
    RAHistoryMenu* __weak weakSelf = self;
    NSMutableArray *section = [NSMutableArray arrayWithObject:BOXSTRING("")];
    
-   for (i = 0; _history && i < content_playlist_size(_history); i ++)
+   for (i = 0; self.history && i < content_playlist_size(self.history); i ++)
    {
        RAMenuItemBasic *item;
        const char *path      = NULL;
@@ -903,7 +907,7 @@ static void RunActionSheet(const char* title, const struct string_list* items, U
 
 - (void)willReloadData
 {
-   int i;
+   size_t i;
    const core_info_list_t* core_list;
    RAFrontendSettingsMenu* __weak weakSelf = self;
    NSMutableArray* cores = (NSMutableArray*)self.coreConfigOptions;
@@ -1151,7 +1155,7 @@ static bool copy_config(const char *src_path, const char *dst_path)
    self.actionRan = true;
    
    if (self.action)
-      _action(coreID);
+      self.action(coreID);
 }
 
 - (void)load:(uint32_t)count coresFromList:(const core_info_t*)list toSection:(NSMutableArray*)array
diff --git a/apple/iOS/platform.h b/apple/iOS/platform.h
index 0bf7845ef6..7479e3c2b4 100644
--- a/apple/iOS/platform.h
+++ b/apple/iOS/platform.h
@@ -37,6 +37,9 @@ const void* apple_get_frontend_settings(void);
 
 @interface RetroArch_iOS : UINavigationController<UIApplicationDelegate, UINavigationControllerDelegate, RetroArch_Platform>
 
+@property (nonatomic) UIWindow* window;
+@property (nonatomic) NSString* documentsDirectory; // e.g. /var/mobile/Documents
+
 + (RetroArch_iOS*)get;
 
 - (void)showGameView;
@@ -45,9 +48,6 @@ const void* apple_get_frontend_settings(void);
 - (void)unloadingCore;
 
 - (void)refreshSystemConfig;
-
-@property (nonatomic) NSString* documentsDirectory; // e.g. /var/mobile/Documents
-
 @end
 
 // modes are: keyboard, icade and btstack
diff --git a/apple/iOS/platform.m b/apple/iOS/platform.m
index b94a801cfb..1d0cd4dd28 100644
--- a/apple/iOS/platform.m
+++ b/apple/iOS/platform.m
@@ -73,7 +73,7 @@ const void* apple_get_frontend_settings(void)
 // Input helpers: This is kept here because it needs objective-c
 static void handle_touch_event(NSArray* touches)
 {
-   int i;
+   NSUInteger i;
    const float scale = [[UIScreen mainScreen] scale];
 
    g_current_input_data.touch_count = 0;
@@ -105,7 +105,7 @@ static void handle_touch_event(NSArray* touches)
 @end
 
 @interface UIApplication(iOS7Keyboard)
-- (id)_keyCommandForEvent:(id)event;
+- (id)_keyCommandForEvent:(UIEvent*)event;
 @end
 
 @interface RApplication : UIApplication
@@ -116,7 +116,7 @@ static void handle_touch_event(NSArray* touches)
 // Keyboard handler for iOS 7
 - (id)_keyCommandForEvent:(UIEvent*)event
 {
-   int i;
+   NSUInteger i;
    // This gets called twice with the same timestamp for each keypress, that's fine for polling
    // but is bad for business with events.
    static double last_time_stamp;
@@ -165,10 +165,6 @@ static void handle_touch_event(NSArray* touches)
 @end
 
 @implementation RetroArch_iOS
-{
-   UIWindow* _window;
-   NSString* _path;
-}
 
 + (RetroArch_iOS*)get
 {
@@ -185,9 +181,9 @@ static void handle_touch_event(NSArray* touches)
    [self setDelegate:self];
 
    // Setup window
-   _window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]];
+   self.window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]];
    [self showPauseMenu:self];
-   [_window makeKeyAndVisible];
+   [self.window makeKeyAndVisible];
 
    // Build system paths and test permissions
    self.documentsDirectory = [NSHomeDirectory() stringByAppendingPathComponent:@"Documents"];
@@ -278,7 +274,7 @@ static void handle_touch_event(NSArray* touches)
    [self setToolbarHidden:true animated:NO];
    [[UIApplication sharedApplication] setStatusBarHidden:true withAnimation:UIStatusBarAnimationNone];
    [[UIApplication sharedApplication] setIdleTimerDisabled:true];
-   [_window setRootViewController:[RAGameView get]];
+   [self.window setRootViewController:[RAGameView get]];
    g_extern.is_paused = false;
 }
 
@@ -287,7 +283,7 @@ static void handle_touch_event(NSArray* touches)
    g_extern.is_paused = true;
    [[UIApplication sharedApplication] setStatusBarHidden:false withAnimation:UIStatusBarAnimationNone];
    [[UIApplication sharedApplication] setIdleTimerDisabled:false];
-   [_window setRootViewController:self];
+   [self.window setRootViewController:self];
 }
 
 - (void)loadingCore:(NSString*)core withFile:(const char*)file
diff --git a/conf/config_file.c b/conf/config_file.c
index 232938510b..323f619250 100644
--- a/conf/config_file.c
+++ b/conf/config_file.c
@@ -84,15 +84,13 @@ static char *getaline(FILE *file)
          cur_size *= 2;
          newline_tmp = (char*)realloc(newline, cur_size + 1);
 
-         if (newline_tmp)
-         {
-            newline = newline_tmp;
-         }
-         else
+         if (!newline_tmp)
          {
             free(newline);
             return NULL;
          }
+
+         newline = newline_tmp;
       }
 
       newline[index++] = in;
@@ -174,7 +172,9 @@ static void add_child_list(config_file_t *parent, config_file_t *child)
    /* Rebase tail. */
    if (parent->entries)
    {
-      struct config_entry_list *head = parent->entries;
+      struct config_entry_list *head = 
+         (struct config_entry_list*)parent->entries;
+
       while (head->next)
          head = head->next;
       parent->tail = head;
@@ -329,15 +329,13 @@ static bool parse_line(config_file_t *conf,
          cur_size *= 2;
          key_tmp = (char*)realloc(key, cur_size + 1);
 
-         if (key_tmp)
-         {
-            key = key_tmp;
-         }
-         else
+         if (!key_tmp)
          {
             free(key);
             return false;
          }
+
+         key = key_tmp;
       }
 
       key[index++] = *line++;
diff --git a/driver.c b/driver.c
index b2d73fd4fb..c69246c8d8 100644
--- a/driver.c
+++ b/driver.c
@@ -638,7 +638,8 @@ static void find_video_driver(void)
    }
 #endif
 
-   if (driver.frontend_ctx->get_video_driver)
+   if (driver.frontend_ctx &&
+       driver.frontend_ctx->get_video_driver)
    {
       driver.video = driver.frontend_ctx->get_video_driver();
 
diff --git a/driver.h b/driver.h
index e27cbae24a..db451704d2 100644
--- a/driver.h
+++ b/driver.h
@@ -624,9 +624,9 @@ extern video_driver_t video_xdk_d3d;
 extern video_driver_t video_sdl;
 extern video_driver_t video_sdl2;
 extern video_driver_t video_vg;
-extern video_driver_t video_null;
 extern video_driver_t video_omap;
 extern video_driver_t video_exynos;
+extern video_driver_t video_null;
 
 extern input_driver_t input_android;
 extern input_driver_t input_sdl;
@@ -672,14 +672,14 @@ extern menu_ctx_driver_backend_t menu_ctx_backend_lakka;
 #define check_oneshot_func(trigger_input) check_oneshot(BIND_PRESSED(trigger_input, RARCH_FRAMEADVANCE), BIND_PRESSED(trigger_input, RARCH_REWIND))
 #define check_slowmotion_func(input) check_slowmotion(BIND_PRESSED(input, RARCH_SLOWMOTION))
 #define check_shader_dir_func(trigger_input) check_shader_dir(BIND_PRESSED(trigger_input, RARCH_SHADER_NEXT), BIND_PRESSED(trigger_input, RARCH_SHADER_PREV))
-#define check_enter_menu_func(input, old_input) check_enter_menu(BIND_PRESSED(input, RARCH_MENU_TOGGLE), BIND_PRESSED(old_input, RARCH_MENU_TOGGLE))
+#define check_enter_menu_func(input) BIND_PRESSED(input, RARCH_MENU_TOGGLE)
 #define check_mute_func(input, old_input) check_mute(BIND_PRESSED(input, RARCH_MUTE), BIND_PRESSED(old_input, RARCH_MUTE))
 #define check_volume_func(input, old_input) check_volume(BIND_PRESSED(input, RARCH_VOLUME_UP), BIND_PRESSED(old_input, RARCH_VOLUME_DOWN))
 #define check_fullscreen_func(trigger_input) rarch_check_fullscreen(BIND_PRESSED(trigger_input, RARCH_FULLSCREEN_TOGGLE_KEY))
 #define check_fast_forward_button_func(input, old_input, trigger_input) check_fast_forward_button(BIND_PRESSED(trigger_input, RARCH_FAST_FORWARD_KEY), BIND_PRESSED(input, RARCH_FAST_FORWARD_HOLD_KEY), BIND_PRESSED(old_input, RARCH_FAST_FORWARD_HOLD_KEY))
 #define check_rewind_func(input) check_rewind(BIND_PRESSED(input, RARCH_REWIND))
 #define check_stateslots_func(trigger_input) check_stateslots(BIND_PRESSED(trigger_input, RARCH_STATE_SLOT_PLUS), BIND_PRESSED(trigger_input, RARCH_STATE_SLOT_MINUS))
-#define check_pause_func(input, old_input) check_pause(BIND_PRESSED(input, RARCH_PAUSE_TOGGLE), BIND_PRESSED(old_input, RARCH_PAUSE_TOGGLE), BIND_PRESSED(input, RARCH_FRAMEADVANCE))
+#define check_pause_func(input) check_pause(BIND_PRESSED(input, RARCH_PAUSE_TOGGLE), BIND_PRESSED(input, RARCH_FRAMEADVANCE))
 #define check_quit_key_func(input) BIND_PRESSED(input, RARCH_QUIT_KEY)
 
 #ifdef __cplusplus
diff --git a/dynamic.c b/dynamic.c
index 7d44990284..e36814298f 100644
--- a/dynamic.c
+++ b/dynamic.c
@@ -102,6 +102,7 @@ size_t (*pretro_get_memory_size)(unsigned);
 #endif
 
 static bool *load_no_content_hook;
+static bool ignore_environment_cb;
 
 static bool environ_cb_get_system_info(unsigned cmd, void *data)
 {
@@ -125,6 +126,13 @@ void libretro_get_environment_info(void (*func)(retro_environment_t),
 
    /* load_no_content gets set in this callback. */
    func(environ_cb_get_system_info);
+
+   /* It's possible that we just set get_system_info callback to the currently running core.
+    * Make sure we reset it to the actual environment callback.
+    * Ignore any environment callbacks here in case we're running on the non-current core. */
+   ignore_environment_cb = true;
+   func(rarch_environment_cb);
+   ignore_environment_cb = false;
 }
 
 static dylib_t libretro_get_system_info_lib(const char *path,
@@ -499,6 +507,9 @@ bool rarch_environment_cb(unsigned cmd, void *data)
 {
    unsigned p, id;
 
+   if (ignore_environment_cb)
+      return false;
+
    switch (cmd)
    {
       case RETRO_ENVIRONMENT_GET_OVERSCAN:
@@ -591,6 +602,7 @@ bool rarch_environment_cb(unsigned cmd, void *data)
       case RETRO_ENVIRONMENT_SHUTDOWN:
          RARCH_LOG("Environ SHUTDOWN.\n");
          g_extern.system.shutdown = true;
+         g_extern.core_shutdown_initiated = true;
          break;
 
       case RETRO_ENVIRONMENT_SET_PERFORMANCE_LEVEL:
diff --git a/frontend/frontend.c b/frontend/frontend.c
index e2a00b6398..465bff0dd6 100644
--- a/frontend/frontend.c
+++ b/frontend/frontend.c
@@ -55,50 +55,28 @@
 
 #define MAX_ARGS 32
 
-static retro_keyboard_event_t key_event;
-
-static int main_entry_iterate_shutdown(signature(), args_type() args)
+static int main_entry_iterate_shutdown(signature(),
+      args_type() args)
 {
    (void)args;
 
-   if (!g_settings.load_dummy_on_core_shutdown)
-      return 1;
+   if (g_extern.core_shutdown_initiated 
+         && g_settings.load_dummy_on_core_shutdown)
+   {
+      /* Load dummy core instead of exiting RetroArch completely. */
+      rarch_main_command(RARCH_CMD_PREPARE_DUMMY);
+      g_extern.core_shutdown_initiated = false;
+      return 0;
+   }
 
-   /* Load dummy core instead of exiting RetroArch completely. */
-   rarch_main_command(RARCH_CMD_PREPARE_DUMMY);
-
-   return 0;
+   return 1;
 }
 
 int main_entry_decide(signature(), args_type() args)
 {
-#ifdef HAVE_MENU
-   if (g_extern.system.shutdown)
-      return main_entry_iterate_shutdown(signature_expand(), args);
-   if (g_extern.lifecycle_state & (1ULL << MODE_CLEAR_INPUT))
-      return main_entry_iterate_clear_input(signature_expand(), args);
-   if (g_extern.lifecycle_state & (1ULL << MODE_LOAD_GAME))
-      return main_entry_iterate_load_content(signature_expand(), args);
-   if (g_extern.lifecycle_state & (1ULL << MODE_GAME))
-      return main_entry_iterate_content(signature_expand(), args);
-   if (g_extern.lifecycle_state & (1ULL << MODE_MENU_PREINIT))
-      return main_entry_iterate_menu_preinit(signature_expand(), args);
-   if (g_extern.lifecycle_state & (1ULL << MODE_MENU))
-      return main_entry_iterate_menu(signature_expand(), args);
-
-   return 1;
-#else
-   return main_entry_iterate_content_nomenu(signature_expand(), args);
-#endif
-}
-
-int main_entry_iterate_content(signature(), args_type() args)
-{
+   int ret = 0;
    if (!rarch_main_iterate())
-   {
-      rarch_main_set_state(RARCH_ACTION_STATE_RUNNING_FINISHED);
-      return 0;
-   }
+      return main_entry_iterate_shutdown(signature_expand(), args);
 
    if (driver.frontend_ctx && driver.frontend_ctx->process_events)
       driver.frontend_ctx->process_events(args);
@@ -106,113 +84,6 @@ int main_entry_iterate_content(signature(), args_type() args)
    return 0;
 }
 
-#ifndef HAVE_MENU
-static int main_entry_iterate_content_nomenu(signature(), args_type() args)
-{
-   if (!rarch_main_iterate())
-      return 1;
-
-   return 0;
-}
-#endif
-
-int main_entry_iterate_clear_input(signature(), args_type() args)
-{
-   (void)args;
-
-   rarch_input_poll();
-#ifdef HAVE_MENU
-   if (menu_input())
-      return 0;
-#endif
-
-   /* Restore libretro keyboard callback. */
-   g_extern.system.key_event = key_event;
-   rarch_main_set_state(RARCH_ACTION_STATE_FLUSH_INPUT_FINISHED);
-
-   return 0;
-}
-
-int main_entry_iterate_load_content(signature(), args_type() args)
-{
-#ifdef HAVE_MENU
-   if (!load_menu_content())
-   {
-      /* If content loading fails, we go back to menu. */
-      rarch_main_set_state(RARCH_ACTION_STATE_RUNNING_FINISHED);
-      if (driver.menu)
-         rarch_main_set_state(RARCH_ACTION_STATE_MENU_PREINIT);
-   }
-#endif
-
-   rarch_main_set_state(RARCH_ACTION_STATE_LOAD_CONTENT_FINISHED);
-
-   return 0;
-}
-
-#ifdef HAVE_MENU
-int main_entry_iterate_menu_preinit(signature(), args_type() args)
-{
-   int i;
-
-   /* Menu should always run with vsync on. */
-   rarch_main_command(RARCH_CMD_VIDEO_SET_BLOCKING_STATE);
-
-   /* Stop all rumbling before entering the menu. */
-   for (i = 0; i < MAX_PLAYERS; i++)
-   {
-      driver_set_rumble_state(i, RETRO_RUMBLE_STRONG, 0);
-      driver_set_rumble_state(i, RETRO_RUMBLE_WEAK, 0);
-   }
-
-   rarch_main_command(RARCH_CMD_AUDIO_STOP);
-
-   if (driver.menu)
-   {
-      /* Override keyboard callback to redirect to menu instead.
-       * We'll use this later for something ...
-       * FIXME: This should probably be moved to menu_common somehow. */
-      key_event = g_extern.system.key_event;
-      g_extern.system.key_event = menu_key_event;
-
-      driver.menu->need_refresh = true;
-      driver.menu->old_input_state |= 1ULL << RARCH_MENU_TOGGLE;
-      rarch_main_set_state(RARCH_ACTION_STATE_MENU_PREINIT_FINISHED);
-      rarch_main_set_state(RARCH_ACTION_STATE_MENU_RUNNING);
-   }
-
-   return 0;
-}
-
-int main_entry_iterate_menu(signature(), args_type() args)
-{
-   retro_input_t input, old_state = 0;
-
-   if (menu_iterate())
-   {
-      if (driver.frontend_ctx && driver.frontend_ctx->process_events)
-         driver.frontend_ctx->process_events(args);
-      return 0;
-   }
-
-   rarch_main_set_state(RARCH_ACTION_STATE_MENU_RUNNING_FINISHED);
-   driver_set_nonblock_state(driver.nonblock_state);
-
-   rarch_main_command(RARCH_CMD_AUDIO_START);
-   rarch_main_set_state(RARCH_ACTION_STATE_FLUSH_INPUT);
-
-   input = input_keys_pressed_func(RARCH_QUIT_KEY, RARCH_QUIT_KEY + 1,
-         &old_state);
-
-   if (BIND_PRESSED(input, RARCH_QUIT_KEY) ||
-         !driver.video->alive(driver.video_data))
-      return 1;
-
-   return 0;
-}
-#endif
-
-
 void main_exit(args_type() args)
 {
    g_extern.system.shutdown = false;
diff --git a/frontend/menu/backend/menu_common_backend.c b/frontend/menu/backend/menu_common_backend.c
index 947202feaf..f9d4386268 100644
--- a/frontend/menu/backend/menu_common_backend.c
+++ b/frontend/menu/backend/menu_common_backend.c
@@ -1970,13 +1970,6 @@ static int menu_common_iterate(unsigned action)
    if (driver.video_data && driver.menu_ctx && driver.menu_ctx->set_texture)
       driver.menu_ctx->set_texture(driver.menu);
 
-   if (action == MENU_ACTION_TOGGLE &&
-         g_extern.main_is_init && !g_extern.libretro_dummy)
-   {
-      rarch_main_command(RARCH_CMD_RESUME);
-      return -1;
-   }
-
    if (!strcmp(menu_label, "help"))
       return menu_start_screen_iterate(action);
    else if (!strcmp(menu_label, "message"))
diff --git a/frontend/menu/backend/menu_lakka_backend.c b/frontend/menu/backend/menu_lakka_backend.c
index 1192ef8706..5ff95b972a 100644
--- a/frontend/menu/backend/menu_lakka_backend.c
+++ b/frontend/menu/backend/menu_lakka_backend.c
@@ -35,6 +35,7 @@
 #include "../../../settings_data.h"
 
 #include "../disp/lakka.h"
+#include "../disp/tween.h"
 
 #ifdef HAVE_CONFIG_H
 #include "../../../config.h"
@@ -514,7 +515,7 @@ static int menu_lakka_iterate(unsigned action)
                  (active_category->num_items - 1))))
          {
             add_tween(LAKKA_DELAY, 1.0, &global_alpha, &inOutQuad, NULL);
-            rarch_main_set_state(RARCH_ACTION_STATE_RUNNING_FINISHED);
+            rarch_main_command(RARCH_CMD_QUIT_RETROARCH);
             return -1;
          }
          break;
diff --git a/frontend/menu/disp/glui.c b/frontend/menu/disp/glui.c
index 6f67405e0a..6de08e7257 100644
--- a/frontend/menu/disp/glui.c
+++ b/frontend/menu/disp/glui.c
@@ -162,9 +162,6 @@ static void glui_frame(void)
    if (!driver.menu || !gl)
       return;
 
-   if (g_extern.lifecycle_state & (1ULL << MODE_GAME))
-      return;
-
    line_height = g_settings.video.font_size * 4 / 3;
    glyph_width = line_height / 2;
    glui_margin = gl->win_width / 20 ;
diff --git a/frontend/menu/disp/lakka.c b/frontend/menu/disp/lakka.c
index be13e34009..6ace776ec7 100644
--- a/frontend/menu/disp/lakka.c
+++ b/frontend/menu/disp/lakka.c
@@ -43,6 +43,7 @@
 #include "../../../gfx/fonts/bitmap.h"
 
 #include "lakka.h"
+#include "tween.h"
 
 // Category variables
 menu_category_t *categories;
@@ -71,6 +72,7 @@ float above_subitem_offset;
 float above_item_offset;
 float active_item_factor;
 float under_item_offset;
+float setting_margin_left;
 
 // Font variables
 static void *font;
@@ -117,9 +119,6 @@ struct lakka_texture_item
 
 struct lakka_texture_item textures[TEXTURE_LAST];
 
-static tween_t* tweens = NULL;
-static int numtweens = 0;
-
 static void lakka_responsive(void)
 {
    gl_t *gl = (gl_t*)driver_video_resolve(NULL);
@@ -149,6 +148,7 @@ static void lakka_responsive(void)
       title_margin_top = 50.0;
       label_margin_left = 192;
       label_margin_top = 15;
+      setting_margin_left = 1200;
       strcpy(icon_dir, "256");
       return;
    }
@@ -166,6 +166,7 @@ static void lakka_responsive(void)
       label_margin_left = 144;
       label_margin_top = 11.0;
       strcpy(icon_dir, "192");
+      setting_margin_left = 800;
       return;
    }
 
@@ -181,6 +182,7 @@ static void lakka_responsive(void)
       title_margin_top = 35.0;
       label_margin_left = 85;
       label_margin_top = 8.0;
+      setting_margin_left = 600;
       strcpy(icon_dir, "128");
       return;
    }
@@ -198,6 +200,7 @@ static void lakka_responsive(void)
       label_margin_left = 48;
       label_margin_top = 6.0;
       strcpy(icon_dir, "64");
+      setting_margin_left = 250;
       return;
    }
 
@@ -211,6 +214,7 @@ static void lakka_responsive(void)
    title_margin_top = 30.0;
    label_margin_left = 64;
    label_margin_top = 6.0;
+   setting_margin_left = 400;
    strcpy(icon_dir, "96");
 }
 
@@ -249,104 +253,6 @@ static char *str_replace (const char *string, const char *substr, const char *re
    return newstr;
 }
 
-float inOutQuad(float t, float b, float c, float d)
-{
-   t = t / d * 2;
-   if (t < 1)
-      return c / 2 * pow(t, 2) + b;
-   return -c / 2 * ((t - 1) * (t - 3) - 1) + b;
-}
-
-void add_tween(float duration, float target_value, float* subject,
-      easingFunc easing, tweenCallback callback)
-{
-   tween_t *tween;
-   tween_t *tweens_tmp;
-
-   numtweens++;
-
-   tweens_tmp = (tween_t*)realloc(tweens, numtweens * sizeof(tween_t));
-   if (tweens_tmp != NULL)
-   {
-       tweens = tweens_tmp;
-   }
-   else // realloc failed
-   {
-      if (tweens != NULL)
-      {
-         free(tweens);
-         tweens = NULL;
-      }
-
-      return;
-   }
-   
-   tween = (tween_t*)&tweens[numtweens-1];
-
-   if (!tween)
-      return;
-
-   tween->alive = 1;
-   tween->duration = duration;
-   tween->running_since = 0;
-   tween->initial_value = *subject;
-   tween->target_value = target_value;
-   tween->subject = subject;
-   tween->easing = easing;
-   tween->callback = callback;
-}
-
-static void update_tween(void *data, float dt)
-{
-   tween_t *tween = (tween_t*)data;
-
-   if (!tween)
-      return;
-
-#if 0
-   RARCH_LOG("delta: %f\n", dt);
-   RARCH_LOG("tween running since: %f\n", tween->running_since);
-   RARCH_LOG("tween duration: %f\n", tween->duration);
-#endif
-
-   if (tween->running_since < tween->duration)
-   {
-      tween->running_since += dt;
-
-      if (tween->easing)
-         *tween->subject = tween->easing(
-               tween->running_since,
-               tween->initial_value,
-               tween->target_value - tween->initial_value,
-               tween->duration);
-
-      if (tween->running_since >= tween->duration)
-      {
-         *tween->subject = tween->target_value;
-
-         if (tween->callback)
-            tween->callback();
-      }
-   }
-}
-
-static void update_tweens(float dt)
-{
-   int i, active_tweens;
-
-   active_tweens = 0;
-
-   for(i = 0; i < numtweens; i++)
-   {
-      update_tween(&tweens[i], dt);
-      active_tweens += tweens[i].running_since <
-         tweens[i].duration ? 1 : 0;
-   }
-
-   if (numtweens && !active_tweens)
-      numtweens = 0;
-}
-
 static void lakka_draw_text(const char *str, float x,
       float y, float scale, float alpha)
 {
@@ -545,7 +451,7 @@ static void lakka_draw_subitems(int i, int j)
                snprintf(slot, sizeof(slot), "%d", g_settings.state_slot);
             lakka_draw_text(slot, 
                   margin_left + hspacing * (i+2.25) +
-                  all_categories_x + label_margin_left + 400, 
+                  all_categories_x + label_margin_left + setting_margin_left, 
                   margin_top + subitem->y + label_margin_top, 
                   1, 
                   subitem->alpha);
@@ -559,7 +465,7 @@ static void lakka_draw_subitems(int i, int j)
                sizeof(val));
          lakka_draw_text(val, 
                margin_left + hspacing * (i+2.25) +
-               all_categories_x + label_margin_left + 400, 
+               all_categories_x + label_margin_left + setting_margin_left, 
                margin_top + subitem->y + label_margin_top, 
                1, 
                subitem->alpha);
@@ -739,9 +645,6 @@ static void lakka_context_destroy(void *data)
       font_driver->free(font);
       font_driver = NULL;
    }
-
-   //if (numtweens)
-   //   free(tweens);
 }
 
 void lakka_init_settings(void)
diff --git a/frontend/menu/disp/lakka.h b/frontend/menu/disp/lakka.h
index 5f8e0b42ed..5edb870b39 100644
--- a/frontend/menu/disp/lakka.h
+++ b/frontend/menu/disp/lakka.h
@@ -79,24 +79,6 @@ typedef struct
    menu_item_t *items;
 } menu_category_t;
 
-typedef float (*easingFunc)(float, float, float, float);
-typedef void (*tweenCallback) (void);
-
-typedef struct
-{
-   int    alive;
-   float  duration;
-   float  running_since;
-   float  initial_value;
-   float  target_value;
-   float* subject;
-   easingFunc easing;
-   tweenCallback callback;
-} tween_t;
-
 extern menu_category_t *categories;
 
-void add_tween(float duration, float target_value, float* subject, easingFunc easing, tweenCallback callback);
-float inOutQuad(float t, float b, float c, float d);
-
 #endif /* MENU_DISP_LAKKA_H */
diff --git a/frontend/menu/disp/tween.c b/frontend/menu/disp/tween.c
new file mode 100644
index 0000000000..ccf999ade8
--- /dev/null
+++ b/frontend/menu/disp/tween.c
@@ -0,0 +1,332 @@
+#include "tween.h"
+#include <math.h>
+
+tween_t* tweens = NULL;
+int numtweens = 0;
+
+void add_tween(float duration, float target_value, float* subject,
+      easingFunc easing, tweenCallback callback)
+{
+   tween_t *tween;
+   tween_t *tweens_tmp;
+
+   numtweens++;
+
+   tweens_tmp = (tween_t*)realloc(tweens, numtweens * sizeof(tween_t));
+   if (tweens_tmp != NULL)
+   {
+       tweens = tweens_tmp;
+   }
+   else // realloc failed
+   {
+      if (tweens != NULL)
+      {
+         free(tweens);
+         tweens = NULL;
+      }
+
+      return;
+   }
+   
+   tween = (tween_t*)&tweens[numtweens-1];
+
+   if (!tween)
+      return;
+
+   tween->alive = 1;
+   tween->duration = duration;
+   tween->running_since = 0;
+   tween->initial_value = *subject;
+   tween->target_value = target_value;
+   tween->subject = subject;
+   tween->easing = easing;
+   tween->callback = callback;
+}
+
+void update_tween(void *data, float dt)
+{
+   tween_t *tween = (tween_t*)data;
+
+   if (!tween)
+      return;
+
+   if (tween->running_since < tween->duration)
+   {
+      tween->running_since += dt;
+
+      if (tween->easing)
+         *tween->subject = tween->easing(
+               tween->running_since,
+               tween->initial_value,
+               tween->target_value - tween->initial_value,
+               tween->duration);
+
+      if (tween->running_since >= tween->duration)
+      {
+         *tween->subject = tween->target_value;
+
+         if (tween->callback)
+            tween->callback();
+      }
+   }
+}
+
+void update_tweens(float dt)
+{
+   int i, active_tweens;
+
+   active_tweens = 0;
+
+   for(i = 0; i < numtweens; i++)
+   {
+      update_tween(&tweens[i], dt);
+      active_tweens += tweens[i].running_since < tweens[i].duration ? 1 : 0;
+   }
+
+   if (numtweens && !active_tweens)
+      numtweens = 0;
+}
+
+// linear
+
+float linear(float t, float b, float c, float d)
+{
+   return c * t / d + b;
+}
+
+// quad
+
+float inQuad(float t, float b, float c, float d)
+{
+   return c * pow(t / d, 2) + b;
+}
+
+float outQuad(float t, float b, float c, float d)
+{
+   t = t / d;
+   return -c * t * (t - 2) + b;
+}
+
+float inOutQuad(float t, float b, float c, float d)
+{
+   t = t / d * 2;
+   if (t < 1)
+      return c / 2 * pow(t, 2) + b;
+   return -c / 2 * ((t - 1) * (t - 3) - 1) + b;
+}
+
+float outInQuad(float t, float b, float c, float d)
+{
+   if (t < d / 2)
+      return outQuad(t * 2, b, c / 2, d);
+   return inQuad((t * 2) - d, b + c / 2, c / 2, d);
+}
+
+// cubic
+
+float inCubic(float t, float b, float c, float d)
+{
+   return c * pow(t / d, 3) + b;
+}
+
+float outCubic(float t, float b, float c, float d)
+{
+   return c * (pow(t / d - 1, 3) + 1) + b;
+}
+
+float inOutCubic(float t, float b, float c, float d)
+{
+   t = t / d * 2;
+   if (t < 1)
+      return c / 2 * t * t * t + b;
+   t = t - 2;
+   return c / 2 * (t * t * t + 2) + b;
+}
+
+float outInCubic(float t, float b, float c, float d)
+{
+   if (t < d / 2)
+      return outCubic(t * 2, b, c / 2, d);
+   return inCubic((t * 2) - d, b + c / 2, c / 2, d);
+}
+
+// quart
+
+float inQuart(float t, float b, float c, float d)
+{
+   return c * pow(t / d, 4) + b;
+}
+
+float outQuart(float t, float b, float c, float d)
+{
+   return -c * (pow(t / d - 1, 4) - 1) + b;
+}
+
+float inOutQuart(float t, float b, float c, float d)
+{
+   t = t / d * 2;
+   if (t < 1)
+      return c / 2 * pow(t, 4) + b;
+   return -c / 2 * (pow(t - 2, 4) - 2) + b;
+}
+
+float outInQuart(float t, float b, float c, float d)
+{
+   if (t < d / 2)
+      return outQuart(t * 2, b, c / 2, d);
+   return inQuart((t * 2) - d, b + c / 2, c / 2, d);
+}
+
+// quint
+
+float inQuint(float t, float b, float c, float d)
+{
+   return c * pow(t / d, 5) + b;
+}
+
+float outQuint(float t, float b, float c, float d)
+{
+   return c * (pow(t / d - 1, 5) + 1) + b;
+}
+
+float inOutQuint(float t, float b, float c, float d)
+{
+   t = t / d * 2;
+   if (t < 1)
+      return c / 2 * pow(t, 5) + b;
+   return c / 2 * (pow(t - 2, 5) + 2) + b;
+}
+
+float outInQuint(float t, float b, float c, float d)
+{
+   if (t < d / 2)
+      return outQuint(t * 2, b, c / 2, d);
+   return inQuint((t * 2) - d, b + c / 2, c / 2, d);
+}
+
+// sine
+
+float inSine(float t, float b, float c, float d)
+{
+   return -c * cos(t / d * (M_PI / 2)) + c + b;
+}
+
+float outSine(float t, float b, float c, float d)
+{
+   return c * sin(t / d * (M_PI / 2)) + b;
+}
+
+float inOutSine(float t, float b, float c, float d)
+{
+   return -c / 2 * (cos(M_PI * t / d) - 1) + b;
+}
+
+float outInSine(float t, float b, float c, float d)
+{
+   if (t < d / 2)
+      return outSine(t * 2, b, c / 2, d);
+   return inSine((t * 2) -d, b + c / 2, c / 2, d);
+}
+
+// expo
+
+float inExpo(float t, float b, float c, float d)
+{
+   if (t == 0)
+      return b;
+   return c * pow(2, 10 * (t / d - 1)) + b - c * 0.001;
+}
+
+float outExpo(float t, float b, float c, float d)
+{
+   if (t == d)
+      return b + c;
+   return c * 1.001 * (-pow(2, -10 * t / d) + 1) + b;
+}
+
+float inOutExpo(float t, float b, float c, float d)
+{
+   if (t == 0)
+      return b;
+   if (t == d)
+      return b + c;
+   t = t / d * 2;
+   if (t < 1)
+      return c / 2 * pow(2, 10 * (t - 1)) + b - c * 0.0005;
+   return c / 2 * 1.0005 * (-pow(2, -10 * (t - 1)) + 2) + b;
+}
+
+float outInExpo(float t, float b, float c, float d)
+{
+   if (t < d / 2)
+      return outExpo(t * 2, b, c / 2, d);
+   return inExpo((t * 2) - d, b + c / 2, c / 2, d);
+}
+
+// circ
+
+float inCirc(float t, float b, float c, float d)
+{
+   return(-c * (sqrt(1 - pow(t / d, 2)) - 1) + b);
+}
+
+float outCirc(float t, float b, float c, float d)
+{
+   return(c * sqrt(1 - pow(t / d - 1, 2)) + b);
+}
+
+float inOutCirc(float t, float b, float c, float d)
+{
+   t = t / d * 2;
+   if (t < 1)
+      return -c / 2 * (sqrt(1 - t * t) - 1) + b;
+   t = t - 2;
+   return c / 2 * (sqrt(1 - t * t) + 1) + b;
+}
+
+float outInCirc(float t, float b, float c, float d)
+{
+   if (t < d / 2)
+      return outCirc(t * 2, b, c / 2, d);
+   return inCirc((t * 2) - d, b + c / 2, c / 2, d);
+}
+
+// bounce
+
+float outBounce(float t, float b, float c, float d)
+{
+   t = t / d;
+   if (t < 1 / 2.75)
+      return c * (7.5625 * t * t) + b;
+   if (t < 2 / 2.75)
+   {
+      t = t - (1.5 / 2.75);
+      return c * (7.5625 * t * t + 0.75) + b;
+   }
+   else if (t < 2.5 / 2.75)
+   {
+      t = t - (2.25 / 2.75);
+      return c * (7.5625 * t * t + 0.9375) + b;
+   }
+   t = t - (2.625 / 2.75);
+   return c * (7.5625 * t * t + 0.984375) + b;
+}
+
+float inBounce(float t, float b, float c, float d)
+{
+   return c - outBounce(d - t, 0, c, d) + b;
+}
+
+float inOutBounce(float t, float b, float c, float d)
+{
+   if (t < d / 2)
+      return inBounce(t * 2, 0, c, d) * 0.5 + b;
+   return outBounce(t * 2 - d, 0, c, d) * 0.5 + c * .5 + b;
+}
+
+float outInBounce(float t, float b, float c, float d)
+{
+   if (t < d / 2)
+      return outBounce(t * 2, b, c / 2, d);
+   return inBounce((t * 2) - d, b + c / 2, c / 2, d);
+}
diff --git a/frontend/menu/disp/tween.h b/frontend/menu/disp/tween.h
new file mode 100644
index 0000000000..f4bfaeaf24
--- /dev/null
+++ b/frontend/menu/disp/tween.h
@@ -0,0 +1,78 @@
+/*  RetroArch - A frontend for libretro.
+ *  Copyright (C) 2010-2014 - Hans-Kristian Arntzen
+ *  Copyright (C) 2011-2014 - Daniel De Matteis
+ *  Copyright (C) 2014      - Jean-André Santoni
+ * 
+ *  RetroArch 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 Found-
+ *  ation, either version 3 of the License, or (at your option) any later version.
+ *
+ *  RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
+ *  without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
+ *  PURPOSE.  See the GNU General Public License for more details.
+ *
+ *  You should have received a copy of the GNU General Public License along with RetroArch.
+ *  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef _TWEEN_H
+#define _TWEEN_H
+
+#include <stdlib.h>
+
+typedef float (*easingFunc)(float, float, float, float);
+typedef void (*tweenCallback) (void);
+
+typedef struct
+{
+   int    alive;
+   float  duration;
+   float  running_since;
+   float  initial_value;
+   float  target_value;
+   float* subject;
+   easingFunc easing;
+   tweenCallback callback;
+} tween_t;
+
+void add_tween(float duration, float target_value, float* subject, easingFunc easing, tweenCallback callback);
+void update_tween(void *data, float dt);
+void update_tweens(float dt);
+
+// from https://github.com/kikito/tween.lua/blob/master/tween.lua
+
+float linear(float t, float b, float c, float d);
+float inQuad(float t, float b, float c, float d);
+float outQuad(float t, float b, float c, float d);
+float inOutQuad(float t, float b, float c, float d);
+float outInQuad(float t, float b, float c, float d);
+float inCubic(float t, float b, float c, float d);
+float outCubic(float t, float b, float c, float d);
+float inOutCubic(float t, float b, float c, float d);
+float outInCubic(float t, float b, float c, float d);
+float inQuart(float t, float b, float c, float d);
+float outQuart(float t, float b, float c, float d);
+float inOutQuart(float t, float b, float c, float d);
+float outInQuart(float t, float b, float c, float d);
+float inQuint(float t, float b, float c, float d);
+float outQuint(float t, float b, float c, float d);
+float inOutQuint(float t, float b, float c, float d);
+float outInQuint(float t, float b, float c, float d);
+float inSine(float t, float b, float c, float d);
+float outSine(float t, float b, float c, float d);
+float inOutSine(float t, float b, float c, float d);
+float outInSine(float t, float b, float c, float d);
+float inExpo(float t, float b, float c, float d);
+float outExpo(float t, float b, float c, float d);
+float inOutExpo(float t, float b, float c, float d);
+float outInExpo(float t, float b, float c, float d);
+float inCirc(float t, float b, float c, float d);
+float outCirc(float t, float b, float c, float d);
+float inOutCirc(float t, float b, float c, float d);
+float outInCirc(float t, float b, float c, float d);
+float inBounce(float t, float b, float c, float d);
+float outBounce(float t, float b, float c, float d);
+float inOutBounce(float t, float b, float c, float d);
+float outInBounce(float t, float b, float c, float d);
+
+#endif /* TWEEN_H */
diff --git a/frontend/menu/menu_common.c b/frontend/menu/menu_common.c
index 25e2a57ee1..06c0a15115 100644
--- a/frontend/menu/menu_common.c
+++ b/frontend/menu/menu_common.c
@@ -301,9 +301,10 @@ static unsigned input_frame(uint64_t trigger_state)
    return MENU_ACTION_NOOP;
 }
 
-bool menu_iterate(void)
+bool menu_iterate(retro_input_t input,
+      retro_input_t old_input, retro_input_t trigger_input)
 {
-   retro_input_t old_state = 0, trigger_input = 0;
+   retro_input_t old_state = 0;
    unsigned action = MENU_ACTION_NOOP;
    static bool initial_held = true;
    static bool first_held = false;
@@ -313,31 +314,20 @@ bool menu_iterate(void)
    if (!driver.menu)
       return false;
 
-
-   if (g_extern.lifecycle_state & (1ULL << MODE_MENU_PREINIT))
-   {
-      driver.menu->need_refresh = true;
-      rarch_main_set_state(RARCH_ACTION_STATE_MENU_PREINIT_FINISHED);
-   }
-
-   rarch_input_poll();
-
-   retro_input_t input = input_keys_pressed_func(RARCH_FIRST_META_KEY,
-         RARCH_BIND_LIST_END, &old_state);
-
-   trigger_input = input & ~old_state;
 #ifdef HAVE_OVERLAY
    if (BIND_PRESSED(trigger_input, RARCH_OVERLAY_NEXT))
          input_overlay_next(driver.overlay);
 #endif
    check_fullscreen_func(trigger_input);
 
-   if (check_quit_key_func(input) || !driver.video->alive(driver.video_data))
+   if (check_enter_menu_func(trigger_input) &&
+         g_extern.main_is_init && !g_extern.libretro_dummy)
    {
       rarch_main_command(RARCH_CMD_RESUME);
       return false;
    }
 
+   rarch_input_poll();
    input_state = menu_input();
 
    if (driver.menu->do_held)
diff --git a/frontend/menu/menu_common.h b/frontend/menu/menu_common.h
index 8f699d0d99..76ac35e6b3 100644
--- a/frontend/menu/menu_common.h
+++ b/frontend/menu/menu_common.h
@@ -132,7 +132,7 @@ typedef enum
 
 void *menu_init(const void *data);
 
-bool menu_iterate(void);
+bool menu_iterate(retro_input_t input, retro_input_t old_input, retro_input_t trigger_input);
 
 void menu_free(void *data);
 
diff --git a/general.c b/general.c
new file mode 100644
index 0000000000..6bf19b5f7c
--- /dev/null
+++ b/general.c
@@ -0,0 +1,210 @@
+/*  RetroArch - A frontend for libretro.
+ *  Copyright (C) 2010-2014 - Hans-Kristian Arntzen
+ *  Copyright (C) 2011-2014 - Daniel De Matteis
+ *  Copyright (C) 2012-2014 - Michael Lelli
+ *
+ *  RetroArch 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 Found-
+ *  ation, either version 3 of the License, or (at your option) any later version.
+ *
+ *  RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
+ *  without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
+ *  PURPOSE.  See the GNU General Public License for more details.
+ *
+ *  You should have received a copy of the GNU General Public License along with RetroArch.
+ *  If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "general.h"
+
+void rarch_playlist_push(content_playlist_t *playlist,
+      const char *path)
+{
+   char tmp[PATH_MAX];
+
+   if (!playlist || !g_extern.libretro_dummy)
+      return;
+
+   /* path can be relative here.
+    * Ensure we're pushing absolute path. */
+
+   strlcpy(tmp, path, sizeof(tmp));
+
+   if (*tmp)
+      path_resolve_realpath(tmp, sizeof(tmp));
+
+   if (g_extern.system.no_content || *tmp)
+      content_playlist_push(playlist,
+            *tmp ? tmp : NULL,
+            g_settings.libretro,
+            g_extern.system.info.library_name);
+}
+
+void rarch_playlist_load_content(content_playlist_t *playlist,
+      unsigned index)
+{
+   const char *path      = NULL;
+   const char *core_path = NULL;
+
+   content_playlist_get_index(playlist,
+         index, &path, &core_path, NULL);
+
+   strlcpy(g_settings.libretro, core_path, sizeof(g_settings.libretro));
+   rarch_environment_cb(RETRO_ENVIRONMENT_EXEC, (void*)path);
+
+   rarch_main_command(RARCH_CMD_LOAD_CORE);
+}
+
+void rarch_main_init_wrap(const struct rarch_main_wrap *args,
+      int *argc, char **argv)
+{
+   *argc = 0;
+   argv[(*argc)++] = strdup("retroarch");
+
+   if (!args->no_content)
+   {
+      if (args->content_path)
+      {
+         RARCH_LOG("Using content: %s.\n", args->content_path);
+         argv[(*argc)++] = strdup(args->content_path);
+      }
+      else
+      {
+         RARCH_LOG("No content, starting dummy core.\n");
+         argv[(*argc)++] = strdup("--menu");
+      }
+   }
+
+   if (args->sram_path)
+   {
+      argv[(*argc)++] = strdup("-s");
+      argv[(*argc)++] = strdup(args->sram_path);
+   }
+
+   if (args->state_path)
+   {
+      argv[(*argc)++] = strdup("-S");
+      argv[(*argc)++] = strdup(args->state_path);
+   }
+
+   if (args->config_path)
+   {
+      argv[(*argc)++] = strdup("-c");
+      argv[(*argc)++] = strdup(args->config_path);
+   }
+
+#ifdef HAVE_DYNAMIC
+   if (args->libretro_path)
+   {
+      argv[(*argc)++] = strdup("-L");
+      argv[(*argc)++] = strdup(args->libretro_path);
+   }
+#endif
+
+   if (args->verbose)
+      argv[(*argc)++] = strdup("-v");
+
+#ifdef HAVE_FILE_LOGGER
+   for (i = 0; i < *argc; i++)
+      RARCH_LOG("arg #%d: %s\n", i, argv[i]);
+#endif
+}
+
+/* When selection is presented back, returns 0.
+ * If it can make a decision right now, returns -1. */
+
+int rarch_defer_core(core_info_list_t *core_info, const char *dir,
+      const char *path, char *deferred_path, size_t sizeof_deferred_path)
+{
+   const core_info_t *info = NULL;
+   size_t supported = 0;
+
+   fill_pathname_join(deferred_path, dir, path, sizeof_deferred_path);
+
+   if (path_is_compressed_file(dir))
+   {
+      /* In case of a compressed archive, we have to join with a hash */
+      /* We are going to write at the position of dir: */
+      rarch_assert(strlen(dir) < strlen(deferred_path));
+      deferred_path[strlen(dir)] = '#';
+   }
+
+   if (core_info)
+      core_info_list_get_supported_cores(core_info, deferred_path, &info,
+            &supported);
+
+   /* Can make a decision right now. */
+   if (supported == 1)
+   {
+      strlcpy(g_extern.fullpath, deferred_path,
+            sizeof(g_extern.fullpath));
+      if (path_file_exists(info->path))
+         strlcpy(g_settings.libretro, info->path,
+               sizeof(g_settings.libretro));
+      return -1;
+   }
+   return 0;
+}
+
+/* Quite intrusive and error prone.
+ * Likely to have lots of small bugs.
+ * Cleanly exit the main loop to ensure that all the tiny details
+ * get set properly.
+ *
+ * This should mitigate most of the smaller bugs. */
+
+bool rarch_replace_config(const char *path)
+{
+   /* If config file to be replaced is the same as the 
+    * current config file, exit. */
+   if (!strcmp(path, g_extern.config_path))
+      return false;
+
+   if (g_settings.config_save_on_exit && *g_extern.config_path)
+      config_save_file(g_extern.config_path);
+
+   strlcpy(g_extern.config_path, path, sizeof(g_extern.config_path));
+   g_extern.block_config_read = false;
+   *g_settings.libretro = '\0'; /* Load core in new config. */
+
+   rarch_main_command(RARCH_CMD_PREPARE_DUMMY);
+
+   return true;
+}
+
+void rarch_update_system_info(struct retro_system_info *_info,
+      bool *load_no_content)
+{
+   const core_info_t *info = NULL;
+#if defined(HAVE_DYNAMIC)
+   libretro_free_system_info(_info);
+   if (!(*g_settings.libretro))
+      return;
+
+   libretro_get_system_info(g_settings.libretro, _info,
+         load_no_content);
+#endif
+   if (!g_extern.core_info)
+      return;
+
+   if (!core_info_list_get_info(g_extern.core_info,
+            g_extern.core_info_current, g_settings.libretro))
+      return;
+
+   /* Keep track of info for the currently selected core. */
+   info = (const core_info_t*)g_extern.core_info_current;
+
+   if (!g_extern.verbosity)
+      return;
+
+   RARCH_LOG("[Core Info]:\n");
+   if (info->display_name)
+      RARCH_LOG("Display Name = %s\n", info->display_name);
+   if (info->supported_extensions)
+      RARCH_LOG("Supported Extensions = %s\n",
+            info->supported_extensions);
+   if (info->authors)
+      RARCH_LOG("Authors = %s\n", info->authors);
+   if (info->permissions)
+      RARCH_LOG("Permissions = %s\n", info->permissions);
+}
diff --git a/general.h b/general.h
index 89bc64f6f8..af94efd87a 100644
--- a/general.h
+++ b/general.h
@@ -137,35 +137,27 @@ enum action_state
 {
    RARCH_ACTION_STATE_NONE = 0,
    RARCH_ACTION_STATE_MENU_PREINIT,
-   RARCH_ACTION_STATE_MENU_PREINIT_FINISHED,
    RARCH_ACTION_STATE_LOAD_CONTENT,
-   RARCH_ACTION_STATE_LOAD_CONTENT_FINISHED,
-   RARCH_ACTION_STATE_RUNNING,
-   RARCH_ACTION_STATE_RUNNING_FINISHED,
    RARCH_ACTION_STATE_MENU_RUNNING,
    RARCH_ACTION_STATE_MENU_RUNNING_FINISHED,
    RARCH_ACTION_STATE_EXITSPAWN,
    RARCH_ACTION_STATE_FLUSH_INPUT,
-   RARCH_ACTION_STATE_FLUSH_INPUT_FINISHED,
    RARCH_ACTION_STATE_QUIT,
    RARCH_ACTION_STATE_FORCE_QUIT,
 };
 
 enum menu_enums
 {
-   MODE_GAME = 0,
-   MODE_LOAD_GAME,
+   MODE_NONE = 0,
    MODE_MENU,
    MODE_MENU_WIDESCREEN,
    MODE_MENU_HD,
-   MODE_MENU_PREINIT,
    MODE_EXTLAUNCH_MULTIMAN,
    MODE_EXITSPAWN,
    MODE_EXITSPAWN_START_GAME,
    MODE_EXITSPAWN_MULTIMAN,
    MODE_OSK_ENTRY_SUCCESS,
    MODE_OSK_ENTRY_FAIL,
-   MODE_CLEAR_INPUT,
 };
 
 enum sound_mode_enums
@@ -266,6 +258,7 @@ struct settings
 
       bool allow_rotate;
       bool shared_context;
+      bool force_srgb_disable;
    } video;
 
 #ifdef HAVE_MENU
@@ -436,6 +429,7 @@ struct global
    bool location_active;
    bool osk_active;
    bool force_fullscreen;
+   bool core_shutdown_initiated;
 
    struct string_list *temporary_content;
 
@@ -761,6 +755,8 @@ struct global
 
    /* Config file associated with per-core configs. */
    char core_specific_config_path[PATH_MAX];
+
+   retro_keyboard_event_t frontend_key_event;
 };
 
 struct rarch_main_wrap
diff --git a/gfx/gl.c b/gfx/gl.c
index fe85f30775..51f64f03d4 100644
--- a/gfx/gl.c
+++ b/gfx/gl.c
@@ -516,6 +516,9 @@ static void gl_create_fbo_textures(gl_t *gl)
             RARCH_ERR("[GL]: sRGB FBO was requested, but it is not supported. Falling back to UNORM. Result may have banding!\n");
       }
 
+      if (g_settings.video.force_srgb_disable)
+         srgb_fbo = false;
+
    #ifndef HAVE_OPENGLES2
       if (fp_fbo && gl->has_fp_fbo)
       {
@@ -1874,6 +1877,9 @@ static bool resolve_extensions(gl_t *gl)
 #endif
 #endif
 
+   if (g_settings.video.force_srgb_disable)
+      gl->has_srgb_fbo = false;
+
 #ifdef GL_DEBUG
    // Useful for debugging, but kinda obnoxious otherwise.
    RARCH_LOG("[GL]: Supported extensions:\n");
diff --git a/gfx/shader_parse.c b/gfx/shader_parse.c
index fee5c2825b..765d7ba833 100644
--- a/gfx/shader_parse.c
+++ b/gfx/shader_parse.c
@@ -608,7 +608,7 @@ static void shader_write_variable(config_file_t *conf,
          config_set_hex(conf, wram_buf, info->addr);
          break;
 
-      default:
+      case RARCH_STATE_NONE:
          break;
    }
 }
diff --git a/griffin/griffin.c b/griffin/griffin.c
index c7951012b6..8c3cfe37b8 100644
--- a/griffin/griffin.c
+++ b/griffin/griffin.c
@@ -590,6 +590,7 @@ MAIN
 /*============================================================
 RETROARCH
 ============================================================ */
+#include "../general.c"
 #include "../retroarch.c"
 
 /*============================================================
@@ -662,6 +663,7 @@ MENU
 #ifdef HAVE_LAKKA
 #include "../frontend/menu/backend/menu_lakka_backend.c"
 #include "../frontend/menu/disp/lakka.c"
+#include "../frontend/menu/disp/tween.c"
 #endif
 
 #ifdef HAVE_GLUI
diff --git a/retroarch.c b/retroarch.c
index 421d75ce24..e69144058f 100644
--- a/retroarch.c
+++ b/retroarch.c
@@ -43,6 +43,7 @@
 
 #ifdef HAVE_MENU
 #include "frontend/menu/menu_common.h"
+#include "frontend/menu/menu_input_line_cb.h"
 #endif
 
 #ifdef _WIN32
@@ -2319,51 +2320,64 @@ static void check_slowmotion(bool pressed)
          "Slow motion rewind." : "Slow motion.", 0, 30);
 }
 
-static void check_movie_record(bool pressed)
+static bool check_movie_init(void)
 {
+   char path[PATH_MAX], msg[PATH_MAX];
+   bool ret = true;
+   
    if (g_extern.bsv.movie)
+      return false;
+
+   g_settings.rewind_granularity = 1;
+
+   if (g_settings.state_slot > 0)
    {
-      msg_queue_clear(g_extern.msg_queue);
-      msg_queue_push(g_extern.msg_queue,
-            RETRO_MSG_MOVIE_RECORD_STOPPING, 2, 180);
-      RARCH_LOG(RETRO_LOG_MOVIE_RECORD_STOPPING);
-      deinit_movie();
+      snprintf(path, sizeof(path), "%s%d.bsv",
+            g_extern.bsv.movie_path, g_settings.state_slot);
    }
    else
    {
-      char path[PATH_MAX], msg[PATH_MAX];
-
-      g_settings.rewind_granularity = 1;
-
-      if (g_settings.state_slot > 0)
-      {
-         snprintf(path, sizeof(path), "%s%d.bsv",
-               g_extern.bsv.movie_path, g_settings.state_slot);
-      }
-      else
-      {
-         snprintf(path, sizeof(path), "%s.bsv",
-               g_extern.bsv.movie_path);
-      }
-
-      snprintf(msg, sizeof(msg), "Starting movie record to \"%s\".", path);
-
-      g_extern.bsv.movie = bsv_movie_init(path, RARCH_MOVIE_RECORD);
-      msg_queue_clear(g_extern.msg_queue);
-      msg_queue_push(g_extern.msg_queue, g_extern.bsv.movie ?
-            msg : "Failed to start movie record.", 1, 180);
-
-      if (g_extern.bsv.movie)
-         RARCH_LOG("Starting movie record to \"%s\".\n", path);
-      else
-         RARCH_ERR("Failed to start movie record.\n");
+      snprintf(path, sizeof(path), "%s.bsv",
+            g_extern.bsv.movie_path);
    }
+
+   snprintf(msg, sizeof(msg), "Starting movie record to \"%s\".", path);
+
+   g_extern.bsv.movie = bsv_movie_init(path, RARCH_MOVIE_RECORD);
+
+   if (!g_extern.bsv.movie)
+      ret = false;
+
+   msg_queue_clear(g_extern.msg_queue);
+   msg_queue_push(g_extern.msg_queue, g_extern.bsv.movie ?
+         msg : "Failed to start movie record.", 1, 180);
+
+   if (g_extern.bsv.movie)
+      RARCH_LOG("Starting movie record to \"%s\".\n", path);
+   else
+      RARCH_ERR("Failed to start movie record.\n");
+
+   return ret;
 }
 
-static void check_movie_playback(bool pressed)
+static bool check_movie_record(void)
+{
+   if (!g_extern.bsv.movie)
+      return false;
+
+   msg_queue_clear(g_extern.msg_queue);
+   msg_queue_push(g_extern.msg_queue,
+         RETRO_MSG_MOVIE_RECORD_STOPPING, 2, 180);
+   RARCH_LOG(RETRO_LOG_MOVIE_RECORD_STOPPING);
+   deinit_movie();
+
+   return true;
+}
+
+static bool check_movie_playback(void)
 {
    if (!g_extern.bsv.movie_end)
-      return;
+      return false;
 
    msg_queue_push(g_extern.msg_queue,
          RETRO_MSG_MOVIE_PLAYBACK_ENDED, 1, 180);
@@ -2372,19 +2386,20 @@ static void check_movie_playback(bool pressed)
    deinit_movie();
    g_extern.bsv.movie_end = false;
    g_extern.bsv.movie_playback = false;
+
+   return true;
 }
 
-static void check_movie(void)
+static bool check_movie(void)
 {
    if (g_extern.bsv.movie_playback)
-      check_movie_playback(true);
-   else
-      check_movie_record(true);
+      return check_movie_playback();
+   if (!g_extern.bsv.movie)
+      return check_movie_init();
+   return check_movie_record();
 }
 
-static void check_pause(
-      bool new_state, bool old_state,
-      bool frameadvance_pressed)
+static void check_pause(bool pressed, bool frameadvance_pressed)
 {
    static bool old_focus    = true;
    bool focus               = true;
@@ -2392,12 +2407,12 @@ static void check_pause(
    bool has_set_audio_start = false;
 
    /* FRAMEADVANCE will set us into pause mode. */
-   new_state |= !g_extern.is_paused && frameadvance_pressed;
+   pressed |= !g_extern.is_paused && frameadvance_pressed;
 
    if (g_settings.pause_nonactive)
       focus = driver.video->focus(driver.video_data);
 
-   if (focus && new_state && !old_state)
+   if (focus && pressed)
    {
       g_extern.is_paused = !g_extern.is_paused;
 
@@ -2486,6 +2501,9 @@ static void check_turbo(void)
 
 static void check_shader_dir(bool pressed_next, bool pressed_prev)
 {
+   char msg[PATH_MAX];
+   const char *shader = NULL, *ext = NULL;
+   enum rarch_shader_type type = RARCH_SHADER_NONE;
    bool should_apply = false;
 
    if (!g_extern.shader_dir.list || !driver.video->set_shader)
@@ -2506,13 +2524,12 @@ static void check_shader_dir(bool pressed_next, bool pressed_prev)
          g_extern.shader_dir.ptr--;
    }
 
-   if (should_apply)
+   if (!should_apply)
+      return;
+
    {
-      char msg[512];
-      const char *shader          = 
-         g_extern.shader_dir.list->elems[g_extern.shader_dir.ptr].data;
-      enum rarch_shader_type type = RARCH_SHADER_NONE;
-      const char *ext             = path_get_extension(shader);
+      shader = g_extern.shader_dir.list->elems[g_extern.shader_dir.ptr].data;
+      ext    = path_get_extension(shader);
 
       if (strcmp(ext, "glsl") == 0 || strcmp(ext, "glslp") == 0)
          type = RARCH_SHADER_GLSL;
@@ -2536,7 +2553,7 @@ static void check_shader_dir(bool pressed_next, bool pressed_prev)
 
 void rarch_disk_control_append_image(const char *path)
 {
-   char msg[512];
+   char msg[PATH_MAX];
    unsigned new_index;
    const struct retro_disk_control_callback *control = 
       (const struct retro_disk_control_callback*)&g_extern.system.disk_control;
@@ -2583,7 +2600,7 @@ void rarch_disk_control_append_image(const char *path)
 
 void rarch_disk_control_set_eject(bool new_state, bool log)
 {
-   char msg[256];
+   char msg[PATH_MAX];
    const struct retro_disk_control_callback *control = 
       (const struct retro_disk_control_callback*)&g_extern.system.disk_control;
    bool error = false;
@@ -2621,7 +2638,7 @@ void rarch_disk_control_set_eject(bool new_state, bool log)
 
 void rarch_disk_control_set_index(unsigned next_index)
 {
-   char msg[256];
+   char msg[PATH_MAX];
    unsigned num_disks;
    const struct retro_disk_control_callback *control = 
       (const struct retro_disk_control_callback*)&g_extern.system.disk_control;
@@ -2779,7 +2796,10 @@ static void check_disk_next(
       RARCH_ERR("Got invalid disk index from libretro.\n");
 }
 
-static void do_state_checks(
+/* Checks for stuff like fullscreen, save states, etc.
+ * Return false when RetroArch is paused. */
+
+static bool do_state_checks(
       retro_input_t input, retro_input_t old_input,
       retro_input_t trigger_input)
 {
@@ -2814,10 +2834,10 @@ static void do_state_checks(
    if (g_extern.netplay)
    {
       check_netplay_flip_func(trigger_input);
-      return;
+      return true;
    }
 #endif
-   check_pause_func(input, old_input);
+   check_pause_func(trigger_input);
 
    check_oneshot_func(trigger_input);
 
@@ -2825,7 +2845,7 @@ static void do_state_checks(
       rarch_render_cached_frame();
 
    if (g_extern.is_paused && !g_extern.is_oneshot)
-      return;
+      return false;
 
    check_fast_forward_button_func(input, old_input, trigger_input);
 
@@ -2872,6 +2892,8 @@ static void do_state_checks(
 
    if (BIND_PRESSED(trigger_input, RARCH_RESET))
       rarch_main_command(RARCH_CMD_RESET);
+
+   return true;
 }
 
 static void init_state(void)
@@ -2951,7 +2973,8 @@ static void verify_api_version(void)
 }
 
 /* Make sure we haven't compiled for something we cannot run.
- * Ideally, code would get swapped out depending on CPU support, but this will do for now.
+ * Ideally, code would get swapped out depending on CPU support, 
+ * but this will do for now.
  */
 static void validate_cpu_features(void)
 {
@@ -3096,21 +3119,6 @@ error:
    return 1;
 }
 
-static bool check_enter_menu(bool pressed, bool old_pressed)
-{
-   bool rmenu_toggle = pressed || (g_extern.libretro_dummy && !old_pressed);
-
-   if (rmenu_toggle && !old_pressed)
-   {
-      /* Always go into menu if dummy core is loaded. */
-      rarch_main_set_state(RARCH_ACTION_STATE_MENU_PREINIT);
-      g_extern.system.frame_time_last = 0;
-      return true;
-   }
-
-   return false;
-}
-
 static inline void update_frame_time(void)
 {
    retro_time_t time = 0;
@@ -3169,22 +3177,47 @@ void rarch_main_set_state(unsigned cmd)
    switch (cmd)
    {
       case RARCH_ACTION_STATE_MENU_PREINIT:
-         g_extern.lifecycle_state |= (1ULL << MODE_MENU_PREINIT);
-         break;
-      case RARCH_ACTION_STATE_MENU_PREINIT_FINISHED:
-         g_extern.lifecycle_state &= ~(1ULL << MODE_MENU_PREINIT);
+         {
+            int i;
+
+            /* Menu should always run with vsync on. */
+            rarch_main_command(RARCH_CMD_VIDEO_SET_BLOCKING_STATE);
+
+            /* Stop all rumbling before entering the menu. */
+            for (i = 0; i < MAX_PLAYERS; i++)
+            {
+               driver_set_rumble_state(i, RETRO_RUMBLE_STRONG, 0);
+               driver_set_rumble_state(i, RETRO_RUMBLE_WEAK, 0);
+            }
+
+            rarch_main_command(RARCH_CMD_AUDIO_STOP);
+
+#ifdef HAVE_MENU
+            if (driver.menu)
+            {
+               /* Override keyboard callback to redirect to menu instead.
+                * We'll use this later for something ...
+                * FIXME: This should probably be moved to menu_common somehow. */
+               g_extern.frontend_key_event = g_extern.system.key_event;
+               g_extern.system.key_event = menu_key_event;
+
+               driver.menu->need_refresh = true;
+               driver.menu->old_input_state |= 1ULL << RARCH_MENU_TOGGLE;
+               rarch_main_set_state(RARCH_ACTION_STATE_MENU_RUNNING);
+            }
+#endif
+            g_extern.system.frame_time_last = 0;
+         }
          break;
       case RARCH_ACTION_STATE_LOAD_CONTENT:
-         g_extern.lifecycle_state |= (1ULL << MODE_LOAD_GAME);
-         break;
-      case RARCH_ACTION_STATE_LOAD_CONTENT_FINISHED:
-         g_extern.lifecycle_state &= ~(1ULL << MODE_LOAD_GAME);
-         break;
-      case RARCH_ACTION_STATE_RUNNING:
-         g_extern.lifecycle_state |= (1ULL << MODE_GAME);
-         break;
-      case RARCH_ACTION_STATE_RUNNING_FINISHED:
-         g_extern.lifecycle_state &= ~(1ULL << MODE_GAME);
+#ifdef HAVE_MENU
+         if (!load_menu_content())
+         {
+            /* If content loading fails, we go back to menu. */
+            if (driver.menu)
+               rarch_main_set_state(RARCH_ACTION_STATE_MENU_PREINIT);
+         }
+#endif
          break;
       case RARCH_ACTION_STATE_MENU_RUNNING:
          g_extern.lifecycle_state |= (1ULL << MODE_MENU);
@@ -3196,18 +3229,20 @@ void rarch_main_set_state(unsigned cmd)
          g_extern.lifecycle_state |= (1ULL << MODE_EXITSPAWN);
          break;
       case RARCH_ACTION_STATE_QUIT:
+         g_extern.system.shutdown = true;
          rarch_main_set_state(RARCH_ACTION_STATE_MENU_RUNNING_FINISHED);
-         rarch_main_set_state(RARCH_ACTION_STATE_RUNNING_FINISHED);
          break;
       case RARCH_ACTION_STATE_FORCE_QUIT:
          g_extern.lifecycle_state = 0;
          rarch_main_set_state(RARCH_ACTION_STATE_QUIT);
          break;
       case RARCH_ACTION_STATE_FLUSH_INPUT:
-         g_extern.lifecycle_state |= (1ULL << MODE_CLEAR_INPUT);
-         break;
-      case RARCH_ACTION_STATE_FLUSH_INPUT_FINISHED:
-         g_extern.lifecycle_state &= ~(1ULL << MODE_CLEAR_INPUT);
+         rarch_input_poll();
+#ifdef HAVE_MENU
+         menu_input();
+#endif
+         /* Restore libretro keyboard callback. */
+         g_extern.system.key_event = g_extern.frontend_key_event;
          break;
       case RARCH_ACTION_STATE_NONE:
       default:
@@ -3238,11 +3273,13 @@ void rarch_main_command(unsigned cmd)
       case RARCH_CMD_LOAD_CORE:
 #ifdef HAVE_MENU
          if (driver.menu)
-            rarch_update_system_info(&g_extern.menu.info, &driver.menu->load_no_content);
+            rarch_update_system_info(&g_extern.menu.info,
+                  &driver.menu->load_no_content);
 #endif
          break;
       case RARCH_CMD_LOAD_STATE:
-         /* Disallow savestate load when we absolutely cannot change game state. */
+         /* Disallow savestate load when we absolutely 
+          * cannot change game state. */
          if (g_extern.bsv.movie)
             return;
 
@@ -3251,7 +3288,6 @@ void rarch_main_command(unsigned cmd)
             return;
 #endif
          main_state(cmd);
-         rarch_main_set_state(RARCH_ACTION_STATE_RUNNING);
          break;
       case RARCH_CMD_RESET:
          RARCH_LOG(RETRO_LOG_RESETTING_CONTENT);
@@ -3261,14 +3297,12 @@ void rarch_main_command(unsigned cmd)
          /* bSNES since v073r01 resets controllers to JOYPAD
           * after a reset, so just enforce it here. */
          init_controllers();
-         rarch_main_set_state(RARCH_ACTION_STATE_RUNNING);
          break;
       case RARCH_CMD_SAVE_STATE:
          if (g_settings.savestate_auto_index)
             g_settings.state_slot++;
 
          main_state(cmd);
-         rarch_main_set_state(RARCH_ACTION_STATE_RUNNING);
          break;
       case RARCH_CMD_TAKE_SCREENSHOT:
          take_screenshot();
@@ -3436,7 +3470,6 @@ void rarch_main_command(unsigned cmd)
 #ifdef HAVE_MENU
          rarch_main_set_state(RARCH_ACTION_STATE_MENU_RUNNING_FINISHED);
 #endif
-         rarch_main_set_state(RARCH_ACTION_STATE_RUNNING);
          break;
       case RARCH_CMD_RESTART_RETROARCH:
 #if defined(GEKKO) && defined(HW_RVL)
@@ -3444,7 +3477,6 @@ void rarch_main_command(unsigned cmd)
                SALAMANDER_FILE,
                sizeof(g_extern.fullpath));
 #endif
-         rarch_main_set_state(RARCH_ACTION_STATE_RUNNING_FINISHED);
          rarch_main_set_state(RARCH_ACTION_STATE_EXITSPAWN);
          break;
       case RARCH_CMD_MENU_SAVE_CONFIG:
@@ -3581,16 +3613,32 @@ bool rarch_main_iterate(void)
 
    trigger_input = input & ~old_input;
 
-   /* SHUTDOWN on consoles should exit RetroArch completely. */
-   if (g_extern.system.shutdown)
-      return false;
-
    /* Time to drop? */
-   if (check_quit_key_func(input) || !driver.video->alive(driver.video_data))
+   if (
+         g_extern.system.shutdown ||
+         check_quit_key_func(input) ||
+         !driver.video->alive(driver.video_data))
       return false;
 
-   if (check_enter_menu_func(input, old_input))
-      return false; /* Enter menu, don't exit. */
+   if (g_extern.lifecycle_state & (1ULL << MODE_MENU))
+   {
+      if (!menu_iterate(input, old_input, trigger_input))
+      {
+         rarch_main_set_state(RARCH_ACTION_STATE_MENU_RUNNING_FINISHED);
+         driver_set_nonblock_state(driver.nonblock_state);
+
+         rarch_main_command(RARCH_CMD_AUDIO_START);
+         rarch_main_set_state(RARCH_ACTION_STATE_FLUSH_INPUT);
+      }
+      return true;
+   }
+
+   if (check_enter_menu_func(trigger_input) || (g_extern.libretro_dummy))
+   {
+      /* Always go into menu if dummy core is loaded. */
+      rarch_main_set_state(RARCH_ACTION_STATE_MENU_PREINIT);
+      return true; /* Enter menu on next run. */
+   }
 
    if (g_extern.exec)
    {
@@ -3598,10 +3646,7 @@ bool rarch_main_iterate(void)
       return false;
    }
 
-   /* Checks for stuff like fullscreen, save states, etc. */
-   do_state_checks(input, old_input, trigger_input);
-
-   if (g_extern.is_paused && !g_extern.is_oneshot)
+   if (!do_state_checks(input, old_input, trigger_input))
    {
       rarch_input_poll();
       rarch_sleep(10);
@@ -3723,195 +3768,3 @@ void rarch_main_deinit(void)
 
    g_extern.main_is_init = false;
 }
-
-void rarch_playlist_push(content_playlist_t *playlist,
-      const char *path)
-{
-   char tmp[PATH_MAX];
-
-   if (!playlist || !g_extern.libretro_dummy)
-      return;
-
-   /* path can be relative here.
-    * Ensure we're pushing absolute path. */
-
-   strlcpy(tmp, path, sizeof(tmp));
-
-   if (*tmp)
-      path_resolve_realpath(tmp, sizeof(tmp));
-
-   if (g_extern.system.no_content || *tmp)
-      content_playlist_push(playlist,
-            *tmp ? tmp : NULL,
-            g_settings.libretro,
-            g_extern.system.info.library_name);
-}
-
-void rarch_playlist_load_content(content_playlist_t *playlist,
-      unsigned index)
-{
-   const char *path      = NULL;
-   const char *core_path = NULL;
-
-   content_playlist_get_index(playlist,
-         index, &path, &core_path, NULL);
-
-   strlcpy(g_settings.libretro, core_path, sizeof(g_settings.libretro));
-   rarch_environment_cb(RETRO_ENVIRONMENT_EXEC, (void*)path);
-
-   rarch_main_command(RARCH_CMD_LOAD_CORE);
-}
-
-void rarch_main_init_wrap(const struct rarch_main_wrap *args,
-      int *argc, char **argv)
-{
-   *argc = 0;
-   argv[(*argc)++] = strdup("retroarch");
-
-   if (!args->no_content)
-   {
-      if (args->content_path)
-      {
-         RARCH_LOG("Using content: %s.\n", args->content_path);
-         argv[(*argc)++] = strdup(args->content_path);
-      }
-      else
-      {
-         RARCH_LOG("No content, starting dummy core.\n");
-         argv[(*argc)++] = strdup("--menu");
-      }
-   }
-
-   if (args->sram_path)
-   {
-      argv[(*argc)++] = strdup("-s");
-      argv[(*argc)++] = strdup(args->sram_path);
-   }
-
-   if (args->state_path)
-   {
-      argv[(*argc)++] = strdup("-S");
-      argv[(*argc)++] = strdup(args->state_path);
-   }
-
-   if (args->config_path)
-   {
-      argv[(*argc)++] = strdup("-c");
-      argv[(*argc)++] = strdup(args->config_path);
-   }
-
-#ifdef HAVE_DYNAMIC
-   if (args->libretro_path)
-   {
-      argv[(*argc)++] = strdup("-L");
-      argv[(*argc)++] = strdup(args->libretro_path);
-   }
-#endif
-
-   if (args->verbose)
-      argv[(*argc)++] = strdup("-v");
-
-#ifdef HAVE_FILE_LOGGER
-   for (i = 0; i < *argc; i++)
-      RARCH_LOG("arg #%d: %s\n", i, argv[i]);
-#endif
-}
-
-/* When selection is presented back, returns 0.
- * If it can make a decision right now, returns -1. */
-
-int rarch_defer_core(core_info_list_t *core_info, const char *dir,
-      const char *path, char *deferred_path, size_t sizeof_deferred_path)
-{
-   const core_info_t *info = NULL;
-   size_t supported = 0;
-
-   fill_pathname_join(deferred_path, dir, path, sizeof_deferred_path);
-
-   if (path_is_compressed_file(dir))
-   {
-      /* In case of a compressed archive, we have to join with a hash */
-      /* We are going to write at the position of dir: */
-      rarch_assert(strlen(dir) < strlen(deferred_path));
-      deferred_path[strlen(dir)] = '#';
-   }
-
-   if (core_info)
-      core_info_list_get_supported_cores(core_info, deferred_path, &info,
-            &supported);
-
-   /* Can make a decision right now. */
-   if (supported == 1)
-   {
-      strlcpy(g_extern.fullpath, deferred_path,
-            sizeof(g_extern.fullpath));
-
-      if (path_file_exists(info->path))
-         strlcpy(g_settings.libretro, info->path,
-               sizeof(g_settings.libretro));
-      return -1;
-   }
-   return 0;
-}
-
-/* Quite intrusive and error prone.
- * Likely to have lots of small bugs.
- * Cleanly exit the main loop to ensure that all the tiny details
- * get set properly.
- *
- * This should mitigate most of the smaller bugs. */
-
-bool rarch_replace_config(const char *path)
-{
-   /* If config file to be replaced is the same as the 
-    * current config file, exit. */
-   if (!strcmp(path, g_extern.config_path))
-      return false;
-
-   if (g_settings.config_save_on_exit && *g_extern.config_path)
-      config_save_file(g_extern.config_path);
-
-   strlcpy(g_extern.config_path, path, sizeof(g_extern.config_path));
-   g_extern.block_config_read = false;
-   *g_settings.libretro = '\0'; /* Load core in new config. */
-
-   rarch_main_command(RARCH_CMD_PREPARE_DUMMY);
-
-   return true;
-}
-
-void rarch_update_system_info(struct retro_system_info *_info, bool *load_no_content)
-{
-   const core_info_t *info = NULL;
-#if defined(HAVE_DYNAMIC)
-   libretro_free_system_info(_info);
-   if (!(*g_settings.libretro))
-      return;
-
-   libretro_get_system_info(g_settings.libretro, _info,
-         load_no_content);
-#endif
-   if (!g_extern.core_info)
-      return;
-
-   if (!core_info_list_get_info(g_extern.core_info,
-            g_extern.core_info_current, g_settings.libretro))
-      return;
-
-   /* Keep track of info for the currently selected core. */
-   info = (const core_info_t*)g_extern.core_info_current;
-
-   if (!g_extern.verbosity)
-      return;
-
-   RARCH_LOG("[Core Info]:\n");
-   if (info->display_name)
-      RARCH_LOG("Display Name = %s\n", info->display_name);
-   if (info->supported_extensions)
-      RARCH_LOG("Supported Extensions = %s\n",
-            info->supported_extensions);
-   if (info->authors)
-      RARCH_LOG("Authors = %s\n", info->authors);
-   if (info->permissions)
-      RARCH_LOG("Permissions = %s\n", info->permissions);
-}
diff --git a/retroarch.cfg b/retroarch.cfg
index a8019d5269..dc772521e3 100644
--- a/retroarch.cfg
+++ b/retroarch.cfg
@@ -124,6 +124,10 @@
 # Video vsync.
 # video_vsync = true
 
+# Forcibly disable sRGB FBO support. Some Intel OpenGL drivers on Windows
+# have video problems with sRGB FBO support enabled.
+# video_force_srgb_disable = false
+
 # Attempts to hard-synchronize CPU and GPU. Can reduce latency at cost of performance.
 # video_hard_sync = false
 
diff --git a/settings.c b/settings.c
index e1f7f00b12..cfa1ecb5b3 100644
--- a/settings.c
+++ b/settings.c
@@ -306,6 +306,7 @@ void config_set_defaults(void)
       g_settings.video.threaded = g_defaults.settings.video_threaded_enable;
 
    g_settings.video.shared_context = video_shared_context;
+   g_settings.video.force_srgb_disable = false;
 #ifdef GEKKO
    g_settings.video.viwidth = video_viwidth;
 #endif
@@ -430,6 +431,7 @@ void config_set_defaults(void)
    g_extern.console.screen.viewports.custom_vp.x = 0;
    g_extern.console.screen.viewports.custom_vp.y = 0;
 
+
    /* Make sure settings from other configs carry over into defaults 
     * for another config. */
    if (!g_extern.has_set_save_path)
@@ -930,6 +932,8 @@ bool config_load_file(const char *path, bool set_defaults)
    CONFIG_GET_FLOAT(video.msg_pos_y, "video_message_pos_y");
    CONFIG_GET_INT(video.rotation, "video_rotation");
 
+   CONFIG_GET_BOOL(video.force_srgb_disable, "video_force_srgb_disable");
+
 #ifdef RARCH_CONSOLE
    /* TODO - will be refactored later to make it more clean - it's more 
     * important that it works for consoles right now */
@@ -1451,6 +1455,8 @@ bool config_save_file(const char *path)
    config_set_bool(conf,  "video_threaded", g_settings.video.threaded);
    config_set_bool(conf,  "video_shared_context",
          g_settings.video.shared_context);
+   config_set_bool(conf,  "video_force_srgb_disable",
+         g_settings.video.force_srgb_disable);
    config_set_bool(conf,  "video_fullscreen", g_settings.video.fullscreen);
    config_set_float(conf, "video_refresh_rate", g_settings.video.refresh_rate);
    config_set_int(conf,   "video_monitor_index",
diff --git a/settings_data.c b/settings_data.c
index 4660a472f8..0300806e0a 100644
--- a/settings_data.c
+++ b/settings_data.c
@@ -2413,6 +2413,7 @@ rarch_setting_t *setting_data_get_list(void)
       CONFIG_UINT(g_settings.video.fullscreen_y,         "video_fullscreen_y",         "Fullscreen Height",          fullscreen_y, GROUP_NAME, SUBGROUP_NAME, general_write_handler, general_read_handler)
       CONFIG_FLOAT(g_settings.video.refresh_rate,        "video_refresh_rate",         "Refresh Rate",               refresh_rate, "%.3f Hz", GROUP_NAME, SUBGROUP_NAME, general_write_handler, general_read_handler) WITH_RANGE(0, 0, 0.001, true, false)
       CONFIG_FLOAT(g_settings.video.refresh_rate,        "video_refresh_rate_auto",    "Estimated Monitor FPS",      refresh_rate, "%.3f Hz", GROUP_NAME, SUBGROUP_NAME, general_write_handler, general_read_handler)
+      CONFIG_BOOL(g_settings.video.force_srgb_disable,     "video_force_srgb_disable",   "Force-disable sRGB FBO",     false, "OFF", "ON", GROUP_NAME, SUBGROUP_NAME, general_write_handler, general_read_handler) WITH_CMD(RARCH_CMD_REINIT) WITH_FLAGS(SD_FLAG_CMD_APPLY_AUTO)
       END_SUB_GROUP()
 
       START_SUB_GROUP("Aspect", GROUP_NAME)