# Copyright 2016 Google Inc. All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""Starboard Linux X64 X11 gcc 4.2 platform configuration for gyp_cobalt."""

import logging
import os
import subprocess
import sys

# Import the shared Linux platform configuration.
sys.path.append(
    os.path.realpath(
        os.path.join(
            os.path.dirname(__file__), os.pardir, os.pardir, os.pardir,
            'shared')))
# pylint: disable=import-self,g-import-not-at-top
import gyp_configuration as shared_configuration
import gyp_utils


class PlatformConfig(shared_configuration.PlatformConfig):
  """Starboard Linux platform configuration."""

  def __init__(self, platform, asan_enabled_by_default=False):
    super(PlatformConfig, self).__init__(
        platform, asan_enabled_by_default, goma_supports_compiler=False)

    # Run the script that ensures gcc 4.2.4 is installed.
    script_path = os.path.dirname(os.path.realpath(__file__))
    subprocess.call(
        os.path.join(script_path, 'download_gcc.sh'), cwd=script_path)

  def GetVariables(self, configuration):
    variables = super(PlatformConfig, self).GetVariables(configuration)
    variables.update({'clang': 0,})
    return variables

  def GetEnvironmentVariables(self):
    env_variables = super(PlatformConfig, self).GetEnvironmentVariables()
    toolchain_bin_dir = os.path.join(gyp_utils.GetToolchainsDir(),
                                     'x86_64-linux-gnu-gcc-4.2.4', 'gcc', 'bin')
    env_variables.update({
        'LIBRARY_PATH':
            '/usr/lib/x86_64-linux-gnu',
        'CC':
            os.path.join(toolchain_bin_dir, 'gcc'),
        'CXX':
            'LIBRARY_PATH=/usr/lib/x86_64-linux-gnu ' + os.path.join(
                toolchain_bin_dir, 'g++'),
    })
    return env_variables


def CreatePlatformConfig():
  try:
    return PlatformConfig('linux-x64x11-gcc-4-2')
  except RuntimeError as e:
    logging.critical(e)
    return None
